Skip to content

Commit c5b0708

Browse files
authored
Fix for operator assignment issue (shader-slang#2951)
* WIP handling LValue coercion via LValueImplicitCast * Need to have the ptr type for the cast. * Casting conversion working on C++. * Make the LValue casts record if in or in/out as we can produce better code if we know the difference. * WIP LValueCast pass * Fix tests so we don't fail because downstream compilers detect use of uninitialized variable. * Do conversions through through tmp for l-value scenarios that can't work other ways. * Fix a typo. * Change diagnostic implicit-cast-lvalue for a type that still exhibits the issue. * Add matrix test. * Added a bit more clarity around LValue casting choices. * Small comment improvements. Improvements based on comments on PR. * Use findOuterGeneric.
1 parent a3ad4dd commit c5b0708

22 files changed

+687
-16
lines changed

build/visual-studio/slang/slang.vcxproj

+2
Original file line numberDiff line numberDiff line change
@@ -411,6 +411,7 @@ IF EXIST ..\..\..\external\slang-glslang\bin\windows-aarch64\release\slang-glsla
411411
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-generic-function.h" />
412412
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-generic-type.h" />
413413
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-generics.h" />
414+
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-l-value-cast.h" />
414415
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-optional-type.h" />
415416
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-reinterpret.h" />
416417
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-result-type.h" />
@@ -609,6 +610,7 @@ IF EXIST ..\..\..\external\slang-glslang\bin\windows-aarch64\release\slang-glsla
609610
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-generic-function.cpp" />
610611
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-generic-type.cpp" />
611612
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-generics.cpp" />
613+
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-l-value-cast.cpp" />
612614
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-optional-type.cpp" />
613615
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-reinterpret.cpp" />
614616
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-result-type.cpp" />

build/visual-studio/slang/slang.vcxproj.filters

+6
Original file line numberDiff line numberDiff line change
@@ -321,6 +321,9 @@
321321
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-generics.h">
322322
<Filter>Header Files</Filter>
323323
</ClInclude>
324+
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-l-value-cast.h">
325+
<Filter>Header Files</Filter>
326+
</ClInclude>
324327
<ClInclude Include="..\..\..\source\slang\slang-ir-lower-optional-type.h">
325328
<Filter>Header Files</Filter>
326329
</ClInclude>
@@ -911,6 +914,9 @@
911914
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-generics.cpp">
912915
<Filter>Source Files</Filter>
913916
</ClCompile>
917+
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-l-value-cast.cpp">
918+
<Filter>Source Files</Filter>
919+
</ClCompile>
914920
<ClCompile Include="..\..\..\source\slang\slang-ir-lower-optional-type.cpp">
915921
<Filter>Source Files</Filter>
916922
</ClCompile>

source/slang/slang-ast-expr.h

+26
Original file line numberDiff line numberDiff line change
@@ -302,6 +302,32 @@ class ImplicitCastExpr : public TypeCastExpr
302302
SLANG_AST_CLASS(ImplicitCastExpr)
303303
};
304304

305+
class LValueImplicitCastExpr : public TypeCastExpr
306+
{
307+
SLANG_AST_CLASS(LValueImplicitCastExpr)
308+
309+
explicit LValueImplicitCastExpr(const TypeCastExpr& rhs) :Super(rhs) {}
310+
};
311+
312+
// To work around situations like int += uint
313+
// where we want to allow an LValue to work with an implicit cast.
314+
// The argument being cast *must* be an LValue.
315+
class OutImplicitCastExpr : public LValueImplicitCastExpr
316+
{
317+
SLANG_AST_CLASS(OutImplicitCastExpr)
318+
319+
/// Allow explict construction from any TypeCastExpr
320+
explicit OutImplicitCastExpr(const TypeCastExpr& rhs) :Super(rhs) {}
321+
};
322+
323+
class InOutImplicitCastExpr : public LValueImplicitCastExpr
324+
{
325+
SLANG_AST_CLASS(InOutImplicitCastExpr)
326+
327+
/// Allow explict construction from any TypeCastExpr
328+
explicit InOutImplicitCastExpr(const TypeCastExpr& rhs) :Super(rhs) {}
329+
};
330+
305331
/// A cast of a value to a super-type of its type.
306332
///
307333
/// The type being cast to is stored as this expression's `type`.

source/slang/slang-check-expr.cpp

+96-11
Original file line numberDiff line numberDiff line change
@@ -1952,6 +1952,51 @@ namespace Slang
19521952
return checkedExpr;
19531953
}
19541954

