mirror of
https://github.com/danog/ton.git
synced 2024-12-11 08:39:51 +01:00
192 lines
8.6 KiB
C++
192 lines
8.6 KiB
C++
/*
|
|
This file is part of TON Blockchain Library.
|
|
|
|
TON Blockchain Library is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
TON Blockchain Library 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 Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with TON Blockchain Library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
Copyright 2017-2019 Telegram Systems LLP
|
|
*/
|
|
#pragma once
|
|
#include "vm/dispatch.h"
|
|
#include <functional>
|
|
#include <utility>
|
|
#include <vector>
|
|
#include <map>
|
|
|
|
namespace vm {
|
|
|
|
typedef std::function<int(const CellSlice&, unsigned, int)> compute_instr_len_func_t;
|
|
//typedef std::function<int(unsigned, int)> compute_arg_instr_len_func_t;
|
|
typedef std::function<std::string(CellSlice&, unsigned, int)> dump_instr_func_t;
|
|
typedef std::function<std::string(CellSlice&, unsigned)> dump_arg_instr_func_t;
|
|
typedef std::function<int(VmState* st, CellSlice&, unsigned, int)> exec_instr_func_t;
|
|
typedef std::function<int(VmState* st, unsigned)> exec_arg_instr_func_t;
|
|
typedef std::function<int(VmState* st)> exec_simple_instr_func_t;
|
|
|
|
enum { max_opcode_bits = 24 };
|
|
const unsigned top_opcode = (1U << max_opcode_bits);
|
|
|
|
class OpcodeInstr {
|
|
unsigned min_opcode, max_opcode;
|
|
|
|
public:
|
|
static constexpr unsigned gas_per_instr = 10, gas_per_bit = 1;
|
|
virtual ~OpcodeInstr() = default;
|
|
virtual int dispatch(VmState* st, CellSlice& cs, unsigned opcode, unsigned bits) const = 0;
|
|
virtual std::string dump(CellSlice& cs, unsigned opcode, unsigned bits) const;
|
|
virtual int instr_len(const CellSlice& cs, unsigned opcode, unsigned bits) const;
|
|
OpcodeInstr(unsigned _min, unsigned _max) : min_opcode(_min), max_opcode(_max) {
|
|
}
|
|
OpcodeInstr(unsigned _opcode, unsigned _bits, bool);
|
|
unsigned get_opcode_min() const {
|
|
return min_opcode;
|
|
}
|
|
unsigned get_opcode_max() const {
|
|
return max_opcode;
|
|
}
|
|
std::pair<unsigned, unsigned> get_opcode_range() const {
|
|
return {min_opcode, max_opcode};
|
|
}
|
|
//static OpcodeInstr* mksimple(unsigned opcode, unsigned opc_bits, std::string _name, exec_instr_func_t exec);
|
|
static OpcodeInstr* mksimple(unsigned opcode, unsigned opc_bits, std::string _name, exec_simple_instr_func_t exec);
|
|
static OpcodeInstr* mkfixed(unsigned opcode, unsigned opc_bits, unsigned arg_bits, dump_arg_instr_func_t dump,
|
|
exec_arg_instr_func_t exec);
|
|
static OpcodeInstr* mkfixedrange(unsigned opcode_min, unsigned opcode_max, unsigned tot_bits, unsigned arg_bits,
|
|
dump_arg_instr_func_t dump, exec_arg_instr_func_t exec);
|
|
static OpcodeInstr* mkext(unsigned opcode, unsigned opc_bits, unsigned arg_bits, dump_instr_func_t dump,
|
|
exec_instr_func_t exec, compute_instr_len_func_t comp_len);
|
|
static OpcodeInstr* mkextrange(unsigned opcode_min, unsigned opcode_max, unsigned tot_bits, unsigned arg_bits,
|
|
dump_instr_func_t dump, exec_instr_func_t exec, compute_instr_len_func_t comp_len);
|
|
};
|
|
|
|
namespace instr {
|
|
|
|
dump_arg_instr_func_t dump_1sr(std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_1sr_l(std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_2sr(std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_2sr_adj(unsigned adj, std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_3sr(std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_3sr_adj(unsigned adj, std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_1c(std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_1c_l_add(int adj, std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_1c_and(unsigned mask, std::string prefix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_2c(std::string prefix, std::string interfix, std::string suffix = "");
|
|
dump_arg_instr_func_t dump_2c_add(unsigned add, std::string prefix, std::string interfix, std::string suffix = "");
|
|
|
|
} // namespace instr
|
|
|
|
class OpcodeTable : public DispatchTable {
|
|
std::map<unsigned, const OpcodeInstr*> instructions;
|
|
std::vector<std::pair<unsigned, const OpcodeInstr*>> instruction_list;
|
|
std::string name;
|
|
Codepage codepage;
|
|
bool final;
|
|
|
|
public:
|
|
OpcodeTable(std::string _name, Codepage cp) : name(_name), codepage(cp), final(false) {
|
|
}
|
|
OpcodeTable(const OpcodeTable&) = delete;
|
|
OpcodeTable(OpcodeTable&&) = delete;
|
|
OpcodeTable& operator=(const OpcodeTable&) = delete;
|
|
OpcodeTable& operator=(OpcodeTable&&) = delete;
|
|
~OpcodeTable() override = default;
|
|
DispatchTable* finalize() override;
|
|
bool is_final() const override {
|
|
return final;
|
|
}
|
|
int dispatch(VmState* st, CellSlice& cs) const override;
|
|
std::string dump_instr(CellSlice& cs) const override;
|
|
int instr_len(const CellSlice& cs) const override;
|
|
bool insert_bool(const OpcodeInstr*);
|
|
OpcodeTable& insert(const OpcodeInstr*);
|
|
|
|
private:
|
|
const OpcodeInstr* lookup_instr(unsigned opcode, unsigned bits) const;
|
|
const OpcodeInstr* lookup_instr(const CellSlice& cs, unsigned& opcode, unsigned& bits) const;
|
|
};
|
|
|
|
class OpcodeInstrDummy : public OpcodeInstr {
|
|
public:
|
|
OpcodeInstrDummy() = delete;
|
|
OpcodeInstrDummy(unsigned _minopc, unsigned _maxopc) : OpcodeInstr(_minopc, _maxopc) {
|
|
}
|
|
~OpcodeInstrDummy() override = default;
|
|
int dispatch(VmState* st, CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
};
|
|
|
|
class OpcodeInstrSimple : public OpcodeInstr {
|
|
unsigned char opc_bits;
|
|
std::string name;
|
|
exec_instr_func_t exec_instr;
|
|
|
|
public:
|
|
OpcodeInstrSimple() = delete;
|
|
OpcodeInstrSimple(unsigned opcode, unsigned _opc_bits, std::string _name, exec_instr_func_t exec);
|
|
~OpcodeInstrSimple() override = default;
|
|
int dispatch(VmState* st, CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
std::string dump(CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
int instr_len(const CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
};
|
|
|
|
class OpcodeInstrSimplest : public OpcodeInstr {
|
|
unsigned char opc_bits;
|
|
std::string name;
|
|
exec_simple_instr_func_t exec_instr;
|
|
|
|
public:
|
|
OpcodeInstrSimplest() = delete;
|
|
OpcodeInstrSimplest(unsigned opcode, unsigned _opc_bits, std::string _name, exec_simple_instr_func_t exec);
|
|
~OpcodeInstrSimplest() override = default;
|
|
int dispatch(VmState* st, CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
std::string dump(CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
int instr_len(const CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
};
|
|
|
|
class OpcodeInstrFixed : public OpcodeInstr {
|
|
unsigned char opc_bits, tot_bits;
|
|
std::string name;
|
|
dump_arg_instr_func_t dump_instr;
|
|
exec_arg_instr_func_t exec_instr;
|
|
|
|
public:
|
|
OpcodeInstrFixed() = delete;
|
|
OpcodeInstrFixed(unsigned opcode, unsigned _opc_bits, unsigned _arg_bits, dump_arg_instr_func_t dump,
|
|
exec_arg_instr_func_t exec);
|
|
OpcodeInstrFixed(unsigned opcode_min, unsigned opcode_max, unsigned _tot_bits, unsigned _arg_bits,
|
|
dump_arg_instr_func_t dump, exec_arg_instr_func_t exec);
|
|
~OpcodeInstrFixed() override = default;
|
|
int dispatch(VmState* st, CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
std::string dump(CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
int instr_len(const CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
};
|
|
|
|
class OpcodeInstrExt : public OpcodeInstr {
|
|
unsigned char opc_bits, tot_bits;
|
|
dump_instr_func_t dump_instr;
|
|
exec_instr_func_t exec_instr;
|
|
compute_instr_len_func_t compute_instr_len;
|
|
|
|
public:
|
|
OpcodeInstrExt() = delete;
|
|
OpcodeInstrExt(unsigned opcode, unsigned _opc_bits, unsigned _arg_bits, dump_instr_func_t dump,
|
|
exec_instr_func_t exec, compute_instr_len_func_t comp_len);
|
|
OpcodeInstrExt(unsigned opcode_min, unsigned opcode_max, unsigned _tot_bits, unsigned _arg_bits,
|
|
dump_instr_func_t dump, exec_instr_func_t exec, compute_instr_len_func_t comp_len);
|
|
~OpcodeInstrExt() override = default;
|
|
int dispatch(VmState* st, CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
std::string dump(CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
int instr_len(const CellSlice& cs, unsigned opcode, unsigned bits) const override;
|
|
};
|
|
|
|
} // namespace vm
|