1*333d2b36SAndroid Build Coastguard Worker// Copyright 2018 Google Inc. All rights reserved. 2*333d2b36SAndroid Build Coastguard Worker// 3*333d2b36SAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License"); 4*333d2b36SAndroid Build Coastguard Worker// you may not use this file except in compliance with the License. 5*333d2b36SAndroid Build Coastguard Worker// You may obtain a copy of the License at 6*333d2b36SAndroid Build Coastguard Worker// 7*333d2b36SAndroid Build Coastguard Worker// http://www.apache.org/licenses/LICENSE-2.0 8*333d2b36SAndroid Build Coastguard Worker// 9*333d2b36SAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software 10*333d2b36SAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS, 11*333d2b36SAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*333d2b36SAndroid Build Coastguard Worker// See the License for the specific language governing permissions and 13*333d2b36SAndroid Build Coastguard Worker// limitations under the License. 14*333d2b36SAndroid Build Coastguard Worker 15*333d2b36SAndroid Build Coastguard Workerpackage makedeps 16*333d2b36SAndroid Build Coastguard Worker 17*333d2b36SAndroid Build Coastguard Workerimport ( 18*333d2b36SAndroid Build Coastguard Worker "bytes" 19*333d2b36SAndroid Build Coastguard Worker "fmt" 20*333d2b36SAndroid Build Coastguard Worker "io" 21*333d2b36SAndroid Build Coastguard Worker "strings" 22*333d2b36SAndroid Build Coastguard Worker 23*333d2b36SAndroid Build Coastguard Worker "android/soong/androidmk/parser" 24*333d2b36SAndroid Build Coastguard Worker) 25*333d2b36SAndroid Build Coastguard Worker 26*333d2b36SAndroid Build Coastguard Workertype Deps struct { 27*333d2b36SAndroid Build Coastguard Worker Output string 28*333d2b36SAndroid Build Coastguard Worker Inputs []string 29*333d2b36SAndroid Build Coastguard Worker} 30*333d2b36SAndroid Build Coastguard Worker 31*333d2b36SAndroid Build Coastguard Workerfunc Parse(filename string, r io.Reader) (*Deps, error) { 32*333d2b36SAndroid Build Coastguard Worker p := parser.NewParser(filename, r) 33*333d2b36SAndroid Build Coastguard Worker nodes, errs := p.Parse() 34*333d2b36SAndroid Build Coastguard Worker 35*333d2b36SAndroid Build Coastguard Worker if len(errs) == 1 { 36*333d2b36SAndroid Build Coastguard Worker return nil, errs[0] 37*333d2b36SAndroid Build Coastguard Worker } else if len(errs) > 1 { 38*333d2b36SAndroid Build Coastguard Worker return nil, fmt.Errorf("many errors: %v", errs) 39*333d2b36SAndroid Build Coastguard Worker } 40*333d2b36SAndroid Build Coastguard Worker 41*333d2b36SAndroid Build Coastguard Worker pos := func(node parser.Node) string { 42*333d2b36SAndroid Build Coastguard Worker return p.Unpack(node.Pos()).String() + ": " 43*333d2b36SAndroid Build Coastguard Worker } 44*333d2b36SAndroid Build Coastguard Worker 45*333d2b36SAndroid Build Coastguard Worker ret := &Deps{} 46*333d2b36SAndroid Build Coastguard Worker 47*333d2b36SAndroid Build Coastguard Worker for _, node := range nodes { 48*333d2b36SAndroid Build Coastguard Worker switch x := node.(type) { 49*333d2b36SAndroid Build Coastguard Worker case *parser.Comment: 50*333d2b36SAndroid Build Coastguard Worker // Do nothing 51*333d2b36SAndroid Build Coastguard Worker case *parser.Rule: 52*333d2b36SAndroid Build Coastguard Worker if x.Recipe != "" { 53*333d2b36SAndroid Build Coastguard Worker return nil, fmt.Errorf("%sunexpected recipe in rule: %v", pos(node), x) 54*333d2b36SAndroid Build Coastguard Worker } 55*333d2b36SAndroid Build Coastguard Worker 56*333d2b36SAndroid Build Coastguard Worker if !x.Target.Const() { 57*333d2b36SAndroid Build Coastguard Worker return nil, fmt.Errorf("%sunsupported variable expansion: %v", pos(node), x.Target.Dump()) 58*333d2b36SAndroid Build Coastguard Worker } 59*333d2b36SAndroid Build Coastguard Worker outputs := x.Target.Words() 60*333d2b36SAndroid Build Coastguard Worker if len(outputs) > 0 { 61*333d2b36SAndroid Build Coastguard Worker ret.Output = outputs[0].Value(nil) 62*333d2b36SAndroid Build Coastguard Worker } else { 63*333d2b36SAndroid Build Coastguard Worker // TODO(b/141372861): put this back 64*333d2b36SAndroid Build Coastguard Worker //return nil, fmt.Errorf("%smissing output: %v", pos(node), x) 65*333d2b36SAndroid Build Coastguard Worker } 66*333d2b36SAndroid Build Coastguard Worker 67*333d2b36SAndroid Build Coastguard Worker if !x.Prerequisites.Const() { 68*333d2b36SAndroid Build Coastguard Worker return nil, fmt.Errorf("%sunsupported variable expansion: %v", pos(node), x.Prerequisites.Dump()) 69*333d2b36SAndroid Build Coastguard Worker } 70*333d2b36SAndroid Build Coastguard Worker for _, input := range x.Prerequisites.Words() { 71*333d2b36SAndroid Build Coastguard Worker ret.Inputs = append(ret.Inputs, input.Value(nil)) 72*333d2b36SAndroid Build Coastguard Worker } 73*333d2b36SAndroid Build Coastguard Worker default: 74*333d2b36SAndroid Build Coastguard Worker return nil, fmt.Errorf("%sunexpected line: %#v", pos(node), node) 75*333d2b36SAndroid Build Coastguard Worker } 76*333d2b36SAndroid Build Coastguard Worker } 77*333d2b36SAndroid Build Coastguard Worker 78*333d2b36SAndroid Build Coastguard Worker return ret, nil 79*333d2b36SAndroid Build Coastguard Worker} 80*333d2b36SAndroid Build Coastguard Worker 81*333d2b36SAndroid Build Coastguard Workerfunc (d *Deps) Print() []byte { 82*333d2b36SAndroid Build Coastguard Worker // We don't really have to escape every \, but it's simpler, 83*333d2b36SAndroid Build Coastguard Worker // and ninja will handle it. 84*333d2b36SAndroid Build Coastguard Worker replacer := strings.NewReplacer(" ", "\\ ", 85*333d2b36SAndroid Build Coastguard Worker ":", "\\:", 86*333d2b36SAndroid Build Coastguard Worker "#", "\\#", 87*333d2b36SAndroid Build Coastguard Worker "$", "$$", 88*333d2b36SAndroid Build Coastguard Worker "\\", "\\\\") 89*333d2b36SAndroid Build Coastguard Worker 90*333d2b36SAndroid Build Coastguard Worker b := &bytes.Buffer{} 91*333d2b36SAndroid Build Coastguard Worker fmt.Fprintf(b, "%s:", replacer.Replace(d.Output)) 92*333d2b36SAndroid Build Coastguard Worker for _, input := range d.Inputs { 93*333d2b36SAndroid Build Coastguard Worker fmt.Fprintf(b, " %s", replacer.Replace(input)) 94*333d2b36SAndroid Build Coastguard Worker } 95*333d2b36SAndroid Build Coastguard Worker fmt.Fprintln(b) 96*333d2b36SAndroid Build Coastguard Worker return b.Bytes() 97*333d2b36SAndroid Build Coastguard Worker} 98