| // Copyright 2015 Google Inc. All rights reserved |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package kati |
| |
| import ( |
| "strings" |
| |
| "github.com/golang/glog" |
| ) |
| |
| type ast interface { |
| eval(*Evaluator) error |
| show() |
| } |
| |
| type assignAST struct { |
| srcpos |
| lhs Value |
| rhs Value |
| op string |
| opt string // "override", "export" |
| } |
| |
| func (ast *assignAST) eval(ev *Evaluator) error { |
| return ev.evalAssign(ast) |
| } |
| |
| func (ast *assignAST) evalRHS(ev *Evaluator, lhs string) (Var, error) { |
| origin := "file" |
| if ast.filename == bootstrapMakefileName { |
| origin = "default" |
| } |
| if ast.opt == "override" { |
| origin = "override" |
| } |
| // TODO(ukai): handle ast.opt == "export" |
| switch ast.op { |
| case ":=": |
| switch v := ast.rhs.(type) { |
| case literal: |
| return &simpleVar{value: []string{v.String()}, origin: origin}, nil |
| case tmpval: |
| return &simpleVar{value: []string{v.String()}, origin: origin}, nil |
| default: |
| var buf evalBuffer |
| buf.resetSep() |
| err := v.Eval(&buf, ev) |
| if err != nil { |
| return nil, err |
| } |
| return &simpleVar{value: []string{buf.String()}, origin: origin}, nil |
| } |
| case "=": |
| return &recursiveVar{expr: ast.rhs, origin: origin}, nil |
| case "+=": |
| prev := ev.lookupVarInCurrentScope(lhs) |
| if !prev.IsDefined() { |
| return &recursiveVar{expr: ast.rhs, origin: origin}, nil |
| } |
| return prev.AppendVar(ev, ast.rhs) |
| case "?=": |
| prev := ev.lookupVarInCurrentScope(lhs) |
| if prev.IsDefined() { |
| return prev, nil |
| } |
| return &recursiveVar{expr: ast.rhs, origin: origin}, nil |
| } |
| return nil, ast.errorf("unknown assign op: %q", ast.op) |
| } |
| |
| func (ast *assignAST) show() { |
| glog.Infof("%s %s %s %q", ast.opt, ast.lhs, ast.op, ast.rhs) |
| } |
| |
| // maybeRuleAST is an ast for rule line. |
| // Note we cannot be sure what this is, until all variables in |expr| |
| // are expanded. |
| type maybeRuleAST struct { |
| srcpos |
| isRule bool // found literal ':' |
| expr Value |
| assign *assignAST // target specific var |
| semi []byte // after ';' if ';' exists |
| } |
| |
| func (ast *maybeRuleAST) eval(ev *Evaluator) error { |
| return ev.evalMaybeRule(ast) |
| } |
| |
| func (ast *maybeRuleAST) show() { |
| glog.Info(ast.expr) |
| } |
| |
| type commandAST struct { |
| srcpos |
| cmd string |
| } |
| |
| func (ast *commandAST) eval(ev *Evaluator) error { |
| return ev.evalCommand(ast) |
| } |
| |
| func (ast *commandAST) show() { |
| glog.Infof("\t%s", strings.Replace(ast.cmd, "\n", `\n`, -1)) |
| } |
| |
| type includeAST struct { |
| srcpos |
| expr string |
| op string |
| } |
| |
| func (ast *includeAST) eval(ev *Evaluator) error { |
| return ev.evalInclude(ast) |
| } |
| |
| func (ast *includeAST) show() { |
| glog.Infof("include %s", ast.expr) |
| } |
| |
| type ifAST struct { |
| srcpos |
| op string |
| lhs Value |
| rhs Value // Empty if |op| is ifdef or ifndef. |
| trueStmts []ast |
| falseStmts []ast |
| } |
| |
| func (ast *ifAST) eval(ev *Evaluator) error { |
| return ev.evalIf(ast) |
| } |
| |
| func (ast *ifAST) show() { |
| // TODO |
| glog.Info("if") |
| } |
| |
| type exportAST struct { |
| srcpos |
| expr []byte |
| hasEqual bool |
| export bool |
| } |
| |
| func (ast *exportAST) eval(ev *Evaluator) error { |
| return ev.evalExport(ast) |
| } |
| |
| func (ast *exportAST) show() { |
| // TODO |
| glog.Info("export") |
| } |
| |
| type vpathAST struct { |
| srcpos |
| expr Value |
| } |
| |
| func (ast *vpathAST) eval(ev *Evaluator) error { |
| return ev.evalVpath(ast) |
| } |
| |
| func (ast *vpathAST) show() { |
| glog.Infof("vpath %s", ast.expr.String()) |
| } |