summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--source/slang/slang-diagnostic-defs.h5
-rw-r--r--source/slang/slang-ir-glsl-legalize.cpp73
-rw-r--r--tests/diagnostics/tessellation-legalize-array-size-too-big.slang47
-rw-r--r--tests/spirv/tess-factor-legalize-array-size.slang55
4 files changed, 180 insertions, 0 deletions
diff --git a/source/slang/slang-diagnostic-defs.h b/source/slang/slang-diagnostic-defs.h
index 03f52a701..789eac88f 100644
--- a/source/slang/slang-diagnostic-defs.h
+++ b/source/slang/slang-diagnostic-defs.h
@@ -606,6 +606,11 @@ DIAGNOSTIC(30019, Error, typeMismatch, "expected an expression of type '$0', got
DIAGNOSTIC(30021, Error, noApplicationFunction, "$0: no overload takes arguments ($1)")
DIAGNOSTIC(30022, Error, invalidTypeCast, "invalid type cast between \"$0\" and \"$1\".")
DIAGNOSTIC(30023, Error, typeHasNoPublicMemberOfName, "\"$0\" does not have public member \"$1\".")
+DIAGNOSTIC(
+ 30024,
+ Error,
+ cannotConvertArrayOfSmallerToLargerSize,
+ "Cannot convert array of size $0 to array of size $1 as this would truncate data")
DIAGNOSTIC(30025, Error, invalidArraySize, "array size must be larger than zero.")
DIAGNOSTIC(
30026,
diff --git a/source/slang/slang-ir-glsl-legalize.cpp b/source/slang/slang-ir-glsl-legalize.cpp
index c11ccc66d..11d5399cf 100644
--- a/source/slang/slang-ir-glsl-legalize.cpp
+++ b/source/slang/slang-ir-glsl-legalize.cpp
@@ -1528,6 +1528,43 @@ ScalarizedVal createSimpleGLSLGlobalVarying(
val = ScalarizedVal::typeAdapter(typeAdapter);
}
+
+ if (auto requiredArrayType = as<IRArrayTypeBase>(systemValueInfo->requiredType))
+ {
+ // Find first array declarator and handle size mismatch
+ for (auto dd = declarator; dd; dd = dd->next)
+ {
+ if (dd->flavor != GlobalVaryingDeclarator::Flavor::array)
+ continue;
+
+ // Compare the array size
+ auto declaredArraySize = dd->elementCount;
+ auto requiredArraySize = requiredArrayType->getElementCount();
+ if (declaredArraySize == requiredArraySize)
+ break;
+
+ auto toSize = getIntVal(requiredArraySize);
+ auto fromSize = getIntVal(declaredArraySize);
+ if (toSize < fromSize)
+ {
+ context->getSink()->diagnose(
+ inVarLayout,
+ Diagnostics::cannotConvertArrayOfSmallerToLargerSize,
+ fromSize,
+ toSize);
+ }
+
+ // Array sizes differ, need type adapter
+ RefPtr<ScalarizedTypeAdapterValImpl> typeAdapter =
+ new ScalarizedTypeAdapterValImpl;
+ typeAdapter->actualType = systemValueInfo->requiredType;
+ typeAdapter->pretendType = builder->getArrayType(inType, declaredArraySize);
+ typeAdapter->val = val;
+
+ val = ScalarizedVal::typeAdapter(typeAdapter);
+ break;
+ }
+ }
}
}
else
@@ -1971,6 +2008,42 @@ ScalarizedVal adaptType(IRBuilder* builder, IRInst* val, IRType* toType, IRType*
val,
builder->getIntValue(builder->getIntType(), 0));
}
+ else if (auto toArray = as<IRArrayTypeBase>(toType))
+ {
+ // If array sizes differ, we need to reshape the array
+ if (fromArray->getElementCount() != toArray->getElementCount())
+ {
+ List<IRInst*> elements;
+
+ // Get array sizes once
+ auto fromSize = getIntVal(fromArray->getElementCount());
+ auto toSize = getIntVal(toArray->getElementCount());
+ SLANG_ASSERT(fromSize <= toSize);
+
+ // Extract elements one at a time up to the source array size
+ for (Index i = 0; i < fromSize; i++)
+ {
+ auto element = builder->emitElementExtract(
+ fromArray->getElementType(),
+ val,
+ builder->getIntValue(builder->getIntType(), i));
+ elements.add(element);
+ }
+
+ if (fromSize < toSize)
+ {
+ // Fill remaining elements with default value up to target size
+ auto elementType = toArray->getElementType();
+ auto defaultValue = builder->emitDefaultConstruct(elementType);
+ for (Index i = fromSize; i < toSize; i++)
+ {
+ elements.add(defaultValue);
+ }
+ }
+
+ val = builder->emitMakeArray(toType, elements.getCount(), elements.getBuffer());
+ }
+ }
}
// TODO: actually consider what needs to go on here...
return ScalarizedVal::value(builder->emitCast(toType, val));
diff --git a/tests/diagnostics/tessellation-legalize-array-size-too-big.slang b/tests/diagnostics/tessellation-legalize-array-size-too-big.slang
new file mode 100644
index 000000000..6153d505e
--- /dev/null
+++ b/tests/diagnostics/tessellation-legalize-array-size-too-big.slang
@@ -0,0 +1,47 @@
+//TEST:SIMPLE(filecheck=CHK):-target glsl -entry hullMain -stage hull -allow-glsl
+//TEST:SIMPLE(filecheck=CHK):-target spirv-asm -entry hullMain -stage hull -allow-glsl
+
+// Tessllation outsie factor must be an array of size 4 or less
+// Tessllation inside factor must be an array of size 2 or less
+
+struct HsOut
+{
+ float2 pos;
+ float2 hm;
+};
+
+struct HscOut
+{
+ //CHK: error 30024: Cannot convert array of size 5 to array of size 4 as this would truncate data
+ float EdgeTessFactor[5] : SV_TessFactor;
+
+ //CHK: error 30024: Cannot convert array of size 3 to array of size 2 as this would truncate data
+ float InsideTessFactor[3] : SV_InsideTessFactor;
+};
+
+[domain("tri")]
+[partitioning("integer")]
+[outputtopology("triangle_ccw")]
+[outputcontrolpoints(4)]
+[patchconstantfunc("constants")]
+HsOut hullMain()
+{
+ HsOut o;
+ o.pos = 1;
+ o.hm = 2;
+ return o;
+}
+
+HscOut constants()
+{
+ HscOut o;
+ o.EdgeTessFactor[0] = 1;
+ o.EdgeTessFactor[1] = 2;
+ o.EdgeTessFactor[2] = 3;
+ o.EdgeTessFactor[3] = 4;
+ o.EdgeTessFactor[4] = 5;
+ o.InsideTessFactor[0] = 0.5;
+ o.InsideTessFactor[1] = 0.3;
+ o.InsideTessFactor[2] = 0.2;
+ return o;
+}
diff --git a/tests/spirv/tess-factor-legalize-array-size.slang b/tests/spirv/tess-factor-legalize-array-size.slang
new file mode 100644
index 000000000..70d020deb
--- /dev/null
+++ b/tests/spirv/tess-factor-legalize-array-size.slang
@@ -0,0 +1,55 @@
+//TEST:SIMPLE(filecheck=GLSL):-target glsl -entry hullMain -stage hull -allow-glsl
+//TEST:SIMPLE(filecheck=SPIRV):-target spirv -entry hullMain -stage hull -allow-glsl
+
+// When targeting SPIRV, the tessellation factors should be an array with size 4 or 2.
+
+// GLSL: gl_TessLevelOuter[3] = 0.0;
+// GLSL: gl_TessLevelInner[1] = 0.0;
+
+// SPIRV: OpDecorate %gl_TessLevelOuter BuiltIn TessLevelOuter
+// SPIRV: OpDecorate %gl_TessLevelInner BuiltIn TessLevelInner
+
+// SPIRV-DAG: OpStore %gl_TessLevelOuter %[[OuterOpCompositeConstruct:[1-9][0-9]*]]
+// SPIRV-DAG: %[[OuterOpCompositeConstruct]] = OpCompositeConstruct %[[OuterOpTypeArray:[a-zA-Z_0-9]*]]
+// SPIRV-DAG: %[[OuterOpTypeArray]] = OpTypeArray %{{[^ ]*}} %[[OuterOpConstant:[a-zA-Z_0-9]*]]
+// SPIRV-DAG: %[[OuterOpConstant]] = OpConstant %{{[^ ]*}} 4
+
+// SPIRV-DAG: OpStore %gl_TessLevelInner %[[InnerOpCompositeConstruct:[1-9][0-9]*]]
+// SPIRV-DAG: %[[InnerOpCompositeConstruct]] = OpCompositeConstruct %[[InnerOpTypeArray:[a-zA-Z_0-9]*]]
+// SPIRV-DAG: %[[InnerOpTypeArray]] = OpTypeArray %{{[^ ]*}} %[[InnerOpConstant:[a-zA-Z_0-9]*]]
+// SPIRV-DAG: %[[InnerOpConstant]] = OpConstant %{{[^ ]*}} 2
+
+struct HsOut
+{
+ float2 pos;
+ float2 hm;
+};
+
+struct HscOut
+{
+ float EdgeTessFactor[3] : SV_TessFactor; // Should be legalized to float[4]
+ float InsideTessFactor[1] : SV_InsideTessFactor; // Should be legalized to float[2]
+};
+
+[domain("tri")]
+[partitioning("integer")]
+[outputtopology("triangle_ccw")]
+[outputcontrolpoints(4)]
+[patchconstantfunc("constants")]
+HsOut hullMain()
+{
+ HsOut o;
+ o.pos = 1;
+ o.hm = 2;
+ return o;
+}
+
+HscOut constants()
+{
+ HscOut o;
+ o.EdgeTessFactor[0] = 1;
+ o.EdgeTessFactor[1] = 2;
+ o.EdgeTessFactor[2] = 3;
+ o.InsideTessFactor[0] = 0.5;
+ return o;
+}