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