diff --git a/lldb/source/Expression/IRInterpreter.cpp b/lldb/source/Expression/IRInterpreter.cpp --- a/lldb/source/Expression/IRInterpreter.cpp +++ b/lldb/source/Expression/IRInterpreter.cpp @@ -283,11 +283,30 @@ return true; // no offset to apply! SmallVector indices(op_cursor, op_end); + SmallVector const_indices; + + for (Value *idx : indices) { + Constant *constant_idx = dyn_cast(idx); + if (!constant_idx) + return false; + + ConstantInt *constant_int = dyn_cast(constant_idx); + if (!constant_int) { + APInt v; + if (!ResolveConstantValue(v, constant_idx)) + return false; + + constant_int = + cast(ConstantInt::get(idx->getType(), v)); + } + + const_indices.push_back(constant_int); + } Type *src_elem_ty = cast(constant_expr)->getSourceElementType(); uint64_t offset = - m_target_data.getIndexedOffsetInType(src_elem_ty, indices); + m_target_data.getIndexedOffsetInType(src_elem_ty, const_indices); const bool is_signed = true; value += APInt(value.getBitWidth(), offset, is_signed); @@ -465,12 +484,18 @@ case Instruction::BitCast: return CanResolveConstant(constant_expr->getOperand(0)); case Instruction::GetElementPtr: { - ConstantExpr::const_op_iterator op_cursor = constant_expr->op_begin(); - Constant *base = dyn_cast(*op_cursor); - if (!base) - return false; - - return CanResolveConstant(base); + // Check that all operands of `getelementptr` can be constant-resolved. + SmallVector operands(constant_expr->op_begin(), + constant_expr->op_end()); + for (Value *op : operands) { + Constant *constant_op = dyn_cast(op); + if (!constant_op) + return false; + if (!CanResolveConstant(constant_op)) { + return false; + } + } + return true; } } } else { diff --git a/lldb/test/API/lang/cpp/static_members/TestCPPStaticMembers.py b/lldb/test/API/lang/cpp/static_members/TestCPPStaticMembers.py --- a/lldb/test/API/lang/cpp/static_members/TestCPPStaticMembers.py +++ b/lldb/test/API/lang/cpp/static_members/TestCPPStaticMembers.py @@ -41,3 +41,31 @@ self.createTestTarget() self.expect("expression s_c", error=True, startstr="error: use of undeclared identifier 's_d'") + + def test_no_crash_in_IR_arithmetic(self): + """ + Test that LLDB doesn't crash on evaluating specific expression involving + pointer arithmetic and taking the address of a static class member. + See https://bugs.llvm.org/show_bug.cgi?id=52449 + """ + self.build() + lldbutil.run_to_source_breakpoint(self, "// stop in main", lldb.SBFileSpec("main.cpp")) + + # This expression contains the following IR code: + # ... i64 ptrtoint (i32* @_ZN1A3s_cE to i64)) ... + expr = "(int*)100 + (long long)(&A::s_c)" + + # The IR interpreter doesn't support non-const operands to the + # `GetElementPtr` IR instruction, so verify that it correctly fails to + # evaluate expression. + opts = lldb.SBExpressionOptions() + opts.SetAllowJIT(False) + value = self.target().EvaluateExpression(expr, opts) + self.assertTrue(value.GetError().Fail()) + self.assertIn( + "Can't evaluate the expression without a running target", + value.GetError().GetCString()) + + # Evaluating the expression via JIT should work fine. + value = self.target().EvaluateExpression(expr) + self.assertTrue(value.GetError().Success()) diff --git a/lldb/test/API/lang/cpp/static_members/main.cpp b/lldb/test/API/lang/cpp/static_members/main.cpp --- a/lldb/test/API/lang/cpp/static_members/main.cpp +++ b/lldb/test/API/lang/cpp/static_members/main.cpp @@ -11,6 +11,14 @@ long A::s_b = 2; int A::s_c = 3; +// class Foo +// { +// public: +// static int Bar; +// }; + +// int Foo::Bar = 10; + int main() { A my_a; my_a.m_a = 1;