diff options
Diffstat (limited to 'java/java.go')
-rw-r--r-- | java/java.go | 2092 |
1 files changed, 211 insertions, 1881 deletions
diff --git a/java/java.go b/java/java.go index bc5bdb192..adb0c5644 100644 --- a/java/java.go +++ b/java/java.go @@ -21,11 +21,9 @@ package java import ( "fmt" "path/filepath" - "strconv" "strings" "github.com/google/blueprint" - "github.com/google/blueprint/pathtools" "github.com/google/blueprint/proptools" "android/soong/android" @@ -36,23 +34,56 @@ import ( ) func init() { - RegisterJavaBuildComponents(android.InitRegistrationContext) + registerJavaBuildComponents(android.InitRegistrationContext) + RegisterJavaSdkMemberTypes() +} + +func registerJavaBuildComponents(ctx android.RegistrationContext) { + ctx.RegisterModuleType("java_defaults", DefaultsFactory) + + ctx.RegisterModuleType("java_library", LibraryFactory) + ctx.RegisterModuleType("java_library_static", LibraryStaticFactory) + ctx.RegisterModuleType("java_library_host", LibraryHostFactory) + ctx.RegisterModuleType("java_binary", BinaryFactory) + ctx.RegisterModuleType("java_binary_host", BinaryHostFactory) + ctx.RegisterModuleType("java_test", TestFactory) + ctx.RegisterModuleType("java_test_helper_library", TestHelperLibraryFactory) + ctx.RegisterModuleType("java_test_host", TestHostFactory) + ctx.RegisterModuleType("java_test_import", JavaTestImportFactory) + ctx.RegisterModuleType("java_import", ImportFactory) + ctx.RegisterModuleType("java_import_host", ImportFactoryHost) + ctx.RegisterModuleType("java_device_for_host", DeviceForHostFactory) + ctx.RegisterModuleType("java_host_for_device", HostForDeviceFactory) + ctx.RegisterModuleType("dex_import", DexImportFactory) + + ctx.FinalDepsMutators(func(ctx android.RegisterMutatorsContext) { + ctx.BottomUp("dexpreopt_tool_deps", dexpreoptToolDepsMutator).Parallel() + }) + + ctx.RegisterSingletonType("logtags", LogtagsSingleton) + ctx.RegisterSingletonType("kythe_java_extract", kytheExtractJavaFactory) +} + +func RegisterJavaSdkMemberTypes() { // Register sdk member types. android.RegisterSdkMemberType(javaHeaderLibsSdkMemberType) + // Export implementation classes jar as part of the sdk. + exportImplementationClassesJar := func(_ android.SdkMemberContext, j *Library) android.Path { + implementationJars := j.ImplementationAndResourcesJars() + if len(implementationJars) != 1 { + panic(fmt.Errorf("there must be only one implementation jar from %q", j.Name())) + } + return implementationJars[0] + } + // Register java implementation libraries for use only in module_exports (not sdk). android.RegisterSdkMemberType(&librarySdkMemberType{ android.SdkMemberTypeBase{ PropertyName: "java_libs", }, - func(_ android.SdkMemberContext, j *Library) android.Path { - implementationJars := j.ImplementationAndResourcesJars() - if len(implementationJars) != 1 { - panic(fmt.Errorf("there must be only one implementation jar from %q", j.Name())) - } - return implementationJars[0] - }, + exportImplementationClassesJar, sdkSnapshotFilePathForJar, copyEverythingToSnapshot, }) @@ -73,19 +104,11 @@ func init() { PropertyName: "java_boot_libs", SupportsSdk: true, }, - func(ctx android.SdkMemberContext, j *Library) android.Path { - // Java boot libs are only provided in the SDK to provide access to their dex implementation - // jar for use by dexpreopting and boot jars package check. They do not need to provide an - // actual implementation jar but the java_import will need a file that exists so just copy an - // empty file. Any attempt to use that file as a jar will cause a build error. - return ctx.SnapshotBuilder().EmptyFile() - }, - func(osPrefix, name string) string { - // Create a special name for the implementation jar to try and provide some useful information - // to a developer that attempts to compile against this. - // TODO(b/175714559): Provide a proper error message in Soong not ninja. - return filepath.Join(osPrefix, "java_boot_libs", "snapshot", "jars", "are", "invalid", name+jarFileSuffix) - }, + // Temporarily export implementation classes jar for java_boot_libs as it is required for the + // hiddenapi processing. + // TODO(b/179354495): Revert once hiddenapi processing has been modularized. + exportImplementationClassesJar, + sdkSnapshotFilePathForJar, onlyCopyJarToSnapshot, }) @@ -95,449 +118,65 @@ func init() { PropertyName: "java_tests", }, }) -} - -func RegisterJavaBuildComponents(ctx android.RegistrationContext) { - ctx.RegisterModuleType("java_defaults", DefaultsFactory) - - ctx.RegisterModuleType("java_library", LibraryFactory) - ctx.RegisterModuleType("java_library_static", LibraryStaticFactory) - ctx.RegisterModuleType("java_library_host", LibraryHostFactory) - ctx.RegisterModuleType("java_binary", BinaryFactory) - ctx.RegisterModuleType("java_binary_host", BinaryHostFactory) - ctx.RegisterModuleType("java_test", TestFactory) - ctx.RegisterModuleType("java_test_helper_library", TestHelperLibraryFactory) - ctx.RegisterModuleType("java_test_host", TestHostFactory) - ctx.RegisterModuleType("java_test_import", JavaTestImportFactory) - ctx.RegisterModuleType("java_import", ImportFactory) - ctx.RegisterModuleType("java_import_host", ImportFactoryHost) - ctx.RegisterModuleType("java_device_for_host", DeviceForHostFactory) - ctx.RegisterModuleType("java_host_for_device", HostForDeviceFactory) - ctx.RegisterModuleType("dex_import", DexImportFactory) - - ctx.FinalDepsMutators(func(ctx android.RegisterMutatorsContext) { - ctx.BottomUp("dexpreopt_tool_deps", dexpreoptToolDepsMutator).Parallel() - }) - - ctx.RegisterSingletonType("logtags", LogtagsSingleton) - ctx.RegisterSingletonType("kythe_java_extract", kytheExtractJavaFactory) -} - -func (j *Module) CheckStableSdkVersion() error { - sdkVersion := j.sdkVersion() - if sdkVersion.stable() { - return nil - } - return fmt.Errorf("non stable SDK %v", sdkVersion) -} - -func (j *Module) checkSdkVersions(ctx android.ModuleContext) { - if j.RequiresStableAPIs(ctx) { - if sc, ok := ctx.Module().(sdkContext); ok { - if !sc.sdkVersion().specified() { - ctx.PropertyErrorf("sdk_version", - "sdk_version must have a value when the module is located at vendor or product(only if PRODUCT_ENFORCE_PRODUCT_PARTITION_INTERFACE is set).") - } - } - } - - ctx.VisitDirectDeps(func(module android.Module) { - tag := ctx.OtherModuleDependencyTag(module) - switch module.(type) { - // TODO(satayev): cover other types as well, e.g. imports - case *Library, *AndroidLibrary: - switch tag { - case bootClasspathTag, libTag, staticLibTag, java9LibTag: - checkLinkType(ctx, j, module.(linkTypeContext), tag.(dependencyTag)) - } - } - }) -} - -func (j *Module) checkPlatformAPI(ctx android.ModuleContext) { - if sc, ok := ctx.Module().(sdkContext); ok { - usePlatformAPI := proptools.Bool(j.deviceProperties.Platform_apis) - sdkVersionSpecified := sc.sdkVersion().specified() - if usePlatformAPI && sdkVersionSpecified { - ctx.PropertyErrorf("platform_apis", "platform_apis must be false when sdk_version is not empty.") - } else if !usePlatformAPI && !sdkVersionSpecified { - ctx.PropertyErrorf("platform_apis", "platform_apis must be true when sdk_version is empty.") - } - - } -} - -// TODO: -// Autogenerated files: -// Renderscript -// Post-jar passes: -// Proguard -// Rmtypedefs -// DroidDoc -// Findbugs - -type CompilerProperties struct { - // list of source files used to compile the Java module. May be .java, .kt, .logtags, .proto, - // or .aidl files. - Srcs []string `android:"path,arch_variant"` - - // list Kotlin of source files containing Kotlin code that should be treated as common code in - // a codebase that supports Kotlin multiplatform. See - // https://kotlinlang.org/docs/reference/multiplatform.html. May be only be .kt files. - Common_srcs []string `android:"path,arch_variant"` - - // list of source files that should not be used to build the Java module. - // This is most useful in the arch/multilib variants to remove non-common files - Exclude_srcs []string `android:"path,arch_variant"` - // list of directories containing Java resources - Java_resource_dirs []string `android:"arch_variant"` - - // list of directories that should be excluded from java_resource_dirs - Exclude_java_resource_dirs []string `android:"arch_variant"` - - // list of files to use as Java resources - Java_resources []string `android:"path,arch_variant"` - - // list of files that should be excluded from java_resources and java_resource_dirs - Exclude_java_resources []string `android:"path,arch_variant"` - - // list of module-specific flags that will be used for javac compiles - Javacflags []string `android:"arch_variant"` - - // list of module-specific flags that will be used for kotlinc compiles - Kotlincflags []string `android:"arch_variant"` - - // list of of java libraries that will be in the classpath - Libs []string `android:"arch_variant"` - - // list of java libraries that will be compiled into the resulting jar - Static_libs []string `android:"arch_variant"` - - // manifest file to be included in resulting jar - Manifest *string `android:"path"` - - // if not blank, run jarjar using the specified rules file - Jarjar_rules *string `android:"path,arch_variant"` - - // If not blank, set the java version passed to javac as -source and -target - Java_version *string - - // If set to true, allow this module to be dexed and installed on devices. Has no - // effect on host modules, which are always considered installable. - Installable *bool - - // If set to true, include sources used to compile the module in to the final jar - Include_srcs *bool - - // If not empty, classes are restricted to the specified packages and their sub-packages. - // This restriction is checked after applying jarjar rules and including static libs. - Permitted_packages []string - - // List of modules to use as annotation processors - Plugins []string - - // List of modules to export to libraries that directly depend on this library as annotation - // processors. Note that if the plugins set generates_api: true this will disable the turbine - // optimization on modules that depend on this module, which will reduce parallelism and cause - // more recompilation. - Exported_plugins []string - - // The number of Java source entries each Javac instance can process - Javac_shard_size *int64 - - // Add host jdk tools.jar to bootclasspath - Use_tools_jar *bool - - Openjdk9 struct { - // List of source files that should only be used when passing -source 1.9 or higher - Srcs []string `android:"path"` - - // List of javac flags that should only be used when passing -source 1.9 or higher - Javacflags []string - } - - // When compiling language level 9+ .java code in packages that are part of - // a system module, patch_module names the module that your sources and - // dependencies should be patched into. The Android runtime currently - // doesn't implement the JEP 261 module system so this option is only - // supported at compile time. It should only be needed to compile tests in - // packages that exist in libcore and which are inconvenient to move - // elsewhere. - Patch_module *string `android:"arch_variant"` - - Jacoco struct { - // List of classes to include for instrumentation with jacoco to collect coverage - // information at runtime when building with coverage enabled. If unset defaults to all - // classes. - // Supports '*' as the last character of an entry in the list as a wildcard match. - // If preceded by '.' it matches all classes in the package and subpackages, otherwise - // it matches classes in the package that have the class name as a prefix. - Include_filter []string - - // List of classes to exclude from instrumentation with jacoco to collect coverage - // information at runtime when building with coverage enabled. Overrides classes selected - // by the include_filter property. - // Supports '*' as the last character of an entry in the list as a wildcard match. - // If preceded by '.' it matches all classes in the package and subpackages, otherwise - // it matches classes in the package that have the class name as a prefix. - Exclude_filter []string - } - - Errorprone struct { - // List of javac flags that should only be used when running errorprone. - Javacflags []string - - // List of java_plugin modules that provide extra errorprone checks. - Extra_check_modules []string - } - - Proto struct { - // List of extra options that will be passed to the proto generator. - Output_params []string - } - - Instrument bool `blueprint:"mutated"` - - // List of files to include in the META-INF/services folder of the resulting jar. - Services []string `android:"path,arch_variant"` - - // If true, package the kotlin stdlib into the jar. Defaults to true. - Static_kotlin_stdlib *bool `android:"arch_variant"` -} - -type CompilerDeviceProperties struct { - // if not blank, set to the version of the sdk to compile against. - // Defaults to compiling against the current platform. - Sdk_version *string - - // if not blank, set the minimum version of the sdk that the compiled artifacts will run against. - // Defaults to sdk_version if not set. - Min_sdk_version *string - - // if not blank, set the targetSdkVersion in the AndroidManifest.xml. - // Defaults to sdk_version if not set. - Target_sdk_version *string - - // Whether to compile against the platform APIs instead of an SDK. - // If true, then sdk_version must be empty. The value of this field - // is ignored when module's type isn't android_app. - Platform_apis *bool - - Aidl struct { - // Top level directories to pass to aidl tool - Include_dirs []string - - // Directories rooted at the Android.bp file to pass to aidl tool - Local_include_dirs []string - - // directories that should be added as include directories for any aidl sources of modules - // that depend on this module, as well as to aidl for this module. - Export_include_dirs []string - - // whether to generate traces (for systrace) for this interface - Generate_traces *bool - - // whether to generate Binder#GetTransaction name method. - Generate_get_transaction_name *bool - - // list of flags that will be passed to the AIDL compiler - Flags []string - } - - // If true, export a copy of the module as a -hostdex module for host testing. - Hostdex *bool - - Target struct { - Hostdex struct { - // Additional required dependencies to add to -hostdex modules. - Required []string - } - } - - // When targeting 1.9 and above, override the modules to use with --system, - // otherwise provides defaults libraries to add to the bootclasspath. - System_modules *string - - // The name of the module as used in build configuration. - // - // Allows a library to separate its actual name from the name used in - // build configuration, e.g.ctx.Config().BootJars(). - ConfigurationName *string `blueprint:"mutated"` - - // set the name of the output - Stem *string - - IsSDKLibrary bool `blueprint:"mutated"` - - // If true, generate the signature file of APK Signing Scheme V4, along side the signed APK file. - // Defaults to false. - V4_signature *bool } -// Functionality common to Module and Import -// -// It is embedded in Module so its functionality can be used by methods in Module -// but it is currently only initialized by Import and Library. -type embeddableInModuleAndImport struct { +// JavaInfo contains information about a java module for use by modules that depend on it. +type JavaInfo struct { + // HeaderJars is a list of jars that can be passed as the javac classpath in order to link + // against this module. If empty, ImplementationJars should be used instead. + HeaderJars android.Paths - // Functionality related to this being used as a component of a java_sdk_library. - EmbeddableSdkLibraryComponent -} + // ImplementationAndResourceJars is a list of jars that contain the implementations of classes + // in the module as well as any resources included in the module. + ImplementationAndResourcesJars android.Paths -func (e *embeddableInModuleAndImport) initModuleAndImport(moduleBase *android.ModuleBase) { - e.initSdkLibraryComponent(moduleBase) -} + // ImplementationJars is a list of jars that contain the implementations of classes in the + //module. + ImplementationJars android.Paths -// Module/Import's DepIsInSameApex(...) delegates to this method. -// -// This cannot implement DepIsInSameApex(...) directly as that leads to ambiguity with -// the one provided by ApexModuleBase. -func (e *embeddableInModuleAndImport) depIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool { - // dependencies other than the static linkage are all considered crossing APEX boundary - if staticLibTag == ctx.OtherModuleDependencyTag(dep) { - return true - } - return false -} - -// Module contains the properties and members used by all java module types -type Module struct { - android.ModuleBase - android.DefaultableModuleBase - android.ApexModuleBase - android.SdkBase - - // Functionality common to Module and Import. - embeddableInModuleAndImport - - properties CompilerProperties - protoProperties android.ProtoProperties - deviceProperties CompilerDeviceProperties - - // jar file containing header classes including static library dependencies, suitable for - // inserting into the bootclasspath/classpath of another compile - headerJarFile android.Path - - // jar file containing implementation classes including static library dependencies but no - // resources - implementationJarFile android.Path - - // jar file containing only resources including from static library dependencies - resourceJar android.Path - - // args and dependencies to package source files into a srcjar - srcJarArgs []string - srcJarDeps android.Paths - - // jar file containing implementation classes and resources including static library - // dependencies - implementationAndResourcesJar android.Path - - // output file containing classes.dex and resources - dexJarFile android.Path + // ResourceJars is a list of jars that contain the resources included in the module. + ResourceJars android.Paths - // output file that contains classes.dex if it should be in the output file - maybeStrippedDexJarFile android.Path - - // output file containing uninstrumented classes that will be instrumented by jacoco - jacocoReportClassesFile android.Path - - // output file of the module, which may be a classes jar or a dex jar - outputFile android.Path - extraOutputFiles android.Paths - - exportAidlIncludeDirs android.Paths - - logtagsSrcs android.Paths - - // installed file for binary dependency - installFile android.Path - - // list of .java files and srcjars that was passed to javac - compiledJavaSrcs android.Paths - compiledSrcJars android.Paths - - // manifest file to use instead of properties.Manifest - overrideManifest android.OptionalPath - - // map of SDK version to class loader context - classLoaderContexts dexpreopt.ClassLoaderContextMap - - // list of plugins that this java module is exporting - exportedPluginJars android.Paths - - // list of plugins that this java module is exporting - exportedPluginClasses []string - - // if true, the exported plugins generate API and require disabling turbine. - exportedDisableTurbine bool - - // list of source files, collected from srcFiles with unique java and all kt files, - // will be used by android.IDEInfo struct - expandIDEInfoCompiledSrcs []string + // AidlIncludeDirs is a list of directories that should be passed to the aidl tool when + // depending on this module. + AidlIncludeDirs android.Paths - // expanded Jarjar_rules - expandJarjarRules android.Path + // SrcJarArgs is a list of arguments to pass to soong_zip to package the sources of this + // module. + SrcJarArgs []string - // list of additional targets for checkbuild - additionalCheckedModules android.Paths + // SrcJarDeps is a list of paths to depend on when packaging the sources of this module. + SrcJarDeps android.Paths - // Extra files generated by the module type to be added as java resources. - extraResources android.Paths + // ExportedPlugins is a list of paths that should be used as annotation processors for any + // module that depends on this module. + ExportedPlugins android.Paths - hiddenAPI - dexer - dexpreopter - usesLibrary - linter + // ExportedPluginClasses is a list of classes that should be run as annotation processors for + // any module that depends on this module. + ExportedPluginClasses []string - // list of the xref extraction files - kytheFiles android.Paths + // ExportedPluginDisableTurbine is true if this module's annotation processors generate APIs, + // requiring disbling turbine for any modules that depend on it. + ExportedPluginDisableTurbine bool - // Collect the module directory for IDE info in java/jdeps.go. - modulePaths []string - - hideApexVariantFromMake bool + // JacocoReportClassesFile is the path to a jar containing uninstrumented classes that will be + // instrumented by jacoco. + JacocoReportClassesFile android.Path } -func (j *Module) addHostProperties() { - j.AddProperties( - &j.properties, - &j.protoProperties, - &j.usesLibraryProperties, - ) -} +var JavaInfoProvider = blueprint.NewProvider(JavaInfo{}) -func (j *Module) addHostAndDeviceProperties() { - j.addHostProperties() - j.AddProperties( - &j.deviceProperties, - &j.dexer.dexProperties, - &j.dexpreoptProperties, - &j.linter.properties, - ) +// SyspropPublicStubInfo contains info about the sysprop public stub library that corresponds to +// the sysprop implementation library. +type SyspropPublicStubInfo struct { + // JavaInfo is the JavaInfoProvider of the sysprop public stub library that corresponds to + // the sysprop implementation library. + JavaInfo JavaInfo } -func (j *Module) OutputFiles(tag string) (android.Paths, error) { - switch tag { - case "": - return append(android.Paths{j.outputFile}, j.extraOutputFiles...), nil - case android.DefaultDistTag: - return android.Paths{j.outputFile}, nil - case ".jar": - return android.Paths{j.implementationAndResourcesJar}, nil - case ".proguard_map": - if j.dexer.proguardDictionary.Valid() { - return android.Paths{j.dexer.proguardDictionary.Path()}, nil - } - return nil, fmt.Errorf("%q was requested, but no output file was found.", tag) - default: - return nil, fmt.Errorf("unsupported module reference tag %q", tag) - } -} - -var _ android.OutputFileProducer = (*Module)(nil) +var SyspropPublicStubInfoProvider = blueprint.NewProvider(SyspropPublicStubInfo{}) // Methods that need to be implemented for a module that is added to apex java_libs property. type ApexDependency interface { @@ -552,18 +191,7 @@ type UsesLibraryDependency interface { ClassLoaderContexts() dexpreopt.ClassLoaderContextMap } -type Dependency interface { - ApexDependency - UsesLibraryDependency - ImplementationJars() android.Paths - ResourceJars() android.Paths - AidlIncludeDirs() android.Paths - ExportedPlugins() (android.Paths, []string, bool) - SrcJarArgs() ([]string, android.Paths) - BaseModuleName() string - JacocoReportClassesFile() android.Path -} - +// TODO(jungjw): Move this to kythe.go once it's created. type xref interface { XrefJavaFiles() android.Paths } @@ -572,24 +200,6 @@ func (j *Module) XrefJavaFiles() android.Paths { return j.kytheFiles } -func InitJavaModule(module android.DefaultableModule, hod android.HostOrDeviceSupported) { - initJavaModule(module, hod, false) -} - -func InitJavaModuleMultiTargets(module android.DefaultableModule, hod android.HostOrDeviceSupported) { - initJavaModule(module, hod, true) -} - -func initJavaModule(module android.DefaultableModule, hod android.HostOrDeviceSupported, multiTargets bool) { - multilib := android.MultilibCommon - if multiTargets { - android.InitAndroidMultiTargetsArchModule(module, hod, multilib) - } else { - android.InitAndroidArchModule(module, hod, multilib) - } - android.InitDefaultableModule(module) -} - type dependencyTag struct { blueprint.BaseDependencyTag name string @@ -620,29 +230,30 @@ func IsJniDepTag(depTag blueprint.DependencyTag) bool { } var ( - dataNativeBinsTag = dependencyTag{name: "dataNativeBins"} - staticLibTag = dependencyTag{name: "staticlib"} - libTag = dependencyTag{name: "javalib"} - java9LibTag = dependencyTag{name: "java9lib"} - pluginTag = dependencyTag{name: "plugin"} - errorpronePluginTag = dependencyTag{name: "errorprone-plugin"} - exportedPluginTag = dependencyTag{name: "exported-plugin"} - bootClasspathTag = dependencyTag{name: "bootclasspath"} - systemModulesTag = dependencyTag{name: "system modules"} - frameworkResTag = dependencyTag{name: "framework-res"} - kotlinStdlibTag = dependencyTag{name: "kotlin-stdlib"} - kotlinAnnotationsTag = dependencyTag{name: "kotlin-annotations"} - proguardRaiseTag = dependencyTag{name: "proguard-raise"} - certificateTag = dependencyTag{name: "certificate"} - instrumentationForTag = dependencyTag{name: "instrumentation_for"} - extraLintCheckTag = dependencyTag{name: "extra-lint-check"} - jniLibTag = dependencyTag{name: "jnilib"} - jniInstallTag = installDependencyTag{name: "jni install"} - binaryInstallTag = installDependencyTag{name: "binary install"} - usesLibTag = makeUsesLibraryDependencyTag(dexpreopt.AnySdkVersion) - usesLibCompat28Tag = makeUsesLibraryDependencyTag(28) - usesLibCompat29Tag = makeUsesLibraryDependencyTag(29) - usesLibCompat30Tag = makeUsesLibraryDependencyTag(30) + dataNativeBinsTag = dependencyTag{name: "dataNativeBins"} + staticLibTag = dependencyTag{name: "staticlib"} + libTag = dependencyTag{name: "javalib"} + java9LibTag = dependencyTag{name: "java9lib"} + pluginTag = dependencyTag{name: "plugin"} + errorpronePluginTag = dependencyTag{name: "errorprone-plugin"} + exportedPluginTag = dependencyTag{name: "exported-plugin"} + bootClasspathTag = dependencyTag{name: "bootclasspath"} + systemModulesTag = dependencyTag{name: "system modules"} + frameworkResTag = dependencyTag{name: "framework-res"} + kotlinStdlibTag = dependencyTag{name: "kotlin-stdlib"} + kotlinAnnotationsTag = dependencyTag{name: "kotlin-annotations"} + proguardRaiseTag = dependencyTag{name: "proguard-raise"} + certificateTag = dependencyTag{name: "certificate"} + instrumentationForTag = dependencyTag{name: "instrumentation_for"} + extraLintCheckTag = dependencyTag{name: "extra-lint-check"} + jniLibTag = dependencyTag{name: "jnilib"} + syspropPublicStubDepTag = dependencyTag{name: "sysprop public stub"} + jniInstallTag = installDependencyTag{name: "jni install"} + binaryInstallTag = installDependencyTag{name: "binary install"} + usesLibTag = makeUsesLibraryDependencyTag(dexpreopt.AnySdkVersion) + usesLibCompat28Tag = makeUsesLibraryDependencyTag(28) + usesLibCompat29Tag = makeUsesLibraryDependencyTag(29) + usesLibCompat30Tag = makeUsesLibraryDependencyTag(30) ) func IsLibDepTag(depTag blueprint.DependencyTag) bool { @@ -694,71 +305,7 @@ type jniLib struct { unstrippedFile android.Path } -func (j *Module) shouldInstrument(ctx android.BaseModuleContext) bool { - return j.properties.Instrument && - ctx.Config().IsEnvTrue("EMMA_INSTRUMENT") && - ctx.DeviceConfig().JavaCoverageEnabledForPath(ctx.ModuleDir()) -} - -func (j *Module) shouldInstrumentStatic(ctx android.BaseModuleContext) bool { - return j.shouldInstrument(ctx) && - (ctx.Config().IsEnvTrue("EMMA_INSTRUMENT_STATIC") || - ctx.Config().UnbundledBuild()) -} - -func (j *Module) shouldInstrumentInApex(ctx android.BaseModuleContext) bool { - // Force enable the instrumentation for java code that is built for APEXes ... - // except for the jacocoagent itself (because instrumenting jacocoagent using jacocoagent - // doesn't make sense) or framework libraries (e.g. libraries found in the InstrumentFrameworkModules list) unless EMMA_INSTRUMENT_FRAMEWORK is true. - apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) - isJacocoAgent := ctx.ModuleName() == "jacocoagent" - if j.DirectlyInAnyApex() && !isJacocoAgent && !apexInfo.IsForPlatform() { - if !inList(ctx.ModuleName(), config.InstrumentFrameworkModules) { - return true - } else if ctx.Config().IsEnvTrue("EMMA_INSTRUMENT_FRAMEWORK") { - return true - } - } - return false -} - -func (j *Module) sdkVersion() sdkSpec { - return sdkSpecFrom(String(j.deviceProperties.Sdk_version)) -} - -func (j *Module) systemModules() string { - return proptools.String(j.deviceProperties.System_modules) -} - -func (j *Module) minSdkVersion() sdkSpec { - if j.deviceProperties.Min_sdk_version != nil { - return sdkSpecFrom(*j.deviceProperties.Min_sdk_version) - } - return j.sdkVersion() -} - -func (j *Module) targetSdkVersion() sdkSpec { - if j.deviceProperties.Target_sdk_version != nil { - return sdkSpecFrom(*j.deviceProperties.Target_sdk_version) - } - return j.sdkVersion() -} - -func (j *Module) MinSdkVersion() string { - return j.minSdkVersion().version.String() -} - -func (j *Module) AvailableFor(what string) bool { - if what == android.AvailableToPlatform && Bool(j.deviceProperties.Hostdex) { - // Exception: for hostdex: true libraries, the platform variant is created - // even if it's not marked as available to platform. In that case, the platform - // variant is used only for the hostdex and not installed to the device. - return true - } - return j.ApexModuleBase.AvailableFor(what) -} - -func sdkDeps(ctx android.BottomUpMutatorContext, sdkContext sdkContext, d dexer) { +func sdkDeps(ctx android.BottomUpMutatorContext, sdkContext android.SdkContext, d dexer) { sdkDep := decodeSdkDep(ctx, sdkContext) if sdkDep.useModule { ctx.AddVariationDependencies(nil, bootClasspathTag, sdkDep.bootclasspath...) @@ -776,174 +323,6 @@ func sdkDeps(ctx android.BottomUpMutatorContext, sdkContext sdkContext, d dexer) } } -func (j *Module) deps(ctx android.BottomUpMutatorContext) { - if ctx.Device() { - j.linter.deps(ctx) - - sdkDeps(ctx, sdkContext(j), j.dexer) - } - - syspropPublicStubs := syspropPublicStubs(ctx.Config()) - - // rewriteSyspropLibs validates if a java module can link against platform's sysprop_library, - // and redirects dependency to public stub depending on the link type. - rewriteSyspropLibs := func(libs []string, prop string) []string { - // make a copy - ret := android.CopyOf(libs) - - for idx, lib := range libs { - stub, ok := syspropPublicStubs[lib] - - if !ok { - continue - } - - linkType, _ := j.getLinkType(ctx.ModuleName()) - // only platform modules can use internal props - if linkType != javaPlatform { - ret[idx] = stub - } - } - - return ret - } - - libDeps := ctx.AddVariationDependencies(nil, libTag, rewriteSyspropLibs(j.properties.Libs, "libs")...) - ctx.AddVariationDependencies(nil, staticLibTag, rewriteSyspropLibs(j.properties.Static_libs, "static_libs")...) - - if ctx.DeviceConfig().VndkVersion() != "" && ctx.Config().EnforceInterPartitionJavaSdkLibrary() { - // Require java_sdk_library at inter-partition java dependency to ensure stable - // interface between partitions. If inter-partition java_library dependency is detected, - // raise build error because java_library doesn't have a stable interface. - // - // Inputs: - // PRODUCT_ENFORCE_INTER_PARTITION_JAVA_SDK_LIBRARY - // if true, enable enforcement - // PRODUCT_INTER_PARTITION_JAVA_LIBRARY_ALLOWLIST - // exception list of java_library names to allow inter-partition dependency - for idx, lib := range j.properties.Libs { - if libDeps[idx] == nil { - continue - } - - if _, ok := syspropPublicStubs[lib]; ok { - continue - } - - if javaDep, ok := libDeps[idx].(javaSdkLibraryEnforceContext); ok { - // java_sdk_library is always allowed at inter-partition dependency. - // So, skip check. - if _, ok := javaDep.(*SdkLibrary); ok { - continue - } - - j.checkPartitionsForJavaDependency(ctx, "libs", javaDep) - } - } - } - - // For library dependencies that are component libraries (like stubs), add the implementation - // as a dependency (dexpreopt needs to be against the implementation library, not stubs). - for _, dep := range libDeps { - if dep != nil { - if component, ok := dep.(SdkLibraryComponentDependency); ok { - if lib := component.OptionalSdkLibraryImplementation(); lib != nil { - ctx.AddVariationDependencies(nil, usesLibTag, *lib) - } - } - } - } - - ctx.AddFarVariationDependencies(ctx.Config().BuildOSCommonTarget.Variations(), pluginTag, j.properties.Plugins...) - ctx.AddFarVariationDependencies(ctx.Config().BuildOSCommonTarget.Variations(), errorpronePluginTag, j.properties.Errorprone.Extra_check_modules...) - ctx.AddFarVariationDependencies(ctx.Config().BuildOSCommonTarget.Variations(), exportedPluginTag, j.properties.Exported_plugins...) - - android.ProtoDeps(ctx, &j.protoProperties) - if j.hasSrcExt(".proto") { - protoDeps(ctx, &j.protoProperties) - } - - if j.hasSrcExt(".kt") { - // TODO(ccross): move this to a mutator pass that can tell if generated sources contain - // Kotlin files - ctx.AddVariationDependencies(nil, kotlinStdlibTag, - "kotlin-stdlib", "kotlin-stdlib-jdk7", "kotlin-stdlib-jdk8") - if len(j.properties.Plugins) > 0 { - ctx.AddVariationDependencies(nil, kotlinAnnotationsTag, "kotlin-annotations") - } - } - - // Framework libraries need special handling in static coverage builds: they should not have - // static dependency on jacoco, otherwise there would be multiple conflicting definitions of - // the same jacoco classes coming from different bootclasspath jars. - if inList(ctx.ModuleName(), config.InstrumentFrameworkModules) { - if ctx.Config().IsEnvTrue("EMMA_INSTRUMENT_FRAMEWORK") { - j.properties.Instrument = true - } - } else if j.shouldInstrumentStatic(ctx) { - ctx.AddVariationDependencies(nil, staticLibTag, "jacocoagent") - } -} - -func hasSrcExt(srcs []string, ext string) bool { - for _, src := range srcs { - if filepath.Ext(src) == ext { - return true - } - } - - return false -} - -func (j *Module) hasSrcExt(ext string) bool { - return hasSrcExt(j.properties.Srcs, ext) -} - -func (j *Module) aidlFlags(ctx android.ModuleContext, aidlPreprocess android.OptionalPath, - aidlIncludeDirs android.Paths) (string, android.Paths) { - - aidlIncludes := android.PathsForModuleSrc(ctx, j.deviceProperties.Aidl.Local_include_dirs) - aidlIncludes = append(aidlIncludes, - android.PathsForModuleSrc(ctx, j.deviceProperties.Aidl.Export_include_dirs)...) - aidlIncludes = append(aidlIncludes, - android.PathsForSource(ctx, j.deviceProperties.Aidl.Include_dirs)...) - - var flags []string - var deps android.Paths - - flags = append(flags, j.deviceProperties.Aidl.Flags...) - - if aidlPreprocess.Valid() { - flags = append(flags, "-p"+aidlPreprocess.String()) - deps = append(deps, aidlPreprocess.Path()) - } else if len(aidlIncludeDirs) > 0 { - flags = append(flags, android.JoinWithPrefix(aidlIncludeDirs.Strings(), "-I")) - } - - if len(j.exportAidlIncludeDirs) > 0 { - flags = append(flags, android.JoinWithPrefix(j.exportAidlIncludeDirs.Strings(), "-I")) - } - - if len(aidlIncludes) > 0 { - flags = append(flags, android.JoinWithPrefix(aidlIncludes.Strings(), "-I")) - } - - flags = append(flags, "-I"+android.PathForModuleSrc(ctx).String()) - if src := android.ExistentPathForSource(ctx, ctx.ModuleDir(), "src"); src.Valid() { - flags = append(flags, "-I"+src.String()) - } - - if Bool(j.deviceProperties.Aidl.Generate_traces) { - flags = append(flags, "-t") - } - - if Bool(j.deviceProperties.Aidl.Generate_get_transaction_name) { - flags = append(flags, "--transaction_names") - } - - return strings.Join(flags, " "), deps -} - type deps struct { classpath classpath java9Classpath classpath @@ -974,265 +353,11 @@ func checkProducesJars(ctx android.ModuleContext, dep android.SourceFileProducer } } -type linkType int - -const ( - // TODO(jiyong) rename these for better readability. Make the allowed - // and disallowed link types explicit - javaCore linkType = iota - javaSdk - javaSystem - javaModule - javaSystemServer - javaPlatform -) - -type linkTypeContext interface { - android.Module - getLinkType(name string) (ret linkType, stubs bool) -} - -func (m *Module) getLinkType(name string) (ret linkType, stubs bool) { - switch name { - case "core.current.stubs", "legacy.core.platform.api.stubs", "stable.core.platform.api.stubs", - "stub-annotations", "private-stub-annotations-jar", - "core-lambda-stubs", "core-generated-annotation-stubs": - return javaCore, true - case "android_stubs_current": - return javaSdk, true - case "android_system_stubs_current": - return javaSystem, true - case "android_module_lib_stubs_current": - return javaModule, true - case "android_system_server_stubs_current": - return javaSystemServer, true - case "android_test_stubs_current": - return javaSystem, true - } - - if stub, linkType := moduleStubLinkType(name); stub { - return linkType, true - } - - ver := m.sdkVersion() - switch ver.kind { - case sdkCore: - return javaCore, false - case sdkSystem: - return javaSystem, false - case sdkPublic: - return javaSdk, false - case sdkModule: - return javaModule, false - case sdkSystemServer: - return javaSystemServer, false - case sdkPrivate, sdkNone, sdkCorePlatform, sdkTest: - return javaPlatform, false - } - - if !ver.valid() { - panic(fmt.Errorf("sdk_version is invalid. got %q", ver.raw)) - } - return javaSdk, false -} - -func checkLinkType(ctx android.ModuleContext, from *Module, to linkTypeContext, tag dependencyTag) { - if ctx.Host() { - return - } - - myLinkType, stubs := from.getLinkType(ctx.ModuleName()) - if stubs { - return - } - otherLinkType, _ := to.getLinkType(ctx.OtherModuleName(to)) - commonMessage := " In order to fix this, consider adjusting sdk_version: OR platform_apis: " + - "property of the source or target module so that target module is built with the same " + - "or smaller API set when compared to the source." - - switch myLinkType { - case javaCore: - if otherLinkType != javaCore { - ctx.ModuleErrorf("compiles against core Java API, but dependency %q is compiling against non-core Java APIs."+commonMessage, - ctx.OtherModuleName(to)) - } - break - case javaSdk: - if otherLinkType != javaCore && otherLinkType != javaSdk { - ctx.ModuleErrorf("compiles against Android API, but dependency %q is compiling against non-public Android API."+commonMessage, - ctx.OtherModuleName(to)) - } - break - case javaSystem: - if otherLinkType == javaPlatform || otherLinkType == javaModule || otherLinkType == javaSystemServer { - ctx.ModuleErrorf("compiles against system API, but dependency %q is compiling against private API."+commonMessage, - ctx.OtherModuleName(to)) - } - break - case javaModule: - if otherLinkType == javaPlatform || otherLinkType == javaSystemServer { - ctx.ModuleErrorf("compiles against module API, but dependency %q is compiling against private API."+commonMessage, - ctx.OtherModuleName(to)) - } - break - case javaSystemServer: - if otherLinkType == javaPlatform { - ctx.ModuleErrorf("compiles against system server API, but dependency %q is compiling against private API."+commonMessage, - ctx.OtherModuleName(to)) - } - break - case javaPlatform: - // no restriction on link-type - break - } -} - -func (j *Module) collectDeps(ctx android.ModuleContext) deps { - var deps deps - - if ctx.Device() { - sdkDep := decodeSdkDep(ctx, sdkContext(j)) - if sdkDep.invalidVersion { - ctx.AddMissingDependencies(sdkDep.bootclasspath) - ctx.AddMissingDependencies(sdkDep.java9Classpath) - } else if sdkDep.useFiles { - // sdkDep.jar is actually equivalent to turbine header.jar. - deps.classpath = append(deps.classpath, sdkDep.jars...) - deps.aidlPreprocess = sdkDep.aidl - } else { - deps.aidlPreprocess = sdkDep.aidl - } - } - - ctx.VisitDirectDeps(func(module android.Module) { - otherName := ctx.OtherModuleName(module) - tag := ctx.OtherModuleDependencyTag(module) - - if IsJniDepTag(tag) { - // Handled by AndroidApp.collectAppDeps - return - } - if tag == certificateTag { - // Handled by AndroidApp.collectAppDeps - return - } - - switch dep := module.(type) { - case SdkLibraryDependency: - switch tag { - case libTag: - deps.classpath = append(deps.classpath, dep.SdkHeaderJars(ctx, j.sdkVersion())...) - case staticLibTag: - ctx.ModuleErrorf("dependency on java_sdk_library %q can only be in libs", otherName) - } - case Dependency: - switch tag { - case bootClasspathTag: - deps.bootClasspath = append(deps.bootClasspath, dep.HeaderJars()...) - case libTag, instrumentationForTag: - deps.classpath = append(deps.classpath, dep.HeaderJars()...) - deps.aidlIncludeDirs = append(deps.aidlIncludeDirs, dep.AidlIncludeDirs()...) - pluginJars, pluginClasses, disableTurbine := dep.ExportedPlugins() - addPlugins(&deps, pluginJars, pluginClasses...) - deps.disableTurbine = deps.disableTurbine || disableTurbine - case java9LibTag: - deps.java9Classpath = append(deps.java9Classpath, dep.HeaderJars()...) - case staticLibTag: - deps.classpath = append(deps.classpath, dep.HeaderJars()...) - deps.staticJars = append(deps.staticJars, dep.ImplementationJars()...) - deps.staticHeaderJars = append(deps.staticHeaderJars, dep.HeaderJars()...) - deps.staticResourceJars = append(deps.staticResourceJars, dep.ResourceJars()...) - deps.aidlIncludeDirs = append(deps.aidlIncludeDirs, dep.AidlIncludeDirs()...) - pluginJars, pluginClasses, disableTurbine := dep.ExportedPlugins() - addPlugins(&deps, pluginJars, pluginClasses...) - // Turbine doesn't run annotation processors, so any module that uses an - // annotation processor that generates API is incompatible with the turbine - // optimization. - deps.disableTurbine = deps.disableTurbine || disableTurbine - case pluginTag: - if plugin, ok := dep.(*Plugin); ok { - if plugin.pluginProperties.Processor_class != nil { - addPlugins(&deps, plugin.ImplementationAndResourcesJars(), *plugin.pluginProperties.Processor_class) - } else { - addPlugins(&deps, plugin.ImplementationAndResourcesJars()) - } - // Turbine doesn't run annotation processors, so any module that uses an - // annotation processor that generates API is incompatible with the turbine - // optimization. - deps.disableTurbine = deps.disableTurbine || Bool(plugin.pluginProperties.Generates_api) - } else { - ctx.PropertyErrorf("plugins", "%q is not a java_plugin module", otherName) - } - case errorpronePluginTag: - if plugin, ok := dep.(*Plugin); ok { - deps.errorProneProcessorPath = append(deps.errorProneProcessorPath, plugin.ImplementationAndResourcesJars()...) - } else { - ctx.PropertyErrorf("plugins", "%q is not a java_plugin module", otherName) - } - case exportedPluginTag: - if plugin, ok := dep.(*Plugin); ok { - j.exportedPluginJars = append(j.exportedPluginJars, plugin.ImplementationAndResourcesJars()...) - if plugin.pluginProperties.Processor_class != nil { - j.exportedPluginClasses = append(j.exportedPluginClasses, *plugin.pluginProperties.Processor_class) - } - // Turbine doesn't run annotation processors, so any module that uses an - // annotation processor that generates API is incompatible with the turbine - // optimization. - j.exportedDisableTurbine = Bool(plugin.pluginProperties.Generates_api) - } else { - ctx.PropertyErrorf("exported_plugins", "%q is not a java_plugin module", otherName) - } - case kotlinStdlibTag: - deps.kotlinStdlib = append(deps.kotlinStdlib, dep.HeaderJars()...) - case kotlinAnnotationsTag: - deps.kotlinAnnotations = dep.HeaderJars() - } - - case android.SourceFileProducer: - switch tag { - case libTag: - checkProducesJars(ctx, dep) - deps.classpath = append(deps.classpath, dep.Srcs()...) - case staticLibTag: - checkProducesJars(ctx, dep) - deps.classpath = append(deps.classpath, dep.Srcs()...) - deps.staticJars = append(deps.staticJars, dep.Srcs()...) - deps.staticHeaderJars = append(deps.staticHeaderJars, dep.Srcs()...) - } - default: - switch tag { - case bootClasspathTag: - // If a system modules dependency has been added to the bootclasspath - // then add its libs to the bootclasspath. - sm := module.(SystemModulesProvider) - deps.bootClasspath = append(deps.bootClasspath, sm.HeaderJars()...) - - case systemModulesTag: - if deps.systemModules != nil { - panic("Found two system module dependencies") - } - sm := module.(SystemModulesProvider) - outputDir, outputDeps := sm.OutputDirAndDeps() - deps.systemModules = &systemModules{outputDir, outputDeps} - } - } - - addCLCFromDep(ctx, module, j.classLoaderContexts) - }) - - return deps -} - -func addPlugins(deps *deps, pluginJars android.Paths, pluginClasses ...string) { - deps.processorPath = append(deps.processorPath, pluginJars...) - deps.processorClasses = append(deps.processorClasses, pluginClasses...) -} - -func getJavaVersion(ctx android.ModuleContext, javaVersion string, sdkContext sdkContext) javaVersion { +func getJavaVersion(ctx android.ModuleContext, javaVersion string, sdkContext android.SdkContext) javaVersion { if javaVersion != "" { return normalizeJavaVersion(ctx, javaVersion) } else if ctx.Device() { - return sdkContext.sdkVersion().defaultJavaLanguageVersion(ctx) + return defaultJavaLanguageVersion(ctx, sdkContext.SdkVersion(ctx)) } else { return JAVA_VERSION_9 } @@ -1287,818 +412,6 @@ func normalizeJavaVersion(ctx android.BaseModuleContext, javaVersion string) jav } } -func (j *Module) collectBuilderFlags(ctx android.ModuleContext, deps deps) javaBuilderFlags { - - var flags javaBuilderFlags - - // javaVersion flag. - flags.javaVersion = getJavaVersion(ctx, String(j.properties.Java_version), sdkContext(j)) - - if ctx.Config().RunErrorProne() { - if config.ErrorProneClasspath == nil && ctx.Config().TestProductVariables == nil { - ctx.ModuleErrorf("cannot build with Error Prone, missing external/error_prone?") - } - - errorProneFlags := []string{ - "-Xplugin:ErrorProne", - "${config.ErrorProneChecks}", - } - errorProneFlags = append(errorProneFlags, j.properties.Errorprone.Javacflags...) - - flags.errorProneExtraJavacFlags = "${config.ErrorProneFlags} " + - "'" + strings.Join(errorProneFlags, " ") + "'" - flags.errorProneProcessorPath = classpath(android.PathsForSource(ctx, config.ErrorProneClasspath)) - } - - // classpath - flags.bootClasspath = append(flags.bootClasspath, deps.bootClasspath...) - flags.classpath = append(flags.classpath, deps.classpath...) - flags.java9Classpath = append(flags.java9Classpath, deps.java9Classpath...) - flags.processorPath = append(flags.processorPath, deps.processorPath...) - flags.errorProneProcessorPath = append(flags.errorProneProcessorPath, deps.errorProneProcessorPath...) - - flags.processors = append(flags.processors, deps.processorClasses...) - flags.processors = android.FirstUniqueStrings(flags.processors) - - if len(flags.bootClasspath) == 0 && ctx.Host() && !flags.javaVersion.usesJavaModules() && - decodeSdkDep(ctx, sdkContext(j)).hasStandardLibs() { - // Give host-side tools a version of OpenJDK's standard libraries - // close to what they're targeting. As of Dec 2017, AOSP is only - // bundling OpenJDK 8 and 9, so nothing < 8 is available. - // - // When building with OpenJDK 8, the following should have no - // effect since those jars would be available by default. - // - // When building with OpenJDK 9 but targeting a version < 1.8, - // putting them on the bootclasspath means that: - // a) code can't (accidentally) refer to OpenJDK 9 specific APIs - // b) references to existing APIs are not reinterpreted in an - // OpenJDK 9-specific way, eg. calls to subclasses of - // java.nio.Buffer as in http://b/70862583 - java8Home := ctx.Config().Getenv("ANDROID_JAVA8_HOME") - flags.bootClasspath = append(flags.bootClasspath, - android.PathForSource(ctx, java8Home, "jre/lib/jce.jar"), - android.PathForSource(ctx, java8Home, "jre/lib/rt.jar")) - if Bool(j.properties.Use_tools_jar) { - flags.bootClasspath = append(flags.bootClasspath, - android.PathForSource(ctx, java8Home, "lib/tools.jar")) - } - } - - // systemModules - flags.systemModules = deps.systemModules - - // aidl flags. - flags.aidlFlags, flags.aidlDeps = j.aidlFlags(ctx, deps.aidlPreprocess, deps.aidlIncludeDirs) - - return flags -} - -func (j *Module) collectJavacFlags( - ctx android.ModuleContext, flags javaBuilderFlags, srcFiles android.Paths) javaBuilderFlags { - // javac flags. - javacFlags := j.properties.Javacflags - - if ctx.Config().MinimizeJavaDebugInfo() && !ctx.Host() { - // For non-host binaries, override the -g flag passed globally to remove - // local variable debug info to reduce disk and memory usage. - javacFlags = append(javacFlags, "-g:source,lines") - } - javacFlags = append(javacFlags, "-Xlint:-dep-ann") - - if flags.javaVersion.usesJavaModules() { - javacFlags = append(javacFlags, j.properties.Openjdk9.Javacflags...) - - if j.properties.Patch_module != nil { - // Manually specify build directory in case it is not under the repo root. - // (javac doesn't seem to expand into symbolic links when searching for patch-module targets, so - // just adding a symlink under the root doesn't help.) - patchPaths := []string{".", ctx.Config().BuildDir()} - - // b/150878007 - // - // Workaround to support *Bazel-executed* JDK9 javac in Bazel's - // execution root for --patch-module. If this javac command line is - // invoked within Bazel's execution root working directory, the top - // level directories (e.g. libcore/, tools/, frameworks/) are all - // symlinks. JDK9 javac does not traverse into symlinks, which causes - // --patch-module to fail source file lookups when invoked in the - // execution root. - // - // Short of patching javac or enumerating *all* directories as possible - // input dirs, manually add the top level dir of the source files to be - // compiled. - topLevelDirs := map[string]bool{} - for _, srcFilePath := range srcFiles { - srcFileParts := strings.Split(srcFilePath.String(), "/") - // Ignore source files that are already in the top level directory - // as well as generated files in the out directory. The out - // directory may be an absolute path, which means srcFileParts[0] is the - // empty string, so check that as well. Note that "out" in Bazel's execution - // root is *not* a symlink, which doesn't cause problems for --patch-modules - // anyway, so it's fine to not apply this workaround for generated - // source files. - if len(srcFileParts) > 1 && - srcFileParts[0] != "" && - srcFileParts[0] != "out" { - topLevelDirs[srcFileParts[0]] = true - } - } - patchPaths = append(patchPaths, android.SortedStringKeys(topLevelDirs)...) - - classPath := flags.classpath.FormJavaClassPath("") - if classPath != "" { - patchPaths = append(patchPaths, classPath) - } - javacFlags = append( - javacFlags, - "--patch-module="+String(j.properties.Patch_module)+"="+strings.Join(patchPaths, ":")) - } - } - - if len(javacFlags) > 0 { - // optimization. - ctx.Variable(pctx, "javacFlags", strings.Join(javacFlags, " ")) - flags.javacFlags = "$javacFlags" - } - - return flags -} - -func (j *Module) compile(ctx android.ModuleContext, aaptSrcJar android.Path) { - j.exportAidlIncludeDirs = android.PathsForModuleSrc(ctx, j.deviceProperties.Aidl.Export_include_dirs) - - deps := j.collectDeps(ctx) - flags := j.collectBuilderFlags(ctx, deps) - - if flags.javaVersion.usesJavaModules() { - j.properties.Srcs = append(j.properties.Srcs, j.properties.Openjdk9.Srcs...) - } - srcFiles := android.PathsForModuleSrcExcludes(ctx, j.properties.Srcs, j.properties.Exclude_srcs) - if hasSrcExt(srcFiles.Strings(), ".proto") { - flags = protoFlags(ctx, &j.properties, &j.protoProperties, flags) - } - - kotlinCommonSrcFiles := android.PathsForModuleSrcExcludes(ctx, j.properties.Common_srcs, nil) - if len(kotlinCommonSrcFiles.FilterOutByExt(".kt")) > 0 { - ctx.PropertyErrorf("common_srcs", "common_srcs must be .kt files") - } - - srcFiles = j.genSources(ctx, srcFiles, flags) - - // Collect javac flags only after computing the full set of srcFiles to - // ensure that the --patch-module lookup paths are complete. - flags = j.collectJavacFlags(ctx, flags, srcFiles) - - srcJars := srcFiles.FilterByExt(".srcjar") - srcJars = append(srcJars, deps.srcJars...) - if aaptSrcJar != nil { - srcJars = append(srcJars, aaptSrcJar) - } - - if j.properties.Jarjar_rules != nil { - j.expandJarjarRules = android.PathForModuleSrc(ctx, *j.properties.Jarjar_rules) - } - - jarName := ctx.ModuleName() + ".jar" - - javaSrcFiles := srcFiles.FilterByExt(".java") - var uniqueSrcFiles android.Paths - set := make(map[string]bool) - for _, v := range javaSrcFiles { - if _, found := set[v.String()]; !found { - set[v.String()] = true - uniqueSrcFiles = append(uniqueSrcFiles, v) - } - } - - // Collect .java files for AIDEGen - j.expandIDEInfoCompiledSrcs = append(j.expandIDEInfoCompiledSrcs, uniqueSrcFiles.Strings()...) - - var kotlinJars android.Paths - - if srcFiles.HasExt(".kt") { - // user defined kotlin flags. - kotlincFlags := j.properties.Kotlincflags - CheckKotlincFlags(ctx, kotlincFlags) - - // Dogfood the JVM_IR backend. - kotlincFlags = append(kotlincFlags, "-Xuse-ir") - - // If there are kotlin files, compile them first but pass all the kotlin and java files - // kotlinc will use the java files to resolve types referenced by the kotlin files, but - // won't emit any classes for them. - kotlincFlags = append(kotlincFlags, "-no-stdlib") - if ctx.Device() { - kotlincFlags = append(kotlincFlags, "-no-jdk") - } - if len(kotlincFlags) > 0 { - // optimization. - ctx.Variable(pctx, "kotlincFlags", strings.Join(kotlincFlags, " ")) - flags.kotlincFlags += "$kotlincFlags" - } - - var kotlinSrcFiles android.Paths - kotlinSrcFiles = append(kotlinSrcFiles, uniqueSrcFiles...) - kotlinSrcFiles = append(kotlinSrcFiles, srcFiles.FilterByExt(".kt")...) - - // Collect .kt files for AIDEGen - j.expandIDEInfoCompiledSrcs = append(j.expandIDEInfoCompiledSrcs, srcFiles.FilterByExt(".kt").Strings()...) - j.expandIDEInfoCompiledSrcs = append(j.expandIDEInfoCompiledSrcs, kotlinCommonSrcFiles.Strings()...) - - flags.classpath = append(flags.classpath, deps.kotlinStdlib...) - flags.classpath = append(flags.classpath, deps.kotlinAnnotations...) - - flags.kotlincClasspath = append(flags.kotlincClasspath, flags.bootClasspath...) - flags.kotlincClasspath = append(flags.kotlincClasspath, flags.classpath...) - - if len(flags.processorPath) > 0 { - // Use kapt for annotation processing - kaptSrcJar := android.PathForModuleOut(ctx, "kapt", "kapt-sources.jar") - kaptResJar := android.PathForModuleOut(ctx, "kapt", "kapt-res.jar") - kotlinKapt(ctx, kaptSrcJar, kaptResJar, kotlinSrcFiles, kotlinCommonSrcFiles, srcJars, flags) - srcJars = append(srcJars, kaptSrcJar) - kotlinJars = append(kotlinJars, kaptResJar) - // Disable annotation processing in javac, it's already been handled by kapt - flags.processorPath = nil - flags.processors = nil - } - - kotlinJar := android.PathForModuleOut(ctx, "kotlin", jarName) - kotlinCompile(ctx, kotlinJar, kotlinSrcFiles, kotlinCommonSrcFiles, srcJars, flags) - if ctx.Failed() { - return - } - - // Make javac rule depend on the kotlinc rule - flags.classpath = append(flags.classpath, kotlinJar) - - kotlinJars = append(kotlinJars, kotlinJar) - // Jar kotlin classes into the final jar after javac - if BoolDefault(j.properties.Static_kotlin_stdlib, true) { - kotlinJars = append(kotlinJars, deps.kotlinStdlib...) - } - } - - jars := append(android.Paths(nil), kotlinJars...) - - // Store the list of .java files that was passed to javac - j.compiledJavaSrcs = uniqueSrcFiles - j.compiledSrcJars = srcJars - - enableSharding := false - var headerJarFileWithoutJarjar android.Path - if ctx.Device() && !ctx.Config().IsEnvFalse("TURBINE_ENABLED") && !deps.disableTurbine { - if j.properties.Javac_shard_size != nil && *(j.properties.Javac_shard_size) > 0 { - enableSharding = true - // Formerly, there was a check here that prevented annotation processors - // from being used when sharding was enabled, as some annotation processors - // do not function correctly in sharded environments. It was removed to - // allow for the use of annotation processors that do function correctly - // with sharding enabled. See: b/77284273. - } - headerJarFileWithoutJarjar, j.headerJarFile = - j.compileJavaHeader(ctx, uniqueSrcFiles, srcJars, deps, flags, jarName, kotlinJars) - if ctx.Failed() { - return - } - } - if len(uniqueSrcFiles) > 0 || len(srcJars) > 0 { - var extraJarDeps android.Paths - if ctx.Config().RunErrorProne() { - // If error-prone is enabled, add an additional rule to compile the java files into - // a separate set of classes (so that they don't overwrite the normal ones and require - // a rebuild when error-prone is turned off). - // TODO(ccross): Once we always compile with javac9 we may be able to conditionally - // enable error-prone without affecting the output class files. - errorprone := android.PathForModuleOut(ctx, "errorprone", jarName) - RunErrorProne(ctx, errorprone, uniqueSrcFiles, srcJars, flags) - extraJarDeps = append(extraJarDeps, errorprone) - } - - if enableSharding { - flags.classpath = append(flags.classpath, headerJarFileWithoutJarjar) - shardSize := int(*(j.properties.Javac_shard_size)) - var shardSrcs []android.Paths - if len(uniqueSrcFiles) > 0 { - shardSrcs = android.ShardPaths(uniqueSrcFiles, shardSize) - for idx, shardSrc := range shardSrcs { - classes := j.compileJavaClasses(ctx, jarName, idx, shardSrc, - nil, flags, extraJarDeps) - jars = append(jars, classes) - } - } - if len(srcJars) > 0 { - classes := j.compileJavaClasses(ctx, jarName, len(shardSrcs), - nil, srcJars, flags, extraJarDeps) - jars = append(jars, classes) - } - } else { - classes := j.compileJavaClasses(ctx, jarName, -1, uniqueSrcFiles, srcJars, flags, extraJarDeps) - jars = append(jars, classes) - } - if ctx.Failed() { - return - } - } - - j.srcJarArgs, j.srcJarDeps = resourcePathsToJarArgs(srcFiles), srcFiles - - var includeSrcJar android.WritablePath - if Bool(j.properties.Include_srcs) { - includeSrcJar = android.PathForModuleOut(ctx, ctx.ModuleName()+".srcjar") - TransformResourcesToJar(ctx, includeSrcJar, j.srcJarArgs, j.srcJarDeps) - } - - dirArgs, dirDeps := ResourceDirsToJarArgs(ctx, j.properties.Java_resource_dirs, - j.properties.Exclude_java_resource_dirs, j.properties.Exclude_java_resources) - fileArgs, fileDeps := ResourceFilesToJarArgs(ctx, j.properties.Java_resources, j.properties.Exclude_java_resources) - extraArgs, extraDeps := resourcePathsToJarArgs(j.extraResources), j.extraResources - - var resArgs []string - var resDeps android.Paths - - resArgs = append(resArgs, dirArgs...) - resDeps = append(resDeps, dirDeps...) - - resArgs = append(resArgs, fileArgs...) - resDeps = append(resDeps, fileDeps...) - - resArgs = append(resArgs, extraArgs...) - resDeps = append(resDeps, extraDeps...) - - if len(resArgs) > 0 { - resourceJar := android.PathForModuleOut(ctx, "res", jarName) - TransformResourcesToJar(ctx, resourceJar, resArgs, resDeps) - j.resourceJar = resourceJar - if ctx.Failed() { - return - } - } - - var resourceJars android.Paths - if j.resourceJar != nil { - resourceJars = append(resourceJars, j.resourceJar) - } - if Bool(j.properties.Include_srcs) { - resourceJars = append(resourceJars, includeSrcJar) - } - resourceJars = append(resourceJars, deps.staticResourceJars...) - - if len(resourceJars) > 1 { - combinedJar := android.PathForModuleOut(ctx, "res-combined", jarName) - TransformJarsToJar(ctx, combinedJar, "for resources", resourceJars, android.OptionalPath{}, - false, nil, nil) - j.resourceJar = combinedJar - } else if len(resourceJars) == 1 { - j.resourceJar = resourceJars[0] - } - - if len(deps.staticJars) > 0 { - jars = append(jars, deps.staticJars...) - } - - manifest := j.overrideManifest - if !manifest.Valid() && j.properties.Manifest != nil { - manifest = android.OptionalPathForPath(android.PathForModuleSrc(ctx, *j.properties.Manifest)) - } - - services := android.PathsForModuleSrc(ctx, j.properties.Services) - if len(services) > 0 { - servicesJar := android.PathForModuleOut(ctx, "services", jarName) - var zipargs []string - for _, file := range services { - serviceFile := file.String() - zipargs = append(zipargs, "-C", filepath.Dir(serviceFile), "-f", serviceFile) - } - rule := zip - args := map[string]string{ - "jarArgs": "-P META-INF/services/ " + strings.Join(proptools.NinjaAndShellEscapeList(zipargs), " "), - } - if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_ZIP") { - rule = zipRE - args["implicits"] = strings.Join(services.Strings(), ",") - } - ctx.Build(pctx, android.BuildParams{ - Rule: rule, - Output: servicesJar, - Implicits: services, - Args: args, - }) - jars = append(jars, servicesJar) - } - - // Combine the classes built from sources, any manifests, and any static libraries into - // classes.jar. If there is only one input jar this step will be skipped. - var outputFile android.ModuleOutPath - - if len(jars) == 1 && !manifest.Valid() { - if moduleOutPath, ok := jars[0].(android.ModuleOutPath); ok { - // Optimization: skip the combine step if there is nothing to do - // TODO(ccross): this leaves any module-info.class files, but those should only come from - // prebuilt dependencies until we support modules in the platform build, so there shouldn't be - // any if len(jars) == 1. - outputFile = moduleOutPath - } else { - combinedJar := android.PathForModuleOut(ctx, "combined", jarName) - ctx.Build(pctx, android.BuildParams{ - Rule: android.Cp, - Input: jars[0], - Output: combinedJar, - }) - outputFile = combinedJar - } - } else { - combinedJar := android.PathForModuleOut(ctx, "combined", jarName) - TransformJarsToJar(ctx, combinedJar, "for javac", jars, manifest, - false, nil, nil) - outputFile = combinedJar - } - - // jarjar implementation jar if necessary - if j.expandJarjarRules != nil { - // Transform classes.jar into classes-jarjar.jar - jarjarFile := android.PathForModuleOut(ctx, "jarjar", jarName) - TransformJarJar(ctx, jarjarFile, outputFile, j.expandJarjarRules) - outputFile = jarjarFile - - // jarjar resource jar if necessary - if j.resourceJar != nil { - resourceJarJarFile := android.PathForModuleOut(ctx, "res-jarjar", jarName) - TransformJarJar(ctx, resourceJarJarFile, j.resourceJar, j.expandJarjarRules) - j.resourceJar = resourceJarJarFile - } - - if ctx.Failed() { - return - } - } - - // Check package restrictions if necessary. - if len(j.properties.Permitted_packages) > 0 { - // Check packages and copy to package-checked file. - pkgckFile := android.PathForModuleOut(ctx, "package-check.stamp") - CheckJarPackages(ctx, pkgckFile, outputFile, j.properties.Permitted_packages) - j.additionalCheckedModules = append(j.additionalCheckedModules, pkgckFile) - - if ctx.Failed() { - return - } - } - - j.implementationJarFile = outputFile - if j.headerJarFile == nil { - j.headerJarFile = j.implementationJarFile - } - - if j.shouldInstrumentInApex(ctx) { - j.properties.Instrument = true - } - - if j.shouldInstrument(ctx) { - outputFile = j.instrument(ctx, flags, outputFile, jarName) - } - - // merge implementation jar with resources if necessary - implementationAndResourcesJar := outputFile - if j.resourceJar != nil { - jars := android.Paths{j.resourceJar, implementationAndResourcesJar} - combinedJar := android.PathForModuleOut(ctx, "withres", jarName) - TransformJarsToJar(ctx, combinedJar, "for resources", jars, manifest, - false, nil, nil) - implementationAndResourcesJar = combinedJar - } - - j.implementationAndResourcesJar = implementationAndResourcesJar - - // Enable dex compilation for the APEX variants, unless it is disabled explicitly - apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) - if j.DirectlyInAnyApex() && !apexInfo.IsForPlatform() { - if j.dexProperties.Compile_dex == nil { - j.dexProperties.Compile_dex = proptools.BoolPtr(true) - } - if j.deviceProperties.Hostdex == nil { - j.deviceProperties.Hostdex = proptools.BoolPtr(true) - } - } - - if ctx.Device() && j.hasCode(ctx) && - (Bool(j.properties.Installable) || Bool(j.dexProperties.Compile_dex)) { - if j.shouldInstrumentStatic(ctx) { - j.dexer.extraProguardFlagFiles = append(j.dexer.extraProguardFlagFiles, - android.PathForSource(ctx, "build/make/core/proguard.jacoco.flags")) - } - // Dex compilation - var dexOutputFile android.ModuleOutPath - dexOutputFile = j.dexer.compileDex(ctx, flags, j.minSdkVersion(), outputFile, jarName) - if ctx.Failed() { - return - } - - configurationName := j.ConfigurationName() - primary := configurationName == ctx.ModuleName() - // If the prebuilt is being used rather than the from source, skip this - // module to prevent duplicated classes - primary = primary && !j.IsReplacedByPrebuilt() - - // Hidden API CSV generation and dex encoding - dexOutputFile = j.hiddenAPI.hiddenAPI(ctx, configurationName, primary, dexOutputFile, j.implementationJarFile, - proptools.Bool(j.dexProperties.Uncompress_dex)) - - // merge dex jar with resources if necessary - if j.resourceJar != nil { - jars := android.Paths{dexOutputFile, j.resourceJar} - combinedJar := android.PathForModuleOut(ctx, "dex-withres", jarName) - TransformJarsToJar(ctx, combinedJar, "for dex resources", jars, android.OptionalPath{}, - false, nil, nil) - if *j.dexProperties.Uncompress_dex { - combinedAlignedJar := android.PathForModuleOut(ctx, "dex-withres-aligned", jarName) - TransformZipAlign(ctx, combinedAlignedJar, combinedJar) - dexOutputFile = combinedAlignedJar - } else { - dexOutputFile = combinedJar - } - } - - j.dexJarFile = dexOutputFile - - // Dexpreopting - j.dexpreopt(ctx, dexOutputFile) - - j.maybeStrippedDexJarFile = dexOutputFile - - outputFile = dexOutputFile - - if ctx.Failed() { - return - } - } else { - outputFile = implementationAndResourcesJar - } - - if ctx.Device() { - lintSDKVersionString := func(sdkSpec sdkSpec) string { - if v := sdkSpec.version; v.isNumbered() { - return v.String() - } else { - return ctx.Config().DefaultAppTargetSdk(ctx).String() - } - } - - j.linter.name = ctx.ModuleName() - j.linter.srcs = srcFiles - j.linter.srcJars = srcJars - j.linter.classpath = append(append(android.Paths(nil), flags.bootClasspath...), flags.classpath...) - j.linter.classes = j.implementationJarFile - j.linter.minSdkVersion = lintSDKVersionString(j.minSdkVersion()) - j.linter.targetSdkVersion = lintSDKVersionString(j.targetSdkVersion()) - j.linter.compileSdkVersion = lintSDKVersionString(j.sdkVersion()) - j.linter.javaLanguageLevel = flags.javaVersion.String() - j.linter.kotlinLanguageLevel = "1.3" - if !apexInfo.IsForPlatform() && ctx.Config().UnbundledBuildApps() { - j.linter.buildModuleReportZip = true - } - j.linter.lint(ctx) - } - - ctx.CheckbuildFile(outputFile) - - // Save the output file with no relative path so that it doesn't end up in a subdirectory when used as a resource - j.outputFile = outputFile.WithoutRel() -} - -func (j *Module) compileJavaClasses(ctx android.ModuleContext, jarName string, idx int, - srcFiles, srcJars android.Paths, flags javaBuilderFlags, extraJarDeps android.Paths) android.WritablePath { - - kzipName := pathtools.ReplaceExtension(jarName, "kzip") - if idx >= 0 { - kzipName = strings.TrimSuffix(jarName, filepath.Ext(jarName)) + strconv.Itoa(idx) + ".kzip" - jarName += strconv.Itoa(idx) - } - - classes := android.PathForModuleOut(ctx, "javac", jarName) - TransformJavaToClasses(ctx, classes, idx, srcFiles, srcJars, flags, extraJarDeps) - - if ctx.Config().EmitXrefRules() { - extractionFile := android.PathForModuleOut(ctx, kzipName) - emitXrefRule(ctx, extractionFile, idx, srcFiles, srcJars, flags, extraJarDeps) - j.kytheFiles = append(j.kytheFiles, extractionFile) - } - - return classes -} - -// Check for invalid kotlinc flags. Only use this for flags explicitly passed by the user, -// since some of these flags may be used internally. -func CheckKotlincFlags(ctx android.ModuleContext, flags []string) { - for _, flag := range flags { - flag = strings.TrimSpace(flag) - - if !strings.HasPrefix(flag, "-") { - ctx.PropertyErrorf("kotlincflags", "Flag `%s` must start with `-`", flag) - } else if strings.HasPrefix(flag, "-Xintellij-plugin-root") { - ctx.PropertyErrorf("kotlincflags", - "Bad flag: `%s`, only use internal compiler for consistency.", flag) - } else if inList(flag, config.KotlincIllegalFlags) { - ctx.PropertyErrorf("kotlincflags", "Flag `%s` already used by build system", flag) - } else if flag == "-include-runtime" { - ctx.PropertyErrorf("kotlincflags", "Bad flag: `%s`, do not include runtime.", flag) - } else { - args := strings.Split(flag, " ") - if args[0] == "-kotlin-home" { - ctx.PropertyErrorf("kotlincflags", - "Bad flag: `%s`, kotlin home already set to default (path to kotlinc in the repo).", flag) - } - } - } -} - -func (j *Module) compileJavaHeader(ctx android.ModuleContext, srcFiles, srcJars android.Paths, - deps deps, flags javaBuilderFlags, jarName string, - extraJars android.Paths) (headerJar, jarjarHeaderJar android.Path) { - - var jars android.Paths - if len(srcFiles) > 0 || len(srcJars) > 0 { - // Compile java sources into turbine.jar. - turbineJar := android.PathForModuleOut(ctx, "turbine", jarName) - TransformJavaToHeaderClasses(ctx, turbineJar, srcFiles, srcJars, flags) - if ctx.Failed() { - return nil, nil - } - jars = append(jars, turbineJar) - } - - jars = append(jars, extraJars...) - - // Combine any static header libraries into classes-header.jar. If there is only - // one input jar this step will be skipped. - jars = append(jars, deps.staticHeaderJars...) - - // we cannot skip the combine step for now if there is only one jar - // since we have to strip META-INF/TRANSITIVE dir from turbine.jar - combinedJar := android.PathForModuleOut(ctx, "turbine-combined", jarName) - TransformJarsToJar(ctx, combinedJar, "for turbine", jars, android.OptionalPath{}, - false, nil, []string{"META-INF/TRANSITIVE"}) - headerJar = combinedJar - jarjarHeaderJar = combinedJar - - if j.expandJarjarRules != nil { - // Transform classes.jar into classes-jarjar.jar - jarjarFile := android.PathForModuleOut(ctx, "turbine-jarjar", jarName) - TransformJarJar(ctx, jarjarFile, headerJar, j.expandJarjarRules) - jarjarHeaderJar = jarjarFile - if ctx.Failed() { - return nil, nil - } - } - - return headerJar, jarjarHeaderJar -} - -func (j *Module) instrument(ctx android.ModuleContext, flags javaBuilderFlags, - classesJar android.Path, jarName string) android.ModuleOutPath { - - specs := j.jacocoModuleToZipCommand(ctx) - - jacocoReportClassesFile := android.PathForModuleOut(ctx, "jacoco-report-classes", jarName) - instrumentedJar := android.PathForModuleOut(ctx, "jacoco", jarName) - - jacocoInstrumentJar(ctx, instrumentedJar, jacocoReportClassesFile, classesJar, specs) - - j.jacocoReportClassesFile = jacocoReportClassesFile - - return instrumentedJar -} - -var _ Dependency = (*Module)(nil) - -func (j *Module) HeaderJars() android.Paths { - if j.headerJarFile == nil { - return nil - } - return android.Paths{j.headerJarFile} -} - -func (j *Module) ImplementationJars() android.Paths { - if j.implementationJarFile == nil { - return nil - } - return android.Paths{j.implementationJarFile} -} - -func (j *Module) DexJarBuildPath() android.Path { - return j.dexJarFile -} - -func (j *Module) DexJarInstallPath() android.Path { - return j.installFile -} - -func (j *Module) ResourceJars() android.Paths { - if j.resourceJar == nil { - return nil - } - return android.Paths{j.resourceJar} -} - -func (j *Module) ImplementationAndResourcesJars() android.Paths { - if j.implementationAndResourcesJar == nil { - return nil - } - return android.Paths{j.implementationAndResourcesJar} -} - -func (j *Module) AidlIncludeDirs() android.Paths { - // exportAidlIncludeDirs is type android.Paths already - return j.exportAidlIncludeDirs -} - -func (j *Module) ClassLoaderContexts() dexpreopt.ClassLoaderContextMap { - return j.classLoaderContexts -} - -// ExportedPlugins returns the list of jars needed to run the exported plugins, the list of -// classes for the plugins, and a boolean for whether turbine needs to be disabled due to plugins -// that generate APIs. -func (j *Module) ExportedPlugins() (android.Paths, []string, bool) { - return j.exportedPluginJars, j.exportedPluginClasses, j.exportedDisableTurbine -} - -func (j *Module) SrcJarArgs() ([]string, android.Paths) { - return j.srcJarArgs, j.srcJarDeps -} - -var _ logtagsProducer = (*Module)(nil) - -func (j *Module) logtags() android.Paths { - return j.logtagsSrcs -} - -// Collect information for opening IDE project files in java/jdeps.go. -func (j *Module) IDEInfo(dpInfo *android.IdeInfo) { - dpInfo.Deps = append(dpInfo.Deps, j.CompilerDeps()...) - dpInfo.Srcs = append(dpInfo.Srcs, j.expandIDEInfoCompiledSrcs...) - dpInfo.SrcJars = append(dpInfo.SrcJars, j.compiledSrcJars.Strings()...) - dpInfo.Aidl_include_dirs = append(dpInfo.Aidl_include_dirs, j.deviceProperties.Aidl.Include_dirs...) - if j.expandJarjarRules != nil { - dpInfo.Jarjar_rules = append(dpInfo.Jarjar_rules, j.expandJarjarRules.String()) - } - dpInfo.Paths = append(dpInfo.Paths, j.modulePaths...) -} - -func (j *Module) CompilerDeps() []string { - jdeps := []string{} - jdeps = append(jdeps, j.properties.Libs...) - jdeps = append(jdeps, j.properties.Static_libs...) - return jdeps -} - -func (j *Module) hasCode(ctx android.ModuleContext) bool { - srcFiles := android.PathsForModuleSrcExcludes(ctx, j.properties.Srcs, j.properties.Exclude_srcs) - return len(srcFiles) > 0 || len(ctx.GetDirectDepsWithTag(staticLibTag)) > 0 -} - -// Implements android.ApexModule -func (j *Module) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool { - return j.depIsInSameApex(ctx, dep) -} - -// Implements android.ApexModule -func (j *Module) ShouldSupportSdkVersion(ctx android.BaseModuleContext, - sdkVersion android.ApiLevel) error { - sdkSpec := j.minSdkVersion() - if !sdkSpec.specified() { - return fmt.Errorf("min_sdk_version is not specified") - } - if sdkSpec.kind == sdkCore { - return nil - } - ver, err := sdkSpec.effectiveVersion(ctx) - if err != nil { - return err - } - if ver.ApiLevel(ctx).GreaterThan(sdkVersion) { - return fmt.Errorf("newer SDK(%v)", ver) - } - return nil -} - -func (j *Module) Stem() string { - return proptools.StringDefault(j.deviceProperties.Stem, j.Name()) -} - -func (j *Module) ConfigurationName() string { - return proptools.StringDefault(j.deviceProperties.ConfigurationName, j.BaseModuleName()) -} - -func (j *Module) JacocoReportClassesFile() android.Path { - return j.jacocoReportClassesFile -} - -func (j *Module) IsInstallable() bool { - return Bool(j.properties.Installable) -} - // // Java libraries (.jar file) // @@ -2146,6 +459,14 @@ func shouldUncompressDex(ctx android.ModuleContext, dexpreopter *dexpreopter) bo } func (j *Library) GenerateAndroidBuildActions(ctx android.ModuleContext) { + // Initialize the hiddenapi structure. Pass in the configuration name rather than the module name + // so the hidden api will encode the <x>.impl java_ library created by java_sdk_library just as it + // would the <x> library if <x> was configured as a boot jar. + j.initHiddenAPI(ctx, j.ConfigurationName()) + + j.sdkVersion = j.SdkVersion(ctx) + j.minSdkVersion = j.MinSdkVersion(ctx) + apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) if !apexInfo.IsForPlatform() { j.hideApexVariantFromMake = true @@ -2462,7 +783,16 @@ func (j *TestHost) DepsMutator(ctx android.BottomUpMutatorContext) { j.deps(ctx) } +func (j *TestHost) AddExtraResource(p android.Path) { + j.extraResources = append(j.extraResources, p) +} + func (j *Test) GenerateAndroidBuildActions(ctx android.ModuleContext) { + if j.testProperties.Test_options.Unit_test == nil && ctx.Host() { + // TODO(b/): Clean temporary heuristic to avoid unexpected onboarding. + defaultUnitTest := !inList("tradefed", j.properties.Libs) && !inList("cts", j.testProperties.Test_suites) + j.testProperties.Test_options.Unit_test = proptools.BoolPtr(defaultUnitTest) + } j.testConfig = tradefed.AutoGenJavaTestConfig(ctx, j.testProperties.Test_config, j.testProperties.Test_config_template, j.testProperties.Test_suites, j.testProperties.Auto_gen_config, j.testProperties.Test_options.Unit_test) @@ -2643,12 +973,23 @@ func TestHostFactory() android.Module { module.AddProperties(&module.testProperties) module.AddProperties(&module.testHostProperties) - module.Module.properties.Installable = proptools.BoolPtr(true) + InitTestHost( + module, + proptools.BoolPtr(true), + nil, + nil) InitJavaModuleMultiTargets(module, android.HostSupported) + return module } +func InitTestHost(th *TestHost, installable *bool, testSuites []string, autoGenConfig *bool) { + th.properties.Installable = installable + th.testProperties.Auto_gen_config = autoGenConfig + th.testProperties.Test_suites = testSuites +} + // // Java Binaries (.jar file plus wrapper script) // @@ -2712,9 +1053,10 @@ func (j *Binary) GenerateAndroidBuildActions(ctx android.ModuleContext) { } func (j *Binary) DepsMutator(ctx android.BottomUpMutatorContext) { - if ctx.Arch().ArchType == android.Common { + if ctx.Arch().ArchType == android.Common || ctx.BazelConversionMode() { j.deps(ctx) - } else { + } + if ctx.Arch().ArchType != android.Common || ctx.BazelConversionMode() { // These dependencies ensure the host installation rules will install the jar file and // the jni libraries when the wrapper is installed. ctx.AddVariationDependencies(nil, jniInstallTag, j.binaryProperties.Jni_libs...) @@ -2769,8 +1111,14 @@ func BinaryHostFactory() android.Module { type ImportProperties struct { Jars []string `android:"path,arch_variant"` + // The version of the SDK that the source prebuilt file was built against. Defaults to the + // current version if not specified. Sdk_version *string + // The minimum version of the SDK that this module supports. Defaults to sdk_version if not + // specified. + Min_sdk_version *string + Installable *bool // List of shared java libs that this module has dependencies to @@ -2819,30 +1167,28 @@ type Import struct { exportAidlIncludeDirs android.Paths hideApexVariantFromMake bool -} -func (j *Import) sdkVersion() sdkSpec { - return sdkSpecFrom(String(j.properties.Sdk_version)) + sdkVersion android.SdkSpec + minSdkVersion android.SdkSpec } -func (j *Import) makeSdkVersion() string { - return j.sdkVersion().raw +func (j *Import) SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec { + return android.SdkSpecFrom(ctx, String(j.properties.Sdk_version)) } -func (j *Import) systemModules() string { +func (j *Import) SystemModules() string { return "none" } -func (j *Import) minSdkVersion() sdkSpec { - return j.sdkVersion() -} - -func (j *Import) targetSdkVersion() sdkSpec { - return j.sdkVersion() +func (j *Import) MinSdkVersion(ctx android.EarlyModuleContext) android.SdkSpec { + if j.properties.Min_sdk_version != nil { + return android.SdkSpecFrom(ctx, *j.properties.Min_sdk_version) + } + return j.SdkVersion(ctx) } -func (j *Import) MinSdkVersion() string { - return j.minSdkVersion().version.String() +func (j *Import) TargetSdkVersion(ctx android.EarlyModuleContext) android.SdkSpec { + return j.SdkVersion(ctx) } func (j *Import) Prebuilt() *android.Prebuilt { @@ -2873,11 +1219,17 @@ func (j *Import) DepsMutator(ctx android.BottomUpMutatorContext) { ctx.AddVariationDependencies(nil, libTag, j.properties.Libs...) if ctx.Device() && Bool(j.dexProperties.Compile_dex) { - sdkDeps(ctx, sdkContext(j), j.dexer) + sdkDeps(ctx, android.SdkContext(j), j.dexer) } } func (j *Import) GenerateAndroidBuildActions(ctx android.ModuleContext) { + j.sdkVersion = j.SdkVersion(ctx) + j.minSdkVersion = j.MinSdkVersion(ctx) + + // Initialize the hiddenapi structure. + j.initHiddenAPI(ctx, j.BaseModuleName()) + if !ctx.Provider(android.ApexInfoProvider).(android.ApexInfo).IsForPlatform() { j.hideApexVariantFromMake = true } @@ -2902,18 +1254,18 @@ func (j *Import) GenerateAndroidBuildActions(ctx android.ModuleContext) { ctx.VisitDirectDeps(func(module android.Module) { tag := ctx.OtherModuleDependencyTag(module) - switch dep := module.(type) { - case Dependency: + if ctx.OtherModuleHasProvider(module, JavaInfoProvider) { + dep := ctx.OtherModuleProvider(module, JavaInfoProvider).(JavaInfo) switch tag { case libTag, staticLibTag: - flags.classpath = append(flags.classpath, dep.HeaderJars()...) + flags.classpath = append(flags.classpath, dep.HeaderJars...) case bootClasspathTag: - flags.bootClasspath = append(flags.bootClasspath, dep.HeaderJars()...) + flags.bootClasspath = append(flags.bootClasspath, dep.HeaderJars...) } - case SdkLibraryDependency: + } else if dep, ok := module.(SdkLibraryDependency); ok { switch tag { case libTag: - flags.classpath = append(flags.classpath, dep.SdkHeaderJars(ctx, j.sdkVersion())...) + flags.classpath = append(flags.classpath, dep.SdkHeaderJars(ctx, j.SdkVersion(ctx))...) } } @@ -2946,14 +1298,16 @@ func (j *Import) GenerateAndroidBuildActions(ctx android.ModuleContext) { // Get the path of the dex implementation jar from the `deapexer` module. di := ctx.OtherModuleProvider(deapexerModule, android.DeapexerProvider).(android.DeapexerInfo) - j.dexJarFile = di.PrebuiltExportPath(j.BaseModuleName(), ".dexjar") - if j.dexJarFile == nil { + if dexOutputPath := di.PrebuiltExportPath(j.BaseModuleName(), ".dexjar"); dexOutputPath != nil { + j.dexJarFile = dexOutputPath + j.hiddenAPIExtractInformation(ctx, dexOutputPath, outputFile) + } else { // This should never happen as a variant for a prebuilt_apex is only created if the // prebuilt_apex has been configured to export the java library dex file. ctx.ModuleErrorf("internal error: no dex implementation jar available from prebuilt_apex %q", deapexerModule.Name()) } } else if Bool(j.dexProperties.Compile_dex) { - sdkDep := decodeSdkDep(ctx, sdkContext(j)) + sdkDep := decodeSdkDep(ctx, android.SdkContext(j)) if sdkDep.invalidVersion { ctx.AddMissingDependencies(sdkDep.bootclasspath) ctx.AddMissingDependencies(sdkDep.java9Classpath) @@ -2971,22 +1325,26 @@ func (j *Import) GenerateAndroidBuildActions(ctx android.ModuleContext) { } j.dexpreopter.uncompressedDex = *j.dexProperties.Uncompress_dex - var dexOutputFile android.ModuleOutPath - dexOutputFile = j.dexer.compileDex(ctx, flags, j.minSdkVersion(), outputFile, jarName) + var dexOutputFile android.OutputPath + dexOutputFile = j.dexer.compileDex(ctx, flags, j.MinSdkVersion(ctx), outputFile, jarName) if ctx.Failed() { return } - configurationName := j.BaseModuleName() - primary := j.Prebuilt().UsePrebuilt() - // Hidden API CSV generation and dex encoding - dexOutputFile = j.hiddenAPI.hiddenAPI(ctx, configurationName, primary, dexOutputFile, outputFile, + dexOutputFile = j.hiddenAPIExtractAndEncode(ctx, dexOutputFile, outputFile, proptools.Bool(j.dexProperties.Uncompress_dex)) j.dexJarFile = dexOutputFile } } + + ctx.SetProvider(JavaInfoProvider, JavaInfo{ + HeaderJars: android.PathsIfNonNil(j.combinedClasspathFile), + ImplementationAndResourcesJars: android.PathsIfNonNil(j.combinedClasspathFile), + ImplementationJars: android.PathsIfNonNil(j.combinedClasspathFile), + AidlIncludeDirs: j.exportAidlIncludeDirs, + }) } func (j *Import) OutputFiles(tag string) (android.Paths, error) { @@ -3000,8 +1358,6 @@ func (j *Import) OutputFiles(tag string) (android.Paths, error) { var _ android.OutputFileProducer = (*Import)(nil) -var _ Dependency = (*Import)(nil) - func (j *Import) HeaderJars() android.Paths { if j.combinedClasspathFile == nil { return nil @@ -3009,17 +1365,6 @@ func (j *Import) HeaderJars() android.Paths { return android.Paths{j.combinedClasspathFile} } -func (j *Import) ImplementationJars() android.Paths { - if j.combinedClasspathFile == nil { - return nil - } - return android.Paths{j.combinedClasspathFile} -} - -func (j *Import) ResourceJars() android.Paths { - return nil -} - func (j *Import) ImplementationAndResourcesJars() android.Paths { if j.combinedClasspathFile == nil { return nil @@ -3035,22 +1380,10 @@ func (j *Import) DexJarInstallPath() android.Path { return nil } -func (j *Import) AidlIncludeDirs() android.Paths { - return j.exportAidlIncludeDirs -} - func (j *Import) ClassLoaderContexts() dexpreopt.ClassLoaderContextMap { return j.classLoaderContexts } -func (j *Import) ExportedPlugins() (android.Paths, []string, bool) { - return nil, nil, false -} - -func (j *Import) SrcJarArgs() ([]string, android.Paths) { - return nil, nil -} - var _ android.ApexModule = (*Import)(nil) // Implements android.ApexModule @@ -3061,7 +1394,20 @@ func (j *Import) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Modu // Implements android.ApexModule func (j *Import) ShouldSupportSdkVersion(ctx android.BaseModuleContext, sdkVersion android.ApiLevel) error { - // Do not check for prebuilts against the min_sdk_version of enclosing APEX + sdkSpec := j.MinSdkVersion(ctx) + if !sdkSpec.Specified() { + return fmt.Errorf("min_sdk_version is not specified") + } + if sdkSpec.Kind == android.SdkCore { + return nil + } + ver, err := sdkSpec.EffectiveVersion(ctx) + if err != nil { + return err + } + if ver.GreaterThan(sdkVersion) { + return fmt.Errorf("newer SDK(%v)", ver) + } return nil } @@ -3156,8 +1502,7 @@ type DexImport struct { properties DexImportProperties - dexJarFile android.Path - maybeStrippedDexJarFile android.Path + dexJarFile android.Path dexpreopter @@ -3244,8 +1589,6 @@ func (j *DexImport) GenerateAndroidBuildActions(ctx android.ModuleContext) { j.dexpreopt(ctx, dexOutputFile) - j.maybeStrippedDexJarFile = dexOutputFile - if apexInfo.IsForPlatform() { ctx.InstallFile(android.PathForModuleInstall(ctx, "framework"), j.Stem()+".jar", dexOutputFile) @@ -3320,16 +1663,12 @@ type Defaults struct { // ], // javacflags: ["-Xlint:all"], // } -func defaultsFactory() android.Module { - return DefaultsFactory() -} - func DefaultsFactory() android.Module { module := &Defaults{} module.AddProperties( - &CompilerProperties{}, - &CompilerDeviceProperties{}, + &CommonProperties{}, + &DeviceProperties{}, &DexProperties{}, &DexpreoptProperties{}, &android.ProtoProperties{}, @@ -3347,6 +1686,7 @@ func DefaultsFactory() android.Module { &android.ApexProperties{}, &RuntimeResourceOverlayProperties{}, &LintProperties{}, + &appTestHelperAppProperties{}, ) android.InitDefaultsModule(module) @@ -3378,16 +1718,6 @@ var BoolDefault = proptools.BoolDefault var String = proptools.String var inList = android.InList -// TODO(b/132357300) Generalize SdkLibrarComponentDependency to non-SDK libraries and merge with -// this interface. -type ProvidesUsesLib interface { - ProvidesUsesLib() *string -} - -func (j *Module) ProvidesUsesLib() *string { - return j.usesLibraryProperties.Provides_uses_lib -} - // Add class loader context (CLC) of a given dependency to the current CLC. func addCLCFromDep(ctx android.ModuleContext, depModule android.Module, clcMap dexpreopt.ClassLoaderContextMap) { |