mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Yul codegen for immutables.
This commit is contained in:
parent
debcc8c056
commit
51ccb1519f
@ -22,6 +22,7 @@
|
|||||||
|
|
||||||
#include <libsolidity/codegen/YulUtilFunctions.h>
|
#include <libsolidity/codegen/YulUtilFunctions.h>
|
||||||
#include <libsolidity/codegen/ABIFunctions.h>
|
#include <libsolidity/codegen/ABIFunctions.h>
|
||||||
|
#include <libsolidity/codegen/CompilerUtils.h>
|
||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
#include <libsolidity/ast/TypeProvider.h>
|
#include <libsolidity/ast/TypeProvider.h>
|
||||||
|
|
||||||
@ -76,6 +77,36 @@ IRVariable const& IRGenerationContext::localVariable(VariableDeclaration const&
|
|||||||
return m_localVariables.at(&_varDecl);
|
return m_localVariables.at(&_varDecl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void IRGenerationContext::registerImmutableVariable(VariableDeclaration const& _variable)
|
||||||
|
{
|
||||||
|
solAssert(_variable.immutable(), "Attempted to register a non-immutable variable as immutable.");
|
||||||
|
solUnimplementedAssert(
|
||||||
|
_variable.annotation().type->isValueType(),
|
||||||
|
"Only immutable variables of value type are supported."
|
||||||
|
);
|
||||||
|
solAssert(m_reservedMemory.has_value(), "Reserved memory has already been reset.");
|
||||||
|
m_immutableVariables[&_variable] = CompilerUtils::generalPurposeMemoryStart + *m_reservedMemory;
|
||||||
|
solAssert(_variable.annotation().type->memoryHeadSize() == 32, "Memory writes might overlap.");
|
||||||
|
*m_reservedMemory += _variable.annotation().type->memoryHeadSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t IRGenerationContext::immutableMemoryOffset(VariableDeclaration const& _variable) const
|
||||||
|
{
|
||||||
|
solAssert(
|
||||||
|
m_immutableVariables.count(&_variable),
|
||||||
|
"Unknown immutable variable: " + _variable.name()
|
||||||
|
);
|
||||||
|
return m_immutableVariables.at(&_variable);
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t IRGenerationContext::reservedMemory()
|
||||||
|
{
|
||||||
|
solAssert(m_reservedMemory.has_value(), "Reserved memory was used before.");
|
||||||
|
size_t reservedMemory = *m_reservedMemory;
|
||||||
|
m_reservedMemory = std::nullopt;
|
||||||
|
return reservedMemory;
|
||||||
|
}
|
||||||
|
|
||||||
void IRGenerationContext::addStateVariable(
|
void IRGenerationContext::addStateVariable(
|
||||||
VariableDeclaration const& _declaration,
|
VariableDeclaration const& _declaration,
|
||||||
u256 _storageOffset,
|
u256 _storageOffset,
|
||||||
|
@ -81,6 +81,17 @@ public:
|
|||||||
bool isLocalVariable(VariableDeclaration const& _varDecl) const { return m_localVariables.count(&_varDecl); }
|
bool isLocalVariable(VariableDeclaration const& _varDecl) const { return m_localVariables.count(&_varDecl); }
|
||||||
IRVariable const& localVariable(VariableDeclaration const& _varDecl);
|
IRVariable const& localVariable(VariableDeclaration const& _varDecl);
|
||||||
|
|
||||||
|
/// Registers an immutable variable of the contract.
|
||||||
|
/// Should only be called at construction time.
|
||||||
|
void registerImmutableVariable(VariableDeclaration const& _varDecl);
|
||||||
|
/// @returns the reserved memory for storing the value of the
|
||||||
|
/// immutable @a _variable during contract creation.
|
||||||
|
size_t immutableMemoryOffset(VariableDeclaration const& _variable) const;
|
||||||
|
/// @returns the reserved memory and resets it to mark it as used.
|
||||||
|
/// Intended to be used only once for initializing the free memory pointer
|
||||||
|
/// to after the area used for immutables.
|
||||||
|
size_t reservedMemory();
|
||||||
|
|
||||||
void addStateVariable(VariableDeclaration const& _varDecl, u256 _storageOffset, unsigned _byteOffset);
|
void addStateVariable(VariableDeclaration const& _varDecl, u256 _storageOffset, unsigned _byteOffset);
|
||||||
bool isStateVariable(VariableDeclaration const& _varDecl) const { return m_stateVariables.count(&_varDecl); }
|
bool isStateVariable(VariableDeclaration const& _varDecl) const { return m_stateVariables.count(&_varDecl); }
|
||||||
std::pair<u256, unsigned> storageLocationOfVariable(VariableDeclaration const& _varDecl) const
|
std::pair<u256, unsigned> storageLocationOfVariable(VariableDeclaration const& _varDecl) const
|
||||||
@ -123,6 +134,12 @@ private:
|
|||||||
OptimiserSettings m_optimiserSettings;
|
OptimiserSettings m_optimiserSettings;
|
||||||
ContractDefinition const* m_mostDerivedContract = nullptr;
|
ContractDefinition const* m_mostDerivedContract = nullptr;
|
||||||
std::map<VariableDeclaration const*, IRVariable> m_localVariables;
|
std::map<VariableDeclaration const*, IRVariable> m_localVariables;
|
||||||
|
/// Memory offsets reserved for the values of immutable variables during contract creation.
|
||||||
|
/// This map is empty in the runtime context.
|
||||||
|
std::map<VariableDeclaration const*, size_t> m_immutableVariables;
|
||||||
|
/// Total amount of reserved memory. Reserved memory is used to store
|
||||||
|
/// immutable variables during contract creation.
|
||||||
|
std::optional<size_t> m_reservedMemory = {0};
|
||||||
/// Storage offsets of state variables
|
/// Storage offsets of state variables
|
||||||
std::map<VariableDeclaration const*, std::pair<u256, unsigned>> m_stateVariables;
|
std::map<VariableDeclaration const*, std::pair<u256, unsigned>> m_stateVariables;
|
||||||
MultiUseYulFunctionCollector m_functions;
|
MultiUseYulFunctionCollector m_functions;
|
||||||
|
@ -114,6 +114,8 @@ string IRGenerator::generate(
|
|||||||
)");
|
)");
|
||||||
|
|
||||||
resetContext(_contract);
|
resetContext(_contract);
|
||||||
|
for (VariableDeclaration const* var: ContractType(_contract).immutableVariables())
|
||||||
|
m_context.registerImmutableVariable(*var);
|
||||||
|
|
||||||
t("CreationObject", m_context.creationObjectName(_contract));
|
t("CreationObject", m_context.creationObjectName(_contract));
|
||||||
t("memoryInit", memoryInit());
|
t("memoryInit", memoryInit());
|
||||||
@ -142,6 +144,7 @@ string IRGenerator::generate(
|
|||||||
t("subObjects", subObjectSources(m_context.subObjectsCreated()));
|
t("subObjects", subObjectSources(m_context.subObjectsCreated()));
|
||||||
|
|
||||||
resetContext(_contract);
|
resetContext(_contract);
|
||||||
|
// Do not register immutables to avoid assignment.
|
||||||
t("RuntimeObject", m_context.runtimeObjectName(_contract));
|
t("RuntimeObject", m_context.runtimeObjectName(_contract));
|
||||||
t("dispatch", dispatchRoutine(_contract));
|
t("dispatch", dispatchRoutine(_contract));
|
||||||
generateQueuedFunctions();
|
generateQueuedFunctions();
|
||||||
@ -200,7 +203,6 @@ string IRGenerator::generateGetter(VariableDeclaration const& _varDecl)
|
|||||||
Type const* type = _varDecl.annotation().type;
|
Type const* type = _varDecl.annotation().type;
|
||||||
|
|
||||||
solAssert(!_varDecl.isConstant(), "");
|
solAssert(!_varDecl.isConstant(), "");
|
||||||
solAssert(!_varDecl.immutable(), "");
|
|
||||||
solAssert(_varDecl.isStateVariable(), "");
|
solAssert(_varDecl.isStateVariable(), "");
|
||||||
|
|
||||||
if (auto const* mappingType = dynamic_cast<MappingType const*>(type))
|
if (auto const* mappingType = dynamic_cast<MappingType const*>(type))
|
||||||
@ -254,17 +256,32 @@ string IRGenerator::generateGetter(VariableDeclaration const& _varDecl)
|
|||||||
solUnimplementedAssert(type->isValueType(), "");
|
solUnimplementedAssert(type->isValueType(), "");
|
||||||
|
|
||||||
return m_context.functionCollector().createFunction(functionName, [&]() {
|
return m_context.functionCollector().createFunction(functionName, [&]() {
|
||||||
pair<u256, unsigned> slot_offset = m_context.storageLocationOfVariable(_varDecl);
|
if (_varDecl.immutable())
|
||||||
|
{
|
||||||
|
solUnimplementedAssert(type->sizeOnStack() == 1, "");
|
||||||
|
return Whiskers(R"(
|
||||||
|
function <functionName>() -> rval {
|
||||||
|
rval := loadimmutable("<id>")
|
||||||
|
}
|
||||||
|
)")
|
||||||
|
("functionName", functionName)
|
||||||
|
("id", to_string(_varDecl.id()))
|
||||||
|
.render();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
pair<u256, unsigned> slot_offset = m_context.storageLocationOfVariable(_varDecl);
|
||||||
|
|
||||||
return Whiskers(R"(
|
return Whiskers(R"(
|
||||||
function <functionName>() -> rval {
|
function <functionName>() -> rval {
|
||||||
rval := <readStorage>(<slot>)
|
rval := <readStorage>(<slot>)
|
||||||
}
|
}
|
||||||
)")
|
)")
|
||||||
("functionName", functionName)
|
("functionName", functionName)
|
||||||
("readStorage", m_utils.readFromStorage(*type, slot_offset.second, false))
|
("readStorage", m_utils.readFromStorage(*type, slot_offset.second, false))
|
||||||
("slot", slot_offset.first.str())
|
("slot", slot_offset.first.str())
|
||||||
.render();
|
.render();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -325,7 +342,7 @@ string IRGenerator::initStateVariables(ContractDefinition const& _contract)
|
|||||||
{
|
{
|
||||||
IRGeneratorForStatements generator{m_context, m_utils};
|
IRGeneratorForStatements generator{m_context, m_utils};
|
||||||
for (VariableDeclaration const* variable: _contract.stateVariables())
|
for (VariableDeclaration const* variable: _contract.stateVariables())
|
||||||
if (!variable->isConstant() && !variable->immutable())
|
if (!variable->isConstant())
|
||||||
generator.initializeStateVar(*variable);
|
generator.initializeStateVar(*variable);
|
||||||
|
|
||||||
return generator.code();
|
return generator.code();
|
||||||
@ -391,10 +408,41 @@ void IRGenerator::generateImplicitConstructors(ContractDefinition const& _contra
|
|||||||
string IRGenerator::deployCode(ContractDefinition const& _contract)
|
string IRGenerator::deployCode(ContractDefinition const& _contract)
|
||||||
{
|
{
|
||||||
Whiskers t(R"X(
|
Whiskers t(R"X(
|
||||||
|
<#loadImmutables>
|
||||||
|
let <var> := mload(<memoryOffset>)
|
||||||
|
</loadImmutables>
|
||||||
|
|
||||||
codecopy(0, dataoffset("<object>"), datasize("<object>"))
|
codecopy(0, dataoffset("<object>"), datasize("<object>"))
|
||||||
|
|
||||||
|
<#storeImmutables>
|
||||||
|
setimmutable("<immutableName>", <var>)
|
||||||
|
</storeImmutables>
|
||||||
|
|
||||||
return(0, datasize("<object>"))
|
return(0, datasize("<object>"))
|
||||||
)X");
|
)X");
|
||||||
t("object", m_context.runtimeObjectName(_contract));
|
t("object", m_context.runtimeObjectName(_contract));
|
||||||
|
|
||||||
|
vector<map<string, string>> loadImmutables;
|
||||||
|
vector<map<string, string>> storeImmutables;
|
||||||
|
|
||||||
|
for (VariableDeclaration const* immutable: ContractType(_contract).immutableVariables())
|
||||||
|
{
|
||||||
|
solUnimplementedAssert(immutable->type()->isValueType(), "");
|
||||||
|
solUnimplementedAssert(immutable->type()->sizeOnStack() == 1, "");
|
||||||
|
string yulVar = m_context.newYulVariable();
|
||||||
|
loadImmutables.emplace_back(map<string, string>{
|
||||||
|
{"var"s, yulVar},
|
||||||
|
{"memoryOffset"s, to_string(m_context.immutableMemoryOffset(*immutable))}
|
||||||
|
});
|
||||||
|
storeImmutables.emplace_back(map<string, string>{
|
||||||
|
{"var"s, yulVar},
|
||||||
|
{"immutableName"s, to_string(immutable->id())}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
t("loadImmutables", std::move(loadImmutables));
|
||||||
|
// reverse order to ease stack strain
|
||||||
|
reverse(storeImmutables.begin(), storeImmutables.end());
|
||||||
|
t("storeImmutables", std::move(storeImmutables));
|
||||||
return t.render();
|
return t.render();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -489,9 +537,9 @@ string IRGenerator::memoryInit()
|
|||||||
// and thus can assume all memory to be zero, including the contents of
|
// and thus can assume all memory to be zero, including the contents of
|
||||||
// the "zero memory area" (the position CompilerUtils::zeroPointer points to).
|
// the "zero memory area" (the position CompilerUtils::zeroPointer points to).
|
||||||
return
|
return
|
||||||
Whiskers{"mstore(<memPtr>, <generalPurposeStart>)"}
|
Whiskers{"mstore(<memPtr>, <freeMemoryStart>)"}
|
||||||
("memPtr", to_string(CompilerUtils::freeMemoryPointer))
|
("memPtr", to_string(CompilerUtils::freeMemoryPointer))
|
||||||
("generalPurposeStart", to_string(CompilerUtils::generalPurposeMemoryStart))
|
("freeMemoryStart", to_string(CompilerUtils::generalPurposeMemoryStart + m_context.reservedMemory()))
|
||||||
.render();
|
.render();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -140,20 +140,21 @@ string IRGeneratorForStatements::code() const
|
|||||||
|
|
||||||
void IRGeneratorForStatements::initializeStateVar(VariableDeclaration const& _varDecl)
|
void IRGeneratorForStatements::initializeStateVar(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
solAssert(m_context.isStateVariable(_varDecl), "Must be a state variable.");
|
solAssert(_varDecl.immutable() || m_context.isStateVariable(_varDecl), "Must be immutable or a state variable.");
|
||||||
solAssert(!_varDecl.isConstant(), "");
|
solAssert(!_varDecl.isConstant(), "");
|
||||||
solAssert(!_varDecl.immutable(), "");
|
if (!_varDecl.value())
|
||||||
if (_varDecl.value())
|
return;
|
||||||
{
|
|
||||||
_varDecl.value()->accept(*this);
|
_varDecl.value()->accept(*this);
|
||||||
writeToLValue(IRLValue{
|
writeToLValue(
|
||||||
*_varDecl.annotation().type,
|
_varDecl.immutable() ?
|
||||||
IRLValue::Storage{
|
IRLValue{*_varDecl.annotation().type, IRLValue::Immutable{&_varDecl}} :
|
||||||
util::toCompactHexWithPrefix(m_context.storageLocationOfVariable(_varDecl).first),
|
IRLValue{*_varDecl.annotation().type, IRLValue::Storage{
|
||||||
m_context.storageLocationOfVariable(_varDecl).second
|
util::toCompactHexWithPrefix(m_context.storageLocationOfVariable(_varDecl).first),
|
||||||
}
|
m_context.storageLocationOfVariable(_varDecl).second
|
||||||
}, *_varDecl.value());
|
}},
|
||||||
}
|
*_varDecl.value()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void IRGeneratorForStatements::initializeLocalVar(VariableDeclaration const& _varDecl)
|
void IRGeneratorForStatements::initializeLocalVar(VariableDeclaration const& _varDecl)
|
||||||
@ -1517,8 +1518,12 @@ void IRGeneratorForStatements::handleVariableReference(
|
|||||||
// If the value is visited twice, `defineExpression` is called twice on
|
// If the value is visited twice, `defineExpression` is called twice on
|
||||||
// the same expression.
|
// the same expression.
|
||||||
solUnimplementedAssert(!_variable.isConstant(), "");
|
solUnimplementedAssert(!_variable.isConstant(), "");
|
||||||
solUnimplementedAssert(!_variable.immutable(), "");
|
if (_variable.isStateVariable() && _variable.immutable())
|
||||||
if (m_context.isLocalVariable(_variable))
|
setLValue(_referencingExpression, IRLValue{
|
||||||
|
*_variable.annotation().type,
|
||||||
|
IRLValue::Immutable{&_variable}
|
||||||
|
});
|
||||||
|
else if (m_context.isLocalVariable(_variable))
|
||||||
setLValue(_referencingExpression, IRLValue{
|
setLValue(_referencingExpression, IRLValue{
|
||||||
*_variable.annotation().type,
|
*_variable.annotation().type,
|
||||||
IRLValue::Stack{m_context.localVariable(_variable)}
|
IRLValue::Stack{m_context.localVariable(_variable)}
|
||||||
@ -1939,6 +1944,18 @@ void IRGeneratorForStatements::writeToLValue(IRLValue const& _lvalue, IRVariable
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
[&](IRLValue::Stack const& _stack) { assign(_stack.variable, _value); },
|
[&](IRLValue::Stack const& _stack) { assign(_stack.variable, _value); },
|
||||||
|
[&](IRLValue::Immutable const& _immutable)
|
||||||
|
{
|
||||||
|
solUnimplementedAssert(_lvalue.type.isValueType(), "");
|
||||||
|
solUnimplementedAssert(_lvalue.type.sizeOnStack() == 1, "");
|
||||||
|
solAssert(_lvalue.type == *_immutable.variable->type(), "");
|
||||||
|
size_t memOffset = m_context.immutableMemoryOffset(*_immutable.variable);
|
||||||
|
|
||||||
|
IRVariable prepared(m_context.newYulVariable(), _lvalue.type);
|
||||||
|
define(prepared, _value);
|
||||||
|
|
||||||
|
m_code << "mstore(" << to_string(memOffset) << ", " << prepared.commaSeparatedList() << ")\n";
|
||||||
|
},
|
||||||
[&](IRLValue::Tuple const& _tuple) {
|
[&](IRLValue::Tuple const& _tuple) {
|
||||||
auto components = std::move(_tuple.components);
|
auto components = std::move(_tuple.components);
|
||||||
for (size_t i = 0; i < components.size(); i++)
|
for (size_t i = 0; i < components.size(); i++)
|
||||||
@ -1994,6 +2011,12 @@ IRVariable IRGeneratorForStatements::readFromLValue(IRLValue const& _lvalue)
|
|||||||
[&](IRLValue::Stack const& _stack) {
|
[&](IRLValue::Stack const& _stack) {
|
||||||
define(result, _stack.variable);
|
define(result, _stack.variable);
|
||||||
},
|
},
|
||||||
|
[&](IRLValue::Immutable const& _immutable) {
|
||||||
|
solUnimplementedAssert(_lvalue.type.isValueType(), "");
|
||||||
|
solUnimplementedAssert(_lvalue.type.sizeOnStack() == 1, "");
|
||||||
|
solAssert(_lvalue.type == *_immutable.variable->type(), "");
|
||||||
|
define(result) << "loadimmutable(\"" << to_string(_immutable.variable->id()) << "\")\n";
|
||||||
|
},
|
||||||
[&](IRLValue::Tuple const&) {
|
[&](IRLValue::Tuple const&) {
|
||||||
solAssert(false, "Attempted to read from tuple lvalue.");
|
solAssert(false, "Attempted to read from tuple lvalue.");
|
||||||
}
|
}
|
||||||
|
@ -35,6 +35,10 @@ struct IRLValue
|
|||||||
{
|
{
|
||||||
IRVariable variable;
|
IRVariable variable;
|
||||||
};
|
};
|
||||||
|
struct Immutable
|
||||||
|
{
|
||||||
|
VariableDeclaration const* variable = nullptr;
|
||||||
|
};
|
||||||
struct Storage
|
struct Storage
|
||||||
{
|
{
|
||||||
std::string const slot;
|
std::string const slot;
|
||||||
@ -59,7 +63,7 @@ struct IRLValue
|
|||||||
{
|
{
|
||||||
std::vector<std::optional<IRLValue>> components;
|
std::vector<std::optional<IRLValue>> components;
|
||||||
};
|
};
|
||||||
std::variant<Stack, Storage, Memory, Tuple> kind;
|
std::variant<Stack, Immutable, Storage, Memory, Tuple> kind;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -4,5 +4,7 @@ contract A {
|
|||||||
return a;
|
return a;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
// ----
|
// ----
|
||||||
// f() -> 2
|
// f() -> 2
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
contract C {
|
contract C {
|
||||||
uint immutable public x = 1;
|
uint immutable public x = 1;
|
||||||
}
|
}
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
// ----
|
// ----
|
||||||
// x() -> 1
|
// x() -> 1
|
||||||
|
@ -26,5 +26,7 @@ contract D is B, C {
|
|||||||
return (a, b, c, d);
|
return (a, b, c, d);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
// ----
|
// ----
|
||||||
// f() -> 4, 3, 2, 1
|
// f() -> 4, 3, 2, 1
|
||||||
|
@ -10,6 +10,8 @@ contract C {
|
|||||||
return z();
|
return z();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
// ----
|
// ----
|
||||||
// f() -> 7
|
// f() -> 7
|
||||||
// callZ() -> 7
|
// callZ() -> 7
|
||||||
|
@ -9,5 +9,7 @@ contract C {
|
|||||||
return (x+x,y);
|
return (x+x,y);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
// ----
|
// ----
|
||||||
// f() -> 84, 23
|
// f() -> 84, 23
|
||||||
|
@ -13,6 +13,8 @@ contract C {
|
|||||||
return (x+x,y);
|
return (x+x,y);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
// ----
|
// ----
|
||||||
// constructor(): 3 ->
|
// constructor(): 3 ->
|
||||||
// f() -> 84, 23
|
// f() -> 84, 23
|
||||||
|
Loading…
Reference in New Issue
Block a user