Jiri Simsa | d7616c9 | 2015-03-24 23:44:30 -0700 | [diff] [blame] | 1 | // Copyright 2015 The Vanadium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style |
| 3 | // license that can be found in the LICENSE file. |
| 4 | |
Todd Wang | 232d649 | 2015-02-25 18:04:54 -0800 | [diff] [blame] | 5 | package parse |
| 6 | |
| 7 | import ( |
| 8 | "fmt" |
| 9 | "math/big" |
| 10 | "strconv" |
| 11 | ) |
| 12 | |
| 13 | // ConstExpr is the interface for all nodes in an expression. |
| 14 | type ConstExpr interface { |
| 15 | String() string |
| 16 | Pos() Pos |
| 17 | } |
| 18 | |
| 19 | // ConstLit represents scalar literals in const expressions. The supported |
| 20 | // types for Lit are: |
| 21 | // string - Represents all string constants. |
| 22 | // *big.Int - Represents all integer constants. |
| 23 | // *big.Rat - Represents all rational constants. |
| 24 | // *BigImag - Represents all imaginary constants. |
| 25 | type ConstLit struct { |
| 26 | Lit interface{} |
| 27 | P Pos |
| 28 | } |
| 29 | |
| 30 | // BigImag represents a literal imaginary number. |
| 31 | type BigImag big.Rat |
| 32 | |
| 33 | // ConstCompositeLit represents composite literals in const expressions. |
| 34 | type ConstCompositeLit struct { |
| 35 | Type Type |
| 36 | KVList []KVLit |
| 37 | P Pos |
| 38 | } |
| 39 | |
| 40 | // KVLit represents a key/value literal in composite literals. |
| 41 | type KVLit struct { |
| 42 | Key ConstExpr |
| 43 | Value ConstExpr |
| 44 | } |
| 45 | |
| 46 | // ConstNamed represents named references to other consts. |
| 47 | type ConstNamed struct { |
| 48 | Name string |
| 49 | P Pos |
| 50 | } |
| 51 | |
| 52 | // ConstIndexed represents an index operation on a composite type. |
| 53 | type ConstIndexed struct { |
| 54 | Expr *ConstNamed |
| 55 | IndexExpr ConstExpr |
| 56 | P Pos |
| 57 | } |
| 58 | |
| 59 | // ConstTypeConv represents explicit type conversions. |
| 60 | type ConstTypeConv struct { |
| 61 | Type Type |
| 62 | Expr ConstExpr |
| 63 | P Pos |
| 64 | } |
| 65 | |
| 66 | // ConstTypeObject represents typeobject; a type used as a value. |
| 67 | type ConstTypeObject struct { |
| 68 | Type Type |
| 69 | P Pos |
| 70 | } |
| 71 | |
| 72 | // ConstUnaryOp represents all unary operations. |
| 73 | type ConstUnaryOp struct { |
| 74 | Op string |
| 75 | Expr ConstExpr |
| 76 | P Pos |
| 77 | } |
| 78 | |
| 79 | // ConstBinaryOp represents all binary operations. |
| 80 | type ConstBinaryOp struct { |
| 81 | Op string |
| 82 | Lexpr ConstExpr |
| 83 | Rexpr ConstExpr |
| 84 | P Pos |
| 85 | } |
| 86 | |
| 87 | // ConstDef represents a user-defined named const. |
| 88 | type ConstDef struct { |
| 89 | NamePos |
| 90 | Expr ConstExpr |
| 91 | } |
| 92 | |
| 93 | // cvString returns a human-readable string representing the const value. |
| 94 | func cvString(val interface{}) string { |
| 95 | switch tv := val.(type) { |
| 96 | case string: |
| 97 | return strconv.Quote(tv) |
| 98 | case *big.Int: |
| 99 | return tv.String() |
| 100 | case *big.Rat: |
| 101 | if tv.IsInt() { |
| 102 | return tv.Num().String() + ".0" |
| 103 | } |
| 104 | fv, _ := tv.Float64() |
| 105 | return strconv.FormatFloat(fv, 'g', -1, 64) |
| 106 | case *BigImag: |
| 107 | return cvString((*big.Rat)(tv)) + "i" |
| 108 | default: |
| 109 | panic(fmt.Errorf("vdl: unhandled const type %T value %v", val, val)) |
| 110 | } |
| 111 | } |
| 112 | |
| 113 | func (c *ConstLit) String() string { |
| 114 | return cvString(c.Lit) |
| 115 | } |
| 116 | func (c *ConstCompositeLit) String() string { |
| 117 | var s string |
| 118 | if c.Type != nil { |
| 119 | s += c.Type.String() |
| 120 | } |
| 121 | s += "{" |
| 122 | for index, kv := range c.KVList { |
| 123 | if index > 0 { |
| 124 | s += ", " |
| 125 | } |
| 126 | if kv.Key != nil { |
| 127 | s += kv.Key.String() + ": " |
| 128 | } |
| 129 | s += kv.Value.String() |
| 130 | } |
| 131 | return s + "}" |
| 132 | } |
| 133 | func (c *ConstNamed) String() string { |
| 134 | return c.Name |
| 135 | } |
| 136 | func (c *ConstIndexed) String() string { |
| 137 | return c.Expr.String() + "[" + c.IndexExpr.String() + "]" |
| 138 | } |
| 139 | func (c *ConstTypeConv) String() string { |
| 140 | return c.Type.String() + "(" + c.Expr.String() + ")" |
| 141 | } |
| 142 | func (c *ConstTypeObject) String() string { |
| 143 | return c.Type.String() |
| 144 | } |
| 145 | func (c *ConstUnaryOp) String() string { |
| 146 | return c.Op + c.Expr.String() |
| 147 | } |
| 148 | func (c *ConstBinaryOp) String() string { |
| 149 | return "(" + c.Lexpr.String() + c.Op + c.Rexpr.String() + ")" |
| 150 | } |
| 151 | func (c *ConstDef) String() string { return fmt.Sprintf("%+v", *c) } |
| 152 | |
| 153 | func (c *ConstLit) Pos() Pos { return c.P } |
| 154 | func (c *ConstCompositeLit) Pos() Pos { return c.P } |
| 155 | func (c *ConstNamed) Pos() Pos { return c.P } |
| 156 | func (c *ConstIndexed) Pos() Pos { return c.P } |
| 157 | func (c *ConstTypeConv) Pos() Pos { return c.P } |
| 158 | func (c *ConstTypeObject) Pos() Pos { return c.P } |
| 159 | func (c *ConstUnaryOp) Pos() Pos { return c.P } |
| 160 | func (c *ConstBinaryOp) Pos() Pos { return c.P } |