1cb93a386Sopenharmony_ci// defines built-in functions supported by SkSL when running on a GPU
2cb93a386Sopenharmony_ci
3cb93a386Sopenharmony_ci$genType radians($genType degrees);
4cb93a386Sopenharmony_ci$genType degrees($genType radians);
5cb93a386Sopenharmony_ci$genType sin($genType angle);
6cb93a386Sopenharmony_ci$genType cos($genType angle);
7cb93a386Sopenharmony_ci$genType tan($genType angle);
8cb93a386Sopenharmony_ci$genType asin($genType x);
9cb93a386Sopenharmony_ci$genType acos($genType x);
10cb93a386Sopenharmony_ci$genType atan($genType y, $genType x);
11cb93a386Sopenharmony_ci$genType atan($genType y_over_x);
12cb93a386Sopenharmony_ci$genType sinh($genType x);
13cb93a386Sopenharmony_ci$genType cosh($genType x);
14cb93a386Sopenharmony_ci$genType tanh($genType x);
15cb93a386Sopenharmony_ci$genType asinh($genType x);
16cb93a386Sopenharmony_ci$genType acosh($genType x);
17cb93a386Sopenharmony_ci$genType atanh($genType x);
18cb93a386Sopenharmony_ci$genType pow($genType x, $genType y);
19cb93a386Sopenharmony_ci$genType exp($genType x);
20cb93a386Sopenharmony_ci$genType log($genType x);
21cb93a386Sopenharmony_ci$genType exp2($genType x);
22cb93a386Sopenharmony_ci$genType log2($genType x);
23cb93a386Sopenharmony_ci$genType sqrt($genType x);
24cb93a386Sopenharmony_ci$genHType radians($genHType degrees);
25cb93a386Sopenharmony_ci$genHType degrees($genHType radians);
26cb93a386Sopenharmony_ci$genHType sin($genHType angle);
27cb93a386Sopenharmony_ci$genHType cos($genHType angle);
28cb93a386Sopenharmony_ci$genHType tan($genHType angle);
29cb93a386Sopenharmony_ci$genHType asin($genHType x);
30cb93a386Sopenharmony_ci$genHType acos($genHType x);
31cb93a386Sopenharmony_ci$genHType atan($genHType y, $genHType x);
32cb93a386Sopenharmony_ci$genHType atan($genHType y_over_x);
33cb93a386Sopenharmony_ci$genHType sinh($genHType x);
34cb93a386Sopenharmony_ci$genHType cosh($genHType x);
35cb93a386Sopenharmony_ci$genHType tanh($genHType x);
36cb93a386Sopenharmony_ci$genHType asinh($genHType x);
37cb93a386Sopenharmony_ci$genHType acosh($genHType x);
38cb93a386Sopenharmony_ci$genHType atanh($genHType x);
39cb93a386Sopenharmony_ci$genHType pow($genHType x, $genHType y);
40cb93a386Sopenharmony_ci$genHType exp($genHType x);
41cb93a386Sopenharmony_ci$genHType log($genHType x);
42cb93a386Sopenharmony_ci$genHType exp2($genHType x);
43cb93a386Sopenharmony_ci$genHType log2($genHType x);
44cb93a386Sopenharmony_ci$genHType sqrt($genHType x);
45cb93a386Sopenharmony_ci$genType inversesqrt($genType x);
46cb93a386Sopenharmony_ci$genHType inversesqrt($genHType x);
47cb93a386Sopenharmony_ci$genType abs($genType x);
48cb93a386Sopenharmony_ci$genHType abs($genHType x);
49cb93a386Sopenharmony_ci$genIType abs($genIType x);
50cb93a386Sopenharmony_ci$genType sign($genType x);
51cb93a386Sopenharmony_ci$genHType sign($genHType x);
52cb93a386Sopenharmony_ci$genIType sign($genIType x);
53cb93a386Sopenharmony_ci$genType floor($genType x);
54cb93a386Sopenharmony_ci$genHType floor($genHType x);
55cb93a386Sopenharmony_ci$genType trunc($genType x);
56cb93a386Sopenharmony_ci$genHType trunc($genHType x);
57cb93a386Sopenharmony_ci$genType round($genType x);
58cb93a386Sopenharmony_ci$genHType round($genHType x);
59cb93a386Sopenharmony_ci$genType roundEven($genType x);
60cb93a386Sopenharmony_ci$genHType roundEven($genHType x);
61cb93a386Sopenharmony_ci$genType ceil($genType x);
62cb93a386Sopenharmony_ci$genHType ceil($genHType x);
63cb93a386Sopenharmony_ci$genType fract($genType x);
64cb93a386Sopenharmony_ci$genHType fract($genHType x);
65cb93a386Sopenharmony_ci$genType mod($genType x, float y);
66cb93a386Sopenharmony_ci$genType mod($genType x, $genType y);
67cb93a386Sopenharmony_ci$genHType mod($genHType x, half y);
68cb93a386Sopenharmony_ci$genHType mod($genHType x, $genHType y);
69cb93a386Sopenharmony_ci$genType modf($genType x, out $genType i);
70cb93a386Sopenharmony_ci$genHType modf($genHType x, out $genHType i);
71cb93a386Sopenharmony_ci$genType min($genType x, $genType y);
72cb93a386Sopenharmony_ci$genType min($genType x, float y);
73cb93a386Sopenharmony_ci$genHType min($genHType x, $genHType y);
74cb93a386Sopenharmony_ci$genHType min($genHType x, half y);
75cb93a386Sopenharmony_ci$genIType min($genIType x, $genIType y);
76cb93a386Sopenharmony_ci$genIType min($genIType x, int y);
77cb93a386Sopenharmony_ci$genType max($genType x, $genType y);
78cb93a386Sopenharmony_ci$genType max($genType x, float y);
79cb93a386Sopenharmony_ci$genHType max($genHType x, $genHType y);
80cb93a386Sopenharmony_ci$genHType max($genHType x, half y);
81cb93a386Sopenharmony_ci$genIType max($genIType x, $genIType y);
82cb93a386Sopenharmony_ci$genIType max($genIType x, int y);
83cb93a386Sopenharmony_ci$genType clamp($genType x, $genType minVal, $genType maxVal);
84cb93a386Sopenharmony_ci$genType clamp($genType x, float minVal, float maxVal);
85cb93a386Sopenharmony_ci$genHType clamp($genHType x, $genHType minVal, $genHType maxVal);
86cb93a386Sopenharmony_ci$genHType clamp($genHType x, half minVal, half maxVal);
87cb93a386Sopenharmony_ci$genIType clamp($genIType x, $genIType minVal, $genIType maxVal);
88cb93a386Sopenharmony_ci$genIType clamp($genIType x, int minVal, int maxVal);
89cb93a386Sopenharmony_ci$genUType clamp($genUType x, $genUType minVal, $genUType maxVal);
90cb93a386Sopenharmony_ci$genUType clamp($genUType x, uint minVal, uint maxVal);
91cb93a386Sopenharmony_ci$genType saturate($genType x);
92cb93a386Sopenharmony_ci$genHType saturate($genHType x);
93cb93a386Sopenharmony_ci$genType mix($genType x, $genType y, $genType a);
94cb93a386Sopenharmony_ci$genType mix($genType x, $genType y, float a);
95cb93a386Sopenharmony_ci$genHType mix($genHType x, $genHType y, $genHType a);
96cb93a386Sopenharmony_ci$genHType mix($genHType x, $genHType y, half a);
97cb93a386Sopenharmony_ci$genType mix($genType x, $genType y, $genBType a);
98cb93a386Sopenharmony_ci$genHType mix($genHType x, $genHType y, $genBType a);
99cb93a386Sopenharmony_ci$genIType mix($genIType x, $genIType y, $genBType a);
100cb93a386Sopenharmony_ci$genBType mix($genBType x, $genBType y, $genBType a);
101cb93a386Sopenharmony_ci$genType step($genType edge, $genType x);
102cb93a386Sopenharmony_ci$genType step(float edge, $genType x);
103cb93a386Sopenharmony_ci$genHType step($genHType edge, $genHType x);
104cb93a386Sopenharmony_ci$genHType step(half edge, $genHType x);
105cb93a386Sopenharmony_ci$genType smoothstep($genType edge0, $genType edge1, $genType x);
106cb93a386Sopenharmony_ci$genType smoothstep(float edge0, float edge1, $genType x);
107cb93a386Sopenharmony_ci$genHType smoothstep($genHType edge0, $genHType edge1, $genHType x);
108cb93a386Sopenharmony_ci$genHType smoothstep(half edge0, half edge1, $genHType x);
109cb93a386Sopenharmony_ci$genBType isnan($genType x);
110cb93a386Sopenharmony_ci$genBType isinf($genType x);
111cb93a386Sopenharmony_ci$genIType floatBitsToInt($genType value);
112cb93a386Sopenharmony_ci$genUType floatBitsToUint($genType value);
113cb93a386Sopenharmony_ci$genType intBitsToFloat($genIType value);
114cb93a386Sopenharmony_ci$genType uintBitsToFloat($genUType value);
115cb93a386Sopenharmony_ci$genType fma($genType a, $genType b, $genType c);
116cb93a386Sopenharmony_ci$genHType fma($genHType a, $genHType b, $genHType c);
117cb93a386Sopenharmony_cisk_has_side_effects $genType frexp($genType x, out $genIType exp);
118cb93a386Sopenharmony_cisk_has_side_effects $genHType frexp($genHType x, out $genIType exp);
119cb93a386Sopenharmony_ci$genType ldexp($genType x, in $genIType exp);
120cb93a386Sopenharmony_ci$genHType ldexp($genHType x, in $genIType exp);
121cb93a386Sopenharmony_ciuint packUnorm2x16(float2 v);
122cb93a386Sopenharmony_ciuint packSnorm2x16(float2 v);
123cb93a386Sopenharmony_ciuint packUnorm4x8(float4 v);
124cb93a386Sopenharmony_ciuint packSnorm4x8(float4 v);
125cb93a386Sopenharmony_cifloat2 unpackUnorm2x16(uint p);
126cb93a386Sopenharmony_cifloat2 unpackSnorm2x16(uint p);
127cb93a386Sopenharmony_cifloat4 unpackUnorm4x8(uint p);
128cb93a386Sopenharmony_cifloat4 unpackSnorm4x8(uint p);
129cb93a386Sopenharmony_ciuint packHalf2x16(float2 v);
130cb93a386Sopenharmony_cifloat2 unpackHalf2x16(uint v);
131cb93a386Sopenharmony_cifloat length($genType x);
132cb93a386Sopenharmony_cihalf length($genHType x);
133cb93a386Sopenharmony_cifloat distance($genType p0, $genType p1);
134cb93a386Sopenharmony_cihalf distance($genHType p0, $genHType p1);
135cb93a386Sopenharmony_cifloat dot($genType x, $genType y);
136cb93a386Sopenharmony_cihalf dot($genHType x, $genHType y);
137cb93a386Sopenharmony_cifloat3 cross(float3 x, float3 y);
138cb93a386Sopenharmony_cihalf3 cross(half3 x, half3 y);
139cb93a386Sopenharmony_ci$genType normalize($genType x);
140cb93a386Sopenharmony_ci$genHType normalize($genHType x);
141cb93a386Sopenharmony_ci$genType faceforward($genType N, $genType I, $genType Nref);
142cb93a386Sopenharmony_ci$genHType faceforward($genHType N, $genHType I, $genHType Nref);
143cb93a386Sopenharmony_ci$genType reflect($genType I, $genType N);
144cb93a386Sopenharmony_ci$genHType reflect($genHType I, $genHType N);
145cb93a386Sopenharmony_ci$genType refract($genType I, $genType N, float eta);
146cb93a386Sopenharmony_ci$genHType refract($genHType I, $genHType N, half eta);
147cb93a386Sopenharmony_ci$mat matrixCompMult($mat x, $mat y);
148cb93a386Sopenharmony_ci$hmat matrixCompMult($hmat x, $hmat y);
149cb93a386Sopenharmony_ci$squareMat outerProduct($vec c, $vec r);
150cb93a386Sopenharmony_cifloat2x3 outerProduct(float3 c, float2 r);
151cb93a386Sopenharmony_cifloat3x2 outerProduct(float2 c, float3 r);
152cb93a386Sopenharmony_cifloat2x4 outerProduct(float4 c, float2 r);
153cb93a386Sopenharmony_cifloat4x2 outerProduct(float2 c, float4 r);
154cb93a386Sopenharmony_cifloat3x4 outerProduct(float4 c, float3 r);
155cb93a386Sopenharmony_cifloat4x3 outerProduct(float3 c, float4 r);
156cb93a386Sopenharmony_ci$squareHMat outerProduct($hvec c, $hvec r);
157cb93a386Sopenharmony_cihalf2x3 outerProduct(half3 c, half2 r);
158cb93a386Sopenharmony_cihalf3x2 outerProduct(half2 c, half3 r);
159cb93a386Sopenharmony_cihalf2x4 outerProduct(half4 c, half2 r);
160cb93a386Sopenharmony_cihalf4x2 outerProduct(half2 c, half4 r);
161cb93a386Sopenharmony_cihalf3x4 outerProduct(half4 c, half3 r);
162cb93a386Sopenharmony_cihalf4x3 outerProduct(half3 c, half4 r);
163cb93a386Sopenharmony_ci$squareMat transpose($squareMat m);
164cb93a386Sopenharmony_cifloat2x3 transpose(float3x2 m);
165cb93a386Sopenharmony_cifloat3x2 transpose(float2x3 m);
166cb93a386Sopenharmony_cifloat2x4 transpose(float4x2 m);
167cb93a386Sopenharmony_cifloat4x2 transpose(float2x4 m);
168cb93a386Sopenharmony_cifloat3x4 transpose(float4x3 m);
169cb93a386Sopenharmony_cifloat4x3 transpose(float3x4 m);
170cb93a386Sopenharmony_ci$squareHMat transpose($squareHMat m);
171cb93a386Sopenharmony_cihalf2x3 transpose(half3x2 m);
172cb93a386Sopenharmony_cihalf3x2 transpose(half2x3 m);
173cb93a386Sopenharmony_cihalf2x4 transpose(half4x2 m);
174cb93a386Sopenharmony_cihalf4x2 transpose(half2x4 m);
175cb93a386Sopenharmony_cihalf3x4 transpose(half4x3 m);
176cb93a386Sopenharmony_cihalf4x3 transpose(half3x4 m);
177cb93a386Sopenharmony_cifloat determinant($squareMat m);
178cb93a386Sopenharmony_cihalf determinant($squareHMat m);
179cb93a386Sopenharmony_ci$squareMat inverse($squareMat m);
180cb93a386Sopenharmony_ci$squareHMat inverse($squareHMat m);
181cb93a386Sopenharmony_ci$bvec lessThan($vec x, $vec y);
182cb93a386Sopenharmony_ci$bvec lessThan($hvec x, $hvec y);
183cb93a386Sopenharmony_ci$bvec lessThan($ivec x, $ivec y);
184cb93a386Sopenharmony_ci$bvec lessThan($svec x, $svec y);
185cb93a386Sopenharmony_ci$bvec lessThan($usvec x, $usvec y);
186cb93a386Sopenharmony_ci$bvec lessThan($uvec x, $uvec y);
187cb93a386Sopenharmony_ci$bvec lessThanEqual($vec x, $vec y);
188cb93a386Sopenharmony_ci$bvec lessThanEqual($hvec x, $hvec y);
189cb93a386Sopenharmony_ci$bvec lessThanEqual($ivec x, $ivec y);
190cb93a386Sopenharmony_ci$bvec lessThanEqual($uvec x, $uvec y);
191cb93a386Sopenharmony_ci$bvec lessThanEqual($svec x, $svec y);
192cb93a386Sopenharmony_ci$bvec lessThanEqual($usvec x, $usvec y);
193cb93a386Sopenharmony_ci$bvec greaterThan($vec x, $vec y);
194cb93a386Sopenharmony_ci$bvec greaterThan($hvec x, $hvec y);
195cb93a386Sopenharmony_ci$bvec greaterThan($ivec x, $ivec y);
196cb93a386Sopenharmony_ci$bvec greaterThan($uvec x, $uvec y);
197cb93a386Sopenharmony_ci$bvec greaterThan($svec x, $svec y);
198cb93a386Sopenharmony_ci$bvec greaterThan($usvec x, $usvec y);
199cb93a386Sopenharmony_ci$bvec greaterThanEqual($vec x, $vec y);
200cb93a386Sopenharmony_ci$bvec greaterThanEqual($hvec x, $hvec y);
201cb93a386Sopenharmony_ci$bvec greaterThanEqual($ivec x, $ivec y);
202cb93a386Sopenharmony_ci$bvec greaterThanEqual($uvec x, $uvec y);
203cb93a386Sopenharmony_ci$bvec greaterThanEqual($svec x, $svec y);
204cb93a386Sopenharmony_ci$bvec greaterThanEqual($usvec x, $usvec y);
205cb93a386Sopenharmony_ci$bvec equal($vec x, $vec y);
206cb93a386Sopenharmony_ci$bvec equal($hvec x, $hvec y);
207cb93a386Sopenharmony_ci$bvec equal($ivec x, $ivec y);
208cb93a386Sopenharmony_ci$bvec equal($uvec x, $uvec y);
209cb93a386Sopenharmony_ci$bvec equal($svec x, $svec y);
210cb93a386Sopenharmony_ci$bvec equal($usvec x, $usvec y);
211cb93a386Sopenharmony_ci$bvec equal($bvec x, $bvec y);
212cb93a386Sopenharmony_ci$bvec notEqual($vec x, $vec y);
213cb93a386Sopenharmony_ci$bvec notEqual($hvec x, $hvec y);
214cb93a386Sopenharmony_ci$bvec notEqual($ivec x, $ivec y);
215cb93a386Sopenharmony_ci$bvec notEqual($uvec x, $uvec y);
216cb93a386Sopenharmony_ci$bvec notEqual($svec x, $svec y);
217cb93a386Sopenharmony_ci$bvec notEqual($usvec x, $usvec y);
218cb93a386Sopenharmony_ci$bvec notEqual($bvec x, $bvec y);
219cb93a386Sopenharmony_cibool any($bvec x);
220cb93a386Sopenharmony_cibool all($bvec x);
221cb93a386Sopenharmony_ci$bvec not($bvec x);
222cb93a386Sopenharmony_ci
223cb93a386Sopenharmony_ci$genIType bitCount($genIType value);
224cb93a386Sopenharmony_ci$genIType bitCount($genUType value);
225cb93a386Sopenharmony_ci$genIType findLSB($genIType value);
226cb93a386Sopenharmony_ci$genIType findLSB($genUType value);
227cb93a386Sopenharmony_ci$genIType findMSB($genIType value);
228cb93a386Sopenharmony_ci$genIType findMSB($genUType value);
229cb93a386Sopenharmony_ci
230cb93a386Sopenharmony_cisampler2D makeSampler2D(texture2D texture, sampler s);
231cb93a386Sopenharmony_ciint2 textureSize(sampler2DRect s);
232cb93a386Sopenharmony_ci
233cb93a386Sopenharmony_cihalf4 sample(sampler1D s, float P);
234cb93a386Sopenharmony_cihalf4 sample(sampler1D s, float P, float bias);
235cb93a386Sopenharmony_cihalf4 sample(sampler2D s, float2 P);
236cb93a386Sopenharmony_ciint4 sample(isampler2D s, float2 P);
237cb93a386Sopenharmony_cihalf4 sample(samplerExternalOES s, float2 P, float bias);
238cb93a386Sopenharmony_cihalf4 sample(samplerExternalOES s, float2 P);
239cb93a386Sopenharmony_ci
240cb93a386Sopenharmony_cihalf4 sample(sampler2DRect s, float2 P);
241cb93a386Sopenharmony_cihalf4 sample(sampler2DRect s, float3 P);
242cb93a386Sopenharmony_ci
243cb93a386Sopenharmony_ci// Currently we do not support the generic types of loading subpassInput so we have some explicit
244cb93a386Sopenharmony_ci// versions that we currently use
245cb93a386Sopenharmony_cihalf4 subpassLoad(subpassInput subpass);
246cb93a386Sopenharmony_cihalf4 subpassLoad(subpassInputMS subpass, int sample);
247cb93a386Sopenharmony_ci
248cb93a386Sopenharmony_cihalf4 sample(sampler1D s, float2 P);
249cb93a386Sopenharmony_cihalf4 sample(sampler1D s, float2 P, float bias);
250cb93a386Sopenharmony_cihalf4 sample(sampler2D s, float3 P);
251cb93a386Sopenharmony_cihalf4 sample(sampler2D s, float3 P, float bias);
252cb93a386Sopenharmony_ci
253cb93a386Sopenharmony_ci$genType dFdx($genType p);
254cb93a386Sopenharmony_ci$genType dFdy($genType p);
255cb93a386Sopenharmony_ci$genHType dFdx($genHType p);
256cb93a386Sopenharmony_ci$genHType dFdy($genHType p);
257cb93a386Sopenharmony_ci$genType fwidth($genType p);
258cb93a386Sopenharmony_ci$genHType fwidth($genHType p);
259cb93a386Sopenharmony_cifloat interpolateAtSample(float interpolant, int sample);
260cb93a386Sopenharmony_cifloat2 interpolateAtSample(float2 interpolant, int sample);
261cb93a386Sopenharmony_cifloat3 interpolateAtSample(float3 interpolant, int sample);
262cb93a386Sopenharmony_cifloat4 interpolateAtSample(float4 interpolant, int sample);
263cb93a386Sopenharmony_cifloat interpolateAtOffset(float interpolant, float2 offset);
264cb93a386Sopenharmony_cifloat2 interpolateAtOffset(float2 interpolant, float2 offset);
265cb93a386Sopenharmony_cifloat3 interpolateAtOffset(float3 interpolant, float2 offset);
266cb93a386Sopenharmony_cifloat4 interpolateAtOffset(float4 interpolant, float2 offset);
267cb93a386Sopenharmony_ci
268cb93a386Sopenharmony_ci// Definitions of functions implementing all of the SkBlendMode blends.
269cb93a386Sopenharmony_ci
270cb93a386Sopenharmony_cihalf4 blend_clear(half4 src, half4 dst) { return half4(0); }
271cb93a386Sopenharmony_ci
272cb93a386Sopenharmony_cihalf4 blend_src(half4 src, half4 dst) { return src; }
273cb93a386Sopenharmony_ci
274cb93a386Sopenharmony_cihalf4 blend_dst(half4 src, half4 dst) { return dst; }
275cb93a386Sopenharmony_ci
276cb93a386Sopenharmony_cihalf4 blend_src_over(half4 src, half4 dst) { return src + (1 - src.a)*dst; }
277cb93a386Sopenharmony_ci
278cb93a386Sopenharmony_cihalf4 blend_dst_over(half4 src, half4 dst) { return (1 - dst.a)*src + dst; }
279cb93a386Sopenharmony_ci
280cb93a386Sopenharmony_cihalf4 blend_src_in(half4 src, half4 dst) { return src*dst.a; }
281cb93a386Sopenharmony_ci
282cb93a386Sopenharmony_cihalf4 blend_dst_in(half4 src, half4 dst) { return dst*src.a; }
283cb93a386Sopenharmony_ci
284cb93a386Sopenharmony_cihalf4 blend_src_out(half4 src, half4 dst) { return (1 - dst.a)*src; }
285cb93a386Sopenharmony_ci
286cb93a386Sopenharmony_cihalf4 blend_dst_out(half4 src, half4 dst) { return (1 - src.a)*dst; }
287cb93a386Sopenharmony_ci
288cb93a386Sopenharmony_cihalf4 blend_src_atop(half4 src, half4 dst) { return dst.a*src + (1 - src.a)*dst; }
289cb93a386Sopenharmony_ci
290cb93a386Sopenharmony_cihalf4 blend_dst_atop(half4 src, half4 dst)  { return  (1 - dst.a) * src + src.a*dst; }
291cb93a386Sopenharmony_ci
292cb93a386Sopenharmony_cihalf4 blend_xor(half4 src, half4 dst) { return (1 - dst.a)*src + (1 - src.a)*dst; }
293cb93a386Sopenharmony_ci
294cb93a386Sopenharmony_cihalf4 blend_plus(half4 src, half4 dst) { return min(src + dst, 1); }
295cb93a386Sopenharmony_ci
296cb93a386Sopenharmony_cihalf4 blend_modulate(half4 src, half4 dst) { return src*dst; }
297cb93a386Sopenharmony_ci
298cb93a386Sopenharmony_cihalf4 blend_screen(half4 src, half4 dst) { return src + (1 - src)*dst; }
299cb93a386Sopenharmony_ci
300cb93a386Sopenharmony_cihalf _blend_overlay_component(half2 s, half2 d) {
301cb93a386Sopenharmony_ci    return (2*d.x <= d.y)
302cb93a386Sopenharmony_ci            ? 2*s.x*d.x
303cb93a386Sopenharmony_ci            : s.y*d.y - 2*(d.y - d.x)*(s.y - s.x);
304cb93a386Sopenharmony_ci}
305cb93a386Sopenharmony_ci
306cb93a386Sopenharmony_cihalf4 blend_overlay(half4 src, half4 dst) {
307cb93a386Sopenharmony_ci    half4 result = half4(_blend_overlay_component(src.ra, dst.ra),
308cb93a386Sopenharmony_ci                         _blend_overlay_component(src.ga, dst.ga),
309cb93a386Sopenharmony_ci                         _blend_overlay_component(src.ba, dst.ba),
310cb93a386Sopenharmony_ci                         src.a + (1 - src.a)*dst.a);
311cb93a386Sopenharmony_ci    result.rgb += dst.rgb*(1 - src.a) + src.rgb*(1 - dst.a);
312cb93a386Sopenharmony_ci    return result;
313cb93a386Sopenharmony_ci}
314cb93a386Sopenharmony_ci
315cb93a386Sopenharmony_cihalf4 blend_darken(half4 src, half4 dst) {
316cb93a386Sopenharmony_ci   half4 result = blend_src_over(src, dst);
317cb93a386Sopenharmony_ci   result.rgb = min(result.rgb, (1 - dst.a)*src.rgb + dst.rgb);
318cb93a386Sopenharmony_ci   return result;
319cb93a386Sopenharmony_ci}
320cb93a386Sopenharmony_ci
321cb93a386Sopenharmony_cihalf4 blend_lighten(half4 src, half4 dst) {
322cb93a386Sopenharmony_ci    half4 result = blend_src_over(src, dst);
323cb93a386Sopenharmony_ci    result.rgb = max(result.rgb, (1 - dst.a)*src.rgb + dst.rgb);
324cb93a386Sopenharmony_ci    return result;
325cb93a386Sopenharmony_ci}
326cb93a386Sopenharmony_ci
327cb93a386Sopenharmony_cihalf _guarded_divide(half n, half d) {
328cb93a386Sopenharmony_ci    return sk_Caps.mustGuardDivisionEvenAfterExplicitZeroCheck
329cb93a386Sopenharmony_ci            ? n/(d + 0.00000001)
330cb93a386Sopenharmony_ci            : n/d;
331cb93a386Sopenharmony_ci}
332cb93a386Sopenharmony_ci
333cb93a386Sopenharmony_cihalf3 _guarded_divide(half3 n, half d) {
334cb93a386Sopenharmony_ci    return sk_Caps.mustGuardDivisionEvenAfterExplicitZeroCheck
335cb93a386Sopenharmony_ci            ? n/(d + 0.00000001)
336cb93a386Sopenharmony_ci            : n/d;
337cb93a386Sopenharmony_ci}
338cb93a386Sopenharmony_ci
339cb93a386Sopenharmony_cihalf _color_dodge_component(half2 s, half2 d) {
340cb93a386Sopenharmony_ci    if (d.x == 0) {
341cb93a386Sopenharmony_ci        return s.x*(1 - d.y);
342cb93a386Sopenharmony_ci    } else {
343cb93a386Sopenharmony_ci        half delta = s.y - s.x;
344cb93a386Sopenharmony_ci        if (delta == 0) {
345cb93a386Sopenharmony_ci             return s.y*d.y + s.x*(1 - d.y) + d.x*(1 - s.y);
346cb93a386Sopenharmony_ci        } else {
347cb93a386Sopenharmony_ci            delta = min(d.y, _guarded_divide(d.x*s.y, delta));
348cb93a386Sopenharmony_ci            return delta*s.y + s.x*(1 - d.y) + d.x*(1 - s.y);
349cb93a386Sopenharmony_ci        }
350cb93a386Sopenharmony_ci    }
351cb93a386Sopenharmony_ci}
352cb93a386Sopenharmony_ci
353cb93a386Sopenharmony_cihalf4 blend_color_dodge(half4 src, half4 dst) {
354cb93a386Sopenharmony_ci    return half4(_color_dodge_component(src.ra, dst.ra),
355cb93a386Sopenharmony_ci                 _color_dodge_component(src.ga, dst.ga),
356cb93a386Sopenharmony_ci                 _color_dodge_component(src.ba, dst.ba),
357cb93a386Sopenharmony_ci                 src.a + (1 - src.a)*dst.a);
358cb93a386Sopenharmony_ci}
359cb93a386Sopenharmony_ci
360cb93a386Sopenharmony_cihalf _color_burn_component(half2 s, half2 d) {
361cb93a386Sopenharmony_ci    if (d.y == d.x) {
362cb93a386Sopenharmony_ci        return s.y*d.y + s.x*(1 - d.y) + d.x*(1 - s.y);
363cb93a386Sopenharmony_ci    } else if (s.x == 0) {
364cb93a386Sopenharmony_ci        return d.x*(1 - s.y);
365cb93a386Sopenharmony_ci    } else {
366cb93a386Sopenharmony_ci        half delta = max(0, d.y - _guarded_divide((d.y - d.x)*s.y, s.x));
367cb93a386Sopenharmony_ci        return delta*s.y + s.x*(1 - d.y) + d.x*(1 - s.y);
368cb93a386Sopenharmony_ci    }
369cb93a386Sopenharmony_ci}
370cb93a386Sopenharmony_ci
371cb93a386Sopenharmony_cihalf4 blend_color_burn(half4 src, half4 dst) {
372cb93a386Sopenharmony_ci    return half4(_color_burn_component(src.ra, dst.ra),
373cb93a386Sopenharmony_ci                 _color_burn_component(src.ga, dst.ga),
374cb93a386Sopenharmony_ci                 _color_burn_component(src.ba, dst.ba),
375cb93a386Sopenharmony_ci                 src.a + (1 - src.a)*dst.a);
376cb93a386Sopenharmony_ci}
377cb93a386Sopenharmony_ci
378cb93a386Sopenharmony_cihalf4 blend_hard_light(half4 src, half4 dst) { return blend_overlay(dst, src); }
379cb93a386Sopenharmony_ci
380cb93a386Sopenharmony_cihalf _soft_light_component(half2 s, half2 d) {
381cb93a386Sopenharmony_ci    if (2*s.x <= s.y) {
382cb93a386Sopenharmony_ci        return _guarded_divide(d.x*d.x*(s.y - 2*s.x), d.y) + (1 - d.y)*s.x + d.x*(-s.y + 2*s.x + 1);
383cb93a386Sopenharmony_ci    } else if (4.0 * d.x <= d.y) {
384cb93a386Sopenharmony_ci        half DSqd = d.x*d.x;
385cb93a386Sopenharmony_ci        half DCub = DSqd*d.x;
386cb93a386Sopenharmony_ci        half DaSqd = d.y*d.y;
387cb93a386Sopenharmony_ci        half DaCub = DaSqd*d.y;
388cb93a386Sopenharmony_ci        return _guarded_divide(DaSqd*(s.x - d.x*(3*s.y - 6*s.x - 1)) + 12*d.y*DSqd*(s.y - 2*s.x)
389cb93a386Sopenharmony_ci                               - 16*DCub * (s.y - 2*s.x) - DaCub*s.x, DaSqd);
390cb93a386Sopenharmony_ci    } else {
391cb93a386Sopenharmony_ci        return d.x*(s.y - 2*s.x + 1) + s.x - sqrt(d.y*d.x)*(s.y - 2*s.x) - d.y*s.x;
392cb93a386Sopenharmony_ci    }
393cb93a386Sopenharmony_ci}
394cb93a386Sopenharmony_ci
395cb93a386Sopenharmony_cihalf4 blend_soft_light(half4 src, half4 dst) {
396cb93a386Sopenharmony_ci    return (dst.a == 0) ? src : half4(_soft_light_component(src.ra, dst.ra),
397cb93a386Sopenharmony_ci                                      _soft_light_component(src.ga, dst.ga),
398cb93a386Sopenharmony_ci                                      _soft_light_component(src.ba, dst.ba),
399cb93a386Sopenharmony_ci                                      src.a + (1 - src.a)*dst.a);
400cb93a386Sopenharmony_ci}
401cb93a386Sopenharmony_ci
402cb93a386Sopenharmony_cihalf4 blend_difference(half4 src, half4 dst) {
403cb93a386Sopenharmony_ci    return half4(src.rgb + dst.rgb - 2*min(src.rgb*dst.a, dst.rgb*src.a),
404cb93a386Sopenharmony_ci                 src.a + (1 - src.a)*dst.a);
405cb93a386Sopenharmony_ci}
406cb93a386Sopenharmony_ci
407cb93a386Sopenharmony_cihalf4 blend_exclusion(half4 src, half4 dst) {
408cb93a386Sopenharmony_ci    return half4(dst.rgb + src.rgb - 2*dst.rgb*src.rgb, src.a + (1 - src.a)*dst.a);
409cb93a386Sopenharmony_ci}
410cb93a386Sopenharmony_ci
411cb93a386Sopenharmony_cihalf4 blend_multiply(half4 src, half4 dst) {
412cb93a386Sopenharmony_ci    return half4((1 - src.a)*dst.rgb + (1 - dst.a)*src.rgb + src.rgb*dst.rgb,
413cb93a386Sopenharmony_ci                 src.a + (1 - src.a)*dst.a);
414cb93a386Sopenharmony_ci}
415cb93a386Sopenharmony_ci
416cb93a386Sopenharmony_cihalf _blend_color_luminance(half3 color) { return dot(half3(0.3, 0.59, 0.11), color); }
417cb93a386Sopenharmony_ci
418cb93a386Sopenharmony_cihalf3 _blend_set_color_luminance(half3 hueSatColor, half alpha, half3 lumColor) {
419cb93a386Sopenharmony_ci    half lum = _blend_color_luminance(lumColor);
420cb93a386Sopenharmony_ci    half3 result = lum - _blend_color_luminance(hueSatColor) + hueSatColor;
421cb93a386Sopenharmony_ci    half minComp = min(min(result.r, result.g), result.b);
422cb93a386Sopenharmony_ci    half maxComp = max(max(result.r, result.g), result.b);
423cb93a386Sopenharmony_ci    if (minComp < 0 && lum != minComp) {
424cb93a386Sopenharmony_ci        result = lum + (result - lum) * _guarded_divide(lum, (lum - minComp));
425cb93a386Sopenharmony_ci    }
426cb93a386Sopenharmony_ci    if (maxComp > alpha && maxComp != lum) {
427cb93a386Sopenharmony_ci        return lum + _guarded_divide((result - lum) * (alpha - lum), (maxComp - lum));
428cb93a386Sopenharmony_ci    } else {
429cb93a386Sopenharmony_ci        return result;
430cb93a386Sopenharmony_ci    }
431cb93a386Sopenharmony_ci}
432cb93a386Sopenharmony_ci
433cb93a386Sopenharmony_cihalf _blend_color_saturation(half3 color) {
434cb93a386Sopenharmony_ci    return max(max(color.r, color.g), color.b) - min(min(color.r, color.g), color.b);
435cb93a386Sopenharmony_ci}
436cb93a386Sopenharmony_ci
437cb93a386Sopenharmony_cihalf3 _blend_set_color_saturation_helper(half3 minMidMax, half sat) {
438cb93a386Sopenharmony_ci    if (minMidMax.r < minMidMax.b) {
439cb93a386Sopenharmony_ci        return half3(0,
440cb93a386Sopenharmony_ci                    _guarded_divide(sat*(minMidMax.g - minMidMax.r), (minMidMax.b - minMidMax.r)),
441cb93a386Sopenharmony_ci                    sat);
442cb93a386Sopenharmony_ci    } else {
443cb93a386Sopenharmony_ci        return half3(0);
444cb93a386Sopenharmony_ci    }
445cb93a386Sopenharmony_ci}
446cb93a386Sopenharmony_ci
447cb93a386Sopenharmony_cihalf3 _blend_set_color_saturation(half3 hueLumColor, half3 satColor) {
448cb93a386Sopenharmony_ci    half sat = _blend_color_saturation(satColor);
449cb93a386Sopenharmony_ci    if (hueLumColor.r <= hueLumColor.g) {
450cb93a386Sopenharmony_ci        if (hueLumColor.g <= hueLumColor.b) {
451cb93a386Sopenharmony_ci            return _blend_set_color_saturation_helper(hueLumColor.rgb, sat);
452cb93a386Sopenharmony_ci        } else if (hueLumColor.r <= hueLumColor.b) {
453cb93a386Sopenharmony_ci            return _blend_set_color_saturation_helper(hueLumColor.rbg, sat).rbg;
454cb93a386Sopenharmony_ci        } else {
455cb93a386Sopenharmony_ci            return _blend_set_color_saturation_helper(hueLumColor.brg, sat).gbr;
456cb93a386Sopenharmony_ci        }
457cb93a386Sopenharmony_ci    } else if (hueLumColor.r <= hueLumColor.b) {
458cb93a386Sopenharmony_ci       return _blend_set_color_saturation_helper(hueLumColor.grb, sat).grb;
459cb93a386Sopenharmony_ci    } else if (hueLumColor.g <= hueLumColor.b) {
460cb93a386Sopenharmony_ci       return _blend_set_color_saturation_helper(hueLumColor.gbr, sat).brg;
461cb93a386Sopenharmony_ci    } else {
462cb93a386Sopenharmony_ci       return _blend_set_color_saturation_helper(hueLumColor.bgr, sat).bgr;
463cb93a386Sopenharmony_ci    }
464cb93a386Sopenharmony_ci}
465cb93a386Sopenharmony_ci
466cb93a386Sopenharmony_cihalf4 blend_hue(half4 src, half4 dst) {
467cb93a386Sopenharmony_ci    half alpha = dst.a*src.a;
468cb93a386Sopenharmony_ci    half3 sda = src.rgb*dst.a;
469cb93a386Sopenharmony_ci    half3 dsa = dst.rgb*src.a;
470cb93a386Sopenharmony_ci    return half4(_blend_set_color_luminance(_blend_set_color_saturation(sda, dsa), alpha, dsa) +
471cb93a386Sopenharmony_ci                 dst.rgb - dsa + src.rgb - sda,
472cb93a386Sopenharmony_ci                 src.a + dst.a - alpha);
473cb93a386Sopenharmony_ci}
474cb93a386Sopenharmony_ci
475cb93a386Sopenharmony_cihalf4 blend_saturation(half4 src, half4 dst) {
476cb93a386Sopenharmony_ci    half alpha = dst.a*src.a;
477cb93a386Sopenharmony_ci    half3 sda = src.rgb*dst.a;
478cb93a386Sopenharmony_ci    half3 dsa = dst.rgb*src.a;
479cb93a386Sopenharmony_ci    return half4(_blend_set_color_luminance(_blend_set_color_saturation(dsa, sda), alpha, dsa) +
480cb93a386Sopenharmony_ci                 dst.rgb - dsa + src.rgb - sda,
481cb93a386Sopenharmony_ci                 src.a + dst.a - alpha);
482cb93a386Sopenharmony_ci}
483cb93a386Sopenharmony_ci
484cb93a386Sopenharmony_cihalf4 blend_color(half4 src, half4 dst)  {
485cb93a386Sopenharmony_ci    half alpha = dst.a*src.a;
486cb93a386Sopenharmony_ci    half3 sda = src.rgb*dst.a;
487cb93a386Sopenharmony_ci    half3 dsa = dst.rgb*src.a;
488cb93a386Sopenharmony_ci    return half4(_blend_set_color_luminance(sda, alpha, dsa) + dst.rgb - dsa + src.rgb - sda,
489cb93a386Sopenharmony_ci                 src.a + dst.a - alpha);
490cb93a386Sopenharmony_ci}
491cb93a386Sopenharmony_ci
492cb93a386Sopenharmony_cihalf4 blend_luminosity(half4 src, half4 dst) {
493cb93a386Sopenharmony_ci    half alpha = dst.a*src.a;
494cb93a386Sopenharmony_ci    half3 sda = src.rgb*dst.a;
495cb93a386Sopenharmony_ci    half3 dsa = dst.rgb*src.a;
496cb93a386Sopenharmony_ci    return half4(_blend_set_color_luminance(dsa, alpha, sda) + dst.rgb - dsa + src.rgb - sda,
497cb93a386Sopenharmony_ci                 src.a + dst.a - alpha);
498cb93a386Sopenharmony_ci}
499cb93a386Sopenharmony_ci
500cb93a386Sopenharmony_ci// The max() guards against division by zero when the incoming color is transparent black
501cb93a386Sopenharmony_cihalf4  unpremul(half4 color)  { return half4(color.rgb / max(color.a, 0.0001), color.a); }
502cb93a386Sopenharmony_cifloat4 unpremul(float4 color) { return float4(color.rgb / max(color.a, 0.0001), color.a); }
503cb93a386Sopenharmony_ci
504cb93a386Sopenharmony_cifloat2 proj(float3 p) { return p.xy / p.z; }
505cb93a386Sopenharmony_ci
506cb93a386Sopenharmony_ci// Implement cross() as a determinant to communicate our intent more clearly to the compiler.
507cb93a386Sopenharmony_ci// NOTE: Due to precision issues, it might be the case that cross(a, a) != 0.
508cb93a386Sopenharmony_cifloat cross(float2 a, float2 b) {
509cb93a386Sopenharmony_ci    return sk_Caps.builtinDeterminantSupport ? determinant(float2x2(a, b))
510cb93a386Sopenharmony_ci                                             : a.x*b.y - a.y*b.x;
511cb93a386Sopenharmony_ci}
512cb93a386Sopenharmony_ci
513cb93a386Sopenharmony_cihalf cross(half2 a, half2 b) {
514cb93a386Sopenharmony_ci    return sk_Caps.builtinDeterminantSupport ? determinant(half2x2(a, b))
515cb93a386Sopenharmony_ci                                             : a.x*b.y - a.y*b.x;
516cb93a386Sopenharmony_ci}
517