# mado: spline.c [src/spline.c](https://github.com/sysprog21/mado/blob/main/src/spline.c) [mado 發音](http://ilrdc.tw/grammar/sound/13/A2-10-8.mp3) ## Flow ## functions and types ### fixed-point types * [Fixed-point operation with MCTS by vax-r](https://hackmd.io/@vax-r/linux2024-homework1#Fixed-point-operation-with-MCTS) * Fixed-points standard ISO/IEC TR 18037 * [Fixed-point types in GCC](https://gcc.gnu.org/onlinedocs/gcc/Fixed-Point.html) 根據 ISO/IEC TR 18037:2008(E) 規格書,其中第 4.2 節 Detailed changes to ISO/IEC 9899:1999 給出基於 C99 擴增的 C 語言新規範。 * Clause 5.2.4.2.3 - Characteristics of fixed-point types 一定點數 $x$ 定義為: $$ x= s\times n \times b^f $$ 其中,$s$ 為正負號(-1 或 1); $p$ 為精度; $n$ 為一非負整數且小於 $b^p$; $b$ 為一基數,在二進位系統中通常為 $2$; $f$ 為一係數,用於表示小數部分的位數。 舉例來說,以 16 位元的有號整數來表示一定點數 $x$,其中 4 個位元用於表示分數部分,11 個位元用於表示整數部分,1 個位元用於表示正負號。$f=-4$、$p=11+4=15$、$0\leq n <2^{15}$。假如有一定點數為 $549$ 其二進位表示為 `0000 0010 0010 0101` 則此定點數 $x$ 實際數值為 $$ x = 549 \times 2^{-4} = 34.3125 = 2^5 + 2^1 + 2^{-2}+2^{-1} $$ * Clause 6.2.6.3 - Fixed-point types * 對有號定點數,可以將位元分成三個部分: 1. 正負號位元(sign bit) 2. 數值位元(value bits) 3. 填充位元(padding bits) 對於數值位元又可以分成: 1. 整數位元(integral bits) 2. 分數位元(fractional bits) 其中,假如數值位元為 $N$ 個且整數位元為 $L$ 個,則分數位元部分為 $N-L$ 個。對於用於表示純分數的定點數,該整數位元為 $L=0$ 個,則每一數值位元用於表示在 $2^{-1}$ 到 $2^{-N}$ 之間不同 $2$ 的冪,所以該定點數的實際數值範圍為 $-1$ 至 $1-2^N$; 對於用於表示帶分數的定點數,則每一數值位元用於表示在 $2^{L-1}$ 到 $2^{N-L}$ 之間不同 $2$ 的冪,所以該定點數的實際數值範圍為 $-2^L$ 至 $2^L-2^{N-L}$。 * 對無號定點數,可以將位元分成兩個部分: 1. 數值位元(value bits) 2. 填充位元(padding bits) 對於數值位元又可以分成: 1. 整數位元(integral bits) 2. 分數位元(fractional bits) 其中,假如數值位元為 $N$ 個且整數位元為 $L$ 個,則分數位元部分為 $N-L$ 個。對於用於表示純分數的定點數,該整數位元為 $L=0$ 個,則每一數值位元用於表示在 $2^{-1}$ 到 $2^{-N}$ 之間不同 $2$ 的冪,所以該定點數的實際數值範圍為 $0$ 至 $1-2^{-N}$; 對於用於表示帶分數的定點數,則每一數值位元用於表示在 $2^{L-1}$ 到 $2^{N-L}$ 之間不同 $2$ 的冪,所以該定點數的實際數值範圍為 $0$ 至 $2^L-2^{N-L}$。 ```c typedef int16_t twin_sfixed_t; /* 12.4 format */ typedef int32_t twin_dfixed_t; /* 24.8 format */ ``` 在 `mado` 中採用的是 ```c /* Geometrical objects */ typedef struct _twin_spoint { twin_sfixed_t x, y; } twin_spoint_t; ``` ### _twin_distance_to_line_squared 一條樣到一直線的距離。輸入有三個 `twin_spoint_t` 輸出為一 `twin_dfixed_t`。 回顧[點到直線距離](https://en.wikipedia.org/wiki/Distance_from_a_point_to_a_line),直線方程的一般式為: $$ Ax+By+C = 0 $$ 假設有兩點 $(x_1,y_1)$ 和 $(x_2,y_2)$ 形成的直線寫成直線方程為 $$ (y-y_1)=\dfrac{y_2-y_1}{x_2-x_1}(x-x_1) \text{ and } (y-y_2)=\dfrac{y_2-y_1}{x_2-x_1}(x-x_2) $$ 整理得到 $$ x(y_2-y_1) - y(x_2-x_1) - x_1(y_2-y_1) + y_1(x_2-x_1) = 0 $$ 其中 $$ \begin{align} A&=(y_2-y_1) \\ B&=(x_1-x_2) \\ C&=(y_1x_2-x_1y_2) \end{align} $$ 則一直線方程外的點 $(x_0,y_0)$ 到該直線方程的距離則為: $$ \dfrac{\lvert Ax_0+By_0+C \rvert}{\sqrt{A^2+B^2}} $$ 將其平方 $$ \dfrac{(Ax_0+By_0+C)^2}{A^2+B^2} $$ ### `_lerp` 為一線性插值函數,lerp 為 liner interploation 的縮寫 在 mado 中使用的曲線插值方式是貝茲曲線,兩點 $P_1$、$P_2$ 間的凸組合構成的插值方式 $$ B(t) = P_1 +(P_2-P_1)t = (1-t)P_1 + tP_2 $$ 其中 $t\in[0,1]$。 把 $t$ 寫成 $\frac{1}{2}$ 的冪,則第二個等式可以表示為 $$ P_1 +(P_2-P_1)2^{-k} $$ 其中 $k$ 為一非負整數。 由於為 $(P_2-P_1)$ 除以 $2$ 的冪,所以改成用位元運算可以寫成 $$ P_1 +((P_2-P_1) >> k) $$ 但在此 $t$ 的選擇就被縮限在 $t=1,2^{-1},\dots,2^{-k},\dots$ 而 ```c static void _lerp(twin_spoint_t *a, twin_spoint_t *b, int shift, twin_spoint_t *result) { result->x = a->x + ((b->x - a->x) >> shift); result->y = a->y + ((b->y - a->y) >> shift); } ``` ### `_de_casteljau` [Linear-time geometric algorithm for evaluating Bézier curves Filip Chudy, Paweł Woźny](https://arxiv.org/abs/1803.06843) 在 mado 中的貝茲曲線是三次貝茲曲線,這裡的三次是基於變數 $t$ 的三次。 根據一開始四個點 $a,b,c,d$ 依照順序兩兩做一次線性插值則可以得到 $$ \begin{align} ab(t) &= (1-t)a + tb \\ bc(t) &= (1-t)b + tc \\ cd(t) &= (1-t)c + td \end{align} $$ 其中 $ab(t),bc(t),cd(t)$ 為不同線性插植中的點,根據一次線性插植中的點做二次插值, $$ \begin{align} abbc(t) &= (1-t)ab(t) + tbc(t) \\ &= (1-t)((1-t)a + tb) + t((1-t)b + tc)\\ &= (1-t)^2a+2t(1-t)b+t^2c \\ bccd(t) &= (1-t)bc(t) + tcd(t) \\ &= (1-t)((1-t)b + tc) + t((1-t)c + td)\\ &= (1-t)^2b+2t(1-t)c+t^2d \\ \end{align} $$ 三次插值為 $$ \begin{align} \text{final}(t) &= (1-t)abbc(t) + tbccd(t) \\ &= (1-t)((1-t)^2a+2t(1-t)b+t^2c)\\ & + t((1-t)^2b+2t(1-t)c+t^2d) \\ &= (1-t)^3+ 3t(1-t)^2b + 3t^2(1-t)c + t^3d \end{align} $$ De Casteljau 的演算法則是藉由,每次迭代算出一線性插植點,再用於下一次迭代中。 * 藍色為初始點。 ![圖片](https://hackmd.io/_uploads/HkJCWHVKA.png) * 令插值函數 $t=2^-1$。紅色為第一次線性插值出來的點,黃色點為以紅色點做線性插值出來的點,黑色點為以黃色點做線性插值出來的點。 ![圖片](https://hackmd.io/_uploads/rkORbSVYR.png) ### `_twin_spline_decompose` [include/twin_private.h](https://github.com/sysprog21/mado/blob/main/include/twin_private.h) ```c #define TWIN_SFIXED_ONE (0x10) #define TWIN_SFIXED_TOLERANCE (TWIN_SFIXED_ONE >> 2) ``` `TWIN_SFIXED_ONE` in binary format is `0000 0000 0001 0000` where four bits for representing the fractional part. And the default tolerance `TWIN_SFIXED_TOLERANCE` is `0000 0000 0000 0100` which is $\frac{1}{2^2}=0.25$ in deciaml in the 12.4 fixed-point representation. `TWIN_SFIXED_TOLERANCE * TWIN_SFIXED_TOLERANCE` 基於容忍量 tolerance 選取 shift 的策略。 數學上的插值函數是建立在一連續定義域跟一對應域中,在電腦中是以二進位數值為定義域跟定義域,更具體的來說是一有限域。詳細的內容可以參考[解讀計算機編碼](https://hackmd.io/@sysprog/binary-representation)。 每執行一次三次插值法,就會多 $5$ 個切割線段 ## flatness https://blend2d.com/research/simplify_and_offset_bezier_curves.pdf ## $$ d(p_0, L)^2 \leq \epsilon^2 $$ 自己選的 $\epsilon$ $$ g(t)=d(p_0, L)^2 = \dfrac{((y_2-y_1)p_{x,0}+(x_1-x_2)p_{y,0}+(y_1x_2-x_1y_2))^2}{(y_2-y_1)^2+(x_1-x_2)^2} \leq \epsilon^2 $$ $$ \begin{align} A&=(y_2-y_1) \\ B&=(x_1-x_2) \\ C&=(y_1x_2-x_1y_2) \end{align} $$ ## 論文 GPU-friendly Stroke Expansion, Google 2024 2024 RAPH LEVIEN and ARMAN UGURAY fig 3. $s$ 為分段圓形弧長,該分段圓形弧長的兩端點形成一線段,弦長 $d$ 為該分段圓形弧長至該線段的最遠距離 $$ d = (1-\cos{\theta})\frac{s}{2\theta} $$ 見 https://en.wikipedia.org/wiki/Circular_segment https://en.wikipedia.org/wiki/Sagitta_(geometry) 對一曲線 $\hat{s}$ 曲線上任意點曲率 $\kappa(s)$ 其原文使用積分表示誤差 $$ d\approx \frac{1}{8} \left(\int^{\hat{s}}_0 \sqrt{|\kappa(s)|} ds\right)^2 $$