1 | /***************************************************************************/ |
2 | /* */ |
3 | /* afangles.c */ |
4 | /* */ |
5 | /* Routines used to compute vector angles with limited accuracy */ |
6 | /* and very high speed. It also contains sorting routines (body). */ |
7 | /* */ |
8 | /* Copyright 2003-2018 by */ |
9 | /* David Turner, Robert Wilhelm, and Werner Lemberg. */ |
10 | /* */ |
11 | /* This file is part of the FreeType project, and may only be used, */ |
12 | /* modified, and distributed under the terms of the FreeType project */ |
13 | /* license, LICENSE.TXT. By continuing to use, modify, or distribute */ |
14 | /* this file you indicate that you have read the license and */ |
15 | /* understand and accept it fully. */ |
16 | /* */ |
17 | /***************************************************************************/ |
18 | |
19 | |
20 | #include "aftypes.h" |
21 | |
22 | |
23 | /* |
24 | * We are not using `af_angle_atan' anymore, but we keep the source |
25 | * code below just in case... |
26 | */ |
27 | |
28 | |
29 | #if 0 |
30 | |
31 | |
32 | /* |
33 | * The trick here is to realize that we don't need a very accurate angle |
34 | * approximation. We are going to use the result of `af_angle_atan' to |
35 | * only compare the sign of angle differences, or check whether its |
36 | * magnitude is very small. |
37 | * |
38 | * The approximation |
39 | * |
40 | * dy * PI / (|dx|+|dy|) |
41 | * |
42 | * should be enough, and much faster to compute. |
43 | */ |
44 | FT_LOCAL_DEF( AF_Angle ) |
45 | af_angle_atan( FT_Fixed dx, |
46 | FT_Fixed dy ) |
47 | { |
48 | AF_Angle angle; |
49 | FT_Fixed ax = dx; |
50 | FT_Fixed ay = dy; |
51 | |
52 | |
53 | if ( ax < 0 ) |
54 | ax = -ax; |
55 | if ( ay < 0 ) |
56 | ay = -ay; |
57 | |
58 | ax += ay; |
59 | |
60 | if ( ax == 0 ) |
61 | angle = 0; |
62 | else |
63 | { |
64 | angle = ( AF_ANGLE_PI2 * dy ) / ( ax + ay ); |
65 | if ( dx < 0 ) |
66 | { |
67 | if ( angle >= 0 ) |
68 | angle = AF_ANGLE_PI - angle; |
69 | else |
70 | angle = -AF_ANGLE_PI - angle; |
71 | } |
72 | } |
73 | |
74 | return angle; |
75 | } |
76 | |
77 | |
78 | #elif 0 |
79 | |
80 | |
81 | /* the following table has been automatically generated with */ |
82 | /* the `mather.py' Python script */ |
83 | |
84 | #define AF_ATAN_BITS 8 |
85 | |
86 | static const FT_Byte af_arctan[1L << AF_ATAN_BITS] = |
87 | { |
88 | 0, 0, 1, 1, 1, 2, 2, 2, |
89 | 3, 3, 3, 3, 4, 4, 4, 5, |
90 | 5, 5, 6, 6, 6, 7, 7, 7, |
91 | 8, 8, 8, 9, 9, 9, 10, 10, |
92 | 10, 10, 11, 11, 11, 12, 12, 12, |
93 | 13, 13, 13, 14, 14, 14, 14, 15, |
94 | 15, 15, 16, 16, 16, 17, 17, 17, |
95 | 18, 18, 18, 18, 19, 19, 19, 20, |
96 | 20, 20, 21, 21, 21, 21, 22, 22, |
97 | 22, 23, 23, 23, 24, 24, 24, 24, |
98 | 25, 25, 25, 26, 26, 26, 26, 27, |
99 | 27, 27, 28, 28, 28, 28, 29, 29, |
100 | 29, 30, 30, 30, 30, 31, 31, 31, |
101 | 31, 32, 32, 32, 33, 33, 33, 33, |
102 | 34, 34, 34, 34, 35, 35, 35, 35, |
103 | 36, 36, 36, 36, 37, 37, 37, 38, |
104 | 38, 38, 38, 39, 39, 39, 39, 40, |
105 | 40, 40, 40, 41, 41, 41, 41, 42, |
106 | 42, 42, 42, 42, 43, 43, 43, 43, |
107 | 44, 44, 44, 44, 45, 45, 45, 45, |
108 | 46, 46, 46, 46, 46, 47, 47, 47, |
109 | 47, 48, 48, 48, 48, 48, 49, 49, |
110 | 49, 49, 50, 50, 50, 50, 50, 51, |
111 | 51, 51, 51, 51, 52, 52, 52, 52, |
112 | 52, 53, 53, 53, 53, 53, 54, 54, |
113 | 54, 54, 54, 55, 55, 55, 55, 55, |
114 | 56, 56, 56, 56, 56, 57, 57, 57, |
115 | 57, 57, 57, 58, 58, 58, 58, 58, |
116 | 59, 59, 59, 59, 59, 59, 60, 60, |
117 | 60, 60, 60, 61, 61, 61, 61, 61, |
118 | 61, 62, 62, 62, 62, 62, 62, 63, |
119 | 63, 63, 63, 63, 63, 64, 64, 64 |
120 | }; |
121 | |
122 | |
123 | FT_LOCAL_DEF( AF_Angle ) |
124 | af_angle_atan( FT_Fixed dx, |
125 | FT_Fixed dy ) |
126 | { |
127 | AF_Angle angle; |
128 | |
129 | |
130 | /* check trivial cases */ |
131 | if ( dy == 0 ) |
132 | { |
133 | angle = 0; |
134 | if ( dx < 0 ) |
135 | angle = AF_ANGLE_PI; |
136 | return angle; |
137 | } |
138 | else if ( dx == 0 ) |
139 | { |
140 | angle = AF_ANGLE_PI2; |
141 | if ( dy < 0 ) |
142 | angle = -AF_ANGLE_PI2; |
143 | return angle; |
144 | } |
145 | |
146 | angle = 0; |
147 | if ( dx < 0 ) |
148 | { |
149 | dx = -dx; |
150 | dy = -dy; |
151 | angle = AF_ANGLE_PI; |
152 | } |
153 | |
154 | if ( dy < 0 ) |
155 | { |
156 | FT_Pos tmp; |
157 | |
158 | |
159 | tmp = dx; |
160 | dx = -dy; |
161 | dy = tmp; |
162 | angle -= AF_ANGLE_PI2; |
163 | } |
164 | |
165 | if ( dx == 0 && dy == 0 ) |
166 | return 0; |
167 | |
168 | if ( dx == dy ) |
169 | angle += AF_ANGLE_PI4; |
170 | else if ( dx > dy ) |
171 | angle += af_arctan[FT_DivFix( dy, dx ) >> ( 16 - AF_ATAN_BITS )]; |
172 | else |
173 | angle += AF_ANGLE_PI2 - |
174 | af_arctan[FT_DivFix( dx, dy ) >> ( 16 - AF_ATAN_BITS )]; |
175 | |
176 | if ( angle > AF_ANGLE_PI ) |
177 | angle -= AF_ANGLE_2PI; |
178 | |
179 | return angle; |
180 | } |
181 | |
182 | |
183 | #endif /* 0 */ |
184 | |
185 | |
186 | FT_LOCAL_DEF( void ) |
187 | af_sort_pos( FT_UInt count, |
188 | FT_Pos* table ) |
189 | { |
190 | FT_UInt i, j; |
191 | FT_Pos swap; |
192 | |
193 | |
194 | for ( i = 1; i < count; i++ ) |
195 | { |
196 | for ( j = i; j > 0; j-- ) |
197 | { |
198 | if ( table[j] >= table[j - 1] ) |
199 | break; |
200 | |
201 | swap = table[j]; |
202 | table[j] = table[j - 1]; |
203 | table[j - 1] = swap; |
204 | } |
205 | } |
206 | } |
207 | |
208 | |
209 | FT_LOCAL_DEF( void ) |
210 | af_sort_and_quantize_widths( FT_UInt* count, |
211 | AF_Width table, |
212 | FT_Pos threshold ) |
213 | { |
214 | FT_UInt i, j; |
215 | FT_UInt cur_idx; |
216 | FT_Pos cur_val; |
217 | FT_Pos sum; |
218 | AF_WidthRec swap; |
219 | |
220 | |
221 | if ( *count == 1 ) |
222 | return; |
223 | |
224 | /* sort */ |
225 | for ( i = 1; i < *count; i++ ) |
226 | { |
227 | for ( j = i; j > 0; j-- ) |
228 | { |
229 | if ( table[j].org >= table[j - 1].org ) |
230 | break; |
231 | |
232 | swap = table[j]; |
233 | table[j] = table[j - 1]; |
234 | table[j - 1] = swap; |
235 | } |
236 | } |
237 | |
238 | cur_idx = 0; |
239 | cur_val = table[cur_idx].org; |
240 | |
241 | /* compute and use mean values for clusters not larger than */ |
242 | /* `threshold'; this is very primitive and might not yield */ |
243 | /* the best result, but normally, using reference character */ |
244 | /* `o', `*count' is 2, so the code below is fully sufficient */ |
245 | for ( i = 1; i < *count; i++ ) |
246 | { |
247 | if ( table[i].org - cur_val > threshold || |
248 | i == *count - 1 ) |
249 | { |
250 | sum = 0; |
251 | |
252 | /* fix loop for end of array */ |
253 | if ( table[i].org - cur_val <= threshold && |
254 | i == *count - 1 ) |
255 | i++; |
256 | |
257 | for ( j = cur_idx; j < i; j++ ) |
258 | { |
259 | sum += table[j].org; |
260 | table[j].org = 0; |
261 | } |
262 | table[cur_idx].org = sum / (FT_Pos)j; |
263 | |
264 | if ( i < *count - 1 ) |
265 | { |
266 | cur_idx = i + 1; |
267 | cur_val = table[cur_idx].org; |
268 | } |
269 | } |
270 | } |
271 | |
272 | cur_idx = 1; |
273 | |
274 | /* compress array to remove zero values */ |
275 | for ( i = 1; i < *count; i++ ) |
276 | { |
277 | if ( table[i].org ) |
278 | table[cur_idx++] = table[i]; |
279 | } |
280 | |
281 | *count = cur_idx; |
282 | } |
283 | |
284 | |
285 | /* END */ |
286 | |