/*
This file is part of cpp-ethereum.
cpp-ethereum is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
cpp-ethereum is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with cpp-ethereum. If not, see .
*/
/** @file Instruction.cpp
* @author Gav Wood
* @date 2014
*/
#include "Instruction.h"
#include
using namespace std;
using namespace eth;
u256s eth::assemble(std::string const& _code, bool _quiet)
{
u256s ret;
map known;
map req;
char const* d = _code.data();
char const* e = _code.data() + _code.size();
while (d != e)
{
// skip to next token
for (; d != e && !isalnum(*d) && *d != '_' && *d != ':'; ++d) {}
if (d == e)
break;
char const* s = d;
for (; d != e && (isalnum(*d) || *d == '_' || *d == ':'); ++d) {}
string t = string(s, d - s);
if (isdigit(t[0]))
try
{
ret.push_back(u256(t));
}
catch (...)
{
cwarn << "Invalid numeric" << t;
}
else if (t.back() == ':')
known[t.substr(0, t.size() - 1)] = ret.size();
else
{
auto it = c_instructions.find(boost::algorithm::to_upper_copy(t));
if (it != c_instructions.end())
ret.push_back((u256)it->second);
else
{
req[ret.size()] = t;
ret.push_back(0);
}
}
}
for (auto i: req)
if (known.count(i.second))
ret[i.first] = known[i.second];
else
cwarn << "Unknown assembler token" << i.second << "at address" << i.first;
return ret;
}
static bool compileLispFragment(char const*& d, char const* e, bool _quiet, unsigned _off, u256s& o_code, vector& o_locs)
{
bool exec = false;
while (d != e)
{
// skip to next token
for (; d != e && !isalnum(*d) && *d != '(' && *d != ')' && *d != '_'; ++d) {}
if (d == e)
break;
switch (*d)
{
case '(':
exec = true;
++d;
break;
case ')':
if (exec)
{
++d;
return true;
}
else
// unexpected - return false as we don't know what to do with it.
return false;
default:
{
char const* s = d;
for (; d != e && (isalnum(*d) || *d == '_'); ++d) {}
string t(s, d - s);
if (isdigit(t[0]))
{
if (exec)
{
cwarn << "Cannot execute numeric" << t;
}
else
{
o_code.push_back(Instruction::PUSH);
try
{
o_code.push_back(u256(t));
}
catch (...)
{
cwarn << "Invalid numeric" << t;
}
}
}
else
{
boost::algorithm::to_upper(t);
if (t == "IF")
{
}
else
{
auto it = c_instructions.find(t);
if (it != c_instructions.end())
{
if (exec)
{
vector>> codes(1);
while (d != e && compileLispFragment(d, e, _quiet, o_code.size(), codes.back().first, codes.back().second))
codes.push_back(pair>());
for (auto it = codes.rbegin(); it != codes.rend(); ++it)
{
for (auto i: it->second)
it->first[i] += o_code.size();
o_code.reserve(o_code.size() + it->first.size());
for (auto i: it->first)
o_code.push_back(i);
}
o_code.push_back((u256)it->second);
}
else
{
o_code.push_back(Instruction::PUSH);
try
{
o_code.push_back((u256)it->second);
}
catch (...)
{
cwarn << "Invalid numeric" << t;
o_code.push_back(u256(t));
}
}
}
else if (!_quiet)
cwarn << "Unknown assembler token" << t;
}
}
if (!exec)
return true;
}
}
}
return false;
}
u256s eth::compileLisp(std::string const& _code, bool _quiet)
{
char const* d = _code.data();
char const* e = _code.data() + _code.size();
u256s ret;
vector locs;
compileLispFragment(d, e, _quiet, 0, ret, locs);
return ret;
}
string eth::disassemble(u256s const& _mem)
{
stringstream ret;
uint numerics = 0;
for (auto it = _mem.begin(); it != _mem.end(); ++it)
{
u256 n = *it;
auto iit = c_instructionInfo.find((Instruction)(uint)n);
if (numerics || iit == c_instructionInfo.end() || (u256)(uint)iit->first != n) // not an instruction or expecting an argument...
{
if (numerics)
numerics--;
ret << "0x" << hex << n << " ";
}
else
{
auto const& ii = iit->second;
ret << ii.name << " ";
numerics = ii.additional;
}
}
return ret.str();
}