// Copyright 2021 Google LLC // // 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 mk2rbc import ( "fmt" "strings" mkparser "android/soong/androidmk/parser" ) // A parsed node for which starlark code will be generated // by calling emit(). type starlarkNode interface { emit(ctx *generationContext) } // Types used to keep processed makefile data: type commentNode struct { text string } func (c *commentNode) emit(gctx *generationContext) { chunks := strings.Split(c.text, "\\\n") gctx.newLine() gctx.write(chunks[0]) // It has '#' at the beginning already. for _, chunk := range chunks[1:] { gctx.newLine() gctx.write("#", chunk) } } type moduleInfo struct { path string // Converted Starlark file path originalPath string // Makefile file path moduleLocalName string optional bool } func (im moduleInfo) entryName() string { return im.moduleLocalName + "_init" } type inheritedModule interface { name() string entryName() string emitSelect(gctx *generationContext) shouldExist() bool } type inheritedStaticModule struct { *moduleInfo loadAlways bool } func (im inheritedStaticModule) name() string { return fmt.Sprintf("%q", MakePath2ModuleName(im.originalPath)) } func (im inheritedStaticModule) emitSelect(_ *generationContext) { } func (im inheritedStaticModule) shouldExist() bool { return im.loadAlways } type inheritedDynamicModule struct { path interpolateExpr candidateModules []*moduleInfo loadAlways bool } func (i inheritedDynamicModule) name() string { return "_varmod" } func (i inheritedDynamicModule) entryName() string { return i.name() + "_init" } func (i inheritedDynamicModule) emitSelect(gctx *generationContext) { gctx.newLine() gctx.writef("_entry = {") gctx.indentLevel++ for _, mi := range i.candidateModules { gctx.newLine() gctx.writef(`"%s": (%q, %s),`, mi.originalPath, mi.moduleLocalName, mi.entryName()) } gctx.indentLevel-- gctx.newLine() gctx.write("}.get(") i.path.emit(gctx) gctx.write(")") gctx.newLine() gctx.writef("(%s, %s) = _entry if _entry else (None, None)", i.name(), i.entryName()) if i.loadAlways { gctx.newLine() gctx.writef("if not %s:", i.entryName()) gctx.indentLevel++ gctx.newLine() gctx.write(`rblf.mkerror("cannot")`) gctx.indentLevel-- } } func (i inheritedDynamicModule) shouldExist() bool { return i.loadAlways } type inheritNode struct { module inheritedModule loadAlways bool } func (inn *inheritNode) emit(gctx *generationContext) { // Unconditional case: // rblf.inherit(handle, , module_init) // Conditional case: // if _init != None: // same as above inn.module.emitSelect(gctx) name := inn.module.name() entry := inn.module.entryName() gctx.newLine() if inn.loadAlways { gctx.writef("%s(handle, %s, %s)", cfnInherit, name, entry) return } gctx.writef("if %s:", entry) gctx.indentLevel++ gctx.newLine() gctx.writef("%s(handle, %s, %s)", cfnInherit, name, entry) gctx.indentLevel-- } type includeNode struct { module inheritedModule loadAlways bool } func (inn *includeNode) emit(gctx *generationContext) { inn.module.emitSelect(gctx) entry := inn.module.entryName() gctx.newLine() if inn.loadAlways { gctx.writef("%s(g, handle)", entry) return } gctx.writef("if %s != None:", entry) gctx.indentLevel++ gctx.newLine() gctx.writef("%s(g, handle)", entry) gctx.indentLevel-- } type assignmentFlavor int const ( // Assignment flavors asgnSet assignmentFlavor = iota // := or = asgnMaybeSet assignmentFlavor = iota // ?= and variable may be unset asgnAppend assignmentFlavor = iota // += and variable has been set before asgnMaybeAppend assignmentFlavor = iota // += and variable may be unset ) type assignmentNode struct { lhs variable value starlarkExpr mkValue *mkparser.MakeString flavor assignmentFlavor location ErrorLocation isTraced bool previous *assignmentNode } func (asgn *assignmentNode) emit(gctx *generationContext) { gctx.newLine() gctx.inAssignment = true asgn.lhs.emitSet(gctx, asgn) gctx.inAssignment = false if asgn.isTraced { gctx.newLine() gctx.tracedCount++ gctx.writef(`print("%s.%d: %s := ", `, gctx.starScript.mkFile, gctx.tracedCount, asgn.lhs.name()) asgn.lhs.emitGet(gctx, true) gctx.writef(")") } } type exprNode struct { expr starlarkExpr } func (exn *exprNode) emit(gctx *generationContext) { gctx.newLine() exn.expr.emit(gctx) } type ifNode struct { isElif bool // true if this is 'elif' statement expr starlarkExpr } func (in *ifNode) emit(gctx *generationContext) { ifElif := "if " if in.isElif { ifElif = "elif " } gctx.newLine() gctx.write(ifElif) in.expr.emit(gctx) gctx.write(":") } type elseNode struct{} func (br *elseNode) emit(gctx *generationContext) { gctx.newLine() gctx.write("else:") } // switchCase represents as single if/elseif/else branch. All the necessary // info about flavor (if/elseif/else) is supposed to be kept in `gate`. type switchCase struct { gate starlarkNode nodes []starlarkNode } func (cb *switchCase) newNode(node starlarkNode) { cb.nodes = append(cb.nodes, node) } func (cb *switchCase) emit(gctx *generationContext) { cb.gate.emit(gctx) gctx.indentLevel++ hasStatements := false emitNode := func(node starlarkNode) { if _, ok := node.(*commentNode); !ok { hasStatements = true } node.emit(gctx) } if len(cb.nodes) > 0 { emitNode(cb.nodes[0]) for _, node := range cb.nodes[1:] { emitNode(node) } if !hasStatements { gctx.emitPass() } } else { gctx.emitPass() } gctx.indentLevel-- } // A single complete if ... elseif ... else ... endif sequences type switchNode struct { ssCases []*switchCase } func (ssw *switchNode) newNode(node starlarkNode) { switch br := node.(type) { case *switchCase: ssw.ssCases = append(ssw.ssCases, br) default: panic(fmt.Errorf("expected switchCase node, got %t", br)) } } func (ssw *switchNode) emit(gctx *generationContext) { if len(ssw.ssCases) == 0 { gctx.emitPass() } else { ssw.ssCases[0].emit(gctx) for _, ssCase := range ssw.ssCases[1:] { ssCase.emit(gctx) } } }