Use the IntrinsicKind to look up SkVM intrinsic calls.
We no longer need to maintain a separate mapping table of intrinsic function names in SkVM. Change-Id: Ia3be89f9d2fc792adad2f06acc2373363497a6b7 Bug: skia:11961 Reviewed-on: https://skia-review.googlesource.com/c/skia/+/405277 Commit-Queue: John Stiles <johnstiles@google.com> Commit-Queue: Ethan Nicholas <ethannicholas@google.com> Auto-Submit: John Stiles <johnstiles@google.com> Reviewed-by: Ethan Nicholas <ethannicholas@google.com>
This commit is contained in:
parent
d7a4c05aba
commit
032fcba498
@ -931,73 +931,12 @@ Value SkVMGenerator::writeMatrixInverse4x4(const Value& m) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
||||||
static std::unordered_map<String, Intrinsic> intrinsics {
|
IntrinsicKind intrinsicKind = c.function().intrinsicKind();
|
||||||
{ "radians", Intrinsic::kRadians },
|
SkASSERT(intrinsicKind != kNotIntrinsic);
|
||||||
{ "degrees", Intrinsic::kDegrees },
|
|
||||||
{ "sin", Intrinsic::kSin },
|
|
||||||
{ "cos", Intrinsic::kCos },
|
|
||||||
{ "tan", Intrinsic::kTan },
|
|
||||||
{ "asin", Intrinsic::kASin },
|
|
||||||
{ "acos", Intrinsic::kACos },
|
|
||||||
{ "atan", Intrinsic::kATan },
|
|
||||||
|
|
||||||
{ "pow", Intrinsic::kPow },
|
|
||||||
{ "exp", Intrinsic::kExp },
|
|
||||||
{ "log", Intrinsic::kLog },
|
|
||||||
{ "exp2", Intrinsic::kExp2 },
|
|
||||||
{ "log2", Intrinsic::kLog2 },
|
|
||||||
{ "sqrt", Intrinsic::kSqrt },
|
|
||||||
{ "inversesqrt", Intrinsic::kInverseSqrt },
|
|
||||||
|
|
||||||
{ "abs", Intrinsic::kAbs },
|
|
||||||
{ "sign", Intrinsic::kSign },
|
|
||||||
{ "floor", Intrinsic::kFloor },
|
|
||||||
{ "ceil", Intrinsic::kCeil },
|
|
||||||
{ "fract", Intrinsic::kFract },
|
|
||||||
{ "mod", Intrinsic::kMod },
|
|
||||||
|
|
||||||
{ "min", Intrinsic::kMin },
|
|
||||||
{ "max", Intrinsic::kMax },
|
|
||||||
{ "clamp", Intrinsic::kClamp },
|
|
||||||
{ "saturate", Intrinsic::kSaturate },
|
|
||||||
{ "mix", Intrinsic::kMix },
|
|
||||||
{ "step", Intrinsic::kStep },
|
|
||||||
{ "smoothstep", Intrinsic::kSmoothstep },
|
|
||||||
|
|
||||||
{ "length", Intrinsic::kLength },
|
|
||||||
{ "distance", Intrinsic::kDistance },
|
|
||||||
{ "dot", Intrinsic::kDot },
|
|
||||||
{ "cross", Intrinsic::kCross },
|
|
||||||
{ "normalize", Intrinsic::kNormalize },
|
|
||||||
{ "faceforward", Intrinsic::kFaceforward },
|
|
||||||
{ "reflect", Intrinsic::kReflect },
|
|
||||||
{ "refract", Intrinsic::kRefract },
|
|
||||||
|
|
||||||
{ "matrixCompMult", Intrinsic::kMatrixCompMult },
|
|
||||||
{ "inverse", Intrinsic::kInverse },
|
|
||||||
|
|
||||||
{ "lessThan", Intrinsic::kLessThan },
|
|
||||||
{ "lessThanEqual", Intrinsic::kLessThanEqual },
|
|
||||||
{ "greaterThan", Intrinsic::kGreaterThan },
|
|
||||||
{ "greaterThanEqual", Intrinsic::kGreaterThanEqual },
|
|
||||||
{ "equal", Intrinsic::kEqual },
|
|
||||||
{ "notEqual", Intrinsic::kNotEqual },
|
|
||||||
|
|
||||||
{ "any", Intrinsic::kAny },
|
|
||||||
{ "all", Intrinsic::kAll },
|
|
||||||
{ "not", Intrinsic::kNot },
|
|
||||||
|
|
||||||
{ "sample", Intrinsic::kSample } };
|
|
||||||
|
|
||||||
auto found = intrinsics.find(c.function().name());
|
|
||||||
if (found == intrinsics.end()) {
|
|
||||||
SkDEBUGFAILF("Missing intrinsic: '%s'", String(c.function().name()).c_str());
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
const size_t nargs = c.arguments().size();
|
const size_t nargs = c.arguments().size();
|
||||||
|
|
||||||
if (found->second == Intrinsic::kSample) {
|
if (intrinsicKind == k_sample_IntrinsicKind) {
|
||||||
// Sample is very special, the first argument is a child (shader/colorFilter), which can't
|
// Sample is very special, the first argument is a child (shader/colorFilter), which can't
|
||||||
// be evaluated
|
// be evaluated
|
||||||
SkASSERT(nargs == 2);
|
SkASSERT(nargs == 2);
|
||||||
@ -1083,70 +1022,70 @@ Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
|||||||
return result;
|
return result;
|
||||||
};
|
};
|
||||||
|
|
||||||
switch (found->second) {
|
switch (intrinsicKind) {
|
||||||
case Intrinsic::kRadians:
|
case k_radians_IntrinsicKind:
|
||||||
return unary(args[0], [](skvm::F32 deg) { return deg * (SK_FloatPI / 180); });
|
return unary(args[0], [](skvm::F32 deg) { return deg * (SK_FloatPI / 180); });
|
||||||
case Intrinsic::kDegrees:
|
case k_degrees_IntrinsicKind:
|
||||||
return unary(args[0], [](skvm::F32 rad) { return rad * (180 / SK_FloatPI); });
|
return unary(args[0], [](skvm::F32 rad) { return rad * (180 / SK_FloatPI); });
|
||||||
|
|
||||||
case Intrinsic::kSin: return unary(args[0], skvm::approx_sin);
|
case k_sin_IntrinsicKind: return unary(args[0], skvm::approx_sin);
|
||||||
case Intrinsic::kCos: return unary(args[0], skvm::approx_cos);
|
case k_cos_IntrinsicKind: return unary(args[0], skvm::approx_cos);
|
||||||
case Intrinsic::kTan: return unary(args[0], skvm::approx_tan);
|
case k_tan_IntrinsicKind: return unary(args[0], skvm::approx_tan);
|
||||||
|
|
||||||
case Intrinsic::kASin: return unary(args[0], skvm::approx_asin);
|
case k_asin_IntrinsicKind: return unary(args[0], skvm::approx_asin);
|
||||||
case Intrinsic::kACos: return unary(args[0], skvm::approx_acos);
|
case k_acos_IntrinsicKind: return unary(args[0], skvm::approx_acos);
|
||||||
|
|
||||||
case Intrinsic::kATan: return nargs == 1 ? unary(args[0], skvm::approx_atan)
|
case k_atan_IntrinsicKind: return nargs == 1 ? unary(args[0], skvm::approx_atan)
|
||||||
: binary(skvm::approx_atan2);
|
: binary(skvm::approx_atan2);
|
||||||
|
|
||||||
case Intrinsic::kPow:
|
case k_pow_IntrinsicKind:
|
||||||
return binary([](skvm::F32 x, skvm::F32 y) { return skvm::approx_powf(x, y); });
|
return binary([](skvm::F32 x, skvm::F32 y) { return skvm::approx_powf(x, y); });
|
||||||
case Intrinsic::kExp: return unary(args[0], skvm::approx_exp);
|
case k_exp_IntrinsicKind: return unary(args[0], skvm::approx_exp);
|
||||||
case Intrinsic::kLog: return unary(args[0], skvm::approx_log);
|
case k_log_IntrinsicKind: return unary(args[0], skvm::approx_log);
|
||||||
case Intrinsic::kExp2: return unary(args[0], skvm::approx_pow2);
|
case k_exp2_IntrinsicKind: return unary(args[0], skvm::approx_pow2);
|
||||||
case Intrinsic::kLog2: return unary(args[0], skvm::approx_log2);
|
case k_log2_IntrinsicKind: return unary(args[0], skvm::approx_log2);
|
||||||
|
|
||||||
case Intrinsic::kSqrt: return unary(args[0], skvm::sqrt);
|
case k_sqrt_IntrinsicKind: return unary(args[0], skvm::sqrt);
|
||||||
case Intrinsic::kInverseSqrt:
|
case k_inversesqrt_IntrinsicKind:
|
||||||
return unary(args[0], [](skvm::F32 x) { return 1.0f / skvm::sqrt(x); });
|
return unary(args[0], [](skvm::F32 x) { return 1.0f / skvm::sqrt(x); });
|
||||||
|
|
||||||
case Intrinsic::kAbs: return unary(args[0], skvm::abs);
|
case k_abs_IntrinsicKind: return unary(args[0], skvm::abs);
|
||||||
case Intrinsic::kSign:
|
case k_sign_IntrinsicKind:
|
||||||
return unary(args[0], [](skvm::F32 x) { return select(x < 0, -1.0f,
|
return unary(args[0], [](skvm::F32 x) { return select(x < 0, -1.0f,
|
||||||
select(x > 0, +1.0f, 0.0f)); });
|
select(x > 0, +1.0f, 0.0f)); });
|
||||||
case Intrinsic::kFloor: return unary(args[0], skvm::floor);
|
case k_floor_IntrinsicKind: return unary(args[0], skvm::floor);
|
||||||
case Intrinsic::kCeil: return unary(args[0], skvm::ceil);
|
case k_ceil_IntrinsicKind: return unary(args[0], skvm::ceil);
|
||||||
case Intrinsic::kFract: return unary(args[0], skvm::fract);
|
case k_fract_IntrinsicKind: return unary(args[0], skvm::fract);
|
||||||
case Intrinsic::kMod:
|
case k_mod_IntrinsicKind:
|
||||||
return binary([](skvm::F32 x, skvm::F32 y) { return x - y*skvm::floor(x / y); });
|
return binary([](skvm::F32 x, skvm::F32 y) { return x - y*skvm::floor(x / y); });
|
||||||
|
|
||||||
case Intrinsic::kMin:
|
case k_min_IntrinsicKind:
|
||||||
return binary([](skvm::F32 x, skvm::F32 y) { return skvm::min(x, y); });
|
return binary([](skvm::F32 x, skvm::F32 y) { return skvm::min(x, y); });
|
||||||
case Intrinsic::kMax:
|
case k_max_IntrinsicKind:
|
||||||
return binary([](skvm::F32 x, skvm::F32 y) { return skvm::max(x, y); });
|
return binary([](skvm::F32 x, skvm::F32 y) { return skvm::max(x, y); });
|
||||||
case Intrinsic::kClamp:
|
case k_clamp_IntrinsicKind:
|
||||||
return ternary(
|
return ternary(
|
||||||
[](skvm::F32 x, skvm::F32 lo, skvm::F32 hi) { return skvm::clamp(x, lo, hi); });
|
[](skvm::F32 x, skvm::F32 lo, skvm::F32 hi) { return skvm::clamp(x, lo, hi); });
|
||||||
case Intrinsic::kSaturate:
|
case k_saturate_IntrinsicKind:
|
||||||
return unary(args[0], [](skvm::F32 x) { return skvm::clamp01(x); });
|
return unary(args[0], [](skvm::F32 x) { return skvm::clamp01(x); });
|
||||||
case Intrinsic::kMix:
|
case k_mix_IntrinsicKind:
|
||||||
return ternary(
|
return ternary(
|
||||||
[](skvm::F32 x, skvm::F32 y, skvm::F32 t) { return skvm::lerp(x, y, t); });
|
[](skvm::F32 x, skvm::F32 y, skvm::F32 t) { return skvm::lerp(x, y, t); });
|
||||||
case Intrinsic::kStep:
|
case k_step_IntrinsicKind:
|
||||||
return binary([](skvm::F32 edge, skvm::F32 x) { return select(x < edge, 0.0f, 1.0f); });
|
return binary([](skvm::F32 edge, skvm::F32 x) { return select(x < edge, 0.0f, 1.0f); });
|
||||||
case Intrinsic::kSmoothstep:
|
case k_smoothstep_IntrinsicKind:
|
||||||
return ternary([](skvm::F32 edge0, skvm::F32 edge1, skvm::F32 x) {
|
return ternary([](skvm::F32 edge0, skvm::F32 edge1, skvm::F32 x) {
|
||||||
skvm::F32 t = skvm::clamp01((x - edge0) / (edge1 - edge0));
|
skvm::F32 t = skvm::clamp01((x - edge0) / (edge1 - edge0));
|
||||||
return t ** t ** (3 - 2 ** t);
|
return t ** t ** (3 - 2 ** t);
|
||||||
});
|
});
|
||||||
|
|
||||||
case Intrinsic::kLength: return skvm::sqrt(dot(args[0], args[0]));
|
case k_length_IntrinsicKind: return skvm::sqrt(dot(args[0], args[0]));
|
||||||
case Intrinsic::kDistance: {
|
case k_distance_IntrinsicKind: {
|
||||||
Value vec = binary([](skvm::F32 x, skvm::F32 y) { return x - y; });
|
Value vec = binary([](skvm::F32 x, skvm::F32 y) { return x - y; });
|
||||||
return skvm::sqrt(dot(vec, vec));
|
return skvm::sqrt(dot(vec, vec));
|
||||||
}
|
}
|
||||||
case Intrinsic::kDot: return dot(args[0], args[1]);
|
case k_dot_IntrinsicKind: return dot(args[0], args[1]);
|
||||||
case Intrinsic::kCross: {
|
case k_cross_IntrinsicKind: {
|
||||||
skvm::F32 ax = f32(args[0][0]), ay = f32(args[0][1]), az = f32(args[0][2]),
|
skvm::F32 ax = f32(args[0][0]), ay = f32(args[0][1]), az = f32(args[0][2]),
|
||||||
bx = f32(args[1][0]), by = f32(args[1][1]), bz = f32(args[1][2]);
|
bx = f32(args[1][0]), by = f32(args[1][1]), bz = f32(args[1][2]);
|
||||||
Value result(3);
|
Value result(3);
|
||||||
@ -1155,11 +1094,11 @@ Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
|||||||
result[2] = ax**by - ay**bx;
|
result[2] = ax**by - ay**bx;
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
case Intrinsic::kNormalize: {
|
case k_normalize_IntrinsicKind: {
|
||||||
skvm::F32 invLen = 1.0f / skvm::sqrt(dot(args[0], args[0]));
|
skvm::F32 invLen = 1.0f / skvm::sqrt(dot(args[0], args[0]));
|
||||||
return unary(args[0], [&](skvm::F32 x) { return x ** invLen; });
|
return unary(args[0], [&](skvm::F32 x) { return x ** invLen; });
|
||||||
}
|
}
|
||||||
case Intrinsic::kFaceforward: {
|
case k_faceforward_IntrinsicKind: {
|
||||||
const Value &N = args[0],
|
const Value &N = args[0],
|
||||||
&I = args[1],
|
&I = args[1],
|
||||||
&Nref = args[2];
|
&Nref = args[2];
|
||||||
@ -1167,7 +1106,7 @@ Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
|||||||
skvm::F32 dotNrefI = dot(Nref, I);
|
skvm::F32 dotNrefI = dot(Nref, I);
|
||||||
return unary(N, [&](skvm::F32 n) { return select(dotNrefI<0, n, -n); });
|
return unary(N, [&](skvm::F32 n) { return select(dotNrefI<0, n, -n); });
|
||||||
}
|
}
|
||||||
case Intrinsic::kReflect: {
|
case k_reflect_IntrinsicKind: {
|
||||||
const Value &I = args[0],
|
const Value &I = args[0],
|
||||||
&N = args[1];
|
&N = args[1];
|
||||||
|
|
||||||
@ -1176,7 +1115,7 @@ Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
|||||||
return i - 2**dotNI**n;
|
return i - 2**dotNI**n;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
case Intrinsic::kRefract: {
|
case k_refract_IntrinsicKind: {
|
||||||
const Value &I = args[0],
|
const Value &I = args[0],
|
||||||
&N = args[1];
|
&N = args[1];
|
||||||
skvm::F32 eta = f32(args[2]);
|
skvm::F32 eta = f32(args[2]);
|
||||||
@ -1188,9 +1127,9 @@ Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
case Intrinsic::kMatrixCompMult:
|
case k_matrixCompMult_IntrinsicKind:
|
||||||
return binary([](skvm::F32 x, skvm::F32 y) { return x ** y; });
|
return binary([](skvm::F32 x, skvm::F32 y) { return x ** y; });
|
||||||
case Intrinsic::kInverse: {
|
case k_inverse_IntrinsicKind: {
|
||||||
switch (args[0].slots()) {
|
switch (args[0].slots()) {
|
||||||
case 4: return this->writeMatrixInverse2x2(args[0]);
|
case 4: return this->writeMatrixInverse2x2(args[0]);
|
||||||
case 9: return this->writeMatrixInverse3x3(args[0]);
|
case 9: return this->writeMatrixInverse3x3(args[0]);
|
||||||
@ -1201,58 +1140,57 @@ Value SkVMGenerator::writeIntrinsicCall(const FunctionCall& c) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case Intrinsic::kLessThan:
|
case k_lessThan_IntrinsicKind:
|
||||||
return nk == Type::NumberKind::kFloat
|
return nk == Type::NumberKind::kFloat
|
||||||
? binary([](skvm::F32 x, skvm::F32 y) { return x < y; })
|
? binary([](skvm::F32 x, skvm::F32 y) { return x < y; })
|
||||||
: binary([](skvm::I32 x, skvm::I32 y) { return x < y; });
|
: binary([](skvm::I32 x, skvm::I32 y) { return x < y; });
|
||||||
case Intrinsic::kLessThanEqual:
|
case k_lessThanEqual_IntrinsicKind:
|
||||||
return nk == Type::NumberKind::kFloat
|
return nk == Type::NumberKind::kFloat
|
||||||
? binary([](skvm::F32 x, skvm::F32 y) { return x <= y; })
|
? binary([](skvm::F32 x, skvm::F32 y) { return x <= y; })
|
||||||
: binary([](skvm::I32 x, skvm::I32 y) { return x <= y; });
|
: binary([](skvm::I32 x, skvm::I32 y) { return x <= y; });
|
||||||
case Intrinsic::kGreaterThan:
|
case k_greaterThan_IntrinsicKind:
|
||||||
return nk == Type::NumberKind::kFloat
|
return nk == Type::NumberKind::kFloat
|
||||||
? binary([](skvm::F32 x, skvm::F32 y) { return x > y; })
|
? binary([](skvm::F32 x, skvm::F32 y) { return x > y; })
|
||||||
: binary([](skvm::I32 x, skvm::I32 y) { return x > y; });
|
: binary([](skvm::I32 x, skvm::I32 y) { return x > y; });
|
||||||
case Intrinsic::kGreaterThanEqual:
|
case k_greaterThanEqual_IntrinsicKind:
|
||||||
return nk == Type::NumberKind::kFloat
|
return nk == Type::NumberKind::kFloat
|
||||||
? binary([](skvm::F32 x, skvm::F32 y) { return x >= y; })
|
? binary([](skvm::F32 x, skvm::F32 y) { return x >= y; })
|
||||||
: binary([](skvm::I32 x, skvm::I32 y) { return x >= y; });
|
: binary([](skvm::I32 x, skvm::I32 y) { return x >= y; });
|
||||||
|
|
||||||
case Intrinsic::kEqual:
|
case k_equal_IntrinsicKind:
|
||||||
return nk == Type::NumberKind::kFloat
|
return nk == Type::NumberKind::kFloat
|
||||||
? binary([](skvm::F32 x, skvm::F32 y) { return x == y; })
|
? binary([](skvm::F32 x, skvm::F32 y) { return x == y; })
|
||||||
: binary([](skvm::I32 x, skvm::I32 y) { return x == y; });
|
: binary([](skvm::I32 x, skvm::I32 y) { return x == y; });
|
||||||
case Intrinsic::kNotEqual:
|
case k_notEqual_IntrinsicKind:
|
||||||
return nk == Type::NumberKind::kFloat
|
return nk == Type::NumberKind::kFloat
|
||||||
? binary([](skvm::F32 x, skvm::F32 y) { return x != y; })
|
? binary([](skvm::F32 x, skvm::F32 y) { return x != y; })
|
||||||
: binary([](skvm::I32 x, skvm::I32 y) { return x != y; });
|
: binary([](skvm::I32 x, skvm::I32 y) { return x != y; });
|
||||||
|
|
||||||
case Intrinsic::kAny: {
|
case k_any_IntrinsicKind: {
|
||||||
skvm::I32 result = i32(args[0][0]);
|
skvm::I32 result = i32(args[0][0]);
|
||||||
for (size_t i = 1; i < args[0].slots(); ++i) {
|
for (size_t i = 1; i < args[0].slots(); ++i) {
|
||||||
result |= i32(args[0][i]);
|
result |= i32(args[0][i]);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
case Intrinsic::kAll: {
|
case k_all_IntrinsicKind: {
|
||||||
skvm::I32 result = i32(args[0][0]);
|
skvm::I32 result = i32(args[0][0]);
|
||||||
for (size_t i = 1; i < args[0].slots(); ++i) {
|
for (size_t i = 1; i < args[0].slots(); ++i) {
|
||||||
result &= i32(args[0][i]);
|
result &= i32(args[0][i]);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
case Intrinsic::kNot: return unary(args[0], [](skvm::I32 x) { return ~x; });
|
case k_not_IntrinsicKind: return unary(args[0], [](skvm::I32 x) { return ~x; });
|
||||||
|
|
||||||
case Intrinsic::kSample:
|
default:
|
||||||
// Handled earlier
|
SkDEBUGFAILF("unsupported intrinsic %s", c.function().description().c_str());
|
||||||
SkASSERT(false);
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
SkUNREACHABLE;
|
SkUNREACHABLE;
|
||||||
}
|
}
|
||||||
|
|
||||||
Value SkVMGenerator::writeFunctionCall(const FunctionCall& f) {
|
Value SkVMGenerator::writeFunctionCall(const FunctionCall& f) {
|
||||||
if (f.function().isBuiltin() && !f.function().definition()) {
|
if (f.function().isIntrinsic() && !f.function().definition()) {
|
||||||
return this->writeIntrinsicCall(f);
|
return this->writeIntrinsicCall(f);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user