Skip to content

Commit

Permalink
Fix the try-catch lowering hack
Browse files Browse the repository at this point in the history
This patch moves the success tag tracking of the try-call ast to its
annotation instead of CompilerContext. This fixes the lowering of nested
try-catch statements.

This fixes CPR-1705.
  • Loading branch information
abinavpp authored and antonbaliasnikov committed Jun 5, 2024
1 parent 33af03f commit 9abea9d
Show file tree
Hide file tree
Showing 5 changed files with 26 additions and 13 deletions.
7 changes: 7 additions & 0 deletions libsolidity/ast/ASTAnnotations.h
Original file line number Diff line number Diff line change
Expand Up @@ -317,6 +317,13 @@ struct FunctionCallAnnotation: ExpressionAnnotation
FunctionCallKind kind = FunctionCallKind::Unset;
/// If true, this is the external call of a try statement.
bool tryCall = false;

// HACK!
// We track the success tag here for the `TryStatement` lowering. This is to avoid the redundant status check and
// the conditional jump. Such patterns can confuse the zksolc translator.
//
// uint32_t since Assembly::new[Push]Tag() asserts that the tag is 32 bits.
std::optional<uint32_t> tryCallSuccessTag;
};

}
5 changes: 0 additions & 5 deletions libsolidity/codegen/CompilerContext.h
Original file line number Diff line number Diff line change
Expand Up @@ -349,11 +349,6 @@ class CompilerContext

RevertStrings revertStrings() const { return m_revertStrings; }

// HACK!
// We track the success tag here for the `TryStatement` lowering. This is to avoid the redundant status check and
// the conditional jump. Such patterns can confuse the zksolc translator.
evmasm::AssemblyItem currTryCallSuccessTag{evmasm::AssemblyItemType::UndefinedItem};

private:
/// Updates source location set in the assembly.
void updateSourceLocation();
Expand Down
12 changes: 9 additions & 3 deletions libsolidity/codegen/ContractCompiler.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -932,7 +932,10 @@ bool ContractCompiler::visit(TryStatement const& _tryStatement)
StackHeightChecker checker(m_context);
CompilerContext::LocationSetter locationSetter(m_context, _tryStatement);

compileExpression(_tryStatement.externalCall());
auto* externalCall = dynamic_cast<FunctionCall const*>(&_tryStatement.externalCall());
solAssert(externalCall && externalCall->annotation().tryCall, "");
compileExpression(*externalCall);

int const returnSize = static_cast<int>(_tryStatement.externalCall().annotation().type->sizeOnStack());

// Stack: [ return values] <success flag>
Expand All @@ -945,8 +948,11 @@ bool ContractCompiler::visit(TryStatement const& _tryStatement)

evmasm::AssemblyItem endTag = m_context.appendJumpToNew();

solAssert(m_context.currTryCallSuccessTag.type() == AssemblyItemType::Tag, "");
m_context << m_context.currTryCallSuccessTag;
auto& successTag = externalCall->annotation().tryCallSuccessTag;
solAssert(successTag, "");
m_context << AssemblyItem(AssemblyItemType::Tag, *successTag);
successTag.reset();

m_context.adjustStackOffset(returnSize);
{
// Success case.
Expand Down
12 changes: 8 additions & 4 deletions libsolidity/codegen/ExpressionCompiler.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -706,7 +706,8 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
case FunctionType::Kind::External:
case FunctionType::Kind::DelegateCall:
_functionCall.expression().accept(*this);
appendExternalFunctionCall(function, arguments, _functionCall.annotation().tryCall);
appendExternalFunctionCall(
function, arguments, _functionCall.annotation().tryCall, &_functionCall.annotation());
break;
case FunctionType::Kind::BareCallCode:
solAssert(false, "Callcode has been removed.");
Expand Down Expand Up @@ -766,7 +767,8 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
// If this is a try call, return "<address> 1" in the success case and
// "0" in the error case.
AssemblyItem errorCase = m_context.appendConditionalJump();
m_context.currTryCallSuccessTag = m_context.appendJumpToNew();
_functionCall.annotation().tryCallSuccessTag
= m_context.appendJumpToNew().data().convert_to<uint32_t>();
m_context.adjustStackOffset(1);
m_context << errorCase;
}
Expand Down Expand Up @@ -2238,7 +2240,8 @@ void ExpressionCompiler::appendShiftOperatorCode(Token _operator, Type const& _v
void ExpressionCompiler::appendExternalFunctionCall(
FunctionType const& _functionType,
vector<ASTPointer<Expression const>> const& _arguments,
bool _tryCall
bool _tryCall,
FunctionCallAnnotation* _annotation
)
{
solAssert(
Expand Down Expand Up @@ -2516,8 +2519,9 @@ void ExpressionCompiler::appendExternalFunctionCall(

if (_tryCall)
{
solAssert(_annotation, "");
// Success branch will reach this, failure branch will directly jump to endTag.
m_context.currTryCallSuccessTag = m_context.appendJumpToNew();
_annotation->tryCallSuccessTag = m_context.appendJumpToNew().data().convert_to<uint32_t>();
m_context.adjustStackOffset(1);
m_context << endTag;
}
Expand Down
3 changes: 2 additions & 1 deletion libsolidity/codegen/ExpressionCompiler.h
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,8 @@ class ExpressionCompiler: private ASTConstVisitor
void appendExternalFunctionCall(
FunctionType const& _functionType,
std::vector<ASTPointer<Expression const>> const& _arguments,
bool _tryCall
bool _tryCall,
FunctionCallAnnotation* _annotation = nullptr
);
/// Appends code that evaluates a single expression and moves the result to memory. The memory offset is
/// expected to be on the stack and is updated by this call.
Expand Down

0 comments on commit 9abea9d

Please sign in to comment.