Skip to content

Commit a050f4a

Browse files
author
Tim Foley
authored
Fix some crashing bugs around local variable declarations. (shader-slang#385)
The basic problem here arises when a local variable is used either before its own declaration: ```hlsl int a = b; ... int b = 0; ``` or when a local variable is used *in* its own decalration: ```hlsl int b = b; ``` In each case, Slang considers the scope of the `{}`-enclosed function body (or nested statement) as a whole, and so the lookup can "see" the declaration even if it is later in the same function. This behavior isn't really correct for HLSL semantics, so the right long-term fix is to change our scoping rules, but for now users really just want the compiler to not crash on code like this, and give an error message that points at the issue. This change makes both of the above examples print an error message saying that variable `b` was used before its declaration, which is accurate to the way that Slang is interpreting those code examples. This is currently treated as a fatal error, so that compilation aborts right away, to avoid all of the downstream crashes that these cases were causing.
1 parent 4cd18ec commit a050f4a

7 files changed

+88
-2
lines changed

source/slang/check.cpp

+28-1
Original file line numberDiff line numberDiff line change
@@ -481,7 +481,33 @@ namespace Slang
481481
if (decl->checkState == DeclCheckState::CheckingHeader)
482482
{
483483
// We tried to reference the same declaration while checking it!
484+
//
485+
// TODO: we should ideally be tracking a "chain" of declarations
486+
// being checked on the stack, so that we can report the full
487+
// chain that leads from this declaration back to itself.
488+
//
484489
sink->diagnose(decl, Diagnostics::cyclicReference, decl);
490+
return;
491+
}
492+
493+
// Hack: if we are somehow referencing a local variable declaration
494+
// before the line of code that defines it, then we need to diagnose
495+
// an error.
496+
//
497+
// TODO: The right answer is that lookup should have been performed in
498+
// the scope that was in place *before* the variable was declared, but
499+
// this is a quick fix that at least alerts the user to how we are
500+
// interpreting their code.
501+
if (auto varDecl = decl.As<Variable>())
502+
{
503+
if (auto parenScope = varDecl->ParentDecl->As<ScopeDecl>())
504+
{
505+
// TODO: This diagnostic should be emitted on the line that is referencing
506+
// the declaration. That requires `EnsureDecl` to take the requesting
507+
// location as a parameter.
508+
sink->diagnose(decl, Diagnostics::localVariableUsedBeforeDeclared, decl);
509+
return;
510+
}
485511
}
486512

487513
if (DeclCheckState::CheckingHeader > decl->checkState)
@@ -2151,7 +2177,7 @@ namespace Slang
21512177
{
21522178
for (auto decl : declGroup->decls)
21532179
{
2154-
checkDecl(decl);
2180+
DeclVisitor::dispatch(decl);
21552181
}
21562182
}
21572183

@@ -2808,6 +2834,7 @@ namespace Slang
28082834

28092835
stmt->varDecl->type.type = getSession()->getIntType();
28102836
addModifier(stmt->varDecl, new ConstModifier());
2837+
stmt->varDecl->SetCheckState(DeclCheckState::Checked);
28112838

28122839
RefPtr<IntVal> rangeBeginVal;
28132840
RefPtr<IntVal> rangeEndVal;

source/slang/diagnostic-defs.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -198,7 +198,8 @@ DIAGNOSTIC(33070, Error, expectedFunction, "expression preceding parenthesis of
198198
DIAGNOSTIC(30300, Error, assocTypeInInterfaceOnly, "'associatedtype' can only be defined in an 'interface'.")
199199
DIAGNOSTIC(30301, Error, globalGenParamInGlobalScopeOnly, "'__generic_param' can only be defined global scope.")
200200
// TODO: need to assign numbers to all these extra diagnostics...
201-
DIAGNOSTIC(39999, Error, cyclicReference, "cyclic reference '$0'.")
201+
DIAGNOSTIC(39999, Fatal, cyclicReference, "cyclic reference '$0'.")
202+
DIAGNOSTIC(39999, Fatal, localVariableUsedBeforeDeclared, "local variable '$0' is being used before its declaration.")
202203

203204
DIAGNOSTIC(39999, Error, expectedIntegerConstantWrongType, "expected integer constant (found: '$0')")
204205
DIAGNOSTIC(39999, Error, expectedIntegerConstantNotConstant, "expression does not evaluate to a compile-time constant")

source/slang/slang.cpp

+6
Original file line numberDiff line numberDiff line change
@@ -1045,6 +1045,12 @@ SLANG_API int spCompile(
10451045
int anyErrors = req->executeActions();
10461046
return anyErrors;
10471047
}
1048+
catch (Slang::AbortCompilationException&)
1049+
{
1050+
// This should only be thrown if we already emitted a fatal error
1051+
// message, so there is no reason to print something else.
1052+
return 1;
1053+
}
10481054
catch (...)
10491055
{
10501056
req->mSink.diagnose(Slang::SourceLoc(), Slang::Diagnostics::compilationAborted);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
//TEST:SIMPLE:-target dxbc -profile ps_5_0 -entry main
2+
3+
// Early versions of the front-end had bugs when local
4+
// variables were used before their definition, or
5+
// were defined using a reference to themselves.
6+
//
7+
// This file tries to ensure that we emit proper error
8+
// diagnostics in these cases.
9+
10+
void usedBeforeDeclared()
11+
{
12+
// b is used before it is declared
13+
float c = d + 1.0;
14+
int d = 0;
15+
}
16+
17+
float4 main()
18+
{
19+
usedBeforeDeclared();
20+
return 0;
21+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
result code = -1
2+
standard error = {
3+
tests/diagnostics/local-used-before-declared.slang(14): fatal error 39999: local variable 'd' is being used before its declaration.
4+
}
5+
standard output = {
6+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
//TEST:SIMPLE:-target dxbc -profile ps_5_0
2+
3+
// Early versions of the front-end had bugs when local
4+
// variables were used before their definition, or
5+
// were defined using a reference to themselves.
6+
//
7+
// This file tries to ensure that we emit proper error
8+
// diagnostics in these cases.
9+
10+
void usedInOwnDeclaration()
11+
{
12+
int e = e;
13+
}
14+
15+
float4 main()
16+
{
17+
usedInOwnDeclaration();
18+
return 0;
19+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
result code = -1
2+
standard error = {
3+
tests/diagnostics/local-used-in-own-declaration.slang(12): fatal error 39999: local variable 'e' is being used before its declaration.
4+
}
5+
standard output = {
6+
}

0 commit comments

Comments
 (0)