Allow reading of immutables during construction time

This commit is contained in:
Marenz 2021-08-05 17:06:19 +02:00 committed by chriseth
parent 729db521a9
commit 121fd40f74
40 changed files with 233 additions and 55 deletions

View File

@ -4,6 +4,7 @@ Language Features:
Compiler Features: Compiler Features:
* Immutable variables can be read at construction time once they are initialized.
Bugfixes: Bugfixes:

View File

@ -76,7 +76,8 @@ Immutable
Variables declared as ``immutable`` are a bit less restricted than those Variables declared as ``immutable`` are a bit less restricted than those
declared as ``constant``: Immutable variables can be assigned an arbitrary declared as ``constant``: Immutable variables can be assigned an arbitrary
value in the constructor of the contract or at the point of their declaration. value in the constructor of the contract or at the point of their declaration.
They cannot be read during construction time and can only be assigned once. They can be assigned only once and can, from that point on, be read even during
construction time.
The contract creation code generated by the compiler will modify the The contract creation code generated by the compiler will modify the
contract's runtime code before it is returned by replacing all references contract's runtime code before it is returned by replacing all references
@ -84,3 +85,14 @@ to immutables by the values assigned to the them. This is important if
you are comparing the you are comparing the
runtime code generated by the compiler with the one actually stored in the runtime code generated by the compiler with the one actually stored in the
blockchain. blockchain.
.. note::
Immutables that are assigned at their declaration are only considered
initialized once the constructor of the contract is executing.
This means you cannot initialize immutables inline with a value
that depends on another immutable. You can do this, however,
inside the constructor of the contract.
This is a safeguard against different interpretations about the order
of state variable initialization and constructor execution, especially
with regards to inheritance.

View File

@ -27,30 +27,31 @@ using namespace solidity::langutil;
void ImmutableValidator::analyze() void ImmutableValidator::analyze()
{ {
m_inConstructionContext = true; m_inCreationContext = true;
auto linearizedContracts = m_currentContract.annotation().linearizedBaseContracts | ranges::views::reverse; auto linearizedContracts = m_currentContract.annotation().linearizedBaseContracts | ranges::views::reverse;
for (ContractDefinition const* contract: linearizedContracts)
for (VariableDeclaration const* stateVar: contract->stateVariables())
if (stateVar->value())
m_initializedStateVariables.emplace(stateVar);
for (ContractDefinition const* contract: linearizedContracts) for (ContractDefinition const* contract: linearizedContracts)
for (VariableDeclaration const* stateVar: contract->stateVariables()) for (VariableDeclaration const* stateVar: contract->stateVariables())
if (stateVar->value()) if (stateVar->value())
stateVar->value()->accept(*this); stateVar->value()->accept(*this);
for (ContractDefinition const* contract: linearizedContracts)
if (contract->constructor())
visitCallableIfNew(*contract->constructor());
for (ContractDefinition const* contract: linearizedContracts) for (ContractDefinition const* contract: linearizedContracts)
for (std::shared_ptr<InheritanceSpecifier> const& inheritSpec: contract->baseContracts()) for (std::shared_ptr<InheritanceSpecifier> const& inheritSpec: contract->baseContracts())
if (auto args = inheritSpec->arguments()) if (auto args = inheritSpec->arguments())
ASTNode::listAccept(*args, *this); ASTNode::listAccept(*args, *this);
m_inConstructionContext = false; for (ContractDefinition const* contract: linearizedContracts)
{
for (VariableDeclaration const* stateVar: contract->stateVariables())
if (stateVar->value())
m_initializedStateVariables.emplace(stateVar);
if (contract->constructor())
visitCallableIfNew(*contract->constructor());
}
m_inCreationContext = false;
for (ContractDefinition const* contract: linearizedContracts) for (ContractDefinition const* contract: linearizedContracts)
{ {
@ -64,6 +65,15 @@ void ImmutableValidator::analyze()
checkAllVariablesInitialized(m_currentContract.location()); checkAllVariablesInitialized(m_currentContract.location());
} }
bool ImmutableValidator::visit(Assignment const& _assignment)
{
// Need to visit values first (rhs) as they might access other immutables.
_assignment.rightHandSide().accept(*this);
_assignment.leftHandSide().accept(*this);
return false;
}
bool ImmutableValidator::visit(FunctionDefinition const& _functionDefinition) bool ImmutableValidator::visit(FunctionDefinition const& _functionDefinition)
{ {
return analyseCallable(_functionDefinition); return analyseCallable(_functionDefinition);
@ -207,19 +217,37 @@ void ImmutableValidator::analyseVariableReference(VariableDeclaration const& _va
"Cannot write to immutable here: Immutable variables cannot be initialized inside an if statement." "Cannot write to immutable here: Immutable variables cannot be initialized inside an if statement."
); );
else if (m_initializedStateVariables.count(&_variableReference)) else if (m_initializedStateVariables.count(&_variableReference))
{
if (!read)
m_errorReporter.typeError( m_errorReporter.typeError(
1574_error, 1574_error,
_expression.location(), _expression.location(),
"Immutable state variable already initialized." "Immutable state variable already initialized."
); );
else
m_errorReporter.typeError(
2718_error,
_expression.location(),
"Immutable variables cannot be modified after initialization."
);
}
else if (read)
m_errorReporter.typeError(
3969_error,
_expression.location(),
"Immutable variables must be initialized using an assignment."
);
m_initializedStateVariables.emplace(&_variableReference); m_initializedStateVariables.emplace(&_variableReference);
} }
if (read && m_inConstructionContext) if (
read &&
m_inCreationContext &&
!m_initializedStateVariables.count(&_variableReference)
)
m_errorReporter.typeError( m_errorReporter.typeError(
7733_error, 7733_error,
_expression.location(), _expression.location(),
"Immutable variables cannot be read during contract creation time, which means " "Immutable variables cannot be read before they are initialized."
"they cannot be read in the constructor or any function or modifier called from it."
); );
} }

