Skip to content

Commit

Permalink
Implement precise square root calculation for fixed point
Browse files Browse the repository at this point in the history
The original fixed point sqrt can not calculate sqrt of numbers
less than 1. Implement a new sqrt calculation that is more
precise and without fixed point multiplication.
  • Loading branch information
marvin0102 committed Jul 24, 2024
1 parent 2d85c45 commit c521fcb
Show file tree
Hide file tree
Showing 2 changed files with 51 additions and 31 deletions.
78 changes: 49 additions & 29 deletions src/fixed.c
Original file line number Diff line number Diff line change
Expand Up @@ -12,39 +12,59 @@

twin_fixed_t twin_fixed_sqrt(twin_fixed_t a)
{
twin_fixed_t max = a, min = 0;

while (max > min) {
twin_fixed_t mid = (max + min) >> 1;
if (mid >= 181 * TWIN_FIXED_ONE) {
max = mid - 1;
continue;
}
twin_fixed_t sqr = twin_fixed_mul(mid, mid);
if (sqr == a)
return mid;
if (sqr < a)
min = mid + 1;
else
max = mid - 1;
if (a <= 0)
return 0;
if (a < TWIN_FIXED_ONE + (1 << (8 - 1)) &&
a > TWIN_FIXED_ONE - (1 << (8 - 1)))
return TWIN_FIXED_ONE;

int offset = 0;
for (twin_fixed_t i = a; !(0x40000000 & i); i <<= 1) {
++offset;
}
return (max + min) >> 1;
offset = (offset & ~1);
a <<= offset;

offset >>= 1;
offset -= (16 >> 1);

twin_fixed_t z = 0;
for (twin_fixed_t m = 1UL << ((31 - __builtin_clz(a)) & ~1UL); m; m >>= 2) {
int b = z + m;
z >>= 1;
if (a >= b)
a -= b, z += m;
}

return (offset >= 0) ? z >> offset : z << (-offset);
}

twin_sfixed_t _twin_sfixed_sqrt(twin_sfixed_t as)
{
twin_dfixed_t max = as, min = 0;
twin_dfixed_t a = twin_sfixed_to_dfixed(as);

while (max > min) {
twin_dfixed_t mid = (max + min) >> 1;
twin_dfixed_t sqr = mid * mid;
if (sqr == a)
return (twin_sfixed_t) mid;
if (sqr < a)
min = mid + 1;
else
max = mid - 1;
if (as <= 0)
return 0;
if (as < TWIN_SFIXED_ONE + (1 << (2 - 1)) &&
as > TWIN_SFIXED_ONE - (1 << (2 - 1)))
return TWIN_SFIXED_ONE;

int offset = 0;
for (twin_sfixed_t i = as; !(0x4000 & i); i <<= 1) {
++offset;
}
return (twin_sfixed_t) ((max + min) >> 1);
offset = (offset & ~1);
as <<= offset;

offset >>= 1;
offset -= (4 >> 1);

twin_sfixed_t z = 0;
for (twin_sfixed_t m = 1UL << ((31 - __builtin_clz(as)) & ~1UL); m;
m >>= 2) {
int16_t b = z + m;
z >>= 1;
if (as >= b)
as -= b, z += m;
}

return (offset >= 0) ? z >> offset : z << (-offset);
}
4 changes: 2 additions & 2 deletions src/image-jpeg.c
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
/*
/*
* Twin - A Tiny Window System
* Copyright (c) 2024 National Cheng Kung University, Taiwan
* All rights reserved.
*/

#include <setjmp.h>
#include <stdio.h>
#include <stdint.h>
#include <stdio.h>

#include <jpeglib.h>

Expand Down

0 comments on commit c521fcb

Please sign in to comment.