Skip to content

Commit 8a7c053

Browse files
committed
wip: int literal type coercion.
1 parent f8b1f4f commit 8a7c053

15 files changed

+26825
-153
lines changed

source/compiler-core/slang-lexer.cpp

+5-5
Original file line numberDiff line numberDiff line change
@@ -678,22 +678,22 @@ IntegerLiteralValue getIntegerLiteralValue(
678678
UnownedStringSlice* outSuffix,
679679
bool* outIsDecimalBase)
680680
{
681-
IntegerLiteralValue value = 0;
681+
uint64_t value = 0;
682682

683683
const UnownedStringSlice content = token.getContent();
684684

685685
char const* cursor = content.begin();
686686
char const* end = content.end();
687687

688-
int base = _readOptionalBase(&cursor);
688+
uint32_t base = _readOptionalBase(&cursor);
689689

690690
for (;;)
691691
{
692-
int digit = _maybeReadDigit(&cursor, base);
692+
int32_t digit = _maybeReadDigit(&cursor, base);
693693
if (digit < 0)
694694
break;
695695

696-
value = value * base + digit;
696+
value = value * base + (uint32_t)digit;
697697
}
698698

699699
if (outSuffix)
@@ -706,7 +706,7 @@ IntegerLiteralValue getIntegerLiteralValue(
706706
*outIsDecimalBase = (base == 10);
707707
}
708708

709-
return value;
709+
return (IntegerLiteralValue)value;
710710
}
711711

712712
FloatingPointLiteralValue getFloatingPointLiteralValue(

source/slang/hlsl.meta.slang.temp.h

+26,494
Large diffs are not rendered by default.

source/slang/slang-ast-builder.cpp

+5
Original file line numberDiff line numberDiff line change
@@ -301,6 +301,11 @@ Type* ASTBuilder::getSpecializedBuiltinType(ArrayView<Val*> genericArgs, const c
301301
return rsType;
302302
}
303303

304+
IntLiteralType* ASTBuilder::getIntLiteralType(ConstantIntVal* val)
305+
{
306+
return getOrCreate<IntLiteralType>(val);
307+
}
308+
304309
PtrType* ASTBuilder::getPtrType(Type* valueType, AddressSpace addrSpace)
305310
{
306311
return dynamicCast<PtrType>(getPtrType(valueType, addrSpace, "PtrType"));

source/slang/slang-ast-builder.h

+3
Original file line numberDiff line numberDiff line change
@@ -499,6 +499,9 @@ class ASTBuilder : public RefObject
499499
Type* getNoneType() { return m_sharedASTBuilder->getNoneType(); }
500500
Type* getEnumTypeType() { return m_sharedASTBuilder->getEnumTypeType(); }
501501
Type* getDiffInterfaceType() { return m_sharedASTBuilder->getDiffInterfaceType(); }
502+
503+
IntLiteralType* getIntLiteralType(ConstantIntVal* val);
504+
502505
// Construct the type `Ptr<valueType>`, where `Ptr`
503506
// is looked up as a builtin type.
504507
PtrType* getPtrType(Type* valueType, AddressSpace addrSpace);

source/slang/slang-ast-expr.h

+1
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,7 @@ class IntegerLiteralExpr : public LiteralExpr
9393
SLANG_AST_CLASS(IntegerLiteralExpr)
9494

9595
IntegerLiteralValue value;
96+
bool isSigned = false;
9697
};
9798

9899
class FloatingPointLiteralExpr : public LiteralExpr

source/slang/slang-ast-type.cpp

+16
Original file line numberDiff line numberDiff line change
@@ -78,6 +78,22 @@ Type* InitializerListType::_createCanonicalTypeOverride()
7878
return this;
7979
}
8080

81+
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! IntLiteralType !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
82+
Type* IntLiteralType::getProperType()
83+
{
84+
return getValue()->getType();
85+
}
86+
87+
void IntLiteralType::_toTextOverride(StringBuilder& out)
88+
{
89+
out << toSlice("int-literal");
90+
}
91+
92+
Type* IntLiteralType::_createCanonicalTypeOverride()
93+
{
94+
return this;
95+
}
96+
8197
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! ErrorType !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
8298

8399
void ErrorType::_toTextOverride(StringBuilder& out)

source/slang/slang-ast-type.h

+14
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,20 @@ class InitializerListType : public Type
3030
Type* _createCanonicalTypeOverride();
3131
};
3232

