1295 lines
		
	
	
		
			46 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			1295 lines
		
	
	
		
			46 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 (
 | |
| 	"fmt"
 | |
| 	"reflect"
 | |
| 	"runtime"
 | |
| 	"strings"
 | |
| 
 | |
| 	"android/soong/android/soongconfig"
 | |
| 	"android/soong/bazel"
 | |
| 
 | |
| 	"github.com/google/blueprint/proptools"
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	registerVariableBuildComponents(InitRegistrationContext)
 | |
| }
 | |
| 
 | |
| func registerVariableBuildComponents(ctx RegistrationContext) {
 | |
| 	ctx.PreDepsMutators(func(ctx RegisterMutatorsContext) {
 | |
| 		ctx.BottomUp("variable", VariableMutator).Parallel()
 | |
| 	})
 | |
| }
 | |
| 
 | |
| var PrepareForTestWithVariables = FixtureRegisterWithContext(registerVariableBuildComponents)
 | |
| 
 | |
| type variableProperties struct {
 | |
| 	Product_variables struct {
 | |
| 		Platform_sdk_version struct {
 | |
| 			Asflags []string
 | |
| 			Cflags  []string
 | |
| 			Cmd     *string
 | |
| 		}
 | |
| 
 | |
| 		Platform_sdk_version_or_codename struct {
 | |
| 			Java_resource_dirs []string
 | |
| 		}
 | |
| 
 | |
| 		Platform_sdk_extension_version struct {
 | |
| 			Cmd *string
 | |
| 		}
 | |
| 
 | |
| 		Platform_version_name struct {
 | |
| 			Base_dir *string
 | |
| 		}
 | |
| 
 | |
| 		// unbundled_build is a catch-all property to annotate modules that don't build in one or
 | |
| 		// more unbundled branches, usually due to dependencies missing from the manifest.
 | |
| 		Unbundled_build struct {
 | |
| 			Enabled *bool `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		// similar to `Unbundled_build`, but `Always_use_prebuilt_sdks` means that it uses prebuilt
 | |
| 		// sdk specifically.
 | |
| 		Always_use_prebuilt_sdks struct {
 | |
| 			Enabled *bool `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		Malloc_not_svelte struct {
 | |
| 			Cflags              []string `android:"arch_variant"`
 | |
| 			Shared_libs         []string `android:"arch_variant"`
 | |
| 			Whole_static_libs   []string `android:"arch_variant"`
 | |
| 			Exclude_static_libs []string `android:"arch_variant"`
 | |
| 			Srcs                []string `android:"arch_variant"`
 | |
| 			Header_libs         []string `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		Malloc_zero_contents struct {
 | |
| 			Cflags []string `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		Malloc_pattern_fill_contents struct {
 | |
| 			Cflags []string `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		Safestack struct {
 | |
| 			Cflags []string `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		Binder32bit struct {
 | |
| 			Cflags []string
 | |
| 		}
 | |
| 
 | |
| 		Override_rs_driver struct {
 | |
| 			Cflags []string
 | |
| 		}
 | |
| 
 | |
| 		// treble_linker_namespaces is true when the system/vendor linker namespace separation is
 | |
| 		// enabled.
 | |
| 		Treble_linker_namespaces struct {
 | |
| 			Cflags []string
 | |
| 		}
 | |
| 		// enforce_vintf_manifest is true when a device is required to have a vintf manifest.
 | |
| 		Enforce_vintf_manifest struct {
 | |
| 			Cflags []string
 | |
| 		}
 | |
| 
 | |
| 		Build_from_text_stub struct {
 | |
| 			Static_libs         []string
 | |
| 			Exclude_static_libs []string
 | |
| 		}
 | |
| 
 | |
| 		// debuggable is true for eng and userdebug builds, and can be used to turn on additional
 | |
| 		// debugging features that don't significantly impact runtime behavior.  userdebug builds
 | |
| 		// are used for dogfooding and performance testing, and should be as similar to user builds
 | |
| 		// as possible.
 | |
| 		Debuggable struct {
 | |
| 			Cflags          []string
 | |
| 			Cppflags        []string
 | |
| 			Init_rc         []string
 | |
| 			Required        []string
 | |
| 			Host_required   []string
 | |
| 			Target_required []string
 | |
| 			Strip           struct {
 | |
| 				All                          *bool
 | |
| 				Keep_symbols                 *bool
 | |
| 				Keep_symbols_and_debug_frame *bool
 | |
| 			}
 | |
| 			Static_libs       []string
 | |
| 			Whole_static_libs []string
 | |
| 			Shared_libs       []string
 | |
| 
 | |
| 			Cmdline []string
 | |
| 
 | |
| 			Srcs         []string
 | |
| 			Exclude_srcs []string
 | |
| 			Cmd          *string
 | |
| 		}
 | |
| 
 | |
| 		// eng is true for -eng builds, and can be used to turn on additional heavyweight debugging
 | |
| 		// features.
 | |
| 		Eng struct {
 | |
| 			Cflags   []string
 | |
| 			Cppflags []string
 | |
| 			Lto      struct {
 | |
| 				Never *bool
 | |
| 			}
 | |
| 			Sanitize struct {
 | |
| 				Address *bool
 | |
| 			}
 | |
| 			Optimize struct {
 | |
| 				Enabled *bool
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		Uml struct {
 | |
| 			Cppflags []string
 | |
| 		}
 | |
| 
 | |
| 		Arc struct {
 | |
| 			Cflags            []string `android:"arch_variant"`
 | |
| 			Exclude_srcs      []string `android:"arch_variant"`
 | |
| 			Header_libs       []string `android:"arch_variant"`
 | |
| 			Include_dirs      []string `android:"arch_variant"`
 | |
| 			Shared_libs       []string `android:"arch_variant"`
 | |
| 			Static_libs       []string `android:"arch_variant"`
 | |
| 			Srcs              []string `android:"arch_variant"`
 | |
| 			Whole_static_libs []string `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		Native_coverage struct {
 | |
| 			Src          *string  `android:"arch_variant"`
 | |
| 			Srcs         []string `android:"arch_variant"`
 | |
| 			Exclude_srcs []string `android:"arch_variant"`
 | |
| 		} `android:"arch_variant"`
 | |
| 
 | |
| 		// release_aidl_use_unfrozen is "true" when a device can
 | |
| 		// use the unfrozen versions of AIDL interfaces.
 | |
| 		Release_aidl_use_unfrozen struct {
 | |
| 			Cflags []string
 | |
| 			Cmd    *string
 | |
| 		}
 | |
| 	} `android:"arch_variant"`
 | |
| }
 | |
| 
 | |
| var defaultProductVariables interface{} = variableProperties{}
 | |
| 
 | |
| type ProductVariables struct {
 | |
| 	// Suffix to add to generated Makefiles
 | |
| 	Make_suffix *string `json:",omitempty"`
 | |
| 
 | |
| 	BuildId         *string `json:",omitempty"`
 | |
| 	BuildNumberFile *string `json:",omitempty"`
 | |
| 
 | |
| 	Platform_version_name                     *string  `json:",omitempty"`
 | |
| 	Platform_sdk_version                      *int     `json:",omitempty"`
 | |
| 	Platform_sdk_codename                     *string  `json:",omitempty"`
 | |
| 	Platform_sdk_version_or_codename          *string  `json:",omitempty"`
 | |
| 	Platform_sdk_final                        *bool    `json:",omitempty"`
 | |
| 	Platform_sdk_extension_version            *int     `json:",omitempty"`
 | |
| 	Platform_base_sdk_extension_version       *int     `json:",omitempty"`
 | |
| 	Platform_version_active_codenames         []string `json:",omitempty"`
 | |
| 	Platform_version_all_preview_codenames    []string `json:",omitempty"`
 | |
| 	Platform_vndk_version                     *string  `json:",omitempty"`
 | |
| 	Platform_systemsdk_versions               []string `json:",omitempty"`
 | |
| 	Platform_security_patch                   *string  `json:",omitempty"`
 | |
| 	Platform_preview_sdk_version              *string  `json:",omitempty"`
 | |
| 	Platform_min_supported_target_sdk_version *string  `json:",omitempty"`
 | |
| 	Platform_base_os                          *string  `json:",omitempty"`
 | |
| 	Platform_version_last_stable              *string  `json:",omitempty"`
 | |
| 	Platform_version_known_codenames          *string  `json:",omitempty"`
 | |
| 
 | |
| 	DeviceName                            *string  `json:",omitempty"`
 | |
| 	DeviceProduct                         *string  `json:",omitempty"`
 | |
| 	DeviceArch                            *string  `json:",omitempty"`
 | |
| 	DeviceArchVariant                     *string  `json:",omitempty"`
 | |
| 	DeviceCpuVariant                      *string  `json:",omitempty"`
 | |
| 	DeviceAbi                             []string `json:",omitempty"`
 | |
| 	DeviceVndkVersion                     *string  `json:",omitempty"`
 | |
| 	DeviceCurrentApiLevelForVendorModules *string  `json:",omitempty"`
 | |
| 	DeviceSystemSdkVersions               []string `json:",omitempty"`
 | |
| 	DeviceMaxPageSizeSupported            *string  `json:",omitempty"`
 | |
| 	DevicePageSizeAgnostic                *bool    `json:",omitempty"`
 | |
| 
 | |
| 	RecoverySnapshotVersion *string `json:",omitempty"`
 | |
| 
 | |
| 	DeviceSecondaryArch        *string  `json:",omitempty"`
 | |
| 	DeviceSecondaryArchVariant *string  `json:",omitempty"`
 | |
| 	DeviceSecondaryCpuVariant  *string  `json:",omitempty"`
 | |
| 	DeviceSecondaryAbi         []string `json:",omitempty"`
 | |
| 
 | |
| 	NativeBridgeArch         *string  `json:",omitempty"`
 | |
| 	NativeBridgeArchVariant  *string  `json:",omitempty"`
 | |
| 	NativeBridgeCpuVariant   *string  `json:",omitempty"`
 | |
| 	NativeBridgeAbi          []string `json:",omitempty"`
 | |
| 	NativeBridgeRelativePath *string  `json:",omitempty"`
 | |
| 
 | |
| 	NativeBridgeSecondaryArch         *string  `json:",omitempty"`
 | |
| 	NativeBridgeSecondaryArchVariant  *string  `json:",omitempty"`
 | |
| 	NativeBridgeSecondaryCpuVariant   *string  `json:",omitempty"`
 | |
| 	NativeBridgeSecondaryAbi          []string `json:",omitempty"`
 | |
| 	NativeBridgeSecondaryRelativePath *string  `json:",omitempty"`
 | |
| 
 | |
| 	HostArch          *string `json:",omitempty"`
 | |
| 	HostSecondaryArch *string `json:",omitempty"`
 | |
| 	HostMusl          *bool   `json:",omitempty"`
 | |
| 
 | |
| 	CrossHost              *string `json:",omitempty"`
 | |
| 	CrossHostArch          *string `json:",omitempty"`
 | |
| 	CrossHostSecondaryArch *string `json:",omitempty"`
 | |
| 
 | |
| 	DeviceResourceOverlays     []string `json:",omitempty"`
 | |
| 	ProductResourceOverlays    []string `json:",omitempty"`
 | |
| 	EnforceRROTargets          []string `json:",omitempty"`
 | |
| 	EnforceRROExcludedOverlays []string `json:",omitempty"`
 | |
| 
 | |
| 	AAPTCharacteristics *string  `json:",omitempty"`
 | |
| 	AAPTConfig          []string `json:",omitempty"`
 | |
| 	AAPTPreferredConfig *string  `json:",omitempty"`
 | |
| 	AAPTPrebuiltDPI     []string `json:",omitempty"`
 | |
| 
 | |
| 	DefaultAppCertificate           *string `json:",omitempty"`
 | |
| 	MainlineSepolicyDevCertificates *string `json:",omitempty"`
 | |
| 
 | |
| 	AppsDefaultVersionName *string `json:",omitempty"`
 | |
| 
 | |
| 	Allow_missing_dependencies   *bool    `json:",omitempty"`
 | |
| 	Unbundled_build              *bool    `json:",omitempty"`
 | |
| 	Unbundled_build_apps         []string `json:",omitempty"`
 | |
| 	Unbundled_build_image        *bool    `json:",omitempty"`
 | |
| 	Always_use_prebuilt_sdks     *bool    `json:",omitempty"`
 | |
| 	Skip_boot_jars_check         *bool    `json:",omitempty"`
 | |
| 	Malloc_not_svelte            *bool    `json:",omitempty"`
 | |
| 	Malloc_zero_contents         *bool    `json:",omitempty"`
 | |
| 	Malloc_pattern_fill_contents *bool    `json:",omitempty"`
 | |
| 	Safestack                    *bool    `json:",omitempty"`
 | |
| 	HostStaticBinaries           *bool    `json:",omitempty"`
 | |
| 	Binder32bit                  *bool    `json:",omitempty"`
 | |
| 	UseGoma                      *bool    `json:",omitempty"`
 | |
| 	UseRBE                       *bool    `json:",omitempty"`
 | |
| 	UseRBEJAVAC                  *bool    `json:",omitempty"`
 | |
| 	UseRBER8                     *bool    `json:",omitempty"`
 | |
| 	UseRBED8                     *bool    `json:",omitempty"`
 | |
| 	Debuggable                   *bool    `json:",omitempty"`
 | |
| 	Eng                          *bool    `json:",omitempty"`
 | |
| 	Treble_linker_namespaces     *bool    `json:",omitempty"`
 | |
| 	Enforce_vintf_manifest       *bool    `json:",omitempty"`
 | |
| 	Uml                          *bool    `json:",omitempty"`
 | |
| 	Arc                          *bool    `json:",omitempty"`
 | |
| 	MinimizeJavaDebugInfo        *bool    `json:",omitempty"`
 | |
| 	Build_from_text_stub         *bool    `json:",omitempty"`
 | |
| 
 | |
| 	Check_elf_files *bool `json:",omitempty"`
 | |
| 
 | |
| 	UncompressPrivAppDex             *bool    `json:",omitempty"`
 | |
| 	ModulesLoadedByPrivilegedModules []string `json:",omitempty"`
 | |
| 
 | |
| 	BootJars     ConfiguredJarList `json:",omitempty"`
 | |
| 	ApexBootJars ConfiguredJarList `json:",omitempty"`
 | |
| 
 | |
| 	IntegerOverflowExcludePaths []string `json:",omitempty"`
 | |
| 
 | |
| 	EnableCFI       *bool    `json:",omitempty"`
 | |
| 	CFIExcludePaths []string `json:",omitempty"`
 | |
| 	CFIIncludePaths []string `json:",omitempty"`
 | |
| 
 | |
| 	DisableScudo *bool `json:",omitempty"`
 | |
| 
 | |
| 	MemtagHeapExcludePaths      []string `json:",omitempty"`
 | |
| 	MemtagHeapAsyncIncludePaths []string `json:",omitempty"`
 | |
| 	MemtagHeapSyncIncludePaths  []string `json:",omitempty"`
 | |
| 
 | |
| 	HWASanIncludePaths []string `json:",omitempty"`
 | |
| 	HWASanExcludePaths []string `json:",omitempty"`
 | |
| 
 | |
| 	VendorPath    *string `json:",omitempty"`
 | |
| 	OdmPath       *string `json:",omitempty"`
 | |
| 	ProductPath   *string `json:",omitempty"`
 | |
| 	SystemExtPath *string `json:",omitempty"`
 | |
| 
 | |
| 	ClangTidy  *bool   `json:",omitempty"`
 | |
| 	TidyChecks *string `json:",omitempty"`
 | |
| 
 | |
| 	JavaCoveragePaths        []string `json:",omitempty"`
 | |
| 	JavaCoverageExcludePaths []string `json:",omitempty"`
 | |
| 
 | |
| 	GcovCoverage                *bool    `json:",omitempty"`
 | |
| 	ClangCoverage               *bool    `json:",omitempty"`
 | |
| 	NativeCoveragePaths         []string `json:",omitempty"`
 | |
| 	NativeCoverageExcludePaths  []string `json:",omitempty"`
 | |
| 	ClangCoverageContinuousMode *bool    `json:",omitempty"`
 | |
| 
 | |
| 	// Set by NewConfig
 | |
| 	Native_coverage *bool `json:",omitempty"`
 | |
| 
 | |
| 	SanitizeHost       []string `json:",omitempty"`
 | |
| 	SanitizeDevice     []string `json:",omitempty"`
 | |
| 	SanitizeDeviceDiag []string `json:",omitempty"`
 | |
| 	SanitizeDeviceArch []string `json:",omitempty"`
 | |
| 
 | |
| 	ArtUseReadBarrier *bool `json:",omitempty"`
 | |
| 
 | |
| 	BtConfigIncludeDir *string `json:",omitempty"`
 | |
| 
 | |
| 	Override_rs_driver *string `json:",omitempty"`
 | |
| 
 | |
| 	DeviceKernelHeaders []string `json:",omitempty"`
 | |
| 
 | |
| 	ExtraVndkVersions []string `json:",omitempty"`
 | |
| 
 | |
| 	NamespacesToExport []string `json:",omitempty"`
 | |
| 
 | |
| 	PgoAdditionalProfileDirs []string `json:",omitempty"`
 | |
| 
 | |
| 	VndkUseCoreVariant         *bool `json:",omitempty"`
 | |
| 	VndkSnapshotBuildArtifacts *bool `json:",omitempty"`
 | |
| 
 | |
| 	DirectedVendorSnapshot bool            `json:",omitempty"`
 | |
| 	VendorSnapshotModules  map[string]bool `json:",omitempty"`
 | |
| 
 | |
| 	DirectedRecoverySnapshot bool            `json:",omitempty"`
 | |
| 	RecoverySnapshotModules  map[string]bool `json:",omitempty"`
 | |
| 
 | |
| 	VendorSnapshotDirsIncluded   []string `json:",omitempty"`
 | |
| 	VendorSnapshotDirsExcluded   []string `json:",omitempty"`
 | |
| 	RecoverySnapshotDirsExcluded []string `json:",omitempty"`
 | |
| 	RecoverySnapshotDirsIncluded []string `json:",omitempty"`
 | |
| 	HostFakeSnapshotEnabled      bool     `json:",omitempty"`
 | |
| 
 | |
| 	MultitreeUpdateMeta bool `json:",omitempty"`
 | |
| 
 | |
| 	BoardVendorSepolicyDirs      []string `json:",omitempty"`
 | |
| 	BoardOdmSepolicyDirs         []string `json:",omitempty"`
 | |
| 	SystemExtPublicSepolicyDirs  []string `json:",omitempty"`
 | |
| 	SystemExtPrivateSepolicyDirs []string `json:",omitempty"`
 | |
| 	BoardSepolicyM4Defs          []string `json:",omitempty"`
 | |
| 
 | |
| 	BoardSepolicyVers       *string `json:",omitempty"`
 | |
| 	PlatformSepolicyVersion *string `json:",omitempty"`
 | |
| 	TotSepolicyVersion      *string `json:",omitempty"`
 | |
| 
 | |
| 	SystemExtSepolicyPrebuiltApiDir *string `json:",omitempty"`
 | |
| 	ProductSepolicyPrebuiltApiDir   *string `json:",omitempty"`
 | |
| 
 | |
| 	PlatformSepolicyCompatVersions []string `json:",omitempty"`
 | |
| 
 | |
| 	VendorVars map[string]map[string]string `json:",omitempty"`
 | |
| 
 | |
| 	Ndk_abis *bool `json:",omitempty"`
 | |
| 
 | |
| 	TrimmedApex                  *bool `json:",omitempty"`
 | |
| 	ForceApexSymlinkOptimization *bool `json:",omitempty"`
 | |
| 	CompressedApex               *bool `json:",omitempty"`
 | |
| 	Aml_abis                     *bool `json:",omitempty"`
 | |
| 
 | |
| 	DexpreoptGlobalConfig *string `json:",omitempty"`
 | |
| 
 | |
| 	WithDexpreopt bool `json:",omitempty"`
 | |
| 
 | |
| 	ManifestPackageNameOverrides   []string `json:",omitempty"`
 | |
| 	CertificateOverrides           []string `json:",omitempty"`
 | |
| 	PackageNameOverrides           []string `json:",omitempty"`
 | |
| 	ConfiguredJarLocationOverrides []string `json:",omitempty"`
 | |
| 
 | |
| 	ApexGlobalMinSdkVersionOverride *string `json:",omitempty"`
 | |
| 
 | |
| 	EnforceSystemCertificate          *bool    `json:",omitempty"`
 | |
| 	EnforceSystemCertificateAllowList []string `json:",omitempty"`
 | |
| 
 | |
| 	ProductHiddenAPIStubs       []string `json:",omitempty"`
 | |
| 	ProductHiddenAPIStubsSystem []string `json:",omitempty"`
 | |
| 	ProductHiddenAPIStubsTest   []string `json:",omitempty"`
 | |
| 
 | |
| 	ProductPublicSepolicyDirs  []string `json:",omitempty"`
 | |
| 	ProductPrivateSepolicyDirs []string `json:",omitempty"`
 | |
| 
 | |
| 	ProductVndkVersion *string `json:",omitempty"`
 | |
| 
 | |
| 	TargetFSConfigGen []string `json:",omitempty"`
 | |
| 
 | |
| 	EnforceProductPartitionInterface *bool `json:",omitempty"`
 | |
| 
 | |
| 	EnforceInterPartitionJavaSdkLibrary *bool    `json:",omitempty"`
 | |
| 	InterPartitionJavaLibraryAllowList  []string `json:",omitempty"`
 | |
| 
 | |
| 	BoardUsesRecoveryAsBoot *bool `json:",omitempty"`
 | |
| 
 | |
| 	BoardKernelBinaries                []string `json:",omitempty"`
 | |
| 	BoardKernelModuleInterfaceVersions []string `json:",omitempty"`
 | |
| 
 | |
| 	BoardMoveRecoveryResourcesToVendorBoot *bool `json:",omitempty"`
 | |
| 
 | |
| 	PrebuiltHiddenApiDir *string `json:",omitempty"`
 | |
| 
 | |
| 	ShippingApiLevel *string `json:",omitempty"`
 | |
| 
 | |
| 	BuildBrokenPluginValidation         []string `json:",omitempty"`
 | |
| 	BuildBrokenClangAsFlags             bool     `json:",omitempty"`
 | |
| 	BuildBrokenClangCFlags              bool     `json:",omitempty"`
 | |
| 	BuildBrokenClangProperty            bool     `json:",omitempty"`
 | |
| 	GenruleSandboxing                   *bool    `json:",omitempty"`
 | |
| 	BuildBrokenEnforceSyspropOwner      bool     `json:",omitempty"`
 | |
| 	BuildBrokenTrebleSyspropNeverallow  bool     `json:",omitempty"`
 | |
| 	BuildBrokenUsesSoongPython2Modules  bool     `json:",omitempty"`
 | |
| 	BuildBrokenVendorPropertyNamespace  bool     `json:",omitempty"`
 | |
| 	BuildBrokenIncorrectPartitionImages bool     `json:",omitempty"`
 | |
| 	BuildBrokenInputDirModules          []string `json:",omitempty"`
 | |
| 
 | |
| 	BuildWarningBadOptionalUsesLibsAllowlist []string `json:",omitempty"`
 | |
| 
 | |
| 	BuildDebugfsRestrictionsEnabled bool `json:",omitempty"`
 | |
| 
 | |
| 	RequiresInsecureExecmemForSwiftshader bool `json:",omitempty"`
 | |
| 
 | |
| 	SelinuxIgnoreNeverallows bool `json:",omitempty"`
 | |
| 
 | |
| 	Release_aidl_use_unfrozen *bool `json:",omitempty"`
 | |
| 
 | |
| 	SepolicyFreezeTestExtraDirs         []string `json:",omitempty"`
 | |
| 	SepolicyFreezeTestExtraPrebuiltDirs []string `json:",omitempty"`
 | |
| 
 | |
| 	GenerateAidlNdkPlatformBackend bool `json:",omitempty"`
 | |
| 
 | |
| 	IgnorePrefer32OnDevice bool `json:",omitempty"`
 | |
| 
 | |
| 	IncludeTags    []string `json:",omitempty"`
 | |
| 	SourceRootDirs []string `json:",omitempty"`
 | |
| 
 | |
| 	AfdoProfiles []string `json:",omitempty"`
 | |
| 
 | |
| 	ProductManufacturer string   `json:",omitempty"`
 | |
| 	ProductBrand        string   `json:",omitempty"`
 | |
| 	BuildVersionTags    []string `json:",omitempty"`
 | |
| 
 | |
| 	ReleaseVersion          string `json:",omitempty"`
 | |
| 	ReleaseAconfigValueSets string `json:",omitempty"`
 | |
| 
 | |
| 	ReleaseAconfigFlagDefaultPermission string `json:",omitempty"`
 | |
| 
 | |
| 	KeepVndk *bool `json:",omitempty"`
 | |
| 
 | |
| 	CheckVendorSeappViolations *bool `json:",omitempty"`
 | |
| 
 | |
| 	// PartitionVarsForBazelMigrationOnlyDoNotUse are extra variables that are used to define the
 | |
| 	// partition images. They should not be read from soong modules.
 | |
| 	PartitionVarsForBazelMigrationOnlyDoNotUse PartitionVariables `json:",omitempty"`
 | |
| 
 | |
| 	NextReleaseHideFlaggedApi *bool `json:",omitempty"`
 | |
| }
 | |
| 
 | |
| type PartitionVariables struct {
 | |
| 	ProductDirectory            string `json:",omitempty"`
 | |
| 	PartitionQualifiedVariables map[string]struct {
 | |
| 		BuildingImage               bool   `json:",omitempty"`
 | |
| 		BoardErofsCompressor        string `json:",omitempty"`
 | |
| 		BoardErofsCompressHints     string `json:",omitempty"`
 | |
| 		BoardErofsPclusterSize      string `json:",omitempty"`
 | |
| 		BoardExtfsInodeCount        string `json:",omitempty"`
 | |
| 		BoardExtfsRsvPct            string `json:",omitempty"`
 | |
| 		BoardF2fsSloadCompressFlags string `json:",omitempty"`
 | |
| 		BoardFileSystemCompress     string `json:",omitempty"`
 | |
| 		BoardFileSystemType         string `json:",omitempty"`
 | |
| 		BoardJournalSize            string `json:",omitempty"`
 | |
| 		BoardPartitionReservedSize  string `json:",omitempty"`
 | |
| 		BoardPartitionSize          string `json:",omitempty"`
 | |
| 		BoardSquashfsBlockSize      string `json:",omitempty"`
 | |
| 		BoardSquashfsCompressor     string `json:",omitempty"`
 | |
| 		BoardSquashfsCompressorOpt  string `json:",omitempty"`
 | |
| 		BoardSquashfsDisable4kAlign string `json:",omitempty"`
 | |
| 		ProductBaseFsPath           string `json:",omitempty"`
 | |
| 		ProductHeadroom             string `json:",omitempty"`
 | |
| 		ProductVerityPartition      string `json:",omitempty"`
 | |
| 
 | |
| 		BoardAvbAddHashtreeFooterArgs string `json:",omitempty"`
 | |
| 		BoardAvbKeyPath               string `json:",omitempty"`
 | |
| 		BoardAvbAlgorithm             string `json:",omitempty"`
 | |
| 		BoardAvbRollbackIndex         string `json:",omitempty"`
 | |
| 		BoardAvbRollbackIndexLocation string `json:",omitempty"`
 | |
| 	}
 | |
| 	TargetUserimagesUseExt2 bool `json:",omitempty"`
 | |
| 	TargetUserimagesUseExt3 bool `json:",omitempty"`
 | |
| 	TargetUserimagesUseExt4 bool `json:",omitempty"`
 | |
| 
 | |
| 	TargetUserimagesSparseExtDisabled      bool `json:",omitempty"`
 | |
| 	TargetUserimagesSparseErofsDisabled    bool `json:",omitempty"`
 | |
| 	TargetUserimagesSparseSquashfsDisabled bool `json:",omitempty"`
 | |
| 	TargetUserimagesSparseF2fsDisabled     bool `json:",omitempty"`
 | |
| 
 | |
| 	BoardErofsCompressor                 string `json:",omitempty"`
 | |
| 	BoardErofsCompressorHints            string `json:",omitempty"`
 | |
| 	BoardErofsPclusterSize               string `json:",omitempty"`
 | |
| 	BoardErofsShareDupBlocks             string `json:",omitempty"`
 | |
| 	BoardErofsUseLegacyCompression       string `json:",omitempty"`
 | |
| 	BoardExt4ShareDupBlocks              string `json:",omitempty"`
 | |
| 	BoardFlashLogicalBlockSize           string `json:",omitempty"`
 | |
| 	BoardFlashEraseBlockSize             string `json:",omitempty"`
 | |
| 	BoardUsesRecoveryAsBoot              bool   `json:",omitempty"`
 | |
| 	BoardBuildGkiBootImageWithoutRamdisk bool   `json:",omitempty"`
 | |
| 	ProductUseDynamicPartitionSize       bool   `json:",omitempty"`
 | |
| 	CopyImagesForTargetFilesZip          bool   `json:",omitempty"`
 | |
| 
 | |
| 	BoardAvbEnable bool `json:",omitempty"`
 | |
| }
 | |
| 
 | |
| func boolPtr(v bool) *bool {
 | |
| 	return &v
 | |
| }
 | |
| 
 | |
| func intPtr(v int) *int {
 | |
| 	return &v
 | |
| }
 | |
| 
 | |
| func stringPtr(v string) *string {
 | |
| 	return &v
 | |
| }
 | |
| 
 | |
| func (v *ProductVariables) SetDefaultConfig() {
 | |
| 	*v = ProductVariables{
 | |
| 		BuildNumberFile: stringPtr("build_number.txt"),
 | |
| 
 | |
| 		Platform_version_name:                  stringPtr("S"),
 | |
| 		Platform_base_sdk_extension_version:    intPtr(30),
 | |
| 		Platform_sdk_version:                   intPtr(30),
 | |
| 		Platform_sdk_codename:                  stringPtr("S"),
 | |
| 		Platform_sdk_final:                     boolPtr(false),
 | |
| 		Platform_version_active_codenames:      []string{"S"},
 | |
| 		Platform_version_all_preview_codenames: []string{"S"},
 | |
| 		Platform_vndk_version:                  stringPtr("S"),
 | |
| 
 | |
| 		HostArch:                   stringPtr("x86_64"),
 | |
| 		HostSecondaryArch:          stringPtr("x86"),
 | |
| 		DeviceName:                 stringPtr("generic_arm64"),
 | |
| 		DeviceProduct:              stringPtr("aosp_arm-eng"),
 | |
| 		DeviceArch:                 stringPtr("arm64"),
 | |
| 		DeviceArchVariant:          stringPtr("armv8-a"),
 | |
| 		DeviceCpuVariant:           stringPtr("generic"),
 | |
| 		DeviceAbi:                  []string{"arm64-v8a"},
 | |
| 		DeviceSecondaryArch:        stringPtr("arm"),
 | |
| 		DeviceSecondaryArchVariant: stringPtr("armv8-a"),
 | |
| 		DeviceSecondaryCpuVariant:  stringPtr("generic"),
 | |
| 		DeviceSecondaryAbi:         []string{"armeabi-v7a", "armeabi"},
 | |
| 		DeviceMaxPageSizeSupported: stringPtr("4096"),
 | |
| 		DevicePageSizeAgnostic:     boolPtr(false),
 | |
| 
 | |
| 		AAPTConfig:          []string{"normal", "large", "xlarge", "hdpi", "xhdpi", "xxhdpi"},
 | |
| 		AAPTPreferredConfig: stringPtr("xhdpi"),
 | |
| 		AAPTCharacteristics: stringPtr("nosdcard"),
 | |
| 		AAPTPrebuiltDPI:     []string{"xhdpi", "xxhdpi"},
 | |
| 
 | |
| 		Malloc_not_svelte:            boolPtr(true),
 | |
| 		Malloc_zero_contents:         boolPtr(true),
 | |
| 		Malloc_pattern_fill_contents: boolPtr(false),
 | |
| 		Safestack:                    boolPtr(false),
 | |
| 		TrimmedApex:                  boolPtr(false),
 | |
| 		Build_from_text_stub:         boolPtr(false),
 | |
| 
 | |
| 		BootJars:     ConfiguredJarList{apexes: []string{}, jars: []string{}},
 | |
| 		ApexBootJars: ConfiguredJarList{apexes: []string{}, jars: []string{}},
 | |
| 	}
 | |
| 
 | |
| 	if runtime.GOOS == "linux" {
 | |
| 		v.CrossHost = stringPtr("windows")
 | |
| 		v.CrossHostArch = stringPtr("x86")
 | |
| 		v.CrossHostSecondaryArch = stringPtr("x86_64")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // ProductConfigContext requires the access to the Module to get product config properties.
 | |
| type ProductConfigContext interface {
 | |
| 	Module() Module
 | |
| }
 | |
| 
 | |
| // ProductConfigOrSoongConfigProperty represents either a soong config variable + its value
 | |
| // or a product config variable. You can get both a ConfigurationAxis and a SelectKey from it
 | |
| // for use in bazel attributes. ProductVariableProperties() will return a map from properties ->
 | |
| // this interface -> property structs for use in bp2build converters
 | |
| type ProductConfigOrSoongConfigProperty interface {
 | |
| 	// Name of the product variable or soong config variable
 | |
| 	Name() string
 | |
| 	// AlwaysEmit returns true for soong config variables but false for product variables. This
 | |
| 	// is intended to indicate if we need to always emit empty lists in the select statements.
 | |
| 	AlwaysEmit() bool
 | |
| 	// ConfigurationAxis returns the bazel.ConfigurationAxis that represents this variable. The
 | |
| 	// configuration axis will change depending on the variable and whether it's arch/os variant
 | |
| 	// as well.
 | |
| 	ConfigurationAxis() bazel.ConfigurationAxis
 | |
| 	// SelectKey returns a string that represents the key of a select branch, however, it is not
 | |
| 	// actually the real label written out to the build file.
 | |
| 	// this.ConfigurationAxis().SelectKey(this.SelectKey()) will give the actual label.
 | |
| 	SelectKey() string
 | |
| }
 | |
| 
 | |
| // ProductConfigProperty represents a product config variable, and if it is arch-variant or not.
 | |
| type ProductConfigProperty struct {
 | |
| 	// The name of the product variable, e.g. "safestack", "malloc_not_svelte",
 | |
| 	// "board"
 | |
| 	name string
 | |
| 
 | |
| 	arch string
 | |
| }
 | |
| 
 | |
| func (p ProductConfigProperty) Name() string {
 | |
| 	return p.name
 | |
| }
 | |
| 
 | |
| func (p ProductConfigProperty) AlwaysEmit() bool {
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func (p ProductConfigProperty) ConfigurationAxis() bazel.ConfigurationAxis {
 | |
| 	return bazel.ProductVariableConfigurationAxis(p.arch != "", p.name+"__"+p.arch)
 | |
| }
 | |
| 
 | |
| func (p ProductConfigProperty) SelectKey() string {
 | |
| 	if p.arch == "" {
 | |
| 		return strings.ToLower(p.name)
 | |
| 	} else {
 | |
| 		return strings.ToLower(p.name + "-" + p.arch)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // SoongConfigProperty represents a soong config variable, its value if it's a string variable,
 | |
| // and if it's dependent on the OS or not
 | |
| type SoongConfigProperty struct {
 | |
| 	name      string
 | |
| 	namespace string
 | |
| 	// Can be an empty string for bool/value soong config variables
 | |
| 	value string
 | |
| 	// If there is a target: field inside a soong config property struct, the os that it selects
 | |
| 	// on will be represented here.
 | |
| 	os string
 | |
| }
 | |
| 
 | |
| func (p SoongConfigProperty) Name() string {
 | |
| 	return p.name
 | |
| }
 | |
| 
 | |
| func (p SoongConfigProperty) AlwaysEmit() bool {
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| func (p SoongConfigProperty) ConfigurationAxis() bazel.ConfigurationAxis {
 | |
| 	return bazel.ProductVariableConfigurationAxis(false, p.namespace+"__"+p.name+"__"+p.os)
 | |
| }
 | |
| 
 | |
| // SelectKey returns the literal string that represents this variable in a BUILD
 | |
| // select statement.
 | |
| func (p SoongConfigProperty) SelectKey() string {
 | |
| 	// p.value being conditions_default can happen with or without a desired os. When not using
 | |
| 	// an os, we want to emit literally just //conditions:default in the select statement, but
 | |
| 	// when using an os, we want to emit namespace__name__conditions_default__os, so that
 | |
| 	// the branch is only taken if the variable is not set, and we're on the desired os.
 | |
| 	// ConfigurationAxis#SelectKey will map the conditions_default result of this function to
 | |
| 	// //conditions:default.
 | |
| 	if p.value == bazel.ConditionsDefaultConfigKey && p.os == "" {
 | |
| 		return bazel.ConditionsDefaultConfigKey
 | |
| 	}
 | |
| 
 | |
| 	parts := []string{p.namespace, p.name}
 | |
| 	if p.value != "" && p.value != bazel.ConditionsDefaultSelectKey {
 | |
| 		parts = append(parts, p.value)
 | |
| 	}
 | |
| 	if p.os != "" {
 | |
| 		parts = append(parts, p.os)
 | |
| 	}
 | |
| 
 | |
| 	// e.g. acme__feature1, android__board__soc_a, my_namespace__my_variables__my_value__my_os
 | |
| 	return strings.ToLower(strings.Join(parts, "__"))
 | |
| }
 | |
| 
 | |
| // ProductConfigProperties is a map of maps to group property values according
 | |
| // their property name and the product config variable they're set under.
 | |
| //
 | |
| // The outer map key is the name of the property, like "cflags".
 | |
| //
 | |
| // The inner map key is a ProductConfigProperty, which is a struct of product
 | |
| // variable name, namespace, and the "full configuration" of the product
 | |
| // variable.
 | |
| //
 | |
| // e.g. product variable name: board, namespace: acme, full config: vendor_chip_foo
 | |
| //
 | |
| // The value of the map is the interface{} representing the value of the
 | |
| // property, like ["-DDEFINES"] for cflags.
 | |
| type ProductConfigProperties map[string]map[ProductConfigOrSoongConfigProperty]interface{}
 | |
| 
 | |
| // ProductVariableProperties returns a ProductConfigProperties containing only the properties which
 | |
| // have been set for the given module.
 | |
| func ProductVariableProperties(ctx ArchVariantContext, module Module) (ProductConfigProperties, []error) {
 | |
| 	var errs []error
 | |
| 	moduleBase := module.base()
 | |
| 
 | |
| 	productConfigProperties := ProductConfigProperties{}
 | |
| 
 | |
| 	if moduleBase.variableProperties != nil {
 | |
| 		productVariablesProperty := proptools.FieldNameForProperty("product_variables")
 | |
| 		if moduleBase.ArchSpecific() {
 | |
| 			for /* axis */ _, configToProps := range moduleBase.GetArchVariantProperties(ctx, moduleBase.variableProperties) {
 | |
| 				for config, props := range configToProps {
 | |
| 					variableValues := reflect.ValueOf(props).Elem().FieldByName(productVariablesProperty)
 | |
| 					productConfigProperties.AddProductConfigProperties(variableValues, config)
 | |
| 				}
 | |
| 			}
 | |
| 		} else {
 | |
| 			variableValues := reflect.ValueOf(moduleBase.variableProperties).Elem().FieldByName(productVariablesProperty)
 | |
| 			productConfigProperties.AddProductConfigProperties(variableValues, "")
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if m, ok := module.(Bazelable); ok && m.namespacedVariableProps() != nil {
 | |
| 		for namespace, namespacedVariableProps := range m.namespacedVariableProps() {
 | |
| 			for _, namespacedVariableProp := range namespacedVariableProps {
 | |
| 				variableValues := reflect.ValueOf(namespacedVariableProp).Elem().FieldByName(soongconfig.SoongConfigProperty)
 | |
| 				err := productConfigProperties.AddSoongConfigProperties(namespace, variableValues)
 | |
| 				if err != nil {
 | |
| 					errs = append(errs, err)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return productConfigProperties, errs
 | |
| }
 | |
| 
 | |
| func (p *ProductConfigProperties) AddProductConfigProperty(
 | |
| 	propertyName, productVariableName, arch string, propertyValue interface{}) {
 | |
| 
 | |
| 	productConfigProp := ProductConfigProperty{
 | |
| 		name: productVariableName, // e.g. size, feature1, feature2, FEATURE3, board
 | |
| 		arch: arch,                // e.g. "", x86, arm64
 | |
| 	}
 | |
| 
 | |
| 	p.AddEitherProperty(propertyName, productConfigProp, propertyValue)
 | |
| }
 | |
| 
 | |
| func (p *ProductConfigProperties) AddSoongConfigProperty(
 | |
| 	propertyName, namespace, variableName, value, os string, propertyValue interface{}) {
 | |
| 
 | |
| 	soongConfigProp := SoongConfigProperty{
 | |
| 		namespace: namespace,
 | |
| 		name:      variableName, // e.g. size, feature1, feature2, FEATURE3, board
 | |
| 		value:     value,
 | |
| 		os:        os, // e.g. android, linux_x86
 | |
| 	}
 | |
| 
 | |
| 	p.AddEitherProperty(propertyName, soongConfigProp, propertyValue)
 | |
| }
 | |
| 
 | |
| func (p *ProductConfigProperties) AddEitherProperty(
 | |
| 	propertyName string, key ProductConfigOrSoongConfigProperty, propertyValue interface{}) {
 | |
| 	if (*p)[propertyName] == nil {
 | |
| 		(*p)[propertyName] = make(map[ProductConfigOrSoongConfigProperty]interface{})
 | |
| 	}
 | |
| 
 | |
| 	if existing, ok := (*p)[propertyName][key]; ok {
 | |
| 		switch dst := existing.(type) {
 | |
| 		case []string:
 | |
| 			src, ok := propertyValue.([]string)
 | |
| 			if !ok {
 | |
| 				panic("Conflicting types")
 | |
| 			}
 | |
| 			dst = append(dst, src...)
 | |
| 			(*p)[propertyName][key] = dst
 | |
| 		default:
 | |
| 			if existing != propertyValue {
 | |
| 				panic(fmt.Errorf("TODO: handle merging value %#v", existing))
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		(*p)[propertyName][key] = propertyValue
 | |
| 	}
 | |
| }
 | |
| 
 | |
| var (
 | |
| 	conditionsDefaultField string = proptools.FieldNameForProperty(bazel.ConditionsDefaultConfigKey)
 | |
| )
 | |
| 
 | |
| // maybeExtractConfigVarProp attempts to read this value as a config var struct
 | |
| // wrapped by interfaces and ptrs. If it's not the right type, the second return
 | |
| // value is false.
 | |
| func maybeExtractConfigVarProp(v reflect.Value) (reflect.Value, bool) {
 | |
| 	if v.Kind() == reflect.Interface {
 | |
| 		// The conditions_default value can be either
 | |
| 		// 1) an ptr to an interface of a struct (bool config variables and product variables)
 | |
| 		// 2) an interface of 1) (config variables with nested structs, like string vars)
 | |
| 		v = v.Elem()
 | |
| 	}
 | |
| 	if v.Kind() != reflect.Ptr {
 | |
| 		return v, false
 | |
| 	}
 | |
| 	v = reflect.Indirect(v)
 | |
| 	if v.Kind() == reflect.Interface {
 | |
| 		// Extract the struct from the interface
 | |
| 		v = v.Elem()
 | |
| 	}
 | |
| 
 | |
| 	if !v.IsValid() {
 | |
| 		return v, false
 | |
| 	}
 | |
| 
 | |
| 	if v.Kind() != reflect.Struct {
 | |
| 		return v, false
 | |
| 	}
 | |
| 	return v, true
 | |
| }
 | |
| 
 | |
| func (productConfigProperties *ProductConfigProperties) AddProductConfigProperties(variableValues reflect.Value, arch string) {
 | |
| 	// Example of product_variables:
 | |
| 	//
 | |
| 	// product_variables: {
 | |
| 	//     malloc_not_svelte: {
 | |
| 	//         shared_libs: ["malloc_not_svelte_shared_lib"],
 | |
| 	//         whole_static_libs: ["malloc_not_svelte_whole_static_lib"],
 | |
| 	//         exclude_static_libs: [
 | |
| 	//             "malloc_not_svelte_static_lib_excludes",
 | |
| 	//             "malloc_not_svelte_whole_static_lib_excludes",
 | |
| 	//         ],
 | |
| 	//     },
 | |
| 	// },
 | |
| 
 | |
| 	for i := 0; i < variableValues.NumField(); i++ {
 | |
| 		// e.g. Platform_sdk_version, Unbundled_build, Malloc_not_svelte, etc.
 | |
| 		productVariableName := variableValues.Type().Field(i).Name
 | |
| 
 | |
| 		variableValue := variableValues.Field(i)
 | |
| 		// Check if any properties were set for the module
 | |
| 		if variableValue.IsZero() {
 | |
| 			// e.g. feature1: {}, malloc_not_svelte: {}
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		for j := 0; j < variableValue.NumField(); j++ {
 | |
| 			property := variableValue.Field(j)
 | |
| 			// e.g. Asflags, Cflags, Enabled, etc.
 | |
| 			propertyName := variableValue.Type().Field(j).Name
 | |
| 			if property.Kind() != reflect.Interface {
 | |
| 				productConfigProperties.AddProductConfigProperty(propertyName, productVariableName, arch, property.Interface())
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| }
 | |
| 
 | |
| func (productConfigProperties *ProductConfigProperties) AddSoongConfigProperties(namespace string, soongConfigVariablesStruct reflect.Value) error {
 | |
| 	//
 | |
| 	// Example of soong_config_variables:
 | |
| 	//
 | |
| 	// soong_config_variables: {
 | |
| 	//      feature1: {
 | |
| 	//          conditions_default: {
 | |
| 	//               ...
 | |
| 	//          },
 | |
| 	//          cflags: ...
 | |
| 	//      },
 | |
| 	//      feature2: {
 | |
| 	//          cflags: ...
 | |
| 	//          conditions_default: {
 | |
| 	//               ...
 | |
| 	//          },
 | |
| 	//      },
 | |
| 	//      board: {
 | |
| 	//         soc_a: {
 | |
| 	//             ...
 | |
| 	//         },
 | |
| 	//         soc_b: {
 | |
| 	//             ...
 | |
| 	//         },
 | |
| 	//         soc_c: {},
 | |
| 	//         conditions_default: {
 | |
| 	//              ...
 | |
| 	//         },
 | |
| 	//      },
 | |
| 	// }
 | |
| 	for i := 0; i < soongConfigVariablesStruct.NumField(); i++ {
 | |
| 		// e.g. feature1, feature2, board
 | |
| 		variableName := soongConfigVariablesStruct.Type().Field(i).Name
 | |
| 		variableStruct := soongConfigVariablesStruct.Field(i)
 | |
| 		// Check if any properties were set for the module
 | |
| 		if variableStruct.IsZero() {
 | |
| 			// e.g. feature1: {}
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		// Unlike product variables, config variables require a few more
 | |
| 		// indirections to extract the struct from the reflect.Value.
 | |
| 		if v, ok := maybeExtractConfigVarProp(variableStruct); ok {
 | |
| 			variableStruct = v
 | |
| 		} else if !v.IsValid() {
 | |
| 			// Skip invalid variables which may not used, else leads to panic
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		for j := 0; j < variableStruct.NumField(); j++ {
 | |
| 			propertyOrStruct := variableStruct.Field(j)
 | |
| 			// propertyOrValueName can either be:
 | |
| 			//  - A property, like: Asflags, Cflags, Enabled, etc.
 | |
| 			//  - A soong config string variable's value, like soc_a, soc_b, soc_c in the example above
 | |
| 			//  - "conditions_default"
 | |
| 			propertyOrValueName := variableStruct.Type().Field(j).Name
 | |
| 
 | |
| 			// If the property wasn't set, no need to pass it along
 | |
| 			if propertyOrStruct.IsZero() {
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			if v, ok := maybeExtractConfigVarProp(propertyOrStruct); ok {
 | |
| 				// The field is a struct, which is used by:
 | |
| 				// 1) soong_config_string_variables
 | |
| 				//
 | |
| 				// soc_a: {
 | |
| 				//     cflags: ...,
 | |
| 				// }
 | |
| 				//
 | |
| 				// soc_b: {
 | |
| 				//     cflags: ...,
 | |
| 				// }
 | |
| 				//
 | |
| 				// 2) conditions_default structs for all soong config variable types.
 | |
| 				//
 | |
| 				// conditions_default: {
 | |
| 				//     cflags: ...,
 | |
| 				//     static_libs: ...
 | |
| 				// }
 | |
| 				//
 | |
| 				// This means that propertyOrValueName is either conditions_default, or a soong
 | |
| 				// config string variable's value.
 | |
| 				field := v
 | |
| 				// Iterate over fields of this struct prop.
 | |
| 				for k := 0; k < field.NumField(); k++ {
 | |
| 					// For product variables, zero values are irrelevant; however, for soong config variables,
 | |
| 					// empty values are relevant because there can also be a conditions default which is not
 | |
| 					// applied for empty variables.
 | |
| 					if field.Field(k).IsZero() && namespace == "" {
 | |
| 						continue
 | |
| 					}
 | |
| 
 | |
| 					propertyName := field.Type().Field(k).Name
 | |
| 					if propertyName == "Target" {
 | |
| 						productConfigProperties.AddSoongConfigPropertiesFromTargetStruct(namespace, variableName, proptools.PropertyNameForField(propertyOrValueName), field.Field(k))
 | |
| 					} else if propertyName == "Arch" || propertyName == "Multilib" {
 | |
| 						return fmt.Errorf("Arch/Multilib are not currently supported in soong config variable structs")
 | |
| 					} else {
 | |
| 						productConfigProperties.AddSoongConfigProperty(propertyName, namespace, variableName, proptools.PropertyNameForField(propertyOrValueName), "", field.Field(k).Interface())
 | |
| 					}
 | |
| 				}
 | |
| 			} else if propertyOrStruct.Kind() != reflect.Interface {
 | |
| 				// If not an interface, then this is not a conditions_default or
 | |
| 				// a struct prop. That is, this is a bool/value config variable.
 | |
| 				if propertyOrValueName == "Target" {
 | |
| 					productConfigProperties.AddSoongConfigPropertiesFromTargetStruct(namespace, variableName, "", propertyOrStruct)
 | |
| 				} else if propertyOrValueName == "Arch" || propertyOrValueName == "Multilib" {
 | |
| 					return fmt.Errorf("Arch/Multilib are not currently supported in soong config variable structs")
 | |
| 				} else {
 | |
| 					productConfigProperties.AddSoongConfigProperty(propertyOrValueName, namespace, variableName, "", "", propertyOrStruct.Interface())
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (productConfigProperties *ProductConfigProperties) AddSoongConfigPropertiesFromTargetStruct(namespace, soongConfigVariableName string, soongConfigVariableValue string, targetStruct reflect.Value) {
 | |
| 	// targetStruct will be a struct with fields like "android", "host", "arm", "x86",
 | |
| 	// "android_arm", etc. The values of each of those fields will be a regular property struct.
 | |
| 	for i := 0; i < targetStruct.NumField(); i++ {
 | |
| 		targetFieldName := targetStruct.Type().Field(i).Name
 | |
| 		archOrOsSpecificStruct := targetStruct.Field(i)
 | |
| 		for j := 0; j < archOrOsSpecificStruct.NumField(); j++ {
 | |
| 			property := archOrOsSpecificStruct.Field(j)
 | |
| 			// e.g. Asflags, Cflags, Enabled, etc.
 | |
| 			propertyName := archOrOsSpecificStruct.Type().Field(j).Name
 | |
| 
 | |
| 			if targetFieldName == "Android" {
 | |
| 				productConfigProperties.AddSoongConfigProperty(propertyName, namespace, soongConfigVariableName, soongConfigVariableValue, "android", property.Interface())
 | |
| 			} else if targetFieldName == "Host" {
 | |
| 				for _, os := range osTypeList {
 | |
| 					if os.Class == Host {
 | |
| 						productConfigProperties.AddSoongConfigProperty(propertyName, namespace, soongConfigVariableName, soongConfigVariableValue, os.Name, property.Interface())
 | |
| 					}
 | |
| 				}
 | |
| 			} else if !archOrOsSpecificStruct.IsZero() {
 | |
| 				// One problem with supporting additional fields is that if multiple branches of
 | |
| 				// "target" overlap, we don't want them to be in the same select statement (aka
 | |
| 				// configuration axis). "android" and "host" are disjoint, so it's ok that we only
 | |
| 				// have 2 axes right now. (soongConfigVariables and soongConfigVariablesPlusOs)
 | |
| 				panic("TODO: support other target types in soong config variable structs: " + targetFieldName)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func VariableMutator(mctx BottomUpMutatorContext) {
 | |
| 	var module Module
 | |
| 	var ok bool
 | |
| 	if module, ok = mctx.Module().(Module); !ok {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// TODO: depend on config variable, create variants, propagate variants up tree
 | |
| 	a := module.base()
 | |
| 
 | |
| 	if a.variableProperties == nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	variableValues := reflect.ValueOf(a.variableProperties).Elem().FieldByName("Product_variables")
 | |
| 
 | |
| 	productVariables := reflect.ValueOf(mctx.Config().productVariables)
 | |
| 
 | |
| 	for i := 0; i < variableValues.NumField(); i++ {
 | |
| 		variableValue := variableValues.Field(i)
 | |
| 		name := variableValues.Type().Field(i).Name
 | |
| 		property := "product_variables." + proptools.PropertyNameForField(name)
 | |
| 
 | |
| 		// Check that the variable was set for the product
 | |
| 		val := productVariables.FieldByName(name)
 | |
| 		if !val.IsValid() || val.Kind() != reflect.Ptr || val.IsNil() {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		val = val.Elem()
 | |
| 
 | |
| 		// For bools, check that the value is true
 | |
| 		if val.Kind() == reflect.Bool && val.Bool() == false {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		// Check if any properties were set for the module
 | |
| 		if variableValue.IsZero() {
 | |
| 			continue
 | |
| 		}
 | |
| 		a.setVariableProperties(mctx, property, variableValue, val.Interface())
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (m *ModuleBase) setVariableProperties(ctx BottomUpMutatorContext,
 | |
| 	prefix string, productVariablePropertyValue reflect.Value, variableValue interface{}) {
 | |
| 
 | |
| 	printfIntoProperties(ctx, prefix, productVariablePropertyValue, variableValue)
 | |
| 
 | |
| 	err := proptools.AppendMatchingProperties(m.GetProperties(),
 | |
| 		productVariablePropertyValue.Addr().Interface(), nil)
 | |
| 	if err != nil {
 | |
| 		if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok {
 | |
| 			ctx.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error())
 | |
| 		} else {
 | |
| 			panic(err)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func printfIntoPropertiesError(ctx BottomUpMutatorContext, prefix string,
 | |
| 	productVariablePropertyValue reflect.Value, i int, err error) {
 | |
| 
 | |
| 	field := productVariablePropertyValue.Type().Field(i).Name
 | |
| 	property := prefix + "." + proptools.PropertyNameForField(field)
 | |
| 	ctx.PropertyErrorf(property, "%s", err)
 | |
| }
 | |
| 
 | |
| func printfIntoProperties(ctx BottomUpMutatorContext, prefix string,
 | |
| 	productVariablePropertyValue reflect.Value, variableValue interface{}) {
 | |
| 
 | |
| 	for i := 0; i < productVariablePropertyValue.NumField(); i++ {
 | |
| 		propertyValue := productVariablePropertyValue.Field(i)
 | |
| 		kind := propertyValue.Kind()
 | |
| 		if kind == reflect.Ptr {
 | |
| 			if propertyValue.IsNil() {
 | |
| 				continue
 | |
| 			}
 | |
| 			propertyValue = propertyValue.Elem()
 | |
| 		}
 | |
| 		switch propertyValue.Kind() {
 | |
| 		case reflect.String:
 | |
| 			err := printfIntoProperty(propertyValue, variableValue)
 | |
| 			if err != nil {
 | |
| 				printfIntoPropertiesError(ctx, prefix, productVariablePropertyValue, i, err)
 | |
| 			}
 | |
| 		case reflect.Slice:
 | |
| 			for j := 0; j < propertyValue.Len(); j++ {
 | |
| 				err := printfIntoProperty(propertyValue.Index(j), variableValue)
 | |
| 				if err != nil {
 | |
| 					printfIntoPropertiesError(ctx, prefix, productVariablePropertyValue, i, err)
 | |
| 				}
 | |
| 			}
 | |
| 		case reflect.Bool:
 | |
| 			// Nothing
 | |
| 		case reflect.Struct:
 | |
| 			printfIntoProperties(ctx, prefix, propertyValue, variableValue)
 | |
| 		default:
 | |
| 			panic(fmt.Errorf("unsupported field kind %q", propertyValue.Kind()))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func printfIntoProperty(propertyValue reflect.Value, variableValue interface{}) error {
 | |
| 	s := propertyValue.String()
 | |
| 
 | |
| 	count := strings.Count(s, "%")
 | |
| 	if count == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	if count > 1 {
 | |
| 		return fmt.Errorf("product variable properties only support a single '%%'")
 | |
| 	}
 | |
| 
 | |
| 	if strings.Contains(s, "%d") {
 | |
| 		switch v := variableValue.(type) {
 | |
| 		case int:
 | |
| 			// Nothing
 | |
| 		case bool:
 | |
| 			if v {
 | |
| 				variableValue = 1
 | |
| 			} else {
 | |
| 				variableValue = 0
 | |
| 			}
 | |
| 		default:
 | |
| 			return fmt.Errorf("unsupported type %T for %%d", variableValue)
 | |
| 		}
 | |
| 	} else if strings.Contains(s, "%s") {
 | |
| 		switch variableValue.(type) {
 | |
| 		case string:
 | |
| 			// Nothing
 | |
| 		default:
 | |
| 			return fmt.Errorf("unsupported type %T for %%s", variableValue)
 | |
| 		}
 | |
| 	} else {
 | |
| 		return fmt.Errorf("unsupported %% in product variable property")
 | |
| 	}
 | |
| 
 | |
| 	propertyValue.Set(reflect.ValueOf(fmt.Sprintf(s, variableValue)))
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| var variablePropTypeMap OncePer
 | |
| 
 | |
| // sliceToTypeArray takes a slice of property structs and returns a reflection created array containing the
 | |
| // reflect.Types of each property struct.  The result can be used as a key in a map.
 | |
| func sliceToTypeArray(s []interface{}) interface{} {
 | |
| 	// Create an array using reflection whose length is the length of the input slice
 | |
| 	ret := reflect.New(reflect.ArrayOf(len(s), reflect.TypeOf(reflect.TypeOf(0)))).Elem()
 | |
| 	for i, e := range s {
 | |
| 		ret.Index(i).Set(reflect.ValueOf(reflect.TypeOf(e)))
 | |
| 	}
 | |
| 	return ret.Interface()
 | |
| }
 | |
| 
 | |
| func initProductVariableModule(m Module) {
 | |
| 	base := m.base()
 | |
| 
 | |
| 	// Allow tests to override the default product variables
 | |
| 	if base.variableProperties == nil {
 | |
| 		base.variableProperties = defaultProductVariables
 | |
| 	}
 | |
| 	// Filter the product variables properties to the ones that exist on this module
 | |
| 	base.variableProperties = createVariableProperties(m.GetProperties(), base.variableProperties)
 | |
| 	if base.variableProperties != nil {
 | |
| 		m.AddProperties(base.variableProperties)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // createVariableProperties takes the list of property structs for a module and returns a property struct that
 | |
| // contains the product variable properties that exist in the property structs, or nil if there are none.  It
 | |
| // caches the result.
 | |
| func createVariableProperties(moduleTypeProps []interface{}, productVariables interface{}) interface{} {
 | |
| 	// Convert the moduleTypeProps to an array of reflect.Types that can be used as a key in the OncePer.
 | |
| 	key := sliceToTypeArray(moduleTypeProps)
 | |
| 
 | |
| 	// Use the variablePropTypeMap OncePer to cache the result for each set of property struct types.
 | |
| 	typ, _ := variablePropTypeMap.Once(NewCustomOnceKey(key), func() interface{} {
 | |
| 		// Compute the filtered property struct type.
 | |
| 		return createVariablePropertiesType(moduleTypeProps, productVariables)
 | |
| 	}).(reflect.Type)
 | |
| 
 | |
| 	if typ == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	// Create a new pointer to a filtered property struct.
 | |
| 	return reflect.New(typ).Interface()
 | |
| }
 | |
| 
 | |
| // createVariablePropertiesType creates a new type that contains only the product variable properties that exist in
 | |
| // a list of property structs.
 | |
| func createVariablePropertiesType(moduleTypeProps []interface{}, productVariables interface{}) reflect.Type {
 | |
| 	typ, _ := proptools.FilterPropertyStruct(reflect.TypeOf(productVariables),
 | |
| 		func(field reflect.StructField, prefix string) (bool, reflect.StructField) {
 | |
| 			// Filter function, returns true if the field should be in the resulting struct
 | |
| 			if prefix == "" {
 | |
| 				// Keep the top level Product_variables field
 | |
| 				return true, field
 | |
| 			}
 | |
| 			_, rest := splitPrefix(prefix)
 | |
| 			if rest == "" {
 | |
| 				// Keep the 2nd level field (i.e. Product_variables.Eng)
 | |
| 				return true, field
 | |
| 			}
 | |
| 
 | |
| 			// Strip off the first 2 levels of the prefix
 | |
| 			_, prefix = splitPrefix(rest)
 | |
| 
 | |
| 			for _, p := range moduleTypeProps {
 | |
| 				if fieldExistsByNameRecursive(reflect.TypeOf(p).Elem(), prefix, field.Name) {
 | |
| 					// Keep any fields that exist in one of the property structs
 | |
| 					return true, field
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			return false, field
 | |
| 		})
 | |
| 	return typ
 | |
| }
 | |
| 
 | |
| func splitPrefix(prefix string) (first, rest string) {
 | |
| 	index := strings.IndexByte(prefix, '.')
 | |
| 	if index == -1 {
 | |
| 		return prefix, ""
 | |
| 	}
 | |
| 	return prefix[:index], prefix[index+1:]
 | |
| }
 | |
| 
 | |
| func fieldExistsByNameRecursive(t reflect.Type, prefix, name string) bool {
 | |
| 	if t.Kind() != reflect.Struct {
 | |
| 		panic(fmt.Errorf("fieldExistsByNameRecursive can only be called on a reflect.Struct"))
 | |
| 	}
 | |
| 
 | |
| 	if prefix != "" {
 | |
| 		split := strings.SplitN(prefix, ".", 2)
 | |
| 		firstPrefix := split[0]
 | |
| 		rest := ""
 | |
| 		if len(split) > 1 {
 | |
| 			rest = split[1]
 | |
| 		}
 | |
| 		f, exists := t.FieldByName(firstPrefix)
 | |
| 		if !exists {
 | |
| 			return false
 | |
| 		}
 | |
| 		ft := f.Type
 | |
| 		if ft.Kind() == reflect.Ptr {
 | |
| 			ft = ft.Elem()
 | |
| 		}
 | |
| 		if ft.Kind() != reflect.Struct {
 | |
| 			panic(fmt.Errorf("field %q in %q is not a struct", firstPrefix, t))
 | |
| 		}
 | |
| 		return fieldExistsByNameRecursive(ft, rest, name)
 | |
| 	} else {
 | |
| 		_, exists := t.FieldByName(name)
 | |
| 		return exists
 | |
| 	}
 | |
| }
 |