Previously in order to make use of dist entries with tag properties it was necessary for each module type to be modified to store the results of GenerateTaggedDistFiles(module) in the AndroidMkEntry.DistFiles property. This change generalizes that mechanism to work across all module types. This change does improve the behavior of a couple of tests where previously a dist with tag property was ignored because tagged dist files were not available now it is used correctly. Some module types do not implement OutputFileProducer interface and so they cannot handle dist with tag properties. This change makes it an error to attempt to do that. That necessitated adding OutputFiles(tag) method to customModule in queryview_test.go as some of the tests in that file used dist entries with tag properties. Follow up changes will remove the code that was made redundant by this. Test: m nothing m dist sdk - before and after this change, compare result to make sure that there are no significant differences. Bug: 174226317 Change-Id: Ifc54d67db10ce0d0fe8179c05b97a2be2113be4e
750 lines
21 KiB
Go
750 lines
21 KiB
Go
// 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 android
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/google/blueprint"
|
|
"github.com/google/blueprint/bootstrap"
|
|
)
|
|
|
|
func init() {
|
|
RegisterAndroidMkBuildComponents(InitRegistrationContext)
|
|
}
|
|
|
|
func RegisterAndroidMkBuildComponents(ctx RegistrationContext) {
|
|
ctx.RegisterSingletonType("androidmk", AndroidMkSingleton)
|
|
}
|
|
|
|
// Deprecated: consider using AndroidMkEntriesProvider instead, especially if you're not going to
|
|
// use the Custom function.
|
|
type AndroidMkDataProvider interface {
|
|
AndroidMk() AndroidMkData
|
|
BaseModuleName() string
|
|
}
|
|
|
|
type AndroidMkData struct {
|
|
Class string
|
|
SubName string
|
|
DistFiles TaggedDistFiles
|
|
OutputFile OptionalPath
|
|
Disabled bool
|
|
Include string
|
|
Required []string
|
|
Host_required []string
|
|
Target_required []string
|
|
|
|
Custom func(w io.Writer, name, prefix, moduleDir string, data AndroidMkData)
|
|
|
|
Extra []AndroidMkExtraFunc
|
|
|
|
Entries AndroidMkEntries
|
|
}
|
|
|
|
type AndroidMkExtraFunc func(w io.Writer, outputFile Path)
|
|
|
|
// Allows modules to customize their Android*.mk output.
|
|
type AndroidMkEntriesProvider interface {
|
|
AndroidMkEntries() []AndroidMkEntries
|
|
BaseModuleName() string
|
|
}
|
|
|
|
type AndroidMkEntries struct {
|
|
Class string
|
|
SubName string
|
|
OverrideName string
|
|
DistFiles TaggedDistFiles
|
|
OutputFile OptionalPath
|
|
Disabled bool
|
|
Include string
|
|
Required []string
|
|
Host_required []string
|
|
Target_required []string
|
|
|
|
header bytes.Buffer
|
|
footer bytes.Buffer
|
|
|
|
ExtraEntries []AndroidMkExtraEntriesFunc
|
|
ExtraFooters []AndroidMkExtraFootersFunc
|
|
|
|
EntryMap map[string][]string
|
|
entryOrder []string
|
|
}
|
|
|
|
type AndroidMkExtraEntriesFunc func(entries *AndroidMkEntries)
|
|
type AndroidMkExtraFootersFunc func(w io.Writer, name, prefix, moduleDir string, entries *AndroidMkEntries)
|
|
|
|
func (a *AndroidMkEntries) SetString(name, value string) {
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
a.EntryMap[name] = []string{value}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) SetPath(name string, path Path) {
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
a.EntryMap[name] = []string{path.String()}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) SetOptionalPath(name string, path OptionalPath) {
|
|
if path.Valid() {
|
|
a.SetPath(name, path.Path())
|
|
}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) AddPath(name string, path Path) {
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
a.EntryMap[name] = append(a.EntryMap[name], path.String())
|
|
}
|
|
|
|
func (a *AndroidMkEntries) AddOptionalPath(name string, path OptionalPath) {
|
|
if path.Valid() {
|
|
a.AddPath(name, path.Path())
|
|
}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) SetPaths(name string, paths Paths) {
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
a.EntryMap[name] = paths.Strings()
|
|
}
|
|
|
|
func (a *AndroidMkEntries) SetOptionalPaths(name string, paths Paths) {
|
|
if len(paths) > 0 {
|
|
a.SetPaths(name, paths)
|
|
}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) AddPaths(name string, paths Paths) {
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
a.EntryMap[name] = append(a.EntryMap[name], paths.Strings()...)
|
|
}
|
|
|
|
func (a *AndroidMkEntries) SetBoolIfTrue(name string, flag bool) {
|
|
if flag {
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
a.EntryMap[name] = []string{"true"}
|
|
}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) SetBool(name string, flag bool) {
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
if flag {
|
|
a.EntryMap[name] = []string{"true"}
|
|
} else {
|
|
a.EntryMap[name] = []string{"false"}
|
|
}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) AddStrings(name string, value ...string) {
|
|
if len(value) == 0 {
|
|
return
|
|
}
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
}
|
|
a.EntryMap[name] = append(a.EntryMap[name], value...)
|
|
}
|
|
|
|
// The contributions to the dist.
|
|
type distContributions struct {
|
|
// List of goals and the dist copy instructions.
|
|
copiesForGoals []*copiesForGoals
|
|
}
|
|
|
|
// getCopiesForGoals returns a copiesForGoals into which copy instructions that
|
|
// must be processed when building one or more of those goals can be added.
|
|
func (d *distContributions) getCopiesForGoals(goals string) *copiesForGoals {
|
|
copiesForGoals := &copiesForGoals{goals: goals}
|
|
d.copiesForGoals = append(d.copiesForGoals, copiesForGoals)
|
|
return copiesForGoals
|
|
}
|
|
|
|
// Associates a list of dist copy instructions with a set of goals for which they
|
|
// should be run.
|
|
type copiesForGoals struct {
|
|
// goals are a space separated list of build targets that will trigger the
|
|
// copy instructions.
|
|
goals string
|
|
|
|
// A list of instructions to copy a module's output files to somewhere in the
|
|
// dist directory.
|
|
copies []distCopy
|
|
}
|
|
|
|
// Adds a copy instruction.
|
|
func (d *copiesForGoals) addCopyInstruction(from Path, dest string) {
|
|
d.copies = append(d.copies, distCopy{from, dest})
|
|
}
|
|
|
|
// Instruction on a path that must be copied into the dist.
|
|
type distCopy struct {
|
|
// The path to copy from.
|
|
from Path
|
|
|
|
// The destination within the dist directory to copy to.
|
|
dest string
|
|
}
|
|
|
|
// Compute the contributions that the module makes to the dist.
|
|
func (a *AndroidMkEntries) getDistContributions(mod blueprint.Module) *distContributions {
|
|
amod := mod.(Module).base()
|
|
name := amod.BaseModuleName()
|
|
|
|
// Collate the set of associated tag/paths available for copying to the dist.
|
|
// Start with an empty (nil) set.
|
|
var availableTaggedDists TaggedDistFiles
|
|
|
|
// Then merge in any that are provided explicitly by the module.
|
|
if a.DistFiles != nil {
|
|
// Merge the DistFiles into the set.
|
|
availableTaggedDists = availableTaggedDists.merge(a.DistFiles)
|
|
}
|
|
|
|
// If no paths have been provided for the DefaultDistTag and the output file is
|
|
// valid then add that as the default dist path.
|
|
if _, ok := availableTaggedDists[DefaultDistTag]; !ok && a.OutputFile.Valid() {
|
|
availableTaggedDists = availableTaggedDists.addPathsForTag(DefaultDistTag, a.OutputFile.Path())
|
|
}
|
|
|
|
// If the distFiles created by GenerateTaggedDistFiles contains paths for the
|
|
// DefaultDistTag then that takes priority so delete any existing paths.
|
|
if _, ok := amod.distFiles[DefaultDistTag]; ok {
|
|
delete(availableTaggedDists, DefaultDistTag)
|
|
}
|
|
|
|
// Finally, merge the distFiles created by GenerateTaggedDistFiles.
|
|
availableTaggedDists = availableTaggedDists.merge(amod.distFiles)
|
|
|
|
if len(availableTaggedDists) == 0 {
|
|
// Nothing dist-able for this module.
|
|
return nil
|
|
}
|
|
|
|
// Collate the contributions this module makes to the dist.
|
|
distContributions := &distContributions{}
|
|
|
|
// Iterate over this module's dist structs, merged from the dist and dists properties.
|
|
for _, dist := range amod.Dists() {
|
|
// Get the list of goals this dist should be enabled for. e.g. sdk, droidcore
|
|
goals := strings.Join(dist.Targets, " ")
|
|
|
|
// Get the tag representing the output files to be dist'd. e.g. ".jar", ".proguard_map"
|
|
var tag string
|
|
if dist.Tag == nil {
|
|
// If the dist struct does not specify a tag, use the default output files tag.
|
|
tag = DefaultDistTag
|
|
} else {
|
|
tag = *dist.Tag
|
|
}
|
|
|
|
// Get the paths of the output files to be dist'd, represented by the tag.
|
|
// Can be an empty list.
|
|
tagPaths := availableTaggedDists[tag]
|
|
if len(tagPaths) == 0 {
|
|
// Nothing to dist for this tag, continue to the next dist.
|
|
continue
|
|
}
|
|
|
|
if len(tagPaths) > 1 && (dist.Dest != nil || dist.Suffix != nil) {
|
|
errorMessage := "%s: Cannot apply dest/suffix for more than one dist " +
|
|
"file for %q goals tag %q in module %s. The list of dist files, " +
|
|
"which should have a single element, is:\n%s"
|
|
panic(fmt.Errorf(errorMessage, mod, goals, tag, name, tagPaths))
|
|
}
|
|
|
|
copiesForGoals := distContributions.getCopiesForGoals(goals)
|
|
|
|
// Iterate over each path adding a copy instruction to copiesForGoals
|
|
for _, path := range tagPaths {
|
|
// It's possible that the Path is nil from errant modules. Be defensive here.
|
|
if path == nil {
|
|
tagName := "default" // for error message readability
|
|
if dist.Tag != nil {
|
|
tagName = *dist.Tag
|
|
}
|
|
panic(fmt.Errorf("Dist file should not be nil for the %s tag in %s", tagName, name))
|
|
}
|
|
|
|
dest := filepath.Base(path.String())
|
|
|
|
if dist.Dest != nil {
|
|
var err error
|
|
if dest, err = validateSafePath(*dist.Dest); err != nil {
|
|
// This was checked in ModuleBase.GenerateBuildActions
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
if dist.Suffix != nil {
|
|
ext := filepath.Ext(dest)
|
|
suffix := *dist.Suffix
|
|
dest = strings.TrimSuffix(dest, ext) + suffix + ext
|
|
}
|
|
|
|
if dist.Dir != nil {
|
|
var err error
|
|
if dest, err = validateSafePath(*dist.Dir, dest); err != nil {
|
|
// This was checked in ModuleBase.GenerateBuildActions
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
copiesForGoals.addCopyInstruction(path, dest)
|
|
}
|
|
}
|
|
|
|
return distContributions
|
|
}
|
|
|
|
// generateDistContributionsForMake generates make rules that will generate the
|
|
// dist according to the instructions in the supplied distContribution.
|
|
func generateDistContributionsForMake(distContributions *distContributions) []string {
|
|
var ret []string
|
|
for _, d := range distContributions.copiesForGoals {
|
|
ret = append(ret, fmt.Sprintf(".PHONY: %s\n", d.goals))
|
|
// Create dist-for-goals calls for each of the copy instructions.
|
|
for _, c := range d.copies {
|
|
ret = append(
|
|
ret,
|
|
fmt.Sprintf("$(call dist-for-goals,%s,%s:%s)\n", d.goals, c.from.String(), c.dest))
|
|
}
|
|
}
|
|
|
|
return ret
|
|
}
|
|
|
|
// Compute the list of Make strings to declare phony goals and dist-for-goals
|
|
// calls from the module's dist and dists properties.
|
|
func (a *AndroidMkEntries) GetDistForGoals(mod blueprint.Module) []string {
|
|
distContributions := a.getDistContributions(mod)
|
|
if distContributions == nil {
|
|
return nil
|
|
}
|
|
|
|
return generateDistContributionsForMake(distContributions)
|
|
}
|
|
|
|
func (a *AndroidMkEntries) fillInEntries(config Config, bpPath string, mod blueprint.Module) {
|
|
a.EntryMap = make(map[string][]string)
|
|
amod := mod.(Module).base()
|
|
name := amod.BaseModuleName()
|
|
if a.OverrideName != "" {
|
|
name = a.OverrideName
|
|
}
|
|
|
|
if a.Include == "" {
|
|
a.Include = "$(BUILD_PREBUILT)"
|
|
}
|
|
a.Required = append(a.Required, amod.commonProperties.Required...)
|
|
a.Host_required = append(a.Host_required, amod.commonProperties.Host_required...)
|
|
a.Target_required = append(a.Target_required, amod.commonProperties.Target_required...)
|
|
|
|
for _, distString := range a.GetDistForGoals(mod) {
|
|
fmt.Fprintf(&a.header, distString)
|
|
}
|
|
|
|
fmt.Fprintln(&a.header, "\ninclude $(CLEAR_VARS)")
|
|
|
|
// Collect make variable assignment entries.
|
|
a.SetString("LOCAL_PATH", filepath.Dir(bpPath))
|
|
a.SetString("LOCAL_MODULE", name+a.SubName)
|
|
a.SetString("LOCAL_MODULE_CLASS", a.Class)
|
|
a.SetString("LOCAL_PREBUILT_MODULE_FILE", a.OutputFile.String())
|
|
a.AddStrings("LOCAL_REQUIRED_MODULES", a.Required...)
|
|
a.AddStrings("LOCAL_HOST_REQUIRED_MODULES", a.Host_required...)
|
|
a.AddStrings("LOCAL_TARGET_REQUIRED_MODULES", a.Target_required...)
|
|
|
|
if am, ok := mod.(ApexModule); ok {
|
|
a.SetBoolIfTrue("LOCAL_NOT_AVAILABLE_FOR_PLATFORM", am.NotAvailableForPlatform())
|
|
}
|
|
|
|
archStr := amod.Arch().ArchType.String()
|
|
host := false
|
|
switch amod.Os().Class {
|
|
case Host:
|
|
if amod.Target().HostCross {
|
|
// Make cannot identify LOCAL_MODULE_HOST_CROSS_ARCH:= common.
|
|
if amod.Arch().ArchType != Common {
|
|
a.SetString("LOCAL_MODULE_HOST_CROSS_ARCH", archStr)
|
|
}
|
|
} else {
|
|
// Make cannot identify LOCAL_MODULE_HOST_ARCH:= common.
|
|
if amod.Arch().ArchType != Common {
|
|
a.SetString("LOCAL_MODULE_HOST_ARCH", archStr)
|
|
}
|
|
}
|
|
host = true
|
|
case Device:
|
|
// Make cannot identify LOCAL_MODULE_TARGET_ARCH:= common.
|
|
if amod.Arch().ArchType != Common {
|
|
if amod.Target().NativeBridge {
|
|
hostArchStr := amod.Target().NativeBridgeHostArchName
|
|
if hostArchStr != "" {
|
|
a.SetString("LOCAL_MODULE_TARGET_ARCH", hostArchStr)
|
|
}
|
|
} else {
|
|
a.SetString("LOCAL_MODULE_TARGET_ARCH", archStr)
|
|
}
|
|
}
|
|
|
|
a.AddStrings("LOCAL_INIT_RC", amod.commonProperties.Init_rc...)
|
|
a.AddStrings("LOCAL_VINTF_FRAGMENTS", amod.commonProperties.Vintf_fragments...)
|
|
a.SetBoolIfTrue("LOCAL_PROPRIETARY_MODULE", Bool(amod.commonProperties.Proprietary))
|
|
if Bool(amod.commonProperties.Vendor) || Bool(amod.commonProperties.Soc_specific) {
|
|
a.SetString("LOCAL_VENDOR_MODULE", "true")
|
|
}
|
|
a.SetBoolIfTrue("LOCAL_ODM_MODULE", Bool(amod.commonProperties.Device_specific))
|
|
a.SetBoolIfTrue("LOCAL_PRODUCT_MODULE", Bool(amod.commonProperties.Product_specific))
|
|
a.SetBoolIfTrue("LOCAL_SYSTEM_EXT_MODULE", Bool(amod.commonProperties.System_ext_specific))
|
|
if amod.commonProperties.Owner != nil {
|
|
a.SetString("LOCAL_MODULE_OWNER", *amod.commonProperties.Owner)
|
|
}
|
|
}
|
|
|
|
if len(amod.noticeFiles) > 0 {
|
|
a.SetString("LOCAL_NOTICE_FILE", strings.Join(amod.noticeFiles.Strings(), " "))
|
|
}
|
|
|
|
if host {
|
|
makeOs := amod.Os().String()
|
|
if amod.Os() == Linux || amod.Os() == LinuxBionic {
|
|
makeOs = "linux"
|
|
}
|
|
a.SetString("LOCAL_MODULE_HOST_OS", makeOs)
|
|
a.SetString("LOCAL_IS_HOST_MODULE", "true")
|
|
}
|
|
|
|
prefix := ""
|
|
if amod.ArchSpecific() {
|
|
switch amod.Os().Class {
|
|
case Host:
|
|
if amod.Target().HostCross {
|
|
prefix = "HOST_CROSS_"
|
|
} else {
|
|
prefix = "HOST_"
|
|
}
|
|
case Device:
|
|
prefix = "TARGET_"
|
|
|
|
}
|
|
|
|
if amod.Arch().ArchType != config.Targets[amod.Os()][0].Arch.ArchType {
|
|
prefix = "2ND_" + prefix
|
|
}
|
|
}
|
|
for _, extra := range a.ExtraEntries {
|
|
extra(a)
|
|
}
|
|
|
|
// Write to footer.
|
|
fmt.Fprintln(&a.footer, "include "+a.Include)
|
|
blueprintDir := filepath.Dir(bpPath)
|
|
for _, footerFunc := range a.ExtraFooters {
|
|
footerFunc(&a.footer, name, prefix, blueprintDir, a)
|
|
}
|
|
}
|
|
|
|
func (a *AndroidMkEntries) write(w io.Writer) {
|
|
if a.Disabled {
|
|
return
|
|
}
|
|
|
|
if !a.OutputFile.Valid() {
|
|
return
|
|
}
|
|
|
|
w.Write(a.header.Bytes())
|
|
for _, name := range a.entryOrder {
|
|
fmt.Fprintln(w, name+" := "+strings.Join(a.EntryMap[name], " "))
|
|
}
|
|
w.Write(a.footer.Bytes())
|
|
}
|
|
|
|
func (a *AndroidMkEntries) FooterLinesForTests() []string {
|
|
return strings.Split(string(a.footer.Bytes()), "\n")
|
|
}
|
|
|
|
func AndroidMkSingleton() Singleton {
|
|
return &androidMkSingleton{}
|
|
}
|
|
|
|
type androidMkSingleton struct{}
|
|
|
|
func (c *androidMkSingleton) GenerateBuildActions(ctx SingletonContext) {
|
|
if !ctx.Config().KatiEnabled() {
|
|
return
|
|
}
|
|
|
|
var androidMkModulesList []blueprint.Module
|
|
|
|
ctx.VisitAllModulesBlueprint(func(module blueprint.Module) {
|
|
androidMkModulesList = append(androidMkModulesList, module)
|
|
})
|
|
|
|
sort.SliceStable(androidMkModulesList, func(i, j int) bool {
|
|
return ctx.ModuleName(androidMkModulesList[i]) < ctx.ModuleName(androidMkModulesList[j])
|
|
})
|
|
|
|
transMk := PathForOutput(ctx, "Android"+String(ctx.Config().productVariables.Make_suffix)+".mk")
|
|
if ctx.Failed() {
|
|
return
|
|
}
|
|
|
|
err := translateAndroidMk(ctx, absolutePath(transMk.String()), androidMkModulesList)
|
|
if err != nil {
|
|
ctx.Errorf(err.Error())
|
|
}
|
|
|
|
ctx.Build(pctx, BuildParams{
|
|
Rule: blueprint.Phony,
|
|
Output: transMk,
|
|
})
|
|
}
|
|
|
|
func translateAndroidMk(ctx SingletonContext, mkFile string, mods []blueprint.Module) error {
|
|
buf := &bytes.Buffer{}
|
|
|
|
fmt.Fprintln(buf, "LOCAL_MODULE_MAKEFILE := $(lastword $(MAKEFILE_LIST))")
|
|
|
|
type_stats := make(map[string]int)
|
|
for _, mod := range mods {
|
|
err := translateAndroidMkModule(ctx, buf, mod)
|
|
if err != nil {
|
|
os.Remove(mkFile)
|
|
return err
|
|
}
|
|
|
|
if amod, ok := mod.(Module); ok && ctx.PrimaryModule(amod) == amod {
|
|
type_stats[ctx.ModuleType(amod)] += 1
|
|
}
|
|
}
|
|
|
|
keys := []string{}
|
|
fmt.Fprintln(buf, "\nSTATS.SOONG_MODULE_TYPE :=")
|
|
for k := range type_stats {
|
|
keys = append(keys, k)
|
|
}
|
|
sort.Strings(keys)
|
|
for _, mod_type := range keys {
|
|
fmt.Fprintln(buf, "STATS.SOONG_MODULE_TYPE +=", mod_type)
|
|
fmt.Fprintf(buf, "STATS.SOONG_MODULE_TYPE.%s := %d\n", mod_type, type_stats[mod_type])
|
|
}
|
|
|
|
// Don't write to the file if it hasn't changed
|
|
if _, err := os.Stat(absolutePath(mkFile)); !os.IsNotExist(err) {
|
|
if data, err := ioutil.ReadFile(absolutePath(mkFile)); err == nil {
|
|
matches := buf.Len() == len(data)
|
|
|
|
if matches {
|
|
for i, value := range buf.Bytes() {
|
|
if value != data[i] {
|
|
matches = false
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
if matches {
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
return ioutil.WriteFile(absolutePath(mkFile), buf.Bytes(), 0666)
|
|
}
|
|
|
|
func translateAndroidMkModule(ctx SingletonContext, w io.Writer, mod blueprint.Module) error {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
panic(fmt.Errorf("%s in translateAndroidMkModule for module %s variant %s",
|
|
r, ctx.ModuleName(mod), ctx.ModuleSubDir(mod)))
|
|
}
|
|
}()
|
|
|
|
switch x := mod.(type) {
|
|
case AndroidMkDataProvider:
|
|
return translateAndroidModule(ctx, w, mod, x)
|
|
case bootstrap.GoBinaryTool:
|
|
return translateGoBinaryModule(ctx, w, mod, x)
|
|
case AndroidMkEntriesProvider:
|
|
return translateAndroidMkEntriesModule(ctx, w, mod, x)
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func translateGoBinaryModule(ctx SingletonContext, w io.Writer, mod blueprint.Module,
|
|
goBinary bootstrap.GoBinaryTool) error {
|
|
|
|
name := ctx.ModuleName(mod)
|
|
fmt.Fprintln(w, ".PHONY:", name)
|
|
fmt.Fprintln(w, name+":", goBinary.InstallPath())
|
|
fmt.Fprintln(w, "")
|
|
|
|
return nil
|
|
}
|
|
|
|
func (data *AndroidMkData) fillInData(config Config, bpPath string, mod blueprint.Module) {
|
|
// Get the preamble content through AndroidMkEntries logic.
|
|
data.Entries = AndroidMkEntries{
|
|
Class: data.Class,
|
|
SubName: data.SubName,
|
|
DistFiles: data.DistFiles,
|
|
OutputFile: data.OutputFile,
|
|
Disabled: data.Disabled,
|
|
Include: data.Include,
|
|
Required: data.Required,
|
|
Host_required: data.Host_required,
|
|
Target_required: data.Target_required,
|
|
}
|
|
data.Entries.fillInEntries(config, bpPath, mod)
|
|
|
|
// copy entries back to data since it is used in Custom
|
|
data.Required = data.Entries.Required
|
|
data.Host_required = data.Entries.Host_required
|
|
data.Target_required = data.Entries.Target_required
|
|
}
|
|
|
|
func translateAndroidModule(ctx SingletonContext, w io.Writer, mod blueprint.Module,
|
|
provider AndroidMkDataProvider) error {
|
|
|
|
amod := mod.(Module).base()
|
|
if shouldSkipAndroidMkProcessing(amod) {
|
|
return nil
|
|
}
|
|
|
|
data := provider.AndroidMk()
|
|
if data.Include == "" {
|
|
data.Include = "$(BUILD_PREBUILT)"
|
|
}
|
|
|
|
data.fillInData(ctx.Config(), ctx.BlueprintFile(mod), mod)
|
|
|
|
prefix := ""
|
|
if amod.ArchSpecific() {
|
|
switch amod.Os().Class {
|
|
case Host:
|
|
if amod.Target().HostCross {
|
|
prefix = "HOST_CROSS_"
|
|
} else {
|
|
prefix = "HOST_"
|
|
}
|
|
case Device:
|
|
prefix = "TARGET_"
|
|
|
|
}
|
|
|
|
if amod.Arch().ArchType != ctx.Config().Targets[amod.Os()][0].Arch.ArchType {
|
|
prefix = "2ND_" + prefix
|
|
}
|
|
}
|
|
|
|
name := provider.BaseModuleName()
|
|
blueprintDir := filepath.Dir(ctx.BlueprintFile(mod))
|
|
|
|
if data.Custom != nil {
|
|
data.Custom(w, name, prefix, blueprintDir, data)
|
|
} else {
|
|
WriteAndroidMkData(w, data)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func WriteAndroidMkData(w io.Writer, data AndroidMkData) {
|
|
if data.Disabled {
|
|
return
|
|
}
|
|
|
|
if !data.OutputFile.Valid() {
|
|
return
|
|
}
|
|
|
|
// write preamble via Entries
|
|
data.Entries.footer = bytes.Buffer{}
|
|
data.Entries.write(w)
|
|
|
|
for _, extra := range data.Extra {
|
|
extra(w, data.OutputFile.Path())
|
|
}
|
|
|
|
fmt.Fprintln(w, "include "+data.Include)
|
|
}
|
|
|
|
func translateAndroidMkEntriesModule(ctx SingletonContext, w io.Writer, mod blueprint.Module,
|
|
provider AndroidMkEntriesProvider) error {
|
|
if shouldSkipAndroidMkProcessing(mod.(Module).base()) {
|
|
return nil
|
|
}
|
|
|
|
for _, entries := range provider.AndroidMkEntries() {
|
|
entries.fillInEntries(ctx.Config(), ctx.BlueprintFile(mod), mod)
|
|
entries.write(w)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func shouldSkipAndroidMkProcessing(module *ModuleBase) bool {
|
|
if !module.commonProperties.NamespaceExportedToMake {
|
|
// TODO(jeffrygaston) do we want to validate that there are no modules being
|
|
// exported to Kati that depend on this module?
|
|
return true
|
|
}
|
|
|
|
return !module.Enabled() ||
|
|
module.commonProperties.SkipInstall ||
|
|
// Make does not understand LinuxBionic
|
|
module.Os() == LinuxBionic
|
|
}
|
|
|
|
func AndroidMkDataPaths(data []DataPath) []string {
|
|
var testFiles []string
|
|
for _, d := range data {
|
|
rel := d.SrcPath.Rel()
|
|
path := d.SrcPath.String()
|
|
if !strings.HasSuffix(path, rel) {
|
|
panic(fmt.Errorf("path %q does not end with %q", path, rel))
|
|
}
|
|
path = strings.TrimSuffix(path, rel)
|
|
testFileString := path + ":" + rel
|
|
if len(d.RelativeInstallPath) > 0 {
|
|
testFileString += ":" + d.RelativeInstallPath
|
|
}
|
|
testFiles = append(testFiles, testFileString)
|
|
}
|
|
return testFiles
|
|
}
|