diff --git a/Changelog.md b/Changelog.md index 741f045a8915..40ff714abb4d 100644 --- a/Changelog.md +++ b/Changelog.md @@ -14,6 +14,10 @@ Breaking changes: * Syntax: Abstract contracts need to be marked explicitly as abstract by using the ``abstract`` keyword. * Inline Assembly: Only strict inline assembly is allowed. * Type checker: Resulting type of exponentiation is equal to the type of the base. Also allow signed types for the base. + * Source mappings: Add "modifier depth" as a fifth field in the source mappings. + + + * Language Feature: When overriding a function or modifier, the new keyword ``override`` must be used. When overriding a function or modifier defined in multiple parallel bases, all bases must be listed in parentheses after the keyword like so: ``override(Base1, Base2)`` Language Features: diff --git a/docs/miscellaneous.rst b/docs/miscellaneous.rst index 0111fa4a19ed..598d07ed9084 100644 --- a/docs/miscellaneous.rst +++ b/docs/miscellaneous.rst @@ -222,7 +222,8 @@ Furthermore, the compiler can also generate a mapping from the bytecode to the range in the source code that generated the instruction. This is again important for static analysis tools that operate on bytecode level and for displaying the current position in the source code inside a debugger -or for breakpoint handling. +or for breakpoint handling. This mapping also contains other information, +like the jump type and the modifier depth (see below). Both kinds of source mappings use integer identifiers to refer to source files. The identifier of a source file is stored in @@ -244,12 +245,17 @@ Where ``s`` is the byte-offset to the start of the range in the source file, index mentioned above. The encoding in the source mapping for the bytecode is more complicated: -It is a list of ``s:l:f:j`` separated by ``;``. Each of these +It is a list of ``s:l:f:j:m`` separated by ``;``. Each of these elements corresponds to an instruction, i.e. you cannot use the byte offset but have to use the instruction offset (push instructions are longer than a single byte). -The fields ``s``, ``l`` and ``f`` are as above and ``j`` can be either +The fields ``s``, ``l`` and ``f`` are as above. ``j`` can be either ``i``, ``o`` or ``-`` signifying whether a jump instruction goes into a function, returns from a function or is a regular jump as part of e.g. a loop. +The last field, ``m``, is an integer that denotes the "modifier depth". This depth +is increased whenever the placeholder statement (``_``) is entered in a modifier +and decreased when it is left again. This allows debuggers to track tricky cases +like the same modifier being used twice or multiple placeholder statements being +used in a single modifier. In order to compress these source mappings especially for bytecode, the following rules are used: diff --git a/libevmasm/Assembly.cpp b/libevmasm/Assembly.cpp index ae8d001d7af6..d609aefed3b9 100644 --- a/libevmasm/Assembly.cpp +++ b/libevmasm/Assembly.cpp @@ -86,6 +86,7 @@ AssemblyItem const& Assembly::append(AssemblyItem const& _i) m_items.emplace_back(_i); if (m_items.back().location().isEmpty() && !m_currentSourceLocation.isEmpty()) m_items.back().setLocation(m_currentSourceLocation); + m_items.back().m_modifierDepth = m_currentModifierDepth; return back(); } diff --git a/libevmasm/Assembly.h b/libevmasm/Assembly.h index ebf032ac1531..f79da0e7f4f9 100644 --- a/libevmasm/Assembly.h +++ b/libevmasm/Assembly.h @@ -181,6 +181,8 @@ class Assembly int m_deposit = 0; langutil::SourceLocation m_currentSourceLocation; +public: + size_t m_currentModifierDepth = 0; }; inline std::ostream& operator<<(std::ostream& _out, Assembly const& _a) diff --git a/libevmasm/AssemblyItem.h b/libevmasm/AssemblyItem.h index 95b09d47f1c0..c1d079bb1d56 100644 --- a/libevmasm/AssemblyItem.h +++ b/libevmasm/AssemblyItem.h @@ -146,6 +146,8 @@ class AssemblyItem std::string toAssemblyText() const; + size_t m_modifierDepth = 0; + private: AssemblyItemType m_type; Instruction m_instruction; ///< Only valid if m_type == Operation diff --git a/libsolidity/codegen/CompilerContext.h b/libsolidity/codegen/CompilerContext.h index 869dd0dc80dd..f396e59de08e 100644 --- a/libsolidity/codegen/CompilerContext.h +++ b/libsolidity/codegen/CompilerContext.h @@ -262,6 +262,8 @@ class CompilerContext ScopeGuard([&]{ _compilerContext.popVisitedNodes(); }) { _compilerContext.pushVisitedNodes(&_node); } }; + void setModifierDepth(size_t _modifierDepth) { m_asm->m_currentModifierDepth = _modifierDepth; } + private: /// Searches the inheritance hierarchy towards the base starting from @a _searchStart and returns /// the first function definition that is overwritten by _function. diff --git a/libsolidity/codegen/ContractCompiler.cpp b/libsolidity/codegen/ContractCompiler.cpp index d54b4ed24dd1..4036a44f7901 100644 --- a/libsolidity/codegen/ContractCompiler.cpp +++ b/libsolidity/codegen/ContractCompiler.cpp @@ -564,8 +564,10 @@ bool ContractCompiler::visit(FunctionDefinition const& _function) m_currentFunction = &_function; m_modifierDepth = -1; m_scopeStackHeight.clear(); + m_context.setModifierDepth(0); appendModifierOrFunctionCode(); + m_context.setModifierDepth(0); solAssert(m_returnTags.empty(), ""); // Now we need to re-shuffle the stack. For this we keep a record of the stack layout @@ -1240,6 +1242,7 @@ void ContractCompiler::appendModifierOrFunctionCode() vector addedVariables; m_modifierDepth++; + m_context.setModifierDepth(m_modifierDepth); if (m_modifierDepth >= m_currentFunction->modifiers().size()) { @@ -1293,6 +1296,7 @@ void ContractCompiler::appendModifierOrFunctionCode() m_context.removeVariable(*var); } m_modifierDepth--; + m_context.setModifierDepth(m_modifierDepth); } void ContractCompiler::appendStackVariableInitialisation(VariableDeclaration const& _variable) diff --git a/libsolidity/interface/CompilerStack.cpp b/libsolidity/interface/CompilerStack.cpp index 2b6c00dcbfd0..f8991f6f2fcf 100644 --- a/libsolidity/interface/CompilerStack.cpp +++ b/libsolidity/interface/CompilerStack.cpp @@ -1305,6 +1305,7 @@ string CompilerStack::computeSourceMapping(eth::AssemblyItems const& _items) con int prevStart = -1; int prevLength = -1; int prevSourceIndex = -1; + size_t prevModifierDepth = -1; char prevJump = 0; for (auto const& item: _items) { @@ -1322,19 +1323,24 @@ string CompilerStack::computeSourceMapping(eth::AssemblyItems const& _items) con jump = 'i'; else if (item.getJumpType() == eth::AssemblyItem::JumpType::OutOfFunction) jump = 'o'; + size_t modifierDepth = item.m_modifierDepth; - unsigned components = 4; - if (jump == prevJump) + unsigned components = 5; + if (modifierDepth == prevModifierDepth) { components--; - if (sourceIndex == prevSourceIndex) + if (jump == prevJump) { components--; - if (length == prevLength) + if (sourceIndex == prevSourceIndex) { components--; - if (location.start == prevStart) + if (length == prevLength) + { components--; + if (location.start == prevStart) + components--; + } } } } @@ -1358,6 +1364,12 @@ string CompilerStack::computeSourceMapping(eth::AssemblyItems const& _items) con ret += ':'; if (jump != prevJump) ret += jump; + if (components-- > 0) + { + ret += ':'; + if (modifierDepth != prevModifierDepth) + ret += to_string(modifierDepth); + } } } } @@ -1367,6 +1379,7 @@ string CompilerStack::computeSourceMapping(eth::AssemblyItems const& _items) con prevLength = length; prevSourceIndex = sourceIndex; prevJump = jump; + prevModifierDepth = modifierDepth; } return ret; }