1
0
mirror of https://github.com/danog/ton.git synced 2024-12-02 09:28:02 +01:00
ton/crypto/vm/opctable.h
2019-09-07 14:33:36 +04:00

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