|
@ -33,13 +33,21 @@ namespace |
|
|
static unsigned const c_depthLimit = 1024; |
|
|
static unsigned const c_depthLimit = 1024; |
|
|
|
|
|
|
|
|
/// Upper bound of stack space needed by single CALL/CREATE execution. Set experimentally.
|
|
|
/// Upper bound of stack space needed by single CALL/CREATE execution. Set experimentally.
|
|
|
static size_t const c_singleExecutionStackSize = 12 * 1024; |
|
|
static size_t const c_singleExecutionStackSize = |
|
|
|
|
|
#ifdef NDEBUG |
|
|
|
|
|
12 * 1024; |
|
|
|
|
|
#else |
|
|
|
|
|
33 * 1024; |
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
/// Standard OSX thread stack limit. Should be reasonable for other platforms too.
|
|
|
/// Standard OSX thread stack limit. Should be reasonable for other platforms too.
|
|
|
static size_t const c_defaultStackSize = 512 * 1024; |
|
|
static size_t const c_defaultStackSize = 512 * 1024; |
|
|
|
|
|
|
|
|
|
|
|
/// Stack overhead prior to allocation.
|
|
|
|
|
|
static size_t const c_entryOverhead = 128 * 1024; |
|
|
|
|
|
|
|
|
/// On what depth execution should be offloaded to additional separated stack space.
|
|
|
/// On what depth execution should be offloaded to additional separated stack space.
|
|
|
static unsigned const c_offloadPoint = c_defaultStackSize / c_singleExecutionStackSize; |
|
|
static unsigned const c_offloadPoint = (c_defaultStackSize - c_entryOverhead) / c_singleExecutionStackSize; |
|
|
|
|
|
|
|
|
void goOnOffloadedStack(Executive& _e, OnOpFunc const& _onOp) |
|
|
void goOnOffloadedStack(Executive& _e, OnOpFunc const& _onOp) |
|
|
{ |
|
|
{ |
|
@ -68,13 +76,12 @@ void go(unsigned _depth, Executive& _e, OnOpFunc const& _onOp) |
|
|
{ |
|
|
{ |
|
|
// If in the offloading point we need to switch to additional separated stack space.
|
|
|
// If in the offloading point we need to switch to additional separated stack space.
|
|
|
// Current stack is too small to handle more CALL/CREATE executions.
|
|
|
// Current stack is too small to handle more CALL/CREATE executions.
|
|
|
// It needs to be done only once as newly allocated stack space it enough to handle
|
|
|
// It needs to be done only once as newly allocated stack space it enough to handle
|
|
|
// the rest of the calls up to the depth limit (c_depthLimit).
|
|
|
// the rest of the calls up to the depth limit (c_depthLimit).
|
|
|
#if __GNUC__ |
|
|
|
|
|
if (_depth == c_offloadPoint) |
|
|
if (_depth == c_offloadPoint) |
|
|
goOnOffloadedStack(_e, _onOp); |
|
|
goOnOffloadedStack(_e, _onOp); |
|
|
else |
|
|
else |
|
|
#endif |
|
|
|
|
|
_e.go(_onOp); |
|
|
_e.go(_onOp); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
@ -84,7 +91,11 @@ bool ExtVM::call(CallParameters& _p) |
|
|
Executive e(m_s, lastHashes, depth + 1); |
|
|
Executive e(m_s, lastHashes, depth + 1); |
|
|
if (!e.call(_p, gasPrice, origin)) |
|
|
if (!e.call(_p, gasPrice, origin)) |
|
|
{ |
|
|
{ |
|
|
|
|
|
#if __clang__ // Enabled for clang only as the problem affects OSX
|
|
|
go(depth, e, _p.onOp); |
|
|
go(depth, e, _p.onOp); |
|
|
|
|
|
#else |
|
|
|
|
|
e.go(_p.onOp); |
|
|
|
|
|
#endif |
|
|
e.accrueSubState(sub); |
|
|
e.accrueSubState(sub); |
|
|
} |
|
|
} |
|
|
_p.gas = e.gas(); |
|
|
_p.gas = e.gas(); |
|
@ -106,4 +117,3 @@ h160 ExtVM::create(u256 _endowment, u256& io_gas, bytesConstRef _code, OnOpFunc |
|
|
io_gas = e.gas(); |
|
|
io_gas = e.gas(); |
|
|
return e.newAddress(); |
|
|
return e.newAddress(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|