Browse Source

Workaround for buggy LLVM ctlz used in counting EXP cost

cl-refactor
Paweł Bylica 10 years ago
parent
commit
a64b4d9f71
  1. 21
      evmjit/libevmjit/GasMeter.cpp

21
evmjit/libevmjit/GasMeter.cpp

@ -217,12 +217,21 @@ void GasMeter::countExp(llvm::Value* _exponent)
// OPT: Can gas update be done in exp algorithm? // OPT: Can gas update be done in exp algorithm?
auto ctlz = llvm::Intrinsic::getDeclaration(getModule(), llvm::Intrinsic::ctlz, Type::Word); auto t = llvm::APInt{256, 1};
auto lz256 = m_builder.CreateCall2(ctlz, _exponent, m_builder.getInt1(false)); auto c = m_builder.CreateSelect(m_builder.CreateICmpUGE(_exponent, Constant::get(t)), m_builder.getInt64(1), m_builder.getInt64(0));
auto lz = m_builder.CreateTrunc(lz256, Type::Gas, "lz"); for (auto i = 2; i <= 32; ++i)
auto sigBits = m_builder.CreateSub(m_builder.getInt64(256), lz, "sigBits"); {
auto sigBytes = m_builder.CreateUDiv(m_builder.CreateAdd(sigBits, m_builder.getInt64(7)), m_builder.getInt64(8)); t <<= 8;
count(m_builder.CreateNUWMul(sigBytes, m_builder.getInt64(c_expByteGas))); c = m_builder.CreateSelect(m_builder.CreateICmpUGE(_exponent, Constant::get(t)), m_builder.getInt64(i), c);
}
// FIXME: Does not work because of LLVM bug: https://llvm.org/bugs/show_bug.cgi?id=22304
// auto ctlz = llvm::Intrinsic::getDeclaration(getModule(), llvm::Intrinsic::ctlz, Type::Word);
// auto lz256 = m_builder.CreateCall2(ctlz, _exponent, m_builder.getInt1(false));
// auto lz = m_builder.CreateTrunc(lz256, Type::Gas, "lz");
// auto sigBits = m_builder.CreateSub(m_builder.getInt64(256), lz, "sigBits");
// auto sigBytes = m_builder.CreateUDiv(m_builder.CreateAdd(sigBits, m_builder.getInt64(7)), m_builder.getInt64(8));
count(m_builder.CreateNUWMul(c, m_builder.getInt64(c_expByteGas)));
} }
void GasMeter::countSStore(Ext& _ext, llvm::Value* _index, llvm::Value* _newValue) void GasMeter::countSStore(Ext& _ext, llvm::Value* _index, llvm::Value* _newValue)

Loading…
Cancel
Save