123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- (*
- coranac.com's awesome atan2 implementation
- Very fast and very small.
- *)
- unit cearn_atan;
- {$mode objfpc}
- {$H+}
- interface
- uses
- ctypes, nds9;
-
- const
- __qran_seed: cint = 42;
- function atan2Lookup(x, y: cint): cuint32;
- function atan2Lerp(x, y: cint): cuint32;
- function sqran(seed: cint): cint;
- function qran(): cint; inline;
- function qran_range(min, max: cint): cint; inline;
- const
- BRAD_PI_SHIFT = 14;
- BRAD_PI = 1 shl BRAD_PI_SHIFT;
- BRAD_HPI = BRAD_PI div 2;
- BRAD_2PI = BRAD_PI * 2;
- ATAN_ONE = $1000;
- ATAN_FP = 12;
- // Some constants for dealing with atanLUT.
- ATANLUT_STRIDE = ATAN_ONE div $80;
- ATANLUT_STRIDE_SHIFT = 5;
- // Arctangents LUT. Interval: [0, 1] (one=128); PI=0x20000
- atanLUT: array [0..130-1] of cushort = (
- $0000, $0146, $028C, $03D2, $0517, $065D, $07A2, $08E7,
- $0A2C, $0B71, $0CB5, $0DF9, $0F3C, $107F, $11C1, $1303,
- $1444, $1585, $16C5, $1804, $1943, $1A80, $1BBD, $1CFA,
- $1E35, $1F6F, $20A9, $21E1, $2319, $2450, $2585, $26BA,
- $27ED, $291F, $2A50, $2B80, $2CAF, $2DDC, $2F08, $3033,
- $315D, $3285, $33AC, $34D2, $35F6, $3719, $383A, $395A,
- $3A78, $3B95, $3CB1, $3DCB, $3EE4, $3FFB, $4110, $4224,
- $4336, $4447, $4556, $4664, $4770, $487A, $4983, $4A8B,
- // 64
- $4B90, $4C94, $4D96, $4E97, $4F96, $5093, $518F, $5289,
- $5382, $5478, $556E, $5661, $5753, $5843, $5932, $5A1E,
- $5B0A, $5BF3, $5CDB, $5DC1, $5EA6, $5F89, $606A, $614A,
- $6228, $6305, $63E0, $64B9, $6591, $6667, $673B, $680E,
- $68E0, $69B0, $6A7E, $6B4B, $6C16, $6CDF, $6DA8, $6E6E,
- $6F33, $6FF7, $70B9, $717A, $7239, $72F6, $73B3, $746D,
- $7527, $75DF, $7695, $774A, $77FE, $78B0, $7961, $7A10,
- $7ABF, $7B6B, $7C17, $7CC1, $7D6A, $7E11, $7EB7, $7F5C,
- // 128
- $8000, $80A2);
- implementation
- // Quick (and very dirty) pseudo-random number generator
- // return random in range [0,8000h>
- function qran(): cint; inline;
- begin
- __qran_seed := 1664525 * __qran_seed + 1013904223;
- result := (__qran_seed shr 16) and $7FFF;
- end;
- function qran_range(min, max: cint): cint; inline;
- begin
- result := (qran() * (max - min) shr 15) + min;
- end;
- // Get the octant a coordinate pair is in.
- procedure OCTANTIFY(var _x, _y, _o: cint); inline;
- var
- _t: cint;
- begin
- repeat
- _o := 0;
- if (_y < 0) then
- begin
- _x := -_x;
- _y := -_y;
- _o := _o + 4;
- end;
- if (_x <= 0) then
- begin
- _t := _x;
- _x := _y;
- _y := -_t;
- _o := _o + 2;
- end;
- if (_x <= _y) then
- begin
- _t := _y - _x;
- _x := _x + _y;
- _y := _t;
- _o := _o + 1;
- end;
- until true;
- end;
- function QDIV(num, den: cint; const bits: cint): cint; inline;
- begin
- while (REG_DIVCNT^ and DIV_BUSY) <> 0 do;
- REG_DIVCNT^ := DIV_64_32;
- REG_DIV_NUMER^ := cint64(num) shl bits;
- REG_DIV_DENOM_L^ := den;
- while (REG_DIVCNT^ and DIV_BUSY) <> 0 do;
- result := REG_DIV_RESULT_L^;
- end;
- function atan2Lerp(x, y: cint): cuint32;
- var
- phi: cint;
- t, fa, fb, h: cuint32;
- begin
- if (y =0) then
- begin
- if x >= 0 then
- result := 0
- else
- result := BRAD_PI;
- exit;
- end;
-
- OCTANTIFY(x, y, phi);
- phi := phi * BRAD_PI div 4;
-
- t := QDIV(y, x, ATAN_FP);
- h := t mod ATANLUT_STRIDE;
-
- fa := atanLUT[t div ATANLUT_STRIDE];
- fb := atanLUT[t div ATANLUT_STRIDE + 1];
-
- result := phi + (fa + SarLongint((fb - fa) * h, ATANLUT_STRIDE_SHIFT)) div 8;
- end;
- function atan2Lookup(x, y: cint): cuint32;
- var
- phi: cint;
- t: cuint32;
- begin
- if (y = 0) then
- begin
- if x >= 0 then
- result := 0
- else
- result := BRAD_PI;
- exit;
- end;
-
- OCTANTIFY(x, y, phi);
- phi := phi * BRAD_PI div 4;
- t := QDIV(y, x, ATAN_FP);
-
- result := phi + atanLUT[t div ATANLUT_STRIDE] div 8;
- end;
- function sqran(seed: cint): cint;
- var
- old: cint;
- begin
- old := __qran_seed;
- __qran_seed := seed;
- result := old;
- end;
- end.
|