33+
// The type of a known int literal that can be coerced to different integer types.
34+
class IntLiteralType : public Type
35+
{
36+
SLANG_AST_CLASS(IntLiteralType)
37+
38+
ConstantIntVal* getValue() { return as<ConstantIntVal>(getOperand(0)); }
39+
Type* getProperType();
40+
IntLiteralType(ConstantIntVal* value) { setOperands(value); }
41+
42+
// Overrides should be public so base classes can access
43+
void _toTextOverride(StringBuilder& out);
44+
Type* _createCanonicalTypeOverride();
45+
};
46+
3347
// The type of an expression that was erroneous
3448
class ErrorType : public Type
3549
{

source/slang/slang-check-constraint.cpp

+19-8
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,7 @@ namespace Slang
5959
Type* SemanticsVisitor::TryJoinVectorAndScalarType(
6060
ConstraintSystem* constraints,
6161
VectorExpressionType* vectorType,
62-
BasicExpressionType* scalarType)
62+
Type* scalarType)
6363
{
6464
// Join( vector<T,N>, S ) -> vetor<Join(T,S), N>
6565
//
@@ -232,13 +232,18 @@ Type* SemanticsVisitor::TryJoinTypes(ConstraintSystem* constraints, QualType lef
232232
if (left->equals(right))
233233
return left;
234234

235+
auto leftBasic = as<BasicExpressionType>(left);
236+
auto leftLit = as<IntLiteralType>(left);
237+
auto rightBasic = as<BasicExpressionType>(right);
238+
auto rightLit = as<IntLiteralType>(right);
239+
235240
// We can join two basic types by picking the "better" of the two
236-
if (auto leftBasic = as<BasicExpressionType>(left))
241+
if (leftBasic || leftLit)
237242
{
238-
if (auto rightBasic = as<BasicExpressionType>(right))
243+
if (rightBasic || rightLit)
239244
{
240-
auto costConvertRightToLeft = getConversionCost(leftBasic, right);
241-
auto costConvertLeftToRight = getConversionCost(rightBasic, left);
245+
auto costConvertRightToLeft = getConversionCost(left, right);
246+
auto costConvertLeftToRight = getConversionCost(right, left);
242247

243248
// Return the one that had lower conversion cost.
244249
if (costConvertRightToLeft > costConvertLeftToRight)
@@ -252,7 +257,7 @@ Type* SemanticsVisitor::TryJoinTypes(ConstraintSystem* constraints, QualType lef
252257
// We can also join a vector and a scalar
253258
if (auto rightVector = as<VectorExpressionType>(right))
254259
{
255-
return TryJoinVectorAndScalarType(constraints, rightVector, leftBasic);
260+
return TryJoinVectorAndScalarType(constraints, rightVector, left);
256261
}
257262
}
258263

@@ -278,12 +283,18 @@ Type* SemanticsVisitor::TryJoinTypes(ConstraintSystem* constraints, QualType lef
278283
}
279284

280285
// We can also join a vector and a scalar
281-
if (auto rightBasic = as<BasicExpressionType>(right))
286+
if (rightBasic || rightLit)
282287
{
283-
return TryJoinVectorAndScalarType(constraints, leftVector, rightBasic);
288+
return TryJoinVectorAndScalarType(constraints, leftVector, right);
284289
}
290+
285291
}
286292

293+
if (leftLit)
294+
left = leftLit->getProperType();
295+
if (rightLit)
296+
right = rightLit->getProperType();
297+
287298
// HACK: trying to work trait types in here...
288299
if (auto leftDeclRefType = as<DeclRefType>(left))
289300
{

source/slang/slang-check-conversion.cpp

+123-2
Original file line numberDiff line numberDiff line change
@@ -1089,6 +1089,109 @@ ConversionCost SemanticsVisitor::getImplicitConversionCostWithKnownArg(
10891089
return candidateCost;
10901090
}
10911091

1092+
bool SemanticsVisitor::coerceIntLitToBaseType(
1093+
IntLiteralType* fromType,
1094+
BasicExpressionType* toType,
1095+
Expr** outToExpr,
1096+
ConversionCost* outCost,
1097+
SourceLoc loc)
1098+
{
1099+
if (toType->getBaseType() == BaseType::Void)
1100+
return false;
1101+
1102+
ConstantIntVal* value = fromType->getValue();
1103+
ConversionCost cost = 0;
1104+
if (!isIntValueInRangeOfType(value->getValue(), toType))
1105+
{
1106+
switch (toType->getBaseType())
1107+
{
1108+
case BaseType::Half:
1109+
cost = kConversionCost_IntegerToHalfConversion;
1110+
break;
1111+
case BaseType::Float:
1112+
case BaseType::Double:
1113+
cost = kConversionCost_IntegerToFloatConversion;
1114+
break;
1115+
default:
1116+
cost = kConversionCost_IntegerTruncate;
1117+
break;
1118+
}
1119+
}
1120+
else
1121+
{
1122+
if (toType != fromType->getProperType())
1123+
{
1124+
switch (toType->getBaseType())
1125+
{
1126+
case BaseType::UInt:
1127+
case BaseType::UInt16:
1128+
case BaseType::UInt8:
1129+
cost = kConversionCost_InRangeIntLitSignedToUnsignedConversion;
1130+
break;
1131+
default:
1132+
cost = kConversionCost_InRangeIntLitConversion;
1133+
break;
1134+
}
1135+
}
1136+
}
1137+
if (outCost)
1138+
*outCost = cost;
1139+
if (outToExpr)
1140+
{
1141+
switch (toType->getBaseType())
1142+
{
1143+
case BaseType::Half:
1144+
case BaseType::Float:
1145+
case BaseType::Double:
1146+
{
1147+
auto floatLit = m_astBuilder->create<FloatingPointLiteralExpr>();
1148+
floatLit->loc = loc;
1149+
floatLit->type = QualType(toType);
1150+
floatLit->value = (FloatingPointLiteralValue)value->getValue();
1151+
*outToExpr = floatLit;
1152+
}
1153+
break;
1154+
default:
1155+
{
1156+
auto intLit = m_astBuilder->create<IntegerLiteralExpr>();
1157+
intLit->loc = loc;
1158+
intLit->type = QualType(toType);
1159+
intLit->value =
1160+
_fixIntegerLiteral(toType->getBaseType(), value->getValue(), nullptr, nullptr);
1161+
*outToExpr = intLit;
1162+
}
1163+
break;
1164+
}
1165+
}
1166+
return true;
1167+
}
1168+
1169+
Expr* SemanticsVisitor::maybeCoerceExprToProperIntType(Expr* expr)
1170+
{
1171+
auto intLitExpr = as<IntegerLiteralExpr>(expr);
1172+
if (!intLitExpr)
1173+
{
1174+
SLANG_ASSERT(!as<IntLiteralType>(expr->type.type));
1175+
return expr;
1176+
}
1177+
auto intLitType = as<IntLiteralType>(expr->type.type);
1178+
expr->type.type = intLitType->getProperType();
1179+
return expr;
1180+
}
1181+
1182+
BasicExpressionType* SemanticsVisitor::getProperTypeForIntLit(IntegerLiteralValue value, bool isSigned)
1183+
{
1184+
if (value >= std::numeric_limits<int32_t>::min() &&
1185+
value <= std::numeric_limits<int32_t>::max())
1186+
return (BasicExpressionType*)m_astBuilder->getBuiltinType(BaseType::Int);
1187+
if (value >= std::numeric_limits<uint32_t>::min() &&
1188+
value <= std::numeric_limits<uint32_t>::max())
1189+
return (BasicExpressionType*)m_astBuilder->getBuiltinType(BaseType::UInt);
1190+
if (isSigned || value >= 0 && value <= std::numeric_limits<int64_t>::max())
1191+
return (BasicExpressionType*)m_astBuilder->getBuiltinType(BaseType::Int64);
1192+
return (BasicExpressionType*)m_astBuilder->getBuiltinType(BaseType::UInt64);
1193+
}
1194+
10921195
bool SemanticsVisitor::_coerce(
10931196
CoercionSite site,
10941197
Type* toType,
@@ -1315,6 +1418,25 @@ bool SemanticsVisitor::_coerce(
13151418
return true;
13161419
}
13171420

1421+
if (auto fromIntLitType = as<IntLiteralType>(fromType))
1422+
{
1423+
auto toBasicType = as<BasicExpressionType>(toType);
1424+
if (!toBasicType)
1425+
{
1426+
if (auto toLitType = as<IntLiteralType>(toType))
1427+
toBasicType = (BasicExpressionType*)toLitType->getProperType();
1428+
}
1429+
if (toBasicType)
1430+
{
1431+
return coerceIntLitToBaseType(
1432+
fromIntLitType,
1433+
toBasicType,
1434+
outToExpr,
1435+
outCost,
1436+
fromExpr ? fromExpr->loc : SourceLoc());
1437+
}
1438+
}
1439+
13181440
// A enum type can be converted into its underlying tag type.
13191441
if (auto enumDecl = isEnumType(fromType))
13201442
{
@@ -1671,8 +1793,7 @@ bool SemanticsVisitor::_coerce(
16711793
}
16721794
else if (cost >= kConversionCost_Default)
16731795
{
1674-
// For general types of implicit conversions, we issue a warning, unless `fromExpr`
1675-
// is a known constant and we know it won't cause a problem.
1796+
// For general types of implicit conversions, we issue a warning.
16761797
bool shouldEmitGeneralWarning = true;
16771798
if (isScalarIntegerType(toType) || isHalfType(toType))
16781799
{

0 commit comments

Comments
 (0)