830c69ca66
GLSL/SkSL assumes that == and != on struct/array types should work. We need to emit equality and inequality operators whenever we find code that compares a struct or array. Structs and arrays can be arbitrarily nested, and either type can contain a matrix. All of these things need custom equality operators in Metal. Therefore, we need to recursively generate comparison operators when any of these types are encountered. For arrays we get lucky, and we can cover all possible array types and sizes with a single templated operator== method. Structs and matrices have no such luck, and are generated separately on a per-type basis. For each of these types, operator== is implemented as an equality check on each field, and operator!= is implemented in terms of operator==. Equality and inequality are always emitted together. (Previously, matrix equality and inequality were emitted and implemented independently, but this is no longer the case.) Change-Id: I69ee01c0a390d7db6bcb2253ed6336ab20cc4d1d Bug: skia:11908, skia:11924 Reviewed-on: https://skia-review.googlesource.com/c/skia/+/402016 Auto-Submit: John Stiles <johnstiles@google.com> Commit-Queue: Brian Osman <brianosman@google.com> Reviewed-by: Brian Osman <brianosman@google.com>
52 lines
1.6 KiB
Metal
52 lines
1.6 KiB
Metal
#include <metal_stdlib>
|
|
#include <simd/simd.h>
|
|
using namespace metal;
|
|
struct S {
|
|
int x;
|
|
int y;
|
|
};
|
|
struct Uniforms {
|
|
float4 colorGreen;
|
|
float4 colorRed;
|
|
};
|
|
struct Inputs {
|
|
};
|
|
struct Outputs {
|
|
float4 sk_FragColor [[color(0)]];
|
|
};
|
|
|
|
template <typename T, size_t N>
|
|
bool operator==(thread const array<T, N>& left, thread const array<T, N>& right) {
|
|
for (size_t index = 0; index < N; ++index) {
|
|
if (!(left[index] == right[index])) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <typename T, size_t N>
|
|
bool operator!=(thread const array<T, N>& left, thread const array<T, N>& right) {
|
|
return !(left == right);
|
|
}
|
|
thread bool operator==(thread const S& left, thread const S& right) {
|
|
return (left.x == right.x) &&
|
|
(left.y == right.y);
|
|
}
|
|
thread bool operator!=(thread const S& left, thread const S& right) {
|
|
return !(left == right);
|
|
}
|
|
|
|
fragment Outputs fragmentMain(Inputs _in [[stage_in]], constant Uniforms& _uniforms [[buffer(0)]], bool _frontFacing [[front_facing]], float4 _fragCoord [[position]]) {
|
|
Outputs _out;
|
|
(void)_out;
|
|
array<float, 4> f1 = array<float, 4>{1.0, 2.0, 3.0, 4.0};
|
|
array<float, 4> f2 = array<float, 4>{1.0, 2.0, 3.0, 4.0};
|
|
array<float, 4> f3 = array<float, 4>{1.0, 2.0, 3.0, -4.0};
|
|
array<S, 3> s1 = array<S, 3>{S{1, 2}, S{3, 4}, S{5, 6}};
|
|
array<S, 3> s2 = array<S, 3>{S{1, 2}, S{0, 0}, S{5, 6}};
|
|
array<S, 3> s3 = array<S, 3>{S{1, 2}, S{3, 4}, S{5, 6}};
|
|
_out.sk_FragColor = ((f1 == f2 && f1 != f3) && s1 != s2) && s3 == s1 ? _uniforms.colorGreen : _uniforms.colorRed;
|
|
return _out;
|
|
}
|