opcodes.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  1. // Copyright 2014 The go-ethereum Authors
  2. // This file is part of the go-ethereum library.
  3. //
  4. // The go-ethereum library is free software: you can redistribute it and/or modify
  5. // it under the terms of the GNU Lesser General Public License as published by
  6. // the Free Software Foundation, either version 3 of the License, or
  7. // (at your option) any later version.
  8. //
  9. // The go-ethereum library is distributed in the hope that it will be useful,
  10. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. // GNU Lesser General Public License for more details.
  13. //
  14. // You should have received a copy of the GNU Lesser General Public License
  15. // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
  16. package vm
  17. import (
  18. "fmt"
  19. )
  20. // OpCode is an EVM opcode
  21. type OpCode byte
  22. // IsPush specifies if an opcode is a PUSH opcode.
  23. func (op OpCode) IsPush() bool {
  24. switch op {
  25. case PUSH1, PUSH2, PUSH3, PUSH4, PUSH5, PUSH6, PUSH7, PUSH8, PUSH9, PUSH10, PUSH11, PUSH12, PUSH13, PUSH14, PUSH15, PUSH16, PUSH17, PUSH18, PUSH19, PUSH20, PUSH21, PUSH22, PUSH23, PUSH24, PUSH25, PUSH26, PUSH27, PUSH28, PUSH29, PUSH30, PUSH31, PUSH32:
  26. return true
  27. }
  28. return false
  29. }
  30. // IsStaticJump specifies if an opcode is JUMP.
  31. func (op OpCode) IsStaticJump() bool {
  32. return op == JUMP
  33. }
  34. // 0x0 range - arithmetic ops.
  35. const (
  36. STOP OpCode = iota
  37. ADD
  38. MUL
  39. SUB
  40. DIV
  41. SDIV
  42. MOD
  43. SMOD
  44. ADDMOD
  45. MULMOD
  46. EXP
  47. SIGNEXTEND
  48. )
  49. // 0x10 range - comparison ops.
  50. const (
  51. LT OpCode = iota + 0x10
  52. GT
  53. SLT
  54. SGT
  55. EQ
  56. ISZERO
  57. AND
  58. OR
  59. XOR
  60. NOT
  61. BYTE
  62. SHL
  63. SHR
  64. SAR
  65. SHA3 OpCode = 0x20
  66. )
  67. // 0x30 range - closure state.
  68. const (
  69. ADDRESS OpCode = 0x30 + iota
  70. BALANCE
  71. ORIGIN
  72. CALLER
  73. CALLVALUE
  74. CALLDATALOAD
  75. CALLDATASIZE
  76. CALLDATACOPY
  77. CODESIZE
  78. CODECOPY
  79. GASPRICE
  80. EXTCODESIZE
  81. EXTCODECOPY
  82. RETURNDATASIZE
  83. RETURNDATACOPY
  84. EXTCODEHASH
  85. )
  86. // 0x40 range - block operations.
  87. const (
  88. BLOCKHASH OpCode = 0x40 + iota
  89. COINBASE
  90. TIMESTAMP
  91. NUMBER
  92. DIFFICULTY
  93. GASLIMIT
  94. CHAINID OpCode = 0x46
  95. SELFBALANCE OpCode = 0x47
  96. )
  97. // 0x50 range - 'storage' and execution.
  98. const (
  99. POP OpCode = 0x50
  100. MLOAD OpCode = 0x51
  101. MSTORE OpCode = 0x52
  102. MSTORE8 OpCode = 0x53
  103. SLOAD OpCode = 0x54
  104. SSTORE OpCode = 0x55
  105. JUMP OpCode = 0x56
  106. JUMPI OpCode = 0x57
  107. PC OpCode = 0x58
  108. MSIZE OpCode = 0x59
  109. GAS OpCode = 0x5a
  110. JUMPDEST OpCode = 0x5b
  111. )
  112. // 0x60 range.
  113. const (
  114. PUSH1 OpCode = 0x60 + iota
  115. PUSH2
  116. PUSH3
  117. PUSH4
  118. PUSH5
  119. PUSH6
  120. PUSH7
  121. PUSH8
  122. PUSH9
  123. PUSH10
  124. PUSH11
  125. PUSH12
  126. PUSH13
  127. PUSH14
  128. PUSH15
  129. PUSH16
  130. PUSH17
  131. PUSH18
  132. PUSH19
  133. PUSH20
  134. PUSH21
  135. PUSH22
  136. PUSH23
  137. PUSH24
  138. PUSH25
  139. PUSH26
  140. PUSH27
  141. PUSH28
  142. PUSH29
  143. PUSH30
  144. PUSH31
  145. PUSH32
  146. DUP1
  147. DUP2
  148. DUP3
  149. DUP4
  150. DUP5
  151. DUP6
  152. DUP7
  153. DUP8
  154. DUP9
  155. DUP10
  156. DUP11
  157. DUP12
  158. DUP13
  159. DUP14
  160. DUP15
  161. DUP16
  162. SWAP1
  163. SWAP2
  164. SWAP3
  165. SWAP4
  166. SWAP5
  167. SWAP6
  168. SWAP7
  169. SWAP8
  170. SWAP9
  171. SWAP10
  172. SWAP11
  173. SWAP12
  174. SWAP13
  175. SWAP14
  176. SWAP15
  177. SWAP16
  178. )
  179. // 0xa0 range - logging ops.
  180. const (
  181. LOG0 OpCode = 0xa0 + iota
  182. LOG1
  183. LOG2
  184. LOG3
  185. LOG4
  186. )
  187. // unofficial opcodes used for parsing.
  188. const (
  189. PUSH OpCode = 0xb0 + iota
  190. DUP
  191. SWAP
  192. )
  193. // 0xf0 range - closures.
  194. const (
  195. CREATE OpCode = 0xf0 + iota
  196. CALL
  197. CALLCODE
  198. RETURN
  199. DELEGATECALL
  200. CREATE2
  201. STATICCALL OpCode = 0xfa
  202. REVERT OpCode = 0xfd
  203. SELFDESTRUCT OpCode = 0xff
  204. )
  205. // Since the opcodes aren't all in order we can't use a regular slice.
  206. var opCodeToString = map[OpCode]string{
  207. // 0x0 range - arithmetic ops.
  208. STOP: "STOP",
  209. ADD: "ADD",
  210. MUL: "MUL",
  211. SUB: "SUB",
  212. DIV: "DIV",
  213. SDIV: "SDIV",
  214. MOD: "MOD",
  215. SMOD: "SMOD",
  216. EXP: "EXP",
  217. NOT: "NOT",
  218. LT: "LT",
  219. GT: "GT",
  220. SLT: "SLT",
  221. SGT: "SGT",
  222. EQ: "EQ",
  223. ISZERO: "ISZERO",
  224. SIGNEXTEND: "SIGNEXTEND",
  225. // 0x10 range - bit ops.
  226. AND: "AND",
  227. OR: "OR",
  228. XOR: "XOR",
  229. BYTE: "BYTE",
  230. SHL: "SHL",
  231. SHR: "SHR",
  232. SAR: "SAR",
  233. ADDMOD: "ADDMOD",
  234. MULMOD: "MULMOD",
  235. // 0x20 range - crypto.
  236. SHA3: "SHA3",
  237. // 0x30 range - closure state.
  238. ADDRESS: "ADDRESS",
  239. BALANCE: "BALANCE",
  240. ORIGIN: "ORIGIN",
  241. CALLER: "CALLER",
  242. CALLVALUE: "CALLVALUE",
  243. CALLDATALOAD: "CALLDATALOAD",
  244. CALLDATASIZE: "CALLDATASIZE",
  245. CALLDATACOPY: "CALLDATACOPY",
  246. CODESIZE: "CODESIZE",
  247. CODECOPY: "CODECOPY",
  248. GASPRICE: "GASPRICE",
  249. EXTCODESIZE: "EXTCODESIZE",
  250. EXTCODECOPY: "EXTCODECOPY",
  251. RETURNDATASIZE: "RETURNDATASIZE",
  252. RETURNDATACOPY: "RETURNDATACOPY",
  253. EXTCODEHASH: "EXTCODEHASH",
  254. // 0x40 range - block operations.
  255. BLOCKHASH: "BLOCKHASH",
  256. COINBASE: "COINBASE",
  257. TIMESTAMP: "TIMESTAMP",
  258. NUMBER: "NUMBER",
  259. DIFFICULTY: "DIFFICULTY",
  260. GASLIMIT: "GASLIMIT",
  261. CHAINID: "CHAINID",
  262. SELFBALANCE: "SELFBALANCE",
  263. // 0x50 range - 'storage' and execution.
  264. POP: "POP",
  265. //DUP: "DUP",
  266. //SWAP: "SWAP",
  267. MLOAD: "MLOAD",
  268. MSTORE: "MSTORE",
  269. MSTORE8: "MSTORE8",
  270. SLOAD: "SLOAD",
  271. SSTORE: "SSTORE",
  272. JUMP: "JUMP",
  273. JUMPI: "JUMPI",
  274. PC: "PC",
  275. MSIZE: "MSIZE",
  276. GAS: "GAS",
  277. JUMPDEST: "JUMPDEST",
  278. // 0x60 range - push.
  279. PUSH1: "PUSH1",
  280. PUSH2: "PUSH2",
  281. PUSH3: "PUSH3",
  282. PUSH4: "PUSH4",
  283. PUSH5: "PUSH5",
  284. PUSH6: "PUSH6",
  285. PUSH7: "PUSH7",
  286. PUSH8: "PUSH8",
  287. PUSH9: "PUSH9",
  288. PUSH10: "PUSH10",
  289. PUSH11: "PUSH11",
  290. PUSH12: "PUSH12",
  291. PUSH13: "PUSH13",
  292. PUSH14: "PUSH14",
  293. PUSH15: "PUSH15",
  294. PUSH16: "PUSH16",
  295. PUSH17: "PUSH17",
  296. PUSH18: "PUSH18",
  297. PUSH19: "PUSH19",
  298. PUSH20: "PUSH20",
  299. PUSH21: "PUSH21",
  300. PUSH22: "PUSH22",
  301. PUSH23: "PUSH23",
  302. PUSH24: "PUSH24",
  303. PUSH25: "PUSH25",
  304. PUSH26: "PUSH26",
  305. PUSH27: "PUSH27",
  306. PUSH28: "PUSH28",
  307. PUSH29: "PUSH29",
  308. PUSH30: "PUSH30",
  309. PUSH31: "PUSH31",
  310. PUSH32: "PUSH32",
  311. DUP1: "DUP1",
  312. DUP2: "DUP2",
  313. DUP3: "DUP3",
  314. DUP4: "DUP4",
  315. DUP5: "DUP5",
  316. DUP6: "DUP6",
  317. DUP7: "DUP7",
  318. DUP8: "DUP8",
  319. DUP9: "DUP9",
  320. DUP10: "DUP10",
  321. DUP11: "DUP11",
  322. DUP12: "DUP12",
  323. DUP13: "DUP13",
  324. DUP14: "DUP14",
  325. DUP15: "DUP15",
  326. DUP16: "DUP16",
  327. SWAP1: "SWAP1",
  328. SWAP2: "SWAP2",
  329. SWAP3: "SWAP3",
  330. SWAP4: "SWAP4",
  331. SWAP5: "SWAP5",
  332. SWAP6: "SWAP6",
  333. SWAP7: "SWAP7",
  334. SWAP8: "SWAP8",
  335. SWAP9: "SWAP9",
  336. SWAP10: "SWAP10",
  337. SWAP11: "SWAP11",
  338. SWAP12: "SWAP12",
  339. SWAP13: "SWAP13",
  340. SWAP14: "SWAP14",
  341. SWAP15: "SWAP15",
  342. SWAP16: "SWAP16",
  343. LOG0: "LOG0",
  344. LOG1: "LOG1",
  345. LOG2: "LOG2",
  346. LOG3: "LOG3",
  347. LOG4: "LOG4",
  348. // 0xf0 range.
  349. CREATE: "CREATE",
  350. CALL: "CALL",
  351. RETURN: "RETURN",
  352. CALLCODE: "CALLCODE",
  353. DELEGATECALL: "DELEGATECALL",
  354. CREATE2: "CREATE2",
  355. STATICCALL: "STATICCALL",
  356. REVERT: "REVERT",
  357. SELFDESTRUCT: "SELFDESTRUCT",
  358. PUSH: "PUSH",
  359. DUP: "DUP",
  360. SWAP: "SWAP",
  361. }
  362. func (op OpCode) String() string {
  363. str := opCodeToString[op]
  364. if len(str) == 0 {
  365. return fmt.Sprintf("opcode 0x%x not defined", int(op))
  366. }
  367. return str
  368. }
  369. var stringToOp = map[string]OpCode{
  370. "STOP": STOP,
  371. "ADD": ADD,
  372. "MUL": MUL,
  373. "SUB": SUB,
  374. "DIV": DIV,
  375. "SDIV": SDIV,
  376. "MOD": MOD,
  377. "SMOD": SMOD,
  378. "EXP": EXP,
  379. "NOT": NOT,
  380. "LT": LT,
  381. "GT": GT,
  382. "SLT": SLT,
  383. "SGT": SGT,
  384. "EQ": EQ,
  385. "ISZERO": ISZERO,
  386. "SIGNEXTEND": SIGNEXTEND,
  387. "AND": AND,
  388. "OR": OR,
  389. "XOR": XOR,
  390. "BYTE": BYTE,
  391. "SHL": SHL,
  392. "SHR": SHR,
  393. "SAR": SAR,
  394. "ADDMOD": ADDMOD,
  395. "MULMOD": MULMOD,
  396. "SHA3": SHA3,
  397. "ADDRESS": ADDRESS,
  398. "BALANCE": BALANCE,
  399. "ORIGIN": ORIGIN,
  400. "CALLER": CALLER,
  401. "CALLVALUE": CALLVALUE,
  402. "CALLDATALOAD": CALLDATALOAD,
  403. "CALLDATASIZE": CALLDATASIZE,
  404. "CALLDATACOPY": CALLDATACOPY,
  405. "CHAINID": CHAINID,
  406. "DELEGATECALL": DELEGATECALL,
  407. "STATICCALL": STATICCALL,
  408. "CODESIZE": CODESIZE,
  409. "CODECOPY": CODECOPY,
  410. "GASPRICE": GASPRICE,
  411. "EXTCODESIZE": EXTCODESIZE,
  412. "EXTCODECOPY": EXTCODECOPY,
  413. "RETURNDATASIZE": RETURNDATASIZE,
  414. "RETURNDATACOPY": RETURNDATACOPY,
  415. "EXTCODEHASH": EXTCODEHASH,
  416. "BLOCKHASH": BLOCKHASH,
  417. "COINBASE": COINBASE,
  418. "TIMESTAMP": TIMESTAMP,
  419. "NUMBER": NUMBER,
  420. "DIFFICULTY": DIFFICULTY,
  421. "GASLIMIT": GASLIMIT,
  422. "SELFBALANCE": SELFBALANCE,
  423. "POP": POP,
  424. "MLOAD": MLOAD,
  425. "MSTORE": MSTORE,
  426. "MSTORE8": MSTORE8,
  427. "SLOAD": SLOAD,
  428. "SSTORE": SSTORE,
  429. "JUMP": JUMP,
  430. "JUMPI": JUMPI,
  431. "PC": PC,
  432. "MSIZE": MSIZE,
  433. "GAS": GAS,
  434. "JUMPDEST": JUMPDEST,
  435. "PUSH1": PUSH1,
  436. "PUSH2": PUSH2,
  437. "PUSH3": PUSH3,
  438. "PUSH4": PUSH4,
  439. "PUSH5": PUSH5,
  440. "PUSH6": PUSH6,
  441. "PUSH7": PUSH7,
  442. "PUSH8": PUSH8,
  443. "PUSH9": PUSH9,
  444. "PUSH10": PUSH10,
  445. "PUSH11": PUSH11,
  446. "PUSH12": PUSH12,
  447. "PUSH13": PUSH13,
  448. "PUSH14": PUSH14,
  449. "PUSH15": PUSH15,
  450. "PUSH16": PUSH16,
  451. "PUSH17": PUSH17,
  452. "PUSH18": PUSH18,
  453. "PUSH19": PUSH19,
  454. "PUSH20": PUSH20,
  455. "PUSH21": PUSH21,
  456. "PUSH22": PUSH22,
  457. "PUSH23": PUSH23,
  458. "PUSH24": PUSH24,
  459. "PUSH25": PUSH25,
  460. "PUSH26": PUSH26,
  461. "PUSH27": PUSH27,
  462. "PUSH28": PUSH28,
  463. "PUSH29": PUSH29,
  464. "PUSH30": PUSH30,
  465. "PUSH31": PUSH31,
  466. "PUSH32": PUSH32,
  467. "DUP1": DUP1,
  468. "DUP2": DUP2,
  469. "DUP3": DUP3,
  470. "DUP4": DUP4,
  471. "DUP5": DUP5,
  472. "DUP6": DUP6,
  473. "DUP7": DUP7,
  474. "DUP8": DUP8,
  475. "DUP9": DUP9,
  476. "DUP10": DUP10,
  477. "DUP11": DUP11,
  478. "DUP12": DUP12,
  479. "DUP13": DUP13,
  480. "DUP14": DUP14,
  481. "DUP15": DUP15,
  482. "DUP16": DUP16,
  483. "SWAP1": SWAP1,
  484. "SWAP2": SWAP2,
  485. "SWAP3": SWAP3,
  486. "SWAP4": SWAP4,
  487. "SWAP5": SWAP5,
  488. "SWAP6": SWAP6,
  489. "SWAP7": SWAP7,
  490. "SWAP8": SWAP8,
  491. "SWAP9": SWAP9,
  492. "SWAP10": SWAP10,
  493. "SWAP11": SWAP11,
  494. "SWAP12": SWAP12,
  495. "SWAP13": SWAP13,
  496. "SWAP14": SWAP14,
  497. "SWAP15": SWAP15,
  498. "SWAP16": SWAP16,
  499. "LOG0": LOG0,
  500. "LOG1": LOG1,
  501. "LOG2": LOG2,
  502. "LOG3": LOG3,
  503. "LOG4": LOG4,
  504. "CREATE": CREATE,
  505. "CREATE2": CREATE2,
  506. "CALL": CALL,
  507. "RETURN": RETURN,
  508. "CALLCODE": CALLCODE,
  509. "REVERT": REVERT,
  510. "SELFDESTRUCT": SELFDESTRUCT,
  511. }
  512. // StringToOp finds the opcode whose name is stored in `str`.
  513. func StringToOp(str string) OpCode {
  514. return stringToOp[str]
  515. }