mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			191 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			191 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-2020 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
 |