Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 1 | package main |
| 2 | |
Fumitoshi Ukai | e1b813c | 2015-03-30 18:38:21 +0900 | [diff] [blame] | 3 | import "fmt" |
| 4 | |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 5 | type AST interface { |
Fumitoshi Ukai | e34c179 | 2015-03-30 17:53:47 +0900 | [diff] [blame] | 6 | eval(*Evaluator) |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 7 | show() |
| 8 | } |
| 9 | |
| 10 | type ASTBase struct { |
Shinichiro Hamaji | 685fecf | 2015-03-30 18:28:12 +0900 | [diff] [blame] | 11 | filename string |
| 12 | lineno int |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 13 | } |
| 14 | |
| 15 | type AssignAST struct { |
| 16 | ASTBase |
Fumitoshi Ukai | e1b813c | 2015-03-30 18:38:21 +0900 | [diff] [blame] | 17 | lhs string |
| 18 | rhs string |
| 19 | op string |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 20 | } |
| 21 | |
Fumitoshi Ukai | e34c179 | 2015-03-30 17:53:47 +0900 | [diff] [blame] | 22 | func (ast *AssignAST) eval(ev *Evaluator) { |
| 23 | ev.evalAssign(ast) |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 24 | } |
| 25 | |
Fumitoshi Ukai | 8edcb79 | 2015-04-02 11:23:23 +0900 | [diff] [blame] | 26 | func (ast *AssignAST) evalRHS(ev *Evaluator, lhs string) Var { |
Fumitoshi Ukai | e1b813c | 2015-03-30 18:38:21 +0900 | [diff] [blame] | 27 | switch ast.op { |
| 28 | case ":=": |
Fumitoshi Ukai | 8edcb79 | 2015-04-02 11:23:23 +0900 | [diff] [blame] | 29 | // TODO: origin |
| 30 | return SimpleVar{value: ev.evalExpr(ast.rhs)} |
Fumitoshi Ukai | e1b813c | 2015-03-30 18:38:21 +0900 | [diff] [blame] | 31 | case "=": |
Fumitoshi Ukai | 8edcb79 | 2015-04-02 11:23:23 +0900 | [diff] [blame] | 32 | return RecursiveVar{expr: ast.rhs} |
Shinichiro Hamaji | 69b7f65 | 2015-03-31 01:01:59 +0900 | [diff] [blame] | 33 | case "+=": |
Fumitoshi Ukai | 8edcb79 | 2015-04-02 11:23:23 +0900 | [diff] [blame] | 34 | prev := ev.LookupVar(lhs) |
Fumitoshi Ukai | c184760 | 2015-04-02 16:18:02 +0900 | [diff] [blame] | 35 | if !prev.IsDefined() { |
| 36 | return RecursiveVar{expr: ast.rhs} |
Fumitoshi Ukai | 8edcb79 | 2015-04-02 11:23:23 +0900 | [diff] [blame] | 37 | } |
Fumitoshi Ukai | c184760 | 2015-04-02 16:18:02 +0900 | [diff] [blame] | 38 | return prev.Append(ev, ast.rhs) |
Shinichiro Hamaji | 69b7f65 | 2015-03-31 01:01:59 +0900 | [diff] [blame] | 39 | case "?=": |
Fumitoshi Ukai | 8edcb79 | 2015-04-02 11:23:23 +0900 | [diff] [blame] | 40 | prev := ev.LookupVar(lhs) |
| 41 | if prev.IsDefined() { |
Shinichiro Hamaji | 69b7f65 | 2015-03-31 01:01:59 +0900 | [diff] [blame] | 42 | return prev |
| 43 | } |
Fumitoshi Ukai | 7b428ed | 2015-04-02 16:41:36 +0900 | [diff] [blame^] | 44 | return RecursiveVar{expr: ast.rhs} |
Shinichiro Hamaji | 69b7f65 | 2015-03-31 01:01:59 +0900 | [diff] [blame] | 45 | default: |
| 46 | panic(fmt.Sprintf("unknown assign op: %q", ast.op)) |
Fumitoshi Ukai | e1b813c | 2015-03-30 18:38:21 +0900 | [diff] [blame] | 47 | } |
| 48 | } |
| 49 | |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 50 | func (ast *AssignAST) show() { |
Fumitoshi Ukai | 8773e5e | 2015-04-01 11:23:18 +0900 | [diff] [blame] | 51 | Log("%s=%q", ast.lhs, ast.rhs) |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 52 | } |
| 53 | |
Shinichiro Hamaji | de82971 | 2015-03-31 18:26:56 +0900 | [diff] [blame] | 54 | // Note we cannot be sure what this is, until all variables in |expr| |
| 55 | // are expanded. |
| 56 | type MaybeRuleAST struct { |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 57 | ASTBase |
Shinichiro Hamaji | de82971 | 2015-03-31 18:26:56 +0900 | [diff] [blame] | 58 | expr string |
Shinichiro Hamaji | ae32b78 | 2015-03-31 14:41:19 +0900 | [diff] [blame] | 59 | cmds []string |
Shinichiro Hamaji | 7c4e325 | 2015-03-30 23:04:25 +0900 | [diff] [blame] | 60 | cmdLineno int |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 61 | } |
| 62 | |
Shinichiro Hamaji | de82971 | 2015-03-31 18:26:56 +0900 | [diff] [blame] | 63 | func (ast *MaybeRuleAST) eval(ev *Evaluator) { |
| 64 | ev.evalMaybeRule(ast) |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 65 | } |
| 66 | |
Shinichiro Hamaji | de82971 | 2015-03-31 18:26:56 +0900 | [diff] [blame] | 67 | func (ast *MaybeRuleAST) show() { |
| 68 | Log("%s", ast.expr) |
Fumitoshi Ukai | 119dc91 | 2015-03-30 16:52:41 +0900 | [diff] [blame] | 69 | for _, cmd := range ast.cmds { |
| 70 | Log("\t%s", cmd) |
| 71 | } |
| 72 | } |
Shinichiro Hamaji | 685fecf | 2015-03-30 18:28:12 +0900 | [diff] [blame] | 73 | |
Shinichiro Hamaji | d7bef60 | 2015-03-30 19:55:32 +0900 | [diff] [blame] | 74 | type IncludeAST struct { |
| 75 | ASTBase |
| 76 | expr string |
| 77 | op string |
| 78 | } |
| 79 | |
| 80 | func (ast *IncludeAST) eval(ev *Evaluator) { |
| 81 | ev.evalInclude(ast) |
| 82 | } |
| 83 | |
| 84 | func (ast *IncludeAST) show() { |
| 85 | Log("include %s", ast.expr) |
| 86 | } |
Shinichiro Hamaji | 497754d | 2015-03-31 02:02:11 +0900 | [diff] [blame] | 87 | |
| 88 | type IfAST struct { |
| 89 | ASTBase |
Shinichiro Hamaji | ae32b78 | 2015-03-31 14:41:19 +0900 | [diff] [blame] | 90 | op string |
| 91 | lhs string |
| 92 | rhs string // Empty if |op| is ifdef or ifndef. |
| 93 | trueStmts []AST |
Shinichiro Hamaji | 497754d | 2015-03-31 02:02:11 +0900 | [diff] [blame] | 94 | falseStmts []AST |
| 95 | } |
| 96 | |
| 97 | func (ast *IfAST) eval(ev *Evaluator) { |
| 98 | ev.evalIf(ast) |
| 99 | } |
| 100 | |
| 101 | func (ast *IfAST) show() { |
| 102 | // TODO |
| 103 | Log("if") |
| 104 | } |