1955+
static bool _canLValueCoerceScalarType(Type* a, Type* b)
1956+
{
1957+
auto basicTypeA = as<BasicExpressionType>(a);
1958+
auto basicTypeB = as<BasicExpressionType>(b);
1959+
1960+
if (basicTypeA && basicTypeB)
1961+
{
1962+
const auto& infoA = BaseTypeInfo::getInfo(basicTypeA->baseType);
1963+
const auto& infoB = BaseTypeInfo::getInfo(basicTypeB->baseType);
1964+
1965+
// TODO(JS): Initially this tries to limit where LValueImplict casts happen.
1966+
// We could in principal allow different sizes, as long as we converted to a temprorary
1967+
// and back again.
1968+
//
1969+
// For now we just stick with the simple case.
1970+
// // We only allow on integer types for now. In effect just allowing any size uint/int conversions
1971+
if (infoA.sizeInBytes == infoB.sizeInBytes &&
1972+
(infoA.flags & infoB.flags & BaseTypeInfo::Flag::Integer))
1973+
{
1974+
return true;
1975+
}
1976+
1977+
}
1978+
return false;
1979+
}
1980+
1981+
static bool _canLValueCoerce(Type* a, Type* b)
1982+
{
1983+
// We can *assume* here that if they are coercable, that dimensions of vectors
1984+
// and matrices match. We might want to assert to be sure...
1985+
SLANG_ASSERT(a != b);
1986+
if (a->astNodeType == b->astNodeType)
1987+
{
1988+
if (auto matA = as<MatrixExpressionType>(a))
1989+
{
1990+
return _canLValueCoerceScalarType(matA->getElementType(), static_cast<MatrixExpressionType*>(b)->getElementType());
1991+
}
1992+
else if (auto vecA = as<VectorExpressionType>(a))
1993+
{
1994+
return _canLValueCoerceScalarType(vecA->getScalarType(), static_cast<VectorExpressionType*>(b)->getScalarType());
1995+
}
1996+
}
1997+
return _canLValueCoerceScalarType(a, b);
1998+
}
1999+
19552000
Expr* SemanticsVisitor::CheckInvokeExprWithCheckedOperands(InvokeExpr *expr)
19562001
{
19572002
auto rs = ResolveInvoke(expr);
@@ -1985,23 +2030,63 @@ namespace Slang
19852030
if( pp < expr->arguments.getCount() )
19862031
{
19872032
auto argExpr = expr->arguments[pp];
1988-
if( !argExpr->type.isLeftValue )
2033+
if( !argExpr->type.isLeftValue)
19892034
{
1990-
getSink()->diagnose(
1991-
argExpr,
1992-
Diagnostics::argumentExpectedLValue,
1993-
pp);
2035+
auto implicitCastExpr = as<ImplicitCastExpr>(argExpr);
19942036

1995-
if( auto implicitCastExpr = as<ImplicitCastExpr>(argExpr) )
2037+
if (implicitCastExpr && _canLValueCoerce(implicitCastExpr->arguments[0]->type, implicitCastExpr->type))
2038+
{
2039+
// This is to work around issues like
2040+
//
2041+
// ```
2042+
// int a = 0;
2043+
// uint b = 1;
2044+
// a += b;
2045+
// ```
2046+
// That strictly speaking it's not allowed, but we are going to allow it for now
2047+
// for situations were the types are uint/int and vector/matrix varieties of those types
2048+
//
2049+
// Then in lowering we are going to insert code to do something like
2050+
// ```
2051+
// var OutType: tmp = arg;
2052+
// f(... tmp);
2053+
// arg = tmp;
2054+
// ```
2055+
2056+
TypeCastExpr* lValueImplicitCast;
2057+
2058+
// We want to record if the cast is being used for `out` or `inout`/`ref` as
2059+
// if it's just `out` we won't need to convert before passing in.
2060+
if (as<OutType>(paramType))
2061+
{
2062+
lValueImplicitCast = getASTBuilder()->create<OutImplicitCastExpr>(*implicitCastExpr);
2063+
}
2064+
else
2065+
{
2066+
lValueImplicitCast = getASTBuilder()->create<InOutImplicitCastExpr>(*implicitCastExpr);
2067+
}
2068+
2069+
// Replace the expression. This should make this situation easier to detect.
2070+
expr->arguments[pp] = lValueImplicitCast;
2071+
}
2072+
else
19962073
{
19972074
getSink()->diagnose(
19982075
argExpr,
1999-
Diagnostics::implicitCastUsedAsLValue,
2000-
implicitCastExpr->arguments[0]->type,
2001-
implicitCastExpr->type);
2076+
Diagnostics::argumentExpectedLValue,
2077+
pp);
2078+
2079+
if(implicitCastExpr)
2080+
{
2081+
getSink()->diagnose(
2082+
argExpr,
2083+
Diagnostics::implicitCastUsedAsLValue,
2084+
implicitCastExpr->arguments[pp]->type,
2085+
implicitCastExpr->type);
2086+
}
2087+
2088+
maybeDiagnoseThisNotLValue(argExpr);
20022089
}
2003-
2004-
maybeDiagnoseThisNotLValue(argExpr);
20052090
}
20062091
}
20072092
else

