Templatize.

This commit is contained in:
chriseth 2016-11-23 14:50:20 +01:00
parent f2872bfa99
commit 612c1726d9

View File

@ -37,6 +37,35 @@ struct OptimiserState
std::back_insert_iterator<AssemblyItems> out; std::back_insert_iterator<AssemblyItems> out;
}; };
template <class Method, size_t Arguments>
struct ApplyRule
{
};
template <class Method>
struct ApplyRule<Method, 3>
{
static bool applyRule(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems> _out)
{
return Method::applySimple(_in[0], _in[1], _in[2], _out);
}
};
template <class Method>
struct ApplyRule<Method, 2>
{
static bool applyRule(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems> _out)
{
return Method::applySimple(_in[0], _in[1], _out);
}
};
template <class Method>
struct ApplyRule<Method, 1>
{
static bool applyRule(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems> _out)
{
return Method::applySimple(_in[0], _out);
}
};
template <class Method, size_t WindowSize> template <class Method, size_t WindowSize>
struct SimplePeepholeOptimizerMethod struct SimplePeepholeOptimizerMethod
{ {
@ -44,7 +73,7 @@ struct SimplePeepholeOptimizerMethod
{ {
if ( if (
_state.i + WindowSize <= _state.items.size() && _state.i + WindowSize <= _state.items.size() &&
Method::applySimple(_state.items.begin() + _state.i, _state.out) ApplyRule<Method, WindowSize>::applyRule(_state.items.begin() + _state.i, _state.out)
) )
{ {
_state.i += WindowSize; _state.i += WindowSize;
@ -57,20 +86,20 @@ struct SimplePeepholeOptimizerMethod
struct Identity: SimplePeepholeOptimizerMethod<Identity, 1> struct Identity: SimplePeepholeOptimizerMethod<Identity, 1>
{ {
static bool applySimple(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems> _out) static bool applySimple(AssemblyItem const& _item, std::back_insert_iterator<AssemblyItems> _out)
{ {
*_out = *_in; *_out = _item;
return true; return true;
} }
}; };
struct PushPop: SimplePeepholeOptimizerMethod<PushPop, 2> struct PushPop: SimplePeepholeOptimizerMethod<PushPop, 2>
{ {
static size_t applySimple(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems>) static size_t applySimple(AssemblyItem const& _push, AssemblyItem const& _pop, std::back_insert_iterator<AssemblyItems>)
{ {
auto t = _in[0].type(); auto t = _push.type();
return _in[1] == Instruction::POP && ( return _pop == Instruction::POP && (
SemanticInformation::isDupInstruction(_in[0]) || SemanticInformation::isDupInstruction(_push) ||
t == Push || t == PushString || t == PushTag || t == PushSub || t == Push || t == PushString || t == PushTag || t == PushSub ||
t == PushSubSize || t == PushProgramSize || t == PushData || t == PushLibraryAddress t == PushSubSize || t == PushProgramSize || t == PushData || t == PushLibraryAddress
); );
@ -104,26 +133,55 @@ struct AddPop: SimplePeepholeOptimizerMethod<AddPop, 2>
struct DoubleSwap: SimplePeepholeOptimizerMethod<DoubleSwap, 2> struct DoubleSwap: SimplePeepholeOptimizerMethod<DoubleSwap, 2>
{ {
static size_t applySimple(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems>) static size_t applySimple(AssemblyItem const& _s1, AssemblyItem const& _s2, std::back_insert_iterator<AssemblyItems>)
{ {
return _in[0] == _in[1] && SemanticInformation::isSwapInstruction(_in[0]); return _s1 == _s2 && SemanticInformation::isSwapInstruction(_s1);
} }
}; };
struct JumpToNext: SimplePeepholeOptimizerMethod<JumpToNext, 3> struct JumpToNext: SimplePeepholeOptimizerMethod<JumpToNext, 3>
{ {
static size_t applySimple(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems> _out) static size_t applySimple(
AssemblyItem const& _pushTag,
AssemblyItem const& _jump,
AssemblyItem const& _tag,
std::back_insert_iterator<AssemblyItems> _out
)
{ {
if ( if (
_in[0].type() == PushTag && _pushTag.type() == PushTag &&
(_in[1] == Instruction::JUMP || _in[1] == Instruction::JUMPI) && (_jump == Instruction::JUMP || _jump == Instruction::JUMPI) &&
_in[2].type() == Tag && _tag.type() == Tag &&
_in[0].data() == _in[2].data() _pushTag.data() == _tag.data()
) )
{ {
if (_in[1] == Instruction::JUMPI) if (_jump == Instruction::JUMPI)
*_out = AssemblyItem(Instruction::POP, _in[1].location()); *_out = AssemblyItem(Instruction::POP, _jump.location());
*_out = _in[2]; *_out = _tag;
return true;
}
else
return false;
}
};
struct TagConjunctions: SimplePeepholeOptimizerMethod<TagConjunctions, 3>
{
static bool applySimple(
AssemblyItem const& _pushTag,
AssemblyItem const& _pushConstant,
AssemblyItem const& _and,
std::back_insert_iterator<AssemblyItems> _out
)
{
if (
_pushTag.type() == PushTag &&
_and == Instruction::AND &&
_pushConstant.type() == Push &&
(_pushConstant.data() & u256(0xFFFFFFFF)) == u256(0xFFFFFFFF)
)
{
*_out = _pushTag;
return true; return true;
} }
else else
@ -162,26 +220,6 @@ struct UnreachableCode
} }
}; };
struct TagConjunctions: SimplePeepholeOptimizerMethod<TagConjunctions, 3>
{
static bool applySimple(AssemblyItems::const_iterator _in, std::back_insert_iterator<AssemblyItems> _out)
{
if (
_in[0].type() == PushTag &&
_in[2] == Instruction::AND &&
_in[1].type() == Push &&
(_in[1].data() & u256(0xFFFFFFFF)) == u256(0xFFFFFFFF)
)
{
*_out = _in[0];
return true;
}
else
return false;
}
};
void applyMethods(OptimiserState&) void applyMethods(OptimiserState&)
{ {
assertThrow(false, OptimizerException, "Peephole optimizer failed to apply identity."); assertThrow(false, OptimizerException, "Peephole optimizer failed to apply identity.");