aa118f95a6
The vendor image will have more than just proprietary modules in it under Treble, so let's stop marking open source code as proprietary just to move it to vendor. Bug: 36452052 Bug: 37134596 Test: compare build.ninja before/after, no changes. Test: Set vendor: true, ensure it works. Change-Id: I44b0ec7007d0e311bdcbd44b238b1ef2d05cc6ff
260 lines
6 KiB
Go
260 lines
6 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/proptools"
|
|
)
|
|
|
|
func init() {
|
|
RegisterSingletonType("androidmk", AndroidMkSingleton)
|
|
}
|
|
|
|
type AndroidMkDataProvider interface {
|
|
AndroidMk() (AndroidMkData, error)
|
|
BaseModuleName() string
|
|
}
|
|
|
|
type AndroidMkData struct {
|
|
Class string
|
|
SubName string
|
|
OutputFile OptionalPath
|
|
Disabled bool
|
|
|
|
Custom func(w io.Writer, name, prefix, moduleDir string) error
|
|
|
|
Extra []func(w io.Writer, outputFile Path) error
|
|
}
|
|
|
|
func AndroidMkSingleton() blueprint.Singleton {
|
|
return &androidMkSingleton{}
|
|
}
|
|
|
|
type androidMkSingleton struct{}
|
|
|
|
func (c *androidMkSingleton) GenerateBuildActions(ctx blueprint.SingletonContext) {
|
|
config := ctx.Config().(Config)
|
|
|
|
if !config.EmbeddedInMake() {
|
|
return
|
|
}
|
|
|
|
ctx.SetNinjaBuildDir(pctx, filepath.Join(config.buildDir, ".."))
|
|
|
|
var androidMkModulesList []Module
|
|
|
|
ctx.VisitAllModules(func(module blueprint.Module) {
|
|
if amod, ok := module.(Module); ok {
|
|
androidMkModulesList = append(androidMkModulesList, amod)
|
|
}
|
|
})
|
|
|
|
sort.Sort(AndroidModulesByName{androidMkModulesList, ctx})
|
|
|
|
transMk := PathForOutput(ctx, "Android"+proptools.String(config.ProductVariables.Make_suffix)+".mk")
|
|
if ctx.Failed() {
|
|
return
|
|
}
|
|
|
|
err := translateAndroidMk(ctx, transMk.String(), androidMkModulesList)
|
|
if err != nil {
|
|
ctx.Errorf(err.Error())
|
|
}
|
|
|
|
ctx.Build(pctx, blueprint.BuildParams{
|
|
Rule: blueprint.Phony,
|
|
Outputs: []string{transMk.String()},
|
|
Optional: true,
|
|
})
|
|
}
|
|
|
|
func translateAndroidMk(ctx blueprint.SingletonContext, mkFile string, mods []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 ctx.PrimaryModule(mod) == mod {
|
|
type_stats[ctx.ModuleType(mod)] += 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(mkFile); !os.IsNotExist(err) {
|
|
if data, err := ioutil.ReadFile(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(mkFile, buf.Bytes(), 0666)
|
|
}
|
|
|
|
func translateAndroidMkModule(ctx blueprint.SingletonContext, w io.Writer, mod blueprint.Module) error {
|
|
provider, ok := mod.(AndroidMkDataProvider)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
|
|
name := provider.BaseModuleName()
|
|
amod := mod.(Module).base()
|
|
|
|
if !amod.Enabled() {
|
|
return nil
|
|
}
|
|
|
|
if amod.commonProperties.SkipInstall {
|
|
return nil
|
|
}
|
|
|
|
data, err := provider.AndroidMk()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Make does not understand LinuxBionic
|
|
if amod.Os() == LinuxBionic {
|
|
return nil
|
|
}
|
|
|
|
if data.SubName != "" {
|
|
name += data.SubName
|
|
}
|
|
|
|
if data.Custom != nil {
|
|
prefix := ""
|
|
if amod.ArchSpecific() {
|
|
switch amod.Os().Class {
|
|
case Host:
|
|
prefix = "HOST_"
|
|
case HostCross:
|
|
prefix = "HOST_CROSS_"
|
|
case Device:
|
|
prefix = "TARGET_"
|
|
|
|
}
|
|
|
|
config := ctx.Config().(Config)
|
|
if amod.Arch().ArchType != config.Targets[amod.Os().Class][0].Arch.ArchType {
|
|
prefix = "2ND_" + prefix
|
|
}
|
|
}
|
|
|
|
return data.Custom(w, name, prefix, filepath.Dir(ctx.BlueprintFile(mod)))
|
|
}
|
|
|
|
if data.Disabled {
|
|
return nil
|
|
}
|
|
|
|
if !data.OutputFile.Valid() {
|
|
return err
|
|
}
|
|
|
|
fmt.Fprintln(w, "\ninclude $(CLEAR_VARS)")
|
|
fmt.Fprintln(w, "LOCAL_PATH :=", filepath.Dir(ctx.BlueprintFile(mod)))
|
|
fmt.Fprintln(w, "LOCAL_MODULE :=", name)
|
|
fmt.Fprintln(w, "LOCAL_MODULE_CLASS :=", data.Class)
|
|
fmt.Fprintln(w, "LOCAL_PREBUILT_MODULE_FILE :=", data.OutputFile.String())
|
|
|
|
if len(amod.commonProperties.Required) > 0 {
|
|
fmt.Fprintln(w, "LOCAL_REQUIRED_MODULES := "+strings.Join(amod.commonProperties.Required, " "))
|
|
}
|
|
|
|
archStr := amod.Arch().ArchType.String()
|
|
host := false
|
|
switch amod.Os().Class {
|
|
case Host:
|
|
fmt.Fprintln(w, "LOCAL_MODULE_HOST_ARCH :=", archStr)
|
|
host = true
|
|
case HostCross:
|
|
fmt.Fprintln(w, "LOCAL_MODULE_HOST_CROSS_ARCH :=", archStr)
|
|
host = true
|
|
case Device:
|
|
fmt.Fprintln(w, "LOCAL_MODULE_TARGET_ARCH :=", archStr)
|
|
|
|
if len(amod.commonProperties.Logtags) > 0 {
|
|
fmt.Fprintln(w, "LOCAL_LOGTAGS_FILES := ", strings.Join(amod.commonProperties.Logtags, " "))
|
|
}
|
|
if len(amod.commonProperties.Init_rc) > 0 {
|
|
fmt.Fprintln(w, "LOCAL_INIT_RC := ", strings.Join(amod.commonProperties.Init_rc, " "))
|
|
}
|
|
if amod.commonProperties.Proprietary {
|
|
fmt.Fprintln(w, "LOCAL_PROPRIETARY_MODULE := true")
|
|
}
|
|
if amod.commonProperties.Vendor {
|
|
fmt.Fprintln(w, "LOCAL_VENDOR_MODULE := true")
|
|
}
|
|
if amod.commonProperties.Owner != "" {
|
|
fmt.Fprintln(w, "LOCAL_MODULE_OWNER :=", amod.commonProperties.Owner)
|
|
}
|
|
}
|
|
|
|
if host {
|
|
fmt.Fprintln(w, "LOCAL_MODULE_HOST_OS :=", amod.Os().String())
|
|
fmt.Fprintln(w, "LOCAL_IS_HOST_MODULE := true")
|
|
}
|
|
|
|
for _, extra := range data.Extra {
|
|
err = extra(w, data.OutputFile.Path())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
fmt.Fprintln(w, "include $(BUILD_PREBUILT)")
|
|
|
|
return err
|
|
}
|