Support DCLA
Bug: 240424572 Test: Manual tests: 1. m --dev-mode-staging com.android.adbd com.android.media.swcodec. 2. verify the DCLA libs from the two apexes have the same size and sha1sum, and also match the libs in bazel-out. 3. empty the DCLA libs list in allowlist.go and repeat step 1 4. repeat step 2 and verify the opposite result 5. build git_master: mainline_modules_bundles-userdebug in ABTD with the cl, then follow go/build-sideload-dcla-locally to download the adbd and swcodec aab files, run the DCLA trimming workflow locally, and verify the symlinks in the two trimmed apexes are identical and also match the lib path in the DCLA apex that was created by the workflow. Change-Id: Ib2f8a29126a54829c0e10eba17b256a79930fd70
This commit is contained in:
@@ -109,12 +109,29 @@ type cqueryRequest interface {
|
||||
|
||||
// Portion of cquery map key to describe target configuration.
|
||||
type configKey struct {
|
||||
arch string
|
||||
osType OsType
|
||||
arch string
|
||||
osType OsType
|
||||
apexKey ApexConfigKey
|
||||
}
|
||||
|
||||
type ApexConfigKey struct {
|
||||
WithinApex bool
|
||||
ApexSdkVersion string
|
||||
}
|
||||
|
||||
func (c ApexConfigKey) String() string {
|
||||
return fmt.Sprintf("%s_%s", withinApexToString(c.WithinApex), c.ApexSdkVersion)
|
||||
}
|
||||
|
||||
func withinApexToString(withinApex bool) string {
|
||||
if withinApex {
|
||||
return "within_apex"
|
||||
}
|
||||
return ""
|
||||
}
|
||||
|
||||
func (c configKey) String() string {
|
||||
return fmt.Sprintf("%s::%s", c.arch, c.osType)
|
||||
return fmt.Sprintf("%s::%s::%s", c.arch, c.osType, c.apexKey)
|
||||
}
|
||||
|
||||
// Map key to describe bazel cquery requests.
|
||||
@@ -182,7 +199,7 @@ type BazelContext interface {
|
||||
// Note that this only implies "bazel mixed build" allowlisting. The caller
|
||||
// should independently verify the module is eligible for Bazel handling
|
||||
// (for example, that it is MixedBuildBuildable).
|
||||
IsModuleNameAllowed(moduleName string) bool
|
||||
IsModuleNameAllowed(moduleName string, withinApex bool) bool
|
||||
|
||||
// Returns the bazel output base (the root directory for all bazel intermediate outputs).
|
||||
OutputBase() string
|
||||
@@ -235,6 +252,8 @@ type mixedBuildBazelContext struct {
|
||||
bazelDisabledModules map[string]bool
|
||||
// Per-module allowlist to opt modules in to bazel handling.
|
||||
bazelEnabledModules map[string]bool
|
||||
// DCLA modules are enabled when used in apex.
|
||||
bazelDclaEnabledModules map[string]bool
|
||||
// If true, modules are bazel-enabled by default, unless present in bazelDisabledModules.
|
||||
modulesDefaultToBazel bool
|
||||
|
||||
@@ -258,10 +277,16 @@ type MockBazelContext struct {
|
||||
LabelToPythonBinary map[string]string
|
||||
LabelToApexInfo map[string]cquery.ApexInfo
|
||||
LabelToCcBinary map[string]cquery.CcUnstrippedInfo
|
||||
|
||||
BazelRequests map[string]bool
|
||||
}
|
||||
|
||||
func (m MockBazelContext) QueueBazelRequest(_ string, _ cqueryRequest, _ configKey) {
|
||||
panic("unimplemented")
|
||||
func (m MockBazelContext) QueueBazelRequest(label string, requestType cqueryRequest, cfgKey configKey) {
|
||||
key := BuildMockBazelContextRequestKey(label, requestType, cfgKey.arch, cfgKey.osType, cfgKey.apexKey)
|
||||
if m.BazelRequests == nil {
|
||||
m.BazelRequests = make(map[string]bool)
|
||||
}
|
||||
m.BazelRequests[key] = true
|
||||
}
|
||||
|
||||
func (m MockBazelContext) GetOutputFiles(label string, _ configKey) ([]string, error) {
|
||||
@@ -272,10 +297,14 @@ func (m MockBazelContext) GetOutputFiles(label string, _ configKey) ([]string, e
|
||||
return result, nil
|
||||
}
|
||||
|
||||
func (m MockBazelContext) GetCcInfo(label string, _ configKey) (cquery.CcInfo, error) {
|
||||
func (m MockBazelContext) GetCcInfo(label string, cfgKey configKey) (cquery.CcInfo, error) {
|
||||
result, ok := m.LabelToCcInfo[label]
|
||||
if !ok {
|
||||
return cquery.CcInfo{}, fmt.Errorf("no target with label %q in LabelToCcInfo", label)
|
||||
key := BuildMockBazelContextResultKey(label, cfgKey.arch, cfgKey.osType, cfgKey.apexKey)
|
||||
result, ok = m.LabelToCcInfo[key]
|
||||
if !ok {
|
||||
return cquery.CcInfo{}, fmt.Errorf("no target with label %q in LabelToCcInfo", label)
|
||||
}
|
||||
}
|
||||
return result, nil
|
||||
}
|
||||
@@ -308,7 +337,7 @@ func (m MockBazelContext) InvokeBazel(_ Config, _ invokeBazelContext) error {
|
||||
panic("unimplemented")
|
||||
}
|
||||
|
||||
func (m MockBazelContext) IsModuleNameAllowed(_ string) bool {
|
||||
func (m MockBazelContext) IsModuleNameAllowed(_ string, _ bool) bool {
|
||||
return true
|
||||
}
|
||||
|
||||
@@ -324,6 +353,26 @@ func (m MockBazelContext) AqueryDepsets() []bazel.AqueryDepset {
|
||||
|
||||
var _ BazelContext = MockBazelContext{}
|
||||
|
||||
func BuildMockBazelContextRequestKey(label string, request cqueryRequest, arch string, osType OsType, apexKey ApexConfigKey) string {
|
||||
cfgKey := configKey{
|
||||
arch: arch,
|
||||
osType: osType,
|
||||
apexKey: apexKey,
|
||||
}
|
||||
|
||||
return strings.Join([]string{label, request.Name(), cfgKey.String()}, "_")
|
||||
}
|
||||
|
||||
func BuildMockBazelContextResultKey(label string, arch string, osType OsType, apexKey ApexConfigKey) string {
|
||||
cfgKey := configKey{
|
||||
arch: arch,
|
||||
osType: osType,
|
||||
apexKey: apexKey,
|
||||
}
|
||||
|
||||
return strings.Join([]string{label, cfgKey.String()}, "_")
|
||||
}
|
||||
|
||||
func (bazelCtx *mixedBuildBazelContext) QueueBazelRequest(label string, requestType cqueryRequest, cfgKey configKey) {
|
||||
key := makeCqueryKey(label, requestType, cfgKey)
|
||||
bazelCtx.requestMutex.Lock()
|
||||
@@ -430,7 +479,7 @@ func (m noopBazelContext) OutputBase() string {
|
||||
return ""
|
||||
}
|
||||
|
||||
func (n noopBazelContext) IsModuleNameAllowed(_ string) bool {
|
||||
func (n noopBazelContext) IsModuleNameAllowed(_ string, _ bool) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
@@ -442,14 +491,15 @@ func (m noopBazelContext) AqueryDepsets() []bazel.AqueryDepset {
|
||||
return []bazel.AqueryDepset{}
|
||||
}
|
||||
|
||||
func addToStringSet(set map[string]bool, items []string) {
|
||||
for _, item := range items {
|
||||
set[item] = true
|
||||
}
|
||||
}
|
||||
|
||||
func GetBazelEnabledAndDisabledModules(buildMode SoongBuildMode, forceEnabled map[string]struct{}) (map[string]bool, map[string]bool) {
|
||||
disabledModules := map[string]bool{}
|
||||
enabledModules := map[string]bool{}
|
||||
addToStringSet := func(set map[string]bool, items []string) {
|
||||
for _, item := range items {
|
||||
set[item] = true
|
||||
}
|
||||
}
|
||||
|
||||
switch buildMode {
|
||||
case BazelProdMode:
|
||||
@@ -537,15 +587,24 @@ func NewBazelContext(c *config) (BazelContext, error) {
|
||||
if c.HasDeviceProduct() {
|
||||
targetProduct = c.DeviceProduct()
|
||||
}
|
||||
|
||||
dclaMixedBuildsEnabledList := []string{}
|
||||
if c.BuildMode == BazelProdMode {
|
||||
dclaMixedBuildsEnabledList = allowlists.ProdDclaMixedBuildsEnabledList
|
||||
} else if c.BuildMode == BazelStagingMode {
|
||||
dclaMixedBuildsEnabledList = append(allowlists.ProdDclaMixedBuildsEnabledList,
|
||||
allowlists.StagingDclaMixedBuildsEnabledList...)
|
||||
}
|
||||
dclaEnabledModules := map[string]bool{}
|
||||
addToStringSet(dclaEnabledModules, dclaMixedBuildsEnabledList)
|
||||
return &mixedBuildBazelContext{
|
||||
bazelRunner: &builtinBazelRunner{},
|
||||
paths: &paths,
|
||||
modulesDefaultToBazel: c.BuildMode == BazelDevMode,
|
||||
bazelEnabledModules: enabledModules,
|
||||
bazelDisabledModules: disabledModules,
|
||||
targetProduct: targetProduct,
|
||||
targetBuildVariant: targetBuildVariant,
|
||||
bazelRunner: &builtinBazelRunner{},
|
||||
paths: &paths,
|
||||
modulesDefaultToBazel: c.BuildMode == BazelDevMode,
|
||||
bazelEnabledModules: enabledModules,
|
||||
bazelDisabledModules: disabledModules,
|
||||
bazelDclaEnabledModules: dclaEnabledModules,
|
||||
targetProduct: targetProduct,
|
||||
targetBuildVariant: targetBuildVariant,
|
||||
}, nil
|
||||
}
|
||||
|
||||
@@ -553,13 +612,17 @@ func (p *bazelPaths) BazelMetricsDir() string {
|
||||
return p.metricsDir
|
||||
}
|
||||
|
||||
func (context *mixedBuildBazelContext) IsModuleNameAllowed(moduleName string) bool {
|
||||
func (context *mixedBuildBazelContext) IsModuleNameAllowed(moduleName string, withinApex bool) bool {
|
||||
if context.bazelDisabledModules[moduleName] {
|
||||
return false
|
||||
}
|
||||
if context.bazelEnabledModules[moduleName] {
|
||||
return true
|
||||
}
|
||||
if withinApex && context.bazelDclaEnabledModules[moduleName] {
|
||||
return true
|
||||
}
|
||||
|
||||
return context.modulesDefaultToBazel
|
||||
}
|
||||
|
||||
@@ -696,21 +759,37 @@ func (context *mixedBuildBazelContext) mainBzlFileContents() []byte {
|
||||
#####################################################
|
||||
# This file is generated by soong_build. Do not edit.
|
||||
#####################################################
|
||||
|
||||
def _config_node_transition_impl(settings, attr):
|
||||
if attr.os == "android" and attr.arch == "target":
|
||||
target = "{PRODUCT}-{VARIANT}"
|
||||
else:
|
||||
target = "{PRODUCT}-{VARIANT}_%s_%s" % (attr.os, attr.arch)
|
||||
return {
|
||||
apex_name = ""
|
||||
if attr.within_apex:
|
||||
# //build/bazel/rules/apex:apex_name has to be set to a non_empty value,
|
||||
# otherwise //build/bazel/rules/apex:non_apex will be true and the
|
||||
# "-D__ANDROID_APEX__" compiler flag will be missing. Apex_name is used
|
||||
# in some validation on bazel side which don't really apply in mixed
|
||||
# build because soong will do the work, so we just set it to a fixed
|
||||
# value here.
|
||||
apex_name = "dcla_apex"
|
||||
outputs = {
|
||||
"//command_line_option:platforms": "@soong_injection//product_config_platforms/products/{PRODUCT}-{VARIANT}:%s" % target,
|
||||
"@//build/bazel/rules/apex:within_apex": attr.within_apex,
|
||||
"@//build/bazel/rules/apex:min_sdk_version": attr.apex_sdk_version,
|
||||
"@//build/bazel/rules/apex:apex_name": apex_name,
|
||||
}
|
||||
|
||||
return outputs
|
||||
|
||||
_config_node_transition = transition(
|
||||
implementation = _config_node_transition_impl,
|
||||
inputs = [],
|
||||
outputs = [
|
||||
"//command_line_option:platforms",
|
||||
"@//build/bazel/rules/apex:within_apex",
|
||||
"@//build/bazel/rules/apex:min_sdk_version",
|
||||
"@//build/bazel/rules/apex:apex_name",
|
||||
],
|
||||
)
|
||||
|
||||
@@ -720,9 +799,11 @@ def _passthrough_rule_impl(ctx):
|
||||
config_node = rule(
|
||||
implementation = _passthrough_rule_impl,
|
||||
attrs = {
|
||||
"arch" : attr.string(mandatory = True),
|
||||
"os" : attr.string(mandatory = True),
|
||||
"deps" : attr.label_list(cfg = _config_node_transition, allow_files = True),
|
||||
"arch" : attr.string(mandatory = True),
|
||||
"os" : attr.string(mandatory = True),
|
||||
"within_apex" : attr.bool(default = False),
|
||||
"apex_sdk_version" : attr.string(mandatory = True),
|
||||
"deps" : attr.label_list(cfg = _config_node_transition, allow_files = True),
|
||||
"_allowlist_function_transition": attr.label(default = "@bazel_tools//tools/allowlists/function_transition_allowlist"),
|
||||
},
|
||||
)
|
||||
@@ -781,6 +862,8 @@ phony_root(name = "phonyroot",
|
||||
config_node(name = "%s",
|
||||
arch = "%s",
|
||||
os = "%s",
|
||||
within_apex = %s,
|
||||
apex_sdk_version = "%s",
|
||||
deps = [%s],
|
||||
testonly = True, # Unblocks testonly deps.
|
||||
)
|
||||
@@ -807,15 +890,28 @@ config_node(name = "%s",
|
||||
for _, configString := range sortedConfigs {
|
||||
labels := labelsByConfig[configString]
|
||||
configTokens := strings.Split(configString, "|")
|
||||
if len(configTokens) != 2 {
|
||||
if len(configTokens) < 2 {
|
||||
panic(fmt.Errorf("Unexpected config string format: %s", configString))
|
||||
}
|
||||
archString := configTokens[0]
|
||||
osString := configTokens[1]
|
||||
withinApex := "False"
|
||||
apexSdkVerString := ""
|
||||
targetString := fmt.Sprintf("%s_%s", osString, archString)
|
||||
if len(configTokens) > 2 {
|
||||
targetString += "_" + configTokens[2]
|
||||
if configTokens[2] == withinApexToString(true) {
|
||||
withinApex = "True"
|
||||
}
|
||||
}
|
||||
if len(configTokens) > 3 {
|
||||
targetString += "_" + configTokens[3]
|
||||
apexSdkVerString = configTokens[3]
|
||||
}
|
||||
allLabels = append(allLabels, fmt.Sprintf("\":%s\"", targetString))
|
||||
labelsString := strings.Join(labels, ",\n ")
|
||||
configNodesSection += fmt.Sprintf(configNodeFormatString, targetString, archString, osString, labelsString)
|
||||
configNodesSection += fmt.Sprintf(configNodeFormatString, targetString, archString, osString, withinApex, apexSdkVerString,
|
||||
labelsString)
|
||||
}
|
||||
|
||||
return []byte(fmt.Sprintf(formatString, configNodesSection, strings.Join(allLabels, ",\n ")))
|
||||
@@ -911,6 +1007,7 @@ def get_arch(target):
|
||||
# Soong treats filegroups, but it may not be the case with manually-written
|
||||
# filegroup BUILD targets.
|
||||
buildoptions = build_options(target)
|
||||
|
||||
if buildoptions == None:
|
||||
# File targets do not have buildoptions. File targets aren't associated with
|
||||
# any specific platform architecture in mixed builds, so use the host.
|
||||
@@ -927,15 +1024,26 @@ def get_arch(target):
|
||||
if not platform_name.startswith("{TARGET_PRODUCT}-{TARGET_BUILD_VARIANT}"):
|
||||
fail("expected platform name of the form '{TARGET_PRODUCT}-{TARGET_BUILD_VARIANT}_android_<arch>' or '{TARGET_PRODUCT}-{TARGET_BUILD_VARIANT}_linux_<arch>', but was " + str(platforms))
|
||||
platform_name = platform_name.removeprefix("{TARGET_PRODUCT}-{TARGET_BUILD_VARIANT}").removeprefix("_")
|
||||
config_key = ""
|
||||
if not platform_name:
|
||||
return "target|android"
|
||||
config_key = "target|android"
|
||||
elif platform_name.startswith("android_"):
|
||||
return platform_name.removeprefix("android_") + "|android"
|
||||
config_key = platform_name.removeprefix("android_") + "|android"
|
||||
elif platform_name.startswith("linux_"):
|
||||
return platform_name.removeprefix("linux_") + "|linux"
|
||||
config_key = platform_name.removeprefix("linux_") + "|linux"
|
||||
else:
|
||||
fail("expected platform name of the form '{TARGET_PRODUCT}-{TARGET_BUILD_VARIANT}_android_<arch>' or '{TARGET_PRODUCT}-{TARGET_BUILD_VARIANT}_linux_<arch>', but was " + str(platforms))
|
||||
|
||||
within_apex = buildoptions.get("//build/bazel/rules/apex:within_apex")
|
||||
apex_sdk_version = buildoptions.get("//build/bazel/rules/apex:min_sdk_version")
|
||||
|
||||
if within_apex:
|
||||
config_key += "|within_apex"
|
||||
if apex_sdk_version != None and len(apex_sdk_version) > 0:
|
||||
config_key += "|" + apex_sdk_version
|
||||
|
||||
return config_key
|
||||
|
||||
def format(target):
|
||||
id_string = str(target.label) + "|" + get_arch(target)
|
||||
|
||||
@@ -1044,8 +1152,12 @@ func (context *mixedBuildBazelContext) runCquery(config Config, ctx invokeBazelC
|
||||
return err
|
||||
}
|
||||
|
||||
cqueryCommandWithFlag := context.createBazelCommand(config, context.paths, bazel.CqueryBuildRootRunName, cqueryCmd,
|
||||
"--output=starlark", "--starlark:file="+absolutePath(cqueryFileRelpath))
|
||||
extraFlags := []string{"--output=starlark", "--starlark:file=" + absolutePath(cqueryFileRelpath)}
|
||||
if Bool(config.productVariables.ClangCoverage) {
|
||||
extraFlags = append(extraFlags, "--collect_code_coverage")
|
||||
}
|
||||
|
||||
cqueryCommandWithFlag := context.createBazelCommand(config, context.paths, bazel.CqueryBuildRootRunName, cqueryCmd, extraFlags...)
|
||||
cqueryOutput, cqueryErrorMessage, cqueryErr := context.issueBazelCommand(cqueryCommandWithFlag, eventHandler)
|
||||
if cqueryErr != nil {
|
||||
return cqueryErr
|
||||
@@ -1325,7 +1437,16 @@ func getConfigString(key cqueryKey) string {
|
||||
// Use host OS, which is currently hardcoded to be linux.
|
||||
osName = "linux"
|
||||
}
|
||||
return arch + "|" + osName
|
||||
keyString := arch + "|" + osName
|
||||
if key.configKey.apexKey.WithinApex {
|
||||
keyString += "|" + withinApexToString(key.configKey.apexKey.WithinApex)
|
||||
}
|
||||
|
||||
if len(key.configKey.apexKey.ApexSdkVersion) > 0 {
|
||||
keyString += "|" + key.configKey.apexKey.ApexSdkVersion
|
||||
}
|
||||
|
||||
return keyString
|
||||
}
|
||||
|
||||
func GetConfigKey(ctx BaseModuleContext) configKey {
|
||||
@@ -1336,6 +1457,19 @@ func GetConfigKey(ctx BaseModuleContext) configKey {
|
||||
}
|
||||
}
|
||||
|
||||
func GetConfigKeyApexVariant(ctx BaseModuleContext, apexKey *ApexConfigKey) configKey {
|
||||
configKey := GetConfigKey(ctx)
|
||||
|
||||
if apexKey != nil {
|
||||
configKey.apexKey = ApexConfigKey{
|
||||
WithinApex: apexKey.WithinApex,
|
||||
ApexSdkVersion: apexKey.ApexSdkVersion,
|
||||
}
|
||||
}
|
||||
|
||||
return configKey
|
||||
}
|
||||
|
||||
func bazelDepsetName(contentHash string) string {
|
||||
return fmt.Sprintf("bazel_depset_%s", contentHash)
|
||||
}
|
||||
|
Reference in New Issue
Block a user