aboutsummaryrefslogtreecommitdiffstats
path: root/AssemblyItem.cpp
blob: e005ece18f3f586966b55fd5ec2e9497437e74d3 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/*
    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 <http://www.gnu.org/licenses/>.
*/
/** @file Assembly.cpp
 * @author Gav Wood <i@gavwood.com>
 * @date 2014
 */

#include "AssemblyItem.h"
#include <fstream>

using namespace std;
using namespace dev;
using namespace dev::eth;

unsigned AssemblyItem::bytesRequired(unsigned _addressLength) const
{
    switch (m_type)
    {
    case Operation:
    case Tag: // 1 byte for the JUMPDEST
        return 1;
    case PushString:
        return 33;
    case Push:
        return 1 + max<unsigned>(1, dev::bytesRequired(m_data));
    case PushSubSize:
    case PushProgramSize:
        return 4;       // worst case: a 16MB program
    case PushTag:
    case PushData:
    case PushSub:
        return 1 + _addressLength;
    default:
        break;
    }
    BOOST_THROW_EXCEPTION(InvalidOpcode());
}

int AssemblyItem::deposit() const
{
    switch (m_type)
    {
    case Operation:
        return instructionInfo(instruction()).ret - instructionInfo(instruction()).args;
    case Push:
    case PushString:
    case PushTag:
    case PushData:
    case PushSub:
    case PushSubSize:
    case PushProgramSize:
        return 1;
    case Tag:
        return 0;
    default:;
    }
    return 0;
}

string AssemblyItem::getJumpTypeAsString() const
{
    switch (m_jumpType)
    {
    case JumpType::IntoFunction:
        return "[in]";
    case JumpType::OutOfFunction:
        return "[out]";
    case JumpType::Ordinary:
    default:
        return "";
    }
}

ostream& dev::eth::operator<<(ostream& _out, AssemblyItem const& _item)
{
    switch (_item.type())
    {
    case Operation:
        _out << " " << instructionInfo(_item.instruction()).name;
        if (_item.instruction() == eth::Instruction::JUMP || _item.instruction() == eth::Instruction::JUMPI)
            _out << "\t" << _item.getJumpTypeAsString();
        break;
    case Push:
        _out << " PUSH " << hex << _item.data();
        break;
    case PushString:
        _out << " PushString"  << hex << (unsigned)_item.data();
        break;
    case PushTag:
        _out << " PushTag " << _item.data();
        break;
    case Tag:
        _out << " Tag " << _item.data();
        break;
    case PushData:
        _out << " PushData " << hex << (unsigned)_item.data();
        break;
    case PushSub:
        _out << " PushSub " << hex << h256(_item.data()).abridgedMiddle();
        break;
    case PushSubSize:
        _out << " PushSubSize " << hex << h256(_item.data()).abridgedMiddle();
        break;
    case PushProgramSize:
        _out << " PushProgramSize";
        break;
    case UndefinedItem:
        _out << " ???";
        break;
    default:
        BOOST_THROW_EXCEPTION(InvalidOpcode());
    }
    return _out;
}