source/slang/slang-compiler.h

+6
Original file line numberDiff line numberDiff line change
@@ -1633,6 +1633,12 @@ namespace Slang
16331633
/// Are we generating code for a CUDA API (CUDA / OptiX)?
16341634
bool isCUDATarget(TargetRequest* targetReq);
16351635

1636+
/// Given a target request returns which (if any) intermediate source language is required
1637+
/// to produce it.
1638+
///
1639+
/// If no intermediate source language is required, will return SourceLanguage::Unknown
1640+
SourceLanguage getIntermediateSourceLanguageForTarget(TargetRequest* req);
1641+
16361642
/// Are resource types "bindless" (implemented as ordinary data) on the given `target`?
16371643
bool areResourceTypesBindlessOnTarget(TargetRequest* target);
16381644

source/slang/slang-emit-cpp.cpp

+14
Original file line numberDiff line numberDiff line change
@@ -1200,6 +1200,20 @@ bool CPPSourceEmitter::tryEmitInstExprImpl(IRInst* inst, const EmitOpInfo& inOut
12001200
{
12011201
return false;
12021202
}
1203+
1204+
case kIROp_InOutImplicitCast:
1205+
case kIROp_OutImplicitCast:
1206+
{
1207+
// We'll just the LValue to be the desired type
1208+
m_writer->emit("reinterpret_cast<");
1209+
emitType(inst->getDataType());
1210+
m_writer->emit(">(");
1211+
1212+
emitOperand(inst->getOperand(0), getInfo(EmitOp::General));
1213+
1214+
m_writer->emit(")");
1215+
return true;
1216+
}
12031217
case kIROp_MakeVector:
12041218
{
12051219
IRType* retType = inst->getFullType();

source/slang/slang-emit.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@
3535
#include "slang-ir-lower-result-type.h"
3636
#include "slang-ir-lower-optional-type.h"
3737
#include "slang-ir-lower-bit-cast.h"
38+
#include "slang-ir-lower-l-value-cast.h"
3839
#include "slang-ir-lower-reinterpret.h"
3940
#include "slang-ir-loop-unroll.h"
4041
#include "slang-ir-metadata.h"
@@ -866,6 +867,9 @@ Result linkAndOptimizeIR(
866867
legalizeUniformBufferLoad(irModule);
867868
}
868869

870+
// Lower all the LValue implict casts (used for out/inout/ref scenarios)
871+
lowerLValueCast(targetRequest, irModule);
872+
869873
// Lower all bit_cast operations on complex types into leaf-level
870874
// bit_cast on basic types.
871875
lowerBitCast(targetRequest, irModule);

source/slang/slang-ir-inst-defs.h

+2
Original file line numberDiff line numberDiff line change
@@ -923,6 +923,8 @@ INST(ExtractTaggedUnionPayload, extractTaggedUnionPayload, 1, 0)
923923

924924
INST(BitCast, bitCast, 1, 0)
925925
INST(Reinterpret, reinterpret, 1, 0)
926+
INST(OutImplicitCast, outImplicitCast, 1, 0)
927+
INST(InOutImplicitCast, inOutImplicitCast, 1, 0)
926928
INST(IntCast, intCast, 1, 0)
927929
INST(FloatCast, floatCast, 1, 0)
928930
INST(CastIntToFloat, castIntToFloat, 1, 0)

source/slang/slang-ir-insts.h

+2
Original file line numberDiff line numberDiff line change
@@ -3302,6 +3302,8 @@ struct IRBuilder
33023302
IRUndefined* emitUndefined(IRType* type);
33033303

33043304
IRInst* emitReinterpret(IRInst* type, IRInst* value);
3305+
IRInst* emitOutImplicitCast(IRInst* type, IRInst* value);
3306+
IRInst* emitInOutImplicitCast(IRInst* type, IRInst* value);
33053307

33063308
IRFunc* createFunc();
33073309
IRGlobalVar* createGlobalVar(

0 commit comments

Comments
 (0)