summaryrefslogtreecommitdiffstats
path: root/source
diff options
context:
space:
mode:
Diffstat (limited to 'source')
-rw-r--r--source/slang/hlsl.meta.slang59
1 files changed, 51 insertions, 8 deletions
diff --git a/source/slang/hlsl.meta.slang b/source/slang/hlsl.meta.slang
index 2ac886f61..66d1cb5e6 100644
--- a/source/slang/hlsl.meta.slang
+++ b/source/slang/hlsl.meta.slang
@@ -9539,6 +9539,39 @@ vector<T,N> faceforward(vector<T,N> n, vector<T,N> i, vector<T,N> ng)
}
}
+// Helper functions for Metal target
+internal int __metal_clz(int value)
+{
+ __target_switch
+ {
+ case metal: __intrinsic_asm "clz";
+ }
+}
+
+internal uint __metal_clz(uint value)
+{
+ __target_switch
+ {
+ case metal: __intrinsic_asm "clz";
+ }
+}
+
+internal int __metal_ctz(int value)
+{
+ __target_switch
+ {
+ case metal: __intrinsic_asm "ctz";
+ }
+}
+
+internal uint __metal_ctz(uint value)
+{
+ __target_switch
+ {
+ case metal: __intrinsic_asm "ctz";
+ }
+}
+
/// Find first set bit starting at high bit and working down.
/// @param value The value to find set bits in.
/// @return The bit index number of the most significant bit,
@@ -9559,7 +9592,14 @@ int firstbithigh(int value)
case cuda: __intrinsic_asm "$P_firstbithigh($0)";
case glsl: __intrinsic_asm "findMSB";
case hlsl: __intrinsic_asm "firstbithigh";
- case metal: __intrinsic_asm "clz";
+ case metal:
+ {
+ if ((int)value < 0)
+ value = ~value;
+ if (value == 0)
+ return ~0u;
+ return 31 - __metal_clz(value);
+ }
case spirv: return spirv_asm {
OpExtInst $$int result glsl450 FindSMsb $value
};
@@ -9576,7 +9616,6 @@ vector<int, N> firstbithigh(vector<int, N> value)
{
case glsl: __intrinsic_asm "findMSB";
case hlsl: __intrinsic_asm "firstbithigh";
- case metal: __intrinsic_asm "clz";
case spirv: return spirv_asm {
OpExtInst $$vector<int, N> result glsl450 FindSMsb $value
};
@@ -9596,7 +9635,14 @@ uint firstbithigh(uint value)
case cuda: __intrinsic_asm "$P_firstbithigh($0)";
case glsl: __intrinsic_asm "findMSB";
case hlsl: __intrinsic_asm "firstbithigh";
- case metal: __intrinsic_asm "clz";
+ case metal:
+ {
+ if ((int)value < 0)
+ value = ~value;
+ if (value == 0)
+ return ~0u;
+ return 31 - __metal_clz(value);
+ }
case spirv: return spirv_asm {
OpExtInst $$uint result glsl450 FindUMsb $value
};
@@ -9613,7 +9659,6 @@ vector<uint,N> firstbithigh(vector<uint,N> value)
{
case glsl: __intrinsic_asm "findMSB";
case hlsl: __intrinsic_asm "firstbithigh";
- case metal: __intrinsic_asm "clz";
case spirv: return spirv_asm {
OpExtInst $$vector<uint,N> result glsl450 FindUMsb $value
};
@@ -9639,7 +9684,7 @@ int firstbitlow(int value)
case cuda: __intrinsic_asm "$P_firstbitlow($0)";
case glsl: __intrinsic_asm "findLSB";
case hlsl: __intrinsic_asm "firstbitlow";
- case metal: __intrinsic_asm "ctz";
+ case metal: return (value==0) ? -1 : __metal_ctz(value);
case spirv: return spirv_asm {
OpExtInst $$int result glsl450 FindILsb $value
};
@@ -9656,7 +9701,6 @@ vector<int,N> firstbitlow(vector<int,N> value)
{
case glsl: __intrinsic_asm "findLSB";
case hlsl: __intrinsic_asm "firstbitlow";
- case metal: __intrinsic_asm "ctz";
case spirv: return spirv_asm {
OpExtInst $$vector<int,N> result glsl450 FindILsb $value
};
@@ -9676,7 +9720,7 @@ uint firstbitlow(uint value)
case cuda: __intrinsic_asm "$P_firstbitlow($0)";
case glsl: __intrinsic_asm "findLSB";
case hlsl: __intrinsic_asm "firstbitlow";
- case metal: __intrinsic_asm "ctz";
+ case metal: return (value==0) ? -1 : __metal_ctz(value);
case spirv: return spirv_asm {
OpExtInst $$uint result glsl450 FindILsb $value
};
@@ -9693,7 +9737,6 @@ vector<uint,N> firstbitlow(vector<uint,N> value)
{
case glsl: __intrinsic_asm "findLSB";
case hlsl: __intrinsic_asm "firstbitlow";
- case metal: __intrinsic_asm "ctz";
case spirv: return spirv_asm {
OpExtInst $$vector<uint,N> result glsl450 FindILsb $value
};