View File

@ -28,8 +28,9 @@ namespace solidity::frontend
/** /**
* Validates access and initialization of immutable variables: * Validates access and initialization of immutable variables:
* must be directly initialized in their respective c'tor * must be directly initialized in their respective c'tor or inline
* can not be read by any function/modifier called by the c'tor (or the c'tor itself) * cannot be read before being initialized
* cannot be read when initializing state variables inline
* must be initialized outside loops (only one initialization) * must be initialized outside loops (only one initialization)
* must be initialized outside ifs (must be initialized unconditionally) * must be initialized outside ifs (must be initialized unconditionally)
* must be initialized exactly once (no multiple statements) * must be initialized exactly once (no multiple statements)
@ -48,6 +49,7 @@ public:
void analyze(); void analyze();
private: private:
bool visit(Assignment const& _assignment);
bool visit(FunctionDefinition const& _functionDefinition); bool visit(FunctionDefinition const& _functionDefinition);
bool visit(ModifierDefinition const& _modifierDefinition); bool visit(ModifierDefinition const& _modifierDefinition);
bool visit(MemberAccess const& _memberAccess); bool visit(MemberAccess const& _memberAccess);
@ -74,7 +76,7 @@ private:
FunctionDefinition const* m_currentConstructor = nullptr; FunctionDefinition const* m_currentConstructor = nullptr;
bool m_inLoop = false; bool m_inLoop = false;
bool m_inBranch = false; bool m_inBranch = false;
bool m_inConstructionContext = false; bool m_inCreationContext = true;
}; };
} }

View File

@ -155,7 +155,13 @@ ImmutableItem::ImmutableItem(CompilerContext& _compilerContext, VariableDeclarat
void ImmutableItem::retrieveValue(SourceLocation const&, bool) const void ImmutableItem::retrieveValue(SourceLocation const&, bool) const
{ {
solUnimplementedAssert(m_dataType->isValueType(), ""); solUnimplementedAssert(m_dataType->isValueType(), "");
solAssert(!m_context.runtimeContext(), "Tried to read immutable at construction time.");
if (m_context.runtimeContext())
CompilerUtils(m_context).loadFromMemory(
static_cast<unsigned>(m_context.immutableMemoryOffset(m_variable)),
*m_dataType
);
else
for (auto&& slotName: m_context.immutableVariableSlotNames(m_variable)) for (auto&& slotName: m_context.immutableVariableSlotNames(m_variable))
m_context.appendImmutable(slotName); m_context.appendImmutable(slotName);
} }

View File

@ -65,13 +65,17 @@ using InternalDispatchMap = std::map<YulArity, DispatchSet>;
class IRGenerationContext class IRGenerationContext
{ {
public: public:
enum class ExecutionContext { Creation, Deployed };
IRGenerationContext( IRGenerationContext(
langutil::EVMVersion _evmVersion, langutil::EVMVersion _evmVersion,
ExecutionContext _executionContext,
RevertStrings _revertStrings, RevertStrings _revertStrings,
OptimiserSettings _optimiserSettings, OptimiserSettings _optimiserSettings,
std::map<std::string, unsigned> _sourceIndices std::map<std::string, unsigned> _sourceIndices
): ):
m_evmVersion(_evmVersion), m_evmVersion(_evmVersion),
m_executionContext(_executionContext),
m_revertStrings(_revertStrings), m_revertStrings(_revertStrings),
m_optimiserSettings(std::move(_optimiserSettings)), m_optimiserSettings(std::move(_optimiserSettings)),
m_sourceIndices(std::move(_sourceIndices)) m_sourceIndices(std::move(_sourceIndices))
@ -139,6 +143,7 @@ public:
YulUtilFunctions utils(); YulUtilFunctions utils();
langutil::EVMVersion evmVersion() const { return m_evmVersion; } langutil::EVMVersion evmVersion() const { return m_evmVersion; }
ExecutionContext executionContext() const { return m_executionContext; }
void setArithmetic(Arithmetic _value) { m_arithmetic = _value; } void setArithmetic(Arithmetic _value) { m_arithmetic = _value; }
Arithmetic arithmetic() const { return m_arithmetic; } Arithmetic arithmetic() const { return m_arithmetic; }
@ -162,8 +167,11 @@ public:
std::map<std::string, unsigned> const& sourceIndices() const { return m_sourceIndices; } std::map<std::string, unsigned> const& sourceIndices() const { return m_sourceIndices; }
bool immutableRegistered(VariableDeclaration const& _varDecl) const { return m_immutableVariables.count(&_varDecl); }
private: private:
langutil::EVMVersion m_evmVersion; langutil::EVMVersion m_evmVersion;
ExecutionContext m_executionContext;
RevertStrings m_revertStrings; RevertStrings m_revertStrings;
OptimiserSettings m_optimiserSettings; OptimiserSettings m_optimiserSettings;
std::map<std::string, unsigned> m_sourceIndices; std::map<std::string, unsigned> m_sourceIndices;

View File

@ -164,7 +164,7 @@ string IRGenerator::generate(
} }
)"); )");
resetContext(_contract); resetContext(_contract, ExecutionContext::Creation);
for (VariableDeclaration const* var: ContractType(_contract).immutableVariables()) for (VariableDeclaration const* var: ContractType(_contract).immutableVariables())
m_context.registerImmutableVariable(*var); m_context.registerImmutableVariable(*var);
@ -211,7 +211,7 @@ string IRGenerator::generate(
bool creationInvolvesAssembly = m_context.inlineAssemblySeen(); bool creationInvolvesAssembly = m_context.inlineAssemblySeen();
t("memoryInitCreation", memoryInit(!creationInvolvesAssembly)); t("memoryInitCreation", memoryInit(!creationInvolvesAssembly));
resetContext(_contract); resetContext(_contract, ExecutionContext::Deployed);
// NOTE: Function pointers can be passed from creation code via storage variables. We need to // NOTE: Function pointers can be passed from creation code via storage variables. We need to
// get all the functions they could point to into the dispatch functions even if they're never // get all the functions they could point to into the dispatch functions even if they're never
@ -1049,7 +1049,7 @@ string IRGenerator::memoryInit(bool _useMemoryGuard)
).render(); ).render();
} }
void IRGenerator::resetContext(ContractDefinition const& _contract) void IRGenerator::resetContext(ContractDefinition const& _contract, ExecutionContext _context)
{ {
solAssert( solAssert(
m_context.functionGenerationQueueEmpty(), m_context.functionGenerationQueueEmpty(),
@ -1063,7 +1063,7 @@ void IRGenerator::resetContext(ContractDefinition const& _contract)
m_context.internalDispatchClean(), m_context.internalDispatchClean(),
"Reset internal dispatch map without consuming it." "Reset internal dispatch map without consuming it."
); );
IRGenerationContext newContext(m_evmVersion, m_context.revertStrings(), m_optimiserSettings, m_context.sourceIndices()); IRGenerationContext newContext(m_evmVersion, _context, m_context.revertStrings(), m_optimiserSettings, m_context.sourceIndices());
newContext.copyFunctionIDsFrom(m_context); newContext.copyFunctionIDsFrom(m_context);
m_context = move(newContext); m_context = move(newContext);

View File

@ -39,6 +39,8 @@ class SourceUnit;
class IRGenerator class IRGenerator
{ {
public: public:
using ExecutionContext = IRGenerationContext::ExecutionContext;
IRGenerator( IRGenerator(
langutil::EVMVersion _evmVersion, langutil::EVMVersion _evmVersion,
RevertStrings _revertStrings, RevertStrings _revertStrings,
@ -47,7 +49,7 @@ public:
): ):
m_evmVersion(_evmVersion), m_evmVersion(_evmVersion),
m_optimiserSettings(_optimiserSettings), m_optimiserSettings(_optimiserSettings),
m_context(_evmVersion, _revertStrings, std::move(_optimiserSettings), std::move(_sourceIndices)), m_context(_evmVersion, ExecutionContext::Creation, _revertStrings, std::move(_optimiserSettings), std::move(_sourceIndices)),
m_utils(_evmVersion, m_context.revertStrings(), m_context.functionCollector()) m_utils(_evmVersion, m_context.revertStrings(), m_context.functionCollector())
{} {}
@ -115,7 +117,7 @@ private:
/// to perform memory optimizations. /// to perform memory optimizations.
std::string memoryInit(bool _useMemoryGuard); std::string memoryInit(bool _useMemoryGuard);
void resetContext(ContractDefinition const& _contract); void resetContext(ContractDefinition const& _contract, ExecutionContext _context);
langutil::EVMVersion const m_evmVersion; langutil::EVMVersion const m_evmVersion;
OptimiserSettings const m_optimiserSettings; OptimiserSettings const m_optimiserSettings;

View File

@ -2934,6 +2934,13 @@ IRVariable IRGeneratorForStatements::readFromLValue(IRLValue const& _lvalue)
solUnimplementedAssert(_lvalue.type.isValueType(), ""); solUnimplementedAssert(_lvalue.type.isValueType(), "");
solUnimplementedAssert(_lvalue.type.sizeOnStack() == 1, ""); solUnimplementedAssert(_lvalue.type.sizeOnStack() == 1, "");
solAssert(_lvalue.type == *_immutable.variable->type(), ""); solAssert(_lvalue.type == *_immutable.variable->type(), "");
if (m_context.executionContext() == IRGenerationContext::ExecutionContext::Creation)
define(result) <<
m_utils.readFromMemory(*_immutable.variable->type()) <<
"(" <<
to_string(m_context.immutableMemoryOffset(*_immutable.variable)) <<
")\n";
else
define(result) << "loadimmutable(\"" << to_string(_immutable.variable->id()) << "\")\n"; define(result) << "loadimmutable(\"" << to_string(_immutable.variable->id()) << "\")\n";
}, },
[&](IRLValue::Tuple const&) { [&](IRLValue::Tuple const&) {

View File

@ -0,0 +1,20 @@
contract C {
uint immutable public a;
uint immutable public b;
uint immutable public c;
uint immutable public d;
constructor() {
a = 1;
b = a;
c = b;
d = c;
}
}
// ====
// compileViaYul: also
// ----
// a() -> 1
// b() -> 1
// c() -> 1
// d() -> 1

View File

@ -0,0 +1,22 @@
contract A {
uint8 immutable a;
uint8 x;
constructor() {
a = 3;
x = readA();
}
function readX() public view returns (uint8) {
return x;
}
function readA() public view returns (uint8) {
return a;
}
}
// ====
// compileViaYul: also
// ----
// readX() -> 3
// readA() -> 3

View File

@ -0,0 +1,17 @@
contract A {
uint8 immutable a;
uint8 x;
constructor() {
a = 3;
x = a;
}
function readX() public view returns (uint8) {
return x;
}
}
// ====
// compileViaYul: also
// ----
// readX() -> 3

View File

@ -7,4 +7,4 @@ contract C {
function f() public pure returns (uint) { return 3 + x; } function f() public pure returns (uint) { return 3 + x; }
} }
// ---- // ----
// TypeError 7733: (136-137): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (136-137): Immutable variables cannot be read before they are initialized.

View File

@ -5,4 +5,4 @@ contract C {
} }
} }
// ---- // ----
// TypeError 7733: (71-72): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (71-72): Immutable variables cannot be read before they are initialized.

View File

@ -11,4 +11,4 @@ contract C {
function f(uint a) internal pure {} function f(uint a) internal pure {}
} }
// ---- // ----
// TypeError 7733: (119-120): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (119-120): Immutable variables cannot be read before they are initialized.

View File

@ -5,4 +5,4 @@ contract C {
} }
} }
// ---- // ----
// TypeError 7733: (63-64): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 3969: (63-64): Immutable variables must be initialized using an assignment.

View File

@ -5,4 +5,4 @@ contract C {
} }
} }
// ---- // ----
// TypeError 7733: (70-71): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 3969: (70-71): Immutable variables must be initialized using an assignment.

View File

@ -5,5 +5,4 @@ contract C {
} }
} }
// ---- // ----
// TypeError 1574: (74-75): Immutable state variable already initialized. // TypeError 2718: (74-75): Immutable variables cannot be modified after initialization.
// TypeError 7733: (74-75): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it.

View File

@ -4,4 +4,4 @@ contract C {
function f() public pure returns (uint) { return 3 + x; } function f() public pure returns (uint) { return 3 + x; }
} }
// ---- // ----
// TypeError 7733: (99-100): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (99-100): Immutable variables cannot be read before they are initialized.

View File

@ -11,3 +11,4 @@ contract C is B(C.f) {
} }
// ---- // ----
// TypeError 1581: (200-201): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor. // TypeError 1581: (200-201): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor.
// TypeError 1574: (109-110): Immutable state variable already initialized.

View File

@ -10,4 +10,4 @@ contract C is B(C.f) {
function f() internal returns(uint) { return x + 2; } function f() internal returns(uint) { return x + 2; }
} }
// ---- // ----
// TypeError 7733: (200-201): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (200-201): Immutable variables cannot be read before they are initialized.

View File

@ -5,4 +5,4 @@ contract C {
} }
} }
// ---- // ----
// TypeError 7733: (63-64): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 3969: (63-64): Immutable variables must be initialized using an assignment.

View File

@ -7,5 +7,5 @@ contract C {
} }
} }
// ---- // ----
// TypeError 7733: (77-78): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 3969: (77-78): Immutable variables must be initialized using an assignment.
// TypeError 7733: (86-87): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 3969: (86-87): Immutable variables must be initialized using an assignment.

View File

@ -5,4 +5,4 @@ contract C {
function f() internal returns(uint) { return x; } function f() internal returns(uint) { return x; }
} }
// ---- // ----
// TypeError 7733: (107-108): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (107-108): Immutable variables cannot be read before they are initialized.

View File

@ -13,4 +13,4 @@ contract C is B(C.y) {
} }
} }
// ---- // ----
// TypeError 7733: (104-107): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (104-107): Immutable variables cannot be read before they are initialized.

View File

@ -16,4 +16,4 @@ contract C is B {
} }
} }
// ---- // ----
// TypeError 7733: (253-254): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (253-254): Immutable variables cannot be read before they are initialized.

View File

@ -11,9 +11,8 @@ contract C is B {
B.readX; B.readX;
} }
function readX() internal override returns(uint) { function readX() internal pure override returns(uint) {
return 3; return 3;
} }
} }
// ---- // ----
// TypeError 7733: (109-110): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it.

View File

@ -11,9 +11,8 @@ contract C is B {
super.readX(); super.readX();
} }
function readX() internal view override returns(uint) { function readX() internal pure override returns(uint) {
return 1; return 1;
} }
} }
// ---- // ----
// TypeError 7733: (114-115): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it.

View File

@ -18,4 +18,4 @@ contract C is B {
} }
} }
// ---- // ----
// TypeError 7733: (245-246): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (245-246): Immutable variables cannot be read before they are initialized.

View File

@ -0,0 +1,7 @@
contract C { constructor(uint) {} }
contract D is C {
uint immutable t;
constructor() C(t=2) {}
}
// ----
// TypeError 1581: (92-93): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor.

View File

@ -0,0 +1,6 @@
contract C { constructor(uint) {} }
contract D is C(D.t = 2) {
uint immutable t;
}
// ----
// TypeError 1581: (52-55): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor.

View File

@ -0,0 +1,7 @@
contract D {
uint immutable t;
modifier m(uint) { _; }
constructor() m(t=2) {}
}
// ----
// TypeError 1581: (77-78): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor.

View File

@ -26,4 +26,4 @@ contract C is A, B {
} }
} }
// ---- // ----
// TypeError 7733: (489-490): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (489-490): Immutable variables cannot be read before they are initialized.

View File

@ -26,4 +26,4 @@ contract C is A, B {
} }
} }
// ---- // ----
// TypeError 7733: (493-494): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (493-494): Immutable variables cannot be read before they are initialized.

View File

@ -17,4 +17,4 @@ contract C is B {
} }
// ---- // ----
// TypeError 7733: (202-203): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (202-203): Immutable variables cannot be read before they are initialized.

View File

@ -0,0 +1,12 @@
abstract contract A {
uint public t;
constructor() { t = f(); }
function f() virtual view internal returns (uint);
}
contract B is A {
uint immutable x = 2;
function f() override view internal returns (uint) { return x; }
}
// ----
// TypeError 7733: (223-224): Immutable variables cannot be read before they are initialized.

View File

@ -0,0 +1,7 @@
contract C {
uint immutable t = 2;
uint x = f();
function f() internal pure returns (uint) { return t; }
}
// ----
// TypeError 7733: (106-107): Immutable variables cannot be read before they are initialized.

View File

@ -0,0 +1,16 @@
contract C {
uint immutable x ;
constructor()
{
readX();
x = 3;
readX();
}
function readX() public view returns(uint) {
return x;
}
}
// ----
// TypeError 7733: (145-146): Immutable variables cannot be read before they are initialized.

View File

@ -3,4 +3,4 @@ contract C {
uint y = x; uint y = x;
} }
// ---- // ----
// TypeError 7733: (52-53): Immutable variables cannot be read during contract creation time, which means they cannot be read in the constructor or any function or modifier called from it. // TypeError 7733: (52-53): Immutable variables cannot be read before they are initialized.

View File

@ -4,5 +4,5 @@ contract C {
uint immutable y = 5; uint immutable y = 5;
} }
// ---- // ----
// TypeError 1581: (62-63): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor.
// TypeError 1581: (66-67): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor. // TypeError 1581: (66-67): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor.
// TypeError 1581: (62-63): Cannot write to immutable here: Immutable variables can only be initialized inline or assigned directly in the constructor.