aboutsummaryrefslogtreecommitdiffstats
path: root/parsing.go
diff options
context:
space:
mode:
authorobscuren <obscuren@obscura.com>2013-12-26 19:45:52 +0800
committerobscuren <obscuren@obscura.com>2013-12-26 19:45:52 +0800
commit5db3335dce766bd679c54ea44f6df08a7ff74762 (patch)
tree75614c847fdc07150351e9aa02353d8f1d23196c /parsing.go
downloadgo-tangerine-5db3335dce766bd679c54ea44f6df08a7ff74762.tar
go-tangerine-5db3335dce766bd679c54ea44f6df08a7ff74762.tar.gz
go-tangerine-5db3335dce766bd679c54ea44f6df08a7ff74762.tar.bz2
go-tangerine-5db3335dce766bd679c54ea44f6df08a7ff74762.tar.lz
go-tangerine-5db3335dce766bd679c54ea44f6df08a7ff74762.tar.xz
go-tangerine-5db3335dce766bd679c54ea44f6df08a7ff74762.tar.zst
go-tangerine-5db3335dce766bd679c54ea44f6df08a7ff74762.zip
Initial commit
Diffstat (limited to 'parsing.go')
-rw-r--r--parsing.go113
1 files changed, 113 insertions, 0 deletions
diff --git a/parsing.go b/parsing.go
new file mode 100644
index 000000000..d5dcce04f
--- /dev/null
+++ b/parsing.go
@@ -0,0 +1,113 @@
+package main
+
+import (
+ "fmt"
+ "strings"
+ "errors"
+ "math/big"
+ "strconv"
+)
+
+// Op codes
+var OpCodes = map[string]string{
+ "STOP": "0",
+ "ADD": "16", // 0x10
+ "SUB": "17", // 0x11
+ "MUL": "18", // 0x12
+ "DIV": "19", // 0x13
+ "SDIV": "20", // 0x14
+ "MOD": "21", // 0x15
+ "SMOD": "22", // 0x16
+ "EXP": "23", // 0x17
+ "NEG": "24", // 0x18
+ "LT": "32", // 0x20
+ "LE": "33", // 0x21
+ "GT": "34", // 0x22
+ "GE": "35", // 0x23
+ "EQ": "36", // 0x24
+ "NOT": "37", // 0x25
+ "SHA256": "48", // 0x30
+ "RIPEMD160": "49", // 0x31
+ "ECMUL": "50", // 0x32
+ "ECADD": "51", // 0x33
+ "SIGN": "52", // 0x34
+ "RECOVER": "53", // 0x35
+ "COPY": "64", // 0x40
+ "ST": "65", // 0x41
+ "LD": "66", // 0x42
+ "SET": "67", // 0x43
+ "JMP": "80", // 0x50
+ "JMPI": "81", // 0x51
+ "IND": "82", // 0x52
+ "EXTRO": "96", // 0x60
+ "BALANCE": "97", // 0x61
+ "MKTX": "112", // 0x70
+ "DATA": "128", // 0x80
+ "DATAN": "129", // 0x81
+ "MYADDRESS": "144", // 0x90
+ "BLKHASH": "145", // 0x91
+ "COINBASE": "146", // 0x92
+ "SUICIDE": "255", // 0xff
+}
+
+
+func CompileInstr(s string) (string, error) {
+ tokens := strings.Split(s, " ")
+ if OpCodes[tokens[0]] == "" {
+ return "", errors.New(fmt.Sprintf("OP not found: %s", tokens[0]))
+ }
+
+ code := OpCodes[tokens[0]] // Replace op codes with the proper numerical equivalent
+ op := new(big.Int)
+ op.SetString(code, 0)
+
+ args := make([]*big.Int, 6)
+ for i, val := range tokens[1:len(tokens)] {
+ num := new(big.Int)
+ num.SetString(val, 0)
+ args[i] = num
+ }
+
+ // Big int equation = op + x * 256 + y * 256**2 + z * 256**3 + a * 256**4 + b * 256**5 + c * 256**6
+ base := new(big.Int)
+ x := new(big.Int)
+ y := new(big.Int)
+ z := new(big.Int)
+ a := new(big.Int)
+ b := new(big.Int)
+ c := new(big.Int)
+
+ if args[0] != nil { x.Mul(args[0], big.NewInt(256)) }
+ if args[1] != nil { y.Mul(args[1], BigPow(256, 2)) }
+ if args[2] != nil { z.Mul(args[2], BigPow(256, 3)) }
+ if args[3] != nil { a.Mul(args[3], BigPow(256, 4)) }
+ if args[4] != nil { b.Mul(args[4], BigPow(256, 5)) }
+ if args[5] != nil { c.Mul(args[5], BigPow(256, 6)) }
+
+ base.Add(op, x)
+ base.Add(base, y)
+ base.Add(base, z)
+ base.Add(base, a)
+ base.Add(base, b)
+ base.Add(base, c)
+
+ return base.String(), nil
+}
+
+func Instr(instr string) (int, []string, error) {
+ base := new(big.Int)
+ base.SetString(instr, 0)
+
+ args := make([]string, 7)
+ for i := 0; i < 7; i++ {
+ // int(int(val) / int(math.Pow(256,float64(i)))) % 256
+ exp := BigPow(256, i)
+ num := new(big.Int)
+ num.Div(base, exp)
+
+ args[i] = num.Mod(num, big.NewInt(256)).String()
+ }
+ op, _ := strconv.Atoi(args[0])
+
+ return op, args[1:7], nil
+}