2015-07-09 03:13:11 +02:00
|
|
|
// 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.
|
|
|
|
|
2016-05-19 00:37:25 +02:00
|
|
|
package android
|
2015-07-09 03:13:11 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2016-02-10 02:43:51 +01:00
|
|
|
"fmt"
|
2015-07-09 03:13:11 +02:00
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"sort"
|
2016-06-02 00:25:32 +02:00
|
|
|
"strings"
|
2015-07-09 03:13:11 +02:00
|
|
|
|
|
|
|
"github.com/google/blueprint"
|
2018-09-28 19:19:18 +02:00
|
|
|
"github.com/google/blueprint/bootstrap"
|
2015-07-09 03:13:11 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2020-03-04 21:15:08 +01:00
|
|
|
RegisterAndroidMkBuildComponents(InitRegistrationContext)
|
|
|
|
}
|
|
|
|
|
|
|
|
func RegisterAndroidMkBuildComponents(ctx RegistrationContext) {
|
|
|
|
ctx.RegisterSingletonType("androidmk", AndroidMkSingleton)
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
// Deprecated: consider using AndroidMkEntriesProvider instead, especially if you're not going to
|
|
|
|
// use the Custom function.
|
2015-07-09 03:13:11 +02:00
|
|
|
type AndroidMkDataProvider interface {
|
2017-08-11 02:00:19 +02:00
|
|
|
AndroidMk() AndroidMkData
|
2016-10-07 01:12:58 +02:00
|
|
|
BaseModuleName() string
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type AndroidMkData struct {
|
2019-04-02 03:37:36 +02:00
|
|
|
Class string
|
|
|
|
SubName string
|
2020-06-15 07:24:19 +02:00
|
|
|
DistFiles TaggedDistFiles
|
2019-04-02 03:37:36 +02:00
|
|
|
OutputFile OptionalPath
|
|
|
|
Disabled bool
|
|
|
|
Include string
|
|
|
|
Required []string
|
|
|
|
Host_required []string
|
|
|
|
Target_required []string
|
2015-07-09 03:13:11 +02:00
|
|
|
|
2017-08-11 02:07:28 +02:00
|
|
|
Custom func(w io.Writer, name, prefix, moduleDir string, data AndroidMkData)
|
2015-07-09 03:13:11 +02:00
|
|
|
|
2017-08-11 01:32:23 +02:00
|
|
|
Extra []AndroidMkExtraFunc
|
2017-08-11 02:07:28 +02:00
|
|
|
|
2020-06-24 16:26:26 +02:00
|
|
|
Entries AndroidMkEntries
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
|
|
|
|
2017-08-11 01:32:23 +02:00
|
|
|
type AndroidMkExtraFunc func(w io.Writer, outputFile Path)
|
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
// Allows modules to customize their Android*.mk output.
|
|
|
|
type AndroidMkEntriesProvider interface {
|
2019-12-03 05:24:29 +01:00
|
|
|
AndroidMkEntries() []AndroidMkEntries
|
2019-04-04 00:47:29 +02:00
|
|
|
BaseModuleName() string
|
|
|
|
}
|
|
|
|
|
|
|
|
type AndroidMkEntries struct {
|
|
|
|
Class string
|
|
|
|
SubName string
|
2020-06-15 07:24:19 +02:00
|
|
|
DistFiles TaggedDistFiles
|
2019-04-04 00:47:29 +02:00
|
|
|
OutputFile OptionalPath
|
|
|
|
Disabled bool
|
|
|
|
Include string
|
|
|
|
Required []string
|
|
|
|
Host_required []string
|
|
|
|
Target_required []string
|
|
|
|
|
|
|
|
header bytes.Buffer
|
|
|
|
footer bytes.Buffer
|
|
|
|
|
2019-08-28 02:33:16 +02:00
|
|
|
ExtraEntries []AndroidMkExtraEntriesFunc
|
2019-08-29 23:56:03 +02:00
|
|
|
ExtraFooters []AndroidMkExtraFootersFunc
|
2019-04-04 00:47:29 +02:00
|
|
|
|
|
|
|
EntryMap map[string][]string
|
|
|
|
entryOrder []string
|
|
|
|
}
|
|
|
|
|
2019-08-28 02:33:16 +02:00
|
|
|
type AndroidMkExtraEntriesFunc func(entries *AndroidMkEntries)
|
2019-08-29 23:56:03 +02:00
|
|
|
type AndroidMkExtraFootersFunc func(w io.Writer, name, prefix, moduleDir string, entries *AndroidMkEntries)
|
2019-08-28 02:33:16 +02:00
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
func (a *AndroidMkEntries) SetString(name, value string) {
|
|
|
|
if _, ok := a.EntryMap[name]; !ok {
|
|
|
|
a.entryOrder = append(a.entryOrder, name)
|
|
|
|
}
|
|
|
|
a.EntryMap[name] = []string{value}
|
|
|
|
}
|
|
|
|
|
2019-09-05 05:17:54 +02:00
|
|
|
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()}
|
|
|
|
}
|
|
|
|
|
2020-07-03 20:56:24 +02:00
|
|
|
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())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
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"}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-05 05:17:54 +02:00
|
|
|
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"}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
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...)
|
|
|
|
}
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
// Compute the list of Make strings to declare phone goals and dist-for-goals
|
|
|
|
// calls from the module's dist and dists properties.
|
|
|
|
func (a *AndroidMkEntries) GetDistForGoals(mod blueprint.Module) []string {
|
2019-04-04 00:47:29 +02:00
|
|
|
amod := mod.(Module).base()
|
|
|
|
name := amod.BaseModuleName()
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
var ret []string
|
|
|
|
|
|
|
|
availableTaggedDists := TaggedDistFiles{}
|
2020-07-21 13:32:19 +02:00
|
|
|
if a.DistFiles != nil {
|
2020-06-15 07:24:19 +02:00
|
|
|
availableTaggedDists = a.DistFiles
|
|
|
|
} else if a.OutputFile.Valid() {
|
|
|
|
availableTaggedDists = MakeDefaultDistFiles(a.OutputFile.Path())
|
2019-04-04 00:47:29 +02:00
|
|
|
}
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
// 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 = ""
|
|
|
|
} else {
|
|
|
|
tag = *dist.Tag
|
2019-04-04 00:47:29 +02:00
|
|
|
}
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
// 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 := "Cannot apply dest/suffix for more than one dist " +
|
|
|
|
"file for %s goals in module %s. The list of dist files, " +
|
|
|
|
"which should have a single element, is:\n%s"
|
|
|
|
panic(fmt.Errorf(errorMessage, goals, name, tagPaths))
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = append(ret, fmt.Sprintf(".PHONY: %s\n", goals))
|
|
|
|
|
|
|
|
// Create dist-for-goals calls for each path in the dist'd files.
|
|
|
|
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 {
|
2019-04-04 00:47:29 +02:00
|
|
|
var err error
|
2020-06-15 07:24:19 +02:00
|
|
|
if dest, err = validateSafePath(*dist.Dest); err != nil {
|
2019-04-04 00:47:29 +02:00
|
|
|
// This was checked in ModuleBase.GenerateBuildActions
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
if dist.Suffix != nil {
|
2019-04-04 00:47:29 +02:00
|
|
|
ext := filepath.Ext(dest)
|
2020-06-15 07:24:19 +02:00
|
|
|
suffix := *dist.Suffix
|
2019-04-04 00:47:29 +02:00
|
|
|
dest = strings.TrimSuffix(dest, ext) + suffix + ext
|
|
|
|
}
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
if dist.Dir != nil {
|
2019-04-04 00:47:29 +02:00
|
|
|
var err error
|
2020-06-15 07:24:19 +02:00
|
|
|
if dest, err = validateSafePath(*dist.Dir, dest); err != nil {
|
2019-04-04 00:47:29 +02:00
|
|
|
// This was checked in ModuleBase.GenerateBuildActions
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
ret = append(
|
|
|
|
ret,
|
|
|
|
fmt.Sprintf("$(call dist-for-goals,%s,%s:%s)\n", goals, path.String(), dest))
|
2019-04-04 00:47:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-15 07:24:19 +02:00
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
|
|
|
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.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)
|
|
|
|
}
|
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
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...)
|
|
|
|
|
mark platform un-availability
A module is marked unavailable for platform when 1) it does not have
"//apex_available:platform" in its apex_available property, or 2)
it depends on another module that is unavailable for platform.
In that case, LOCAL_NOT_AVAILABLE_FOR_PLATFORM is set to true for the
module in the Make world. Later, that flag is used to ensure that there
is no module with the flag is installed to the device.
The reason why this isn't entirely done in Soong is because Soong
doesn't know if a module will be installed to the device or not. To
explain this, let's have an example.
cc_test { name: "mytest", static_libs: ["libfoo"]}
cc_library_static { name: "libfoo", static_libs: ["libbar"]}
cc_library { name: "libbar", apex_available: ["com.android.xxx"]}
Here, libbar is not available for platform, but is used by libfoo which
is available for platform (apex_available defaults to
"//apex_available:platform"). libfoo is again depended on by mytest
which again is available for platform. The use of libbar should be
allowed in the context of test; we don't want to make libbar available
to platform just for the dependency from test because it will allow
non-test uses of the library as well.
Soong by itself can't tell whether libfoo and libbar are used only in the
context of a test. There could be another module depending them, e.g.,
cc_library_shared { name: "mylib", static_libs: ["libfoo"] }
can exist and it might be installed to the device, in which case
we really should trigger an error.
Since Make has the knowledge of what's installed and what's not,
the check should be done there.
Bug: 153073816
Test: m
Test: remove "//apex_available:platform" from libmdnssd (it is currently
installed to /system/lib), and check that `m system_image` fails
Change-Id: Ia304cc5f41f173229e8a154e90cea4dce46dcebe
2020-04-07 09:37:39 +02:00
|
|
|
if am, ok := mod.(ApexModule); ok {
|
|
|
|
a.SetBoolIfTrue("LOCAL_NOT_AVAILABLE_FOR_PLATFORM", am.NotAvailableForPlatform())
|
|
|
|
}
|
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
archStr := amod.Arch().ArchType.String()
|
|
|
|
host := false
|
|
|
|
switch amod.Os().Class {
|
|
|
|
case Host:
|
|
|
|
// Make cannot identify LOCAL_MODULE_HOST_ARCH:= common.
|
2019-10-16 20:03:10 +02:00
|
|
|
if amod.Arch().ArchType != Common {
|
2019-04-04 00:47:29 +02:00
|
|
|
a.SetString("LOCAL_MODULE_HOST_ARCH", archStr)
|
|
|
|
}
|
|
|
|
host = true
|
|
|
|
case HostCross:
|
|
|
|
// Make cannot identify LOCAL_MODULE_HOST_CROSS_ARCH:= common.
|
2019-10-16 20:03:10 +02:00
|
|
|
if amod.Arch().ArchType != Common {
|
2019-04-04 00:47:29 +02:00
|
|
|
a.SetString("LOCAL_MODULE_HOST_CROSS_ARCH", archStr)
|
|
|
|
}
|
|
|
|
host = true
|
|
|
|
case Device:
|
|
|
|
// Make cannot identify LOCAL_MODULE_TARGET_ARCH:= common.
|
2019-10-16 20:03:10 +02:00
|
|
|
if amod.Arch().ArchType != Common {
|
2019-03-26 12:39:31 +01:00
|
|
|
if amod.Target().NativeBridge {
|
2019-07-11 10:23:53 +02:00
|
|
|
hostArchStr := amod.Target().NativeBridgeHostArchName
|
2019-03-26 12:39:31 +01:00
|
|
|
if hostArchStr != "" {
|
|
|
|
a.SetString("LOCAL_MODULE_TARGET_ARCH", hostArchStr)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
a.SetString("LOCAL_MODULE_TARGET_ARCH", archStr)
|
|
|
|
}
|
2019-04-04 00:47:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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))
|
2019-06-25 09:47:17 +02:00
|
|
|
a.SetBoolIfTrue("LOCAL_SYSTEM_EXT_MODULE", Bool(amod.commonProperties.System_ext_specific))
|
2019-04-04 00:47:29 +02:00
|
|
|
if amod.commonProperties.Owner != nil {
|
|
|
|
a.SetString("LOCAL_MODULE_OWNER", *amod.commonProperties.Owner)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-19 05:21:55 +01:00
|
|
|
if len(amod.noticeFiles) > 0 {
|
|
|
|
a.SetString("LOCAL_NOTICE_FILE", strings.Join(amod.noticeFiles.Strings(), " "))
|
2019-04-04 00:47:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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:
|
|
|
|
prefix = "HOST_"
|
|
|
|
case HostCross:
|
|
|
|
prefix = "HOST_CROSS_"
|
|
|
|
case Device:
|
|
|
|
prefix = "TARGET_"
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if amod.Arch().ArchType != config.Targets[amod.Os()][0].Arch.ArchType {
|
|
|
|
prefix = "2ND_" + prefix
|
|
|
|
}
|
|
|
|
}
|
2019-08-28 02:33:16 +02:00
|
|
|
for _, extra := range a.ExtraEntries {
|
|
|
|
extra(a)
|
2019-04-04 00:47:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write to footer.
|
|
|
|
fmt.Fprintln(&a.footer, "include "+a.Include)
|
2019-08-29 23:56:03 +02:00
|
|
|
blueprintDir := filepath.Dir(bpPath)
|
|
|
|
for _, footerFunc := range a.ExtraFooters {
|
|
|
|
footerFunc(&a.footer, name, prefix, blueprintDir, a)
|
|
|
|
}
|
2019-04-04 00:47:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *AndroidMkEntries) write(w io.Writer) {
|
2019-08-29 23:56:03 +02:00
|
|
|
if a.Disabled {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if !a.OutputFile.Valid() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
w.Write(a.header.Bytes())
|
|
|
|
for _, name := range a.entryOrder {
|
|
|
|
fmt.Fprintln(w, name+" := "+strings.Join(a.EntryMap[name], " "))
|
|
|
|
}
|
|
|
|
w.Write(a.footer.Bytes())
|
|
|
|
}
|
|
|
|
|
2019-08-29 23:56:03 +02:00
|
|
|
func (a *AndroidMkEntries) FooterLinesForTests() []string {
|
|
|
|
return strings.Split(string(a.footer.Bytes()), "\n")
|
|
|
|
}
|
|
|
|
|
2017-11-29 02:34:01 +01:00
|
|
|
func AndroidMkSingleton() Singleton {
|
2015-07-09 03:13:11 +02:00
|
|
|
return &androidMkSingleton{}
|
|
|
|
}
|
|
|
|
|
|
|
|
type androidMkSingleton struct{}
|
|
|
|
|
2017-11-29 02:34:01 +01:00
|
|
|
func (c *androidMkSingleton) GenerateBuildActions(ctx SingletonContext) {
|
2017-11-29 09:27:14 +01:00
|
|
|
if !ctx.Config().EmbeddedInMake() {
|
2015-12-11 22:51:06 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-09-28 19:19:18 +02:00
|
|
|
var androidMkModulesList []blueprint.Module
|
2015-07-09 03:13:11 +02:00
|
|
|
|
2018-09-28 19:19:18 +02:00
|
|
|
ctx.VisitAllModulesBlueprint(func(module blueprint.Module) {
|
2017-11-29 02:34:01 +01:00
|
|
|
androidMkModulesList = append(androidMkModulesList, module)
|
2016-01-11 21:55:55 +01:00
|
|
|
})
|
2015-07-09 03:13:11 +02:00
|
|
|
|
2019-01-14 21:47:35 +01:00
|
|
|
sort.SliceStable(androidMkModulesList, func(i, j int) bool {
|
|
|
|
return ctx.ModuleName(androidMkModulesList[i]) < ctx.ModuleName(androidMkModulesList[j])
|
|
|
|
})
|
2015-12-18 03:00:23 +01:00
|
|
|
|
2018-03-10 06:22:06 +01:00
|
|
|
transMk := PathForOutput(ctx, "Android"+String(ctx.Config().productVariables.Make_suffix)+".mk")
|
2015-09-24 00:26:20 +02:00
|
|
|
if ctx.Failed() {
|
|
|
|
return
|
|
|
|
}
|
2015-07-09 03:13:11 +02:00
|
|
|
|
2020-01-11 02:11:46 +01:00
|
|
|
err := translateAndroidMk(ctx, absolutePath(transMk.String()), androidMkModulesList)
|
2015-07-09 03:13:11 +02:00
|
|
|
if err != nil {
|
|
|
|
ctx.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
|
2017-11-29 02:34:01 +01:00
|
|
|
ctx.Build(pctx, BuildParams{
|
|
|
|
Rule: blueprint.Phony,
|
|
|
|
Output: transMk,
|
2015-07-09 03:13:11 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-09-28 19:19:18 +02:00
|
|
|
func translateAndroidMk(ctx SingletonContext, mkFile string, mods []blueprint.Module) error {
|
2015-07-09 03:13:11 +02:00
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
|
2016-02-10 02:43:51 +01:00
|
|
|
fmt.Fprintln(buf, "LOCAL_MODULE_MAKEFILE := $(lastword $(MAKEFILE_LIST))")
|
2015-07-09 03:13:11 +02:00
|
|
|
|
2016-07-26 01:00:20 +02:00
|
|
|
type_stats := make(map[string]int)
|
2015-07-09 03:13:11 +02:00
|
|
|
for _, mod := range mods {
|
|
|
|
err := translateAndroidMkModule(ctx, buf, mod)
|
|
|
|
if err != nil {
|
|
|
|
os.Remove(mkFile)
|
|
|
|
return err
|
|
|
|
}
|
2016-07-26 01:00:20 +02:00
|
|
|
|
2018-09-28 19:19:18 +02:00
|
|
|
if amod, ok := mod.(Module); ok && ctx.PrimaryModule(amod) == amod {
|
|
|
|
type_stats[ctx.ModuleType(amod)] += 1
|
2016-07-26 01:00:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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])
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Don't write to the file if it hasn't changed
|
2020-01-11 02:11:46 +01:00
|
|
|
if _, err := os.Stat(absolutePath(mkFile)); !os.IsNotExist(err) {
|
|
|
|
if data, err := ioutil.ReadFile(absolutePath(mkFile)); err == nil {
|
2015-07-09 03:13:11 +02:00
|
|
|
matches := buf.Len() == len(data)
|
|
|
|
|
|
|
|
if matches {
|
|
|
|
for i, value := range buf.Bytes() {
|
|
|
|
if value != data[i] {
|
|
|
|
matches = false
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if matches {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-11 02:11:46 +01:00
|
|
|
return ioutil.WriteFile(absolutePath(mkFile), buf.Bytes(), 0666)
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
|
|
|
|
2017-11-29 02:34:01 +01:00
|
|
|
func translateAndroidMkModule(ctx SingletonContext, w io.Writer, mod blueprint.Module) error {
|
2018-09-06 01:23:54 +02:00
|
|
|
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)))
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2018-09-28 19:19:18 +02:00
|
|
|
switch x := mod.(type) {
|
|
|
|
case AndroidMkDataProvider:
|
|
|
|
return translateAndroidModule(ctx, w, mod, x)
|
|
|
|
case bootstrap.GoBinaryTool:
|
|
|
|
return translateGoBinaryModule(ctx, w, mod, x)
|
2019-04-04 00:47:29 +02:00
|
|
|
case AndroidMkEntriesProvider:
|
|
|
|
return translateAndroidMkEntriesModule(ctx, w, mod, x)
|
2018-09-28 19:19:18 +02:00
|
|
|
default:
|
2016-02-10 02:43:51 +01:00
|
|
|
return nil
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
2018-09-28 19:19:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-07-11 09:18:47 +02:00
|
|
|
func (data *AndroidMkData) fillInData(config Config, bpPath string, mod blueprint.Module) {
|
2019-04-04 00:47:29 +02:00
|
|
|
// Get the preamble content through AndroidMkEntries logic.
|
2020-06-24 16:26:26 +02:00
|
|
|
data.Entries = AndroidMkEntries{
|
2019-04-04 00:47:29 +02:00
|
|
|
Class: data.Class,
|
|
|
|
SubName: data.SubName,
|
2020-06-15 07:24:19 +02:00
|
|
|
DistFiles: data.DistFiles,
|
2019-04-04 00:47:29 +02:00
|
|
|
OutputFile: data.OutputFile,
|
|
|
|
Disabled: data.Disabled,
|
|
|
|
Include: data.Include,
|
|
|
|
Required: data.Required,
|
|
|
|
Host_required: data.Host_required,
|
|
|
|
Target_required: data.Target_required,
|
2016-06-14 02:19:03 +02:00
|
|
|
}
|
2020-06-24 16:26:26 +02:00
|
|
|
data.Entries.fillInEntries(config, bpPath, mod)
|
2016-06-14 02:19:03 +02:00
|
|
|
|
2019-07-11 09:18:47 +02:00
|
|
|
// copy entries back to data since it is used in Custom
|
2020-06-24 16:26:26 +02:00
|
|
|
data.Required = data.Entries.Required
|
|
|
|
data.Host_required = data.Entries.Host_required
|
|
|
|
data.Target_required = data.Entries.Target_required
|
2019-07-11 09:18:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
2017-08-11 02:07:28 +02:00
|
|
|
prefix := ""
|
|
|
|
if amod.ArchSpecific() {
|
|
|
|
switch amod.Os().Class {
|
|
|
|
case Host:
|
|
|
|
prefix = "HOST_"
|
|
|
|
case HostCross:
|
|
|
|
prefix = "HOST_CROSS_"
|
|
|
|
case Device:
|
|
|
|
prefix = "TARGET_"
|
2016-03-24 21:14:12 +01:00
|
|
|
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
|
|
|
|
2018-10-11 02:02:29 +02:00
|
|
|
if amod.Arch().ArchType != ctx.Config().Targets[amod.Os()][0].Arch.ArchType {
|
2017-08-11 02:07:28 +02:00
|
|
|
prefix = "2ND_" + prefix
|
|
|
|
}
|
2016-02-10 02:43:51 +01:00
|
|
|
}
|
2015-07-09 03:13:11 +02:00
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
name := provider.BaseModuleName()
|
2017-08-11 02:07:28 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2020-06-24 16:26:26 +02:00
|
|
|
// write preamble via Entries
|
|
|
|
data.Entries.footer = bytes.Buffer{}
|
|
|
|
data.Entries.write(w)
|
2017-08-11 02:07:28 +02:00
|
|
|
|
2016-01-04 23:34:37 +01:00
|
|
|
for _, extra := range data.Extra {
|
2017-08-11 01:32:23 +02:00
|
|
|
extra(w, data.OutputFile.Path())
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
|
|
|
|
2017-09-07 22:20:25 +02:00
|
|
|
fmt.Fprintln(w, "include "+data.Include)
|
2015-07-09 03:13:11 +02:00
|
|
|
}
|
2019-04-02 03:37:36 +02:00
|
|
|
|
2019-04-04 00:47:29 +02:00
|
|
|
func translateAndroidMkEntriesModule(ctx SingletonContext, w io.Writer, mod blueprint.Module,
|
|
|
|
provider AndroidMkEntriesProvider) error {
|
|
|
|
if shouldSkipAndroidMkProcessing(mod.(Module).base()) {
|
|
|
|
return nil
|
2019-04-02 03:37:36 +02:00
|
|
|
}
|
2019-04-04 00:47:29 +02:00
|
|
|
|
2019-12-03 05:24:29 +01:00
|
|
|
for _, entries := range provider.AndroidMkEntries() {
|
|
|
|
entries.fillInEntries(ctx.Config(), ctx.BlueprintFile(mod), mod)
|
|
|
|
entries.write(w)
|
|
|
|
}
|
2019-04-04 00:47:29 +02:00
|
|
|
|
|
|
|
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
|
2019-04-02 03:37:36 +02:00
|
|
|
}
|
2019-04-04 00:47:29 +02:00
|
|
|
|
|
|
|
return !module.Enabled() ||
|
|
|
|
module.commonProperties.SkipInstall ||
|
|
|
|
// Make does not understand LinuxBionic
|
|
|
|
module.Os() == LinuxBionic
|
2019-04-02 03:37:36 +02:00
|
|
|
}
|