Install system server jar profiles into the APEX.

After this change, if profile-guided compilation is enabled for an APEX
system server jar, the profile will be installed next to the jar with
the ".prof" suffix, ("javalib/<name>.jar.prof"). This file will later be
used by odrefresh and dexpreopt from prebuilt APEX.

Bug: 241823638
Test: m nothing
Test: -
  1. Patch ag/20581649 PS2.
  2. banchan com.android.btservices x86_64 && m
  3. Check that "javalib/service-bluetooth.jar.prof" exists in the APEX.
Change-Id: Ibcfc6257dade92bd40a6d4b7368148717d0863b9
This commit is contained in:
Jiakai Zhang
2023-02-08 01:19:19 +08:00
parent 2411ffded8
commit 3317ce725d
4 changed files with 82 additions and 18 deletions

View File

@@ -1771,6 +1771,18 @@ func apexFileForJavaModuleWithFile(ctx android.BaseModuleContext, module javaMod
return af return af
} }
func apexFileForJavaModuleProfile(ctx android.BaseModuleContext, module javaModule) *apexFile {
if dexpreopter, ok := module.(java.DexpreopterInterface); ok {
if profilePathOnHost := dexpreopter.ProfilePathOnHost(); profilePathOnHost != nil {
dirInApex := "javalib"
af := newApexFile(ctx, profilePathOnHost, module.BaseModuleName()+"-profile", dirInApex, etc, nil)
af.customStem = module.Stem() + ".jar.prof"
return &af
}
}
return nil
}
// androidApp is an interface to handle all app modules (android_app, android_app_import, etc.) in // androidApp is an interface to handle all app modules (android_app, android_app_import, etc.) in
// the same way. // the same way.
type androidApp interface { type androidApp interface {
@@ -2465,6 +2477,9 @@ func (a *apexBundle) depVisitor(vctx *visitorContext, ctx android.ModuleContext,
case *java.Library, *java.SdkLibrary: case *java.Library, *java.SdkLibrary:
af := apexFileForJavaModule(ctx, child.(javaModule)) af := apexFileForJavaModule(ctx, child.(javaModule))
vctx.filesInfo = append(vctx.filesInfo, af) vctx.filesInfo = append(vctx.filesInfo, af)
if profileAf := apexFileForJavaModuleProfile(ctx, child.(javaModule)); profileAf != nil {
vctx.filesInfo = append(vctx.filesInfo, *profileAf)
}
return true // track transitive dependencies return true // track transitive dependencies
default: default:
ctx.PropertyErrorf("systemserverclasspath_fragments", ctx.PropertyErrorf("systemserverclasspath_fragments",

View File

@@ -31,7 +31,7 @@ func TestSystemserverclasspathFragmentContents(t *testing.T) {
result := android.GroupFixturePreparers( result := android.GroupFixturePreparers(
prepareForTestWithSystemserverclasspathFragment, prepareForTestWithSystemserverclasspathFragment,
prepareForTestWithMyapex, prepareForTestWithMyapex,
dexpreopt.FixtureSetApexSystemServerJars("myapex:foo"), dexpreopt.FixtureSetApexSystemServerJars("myapex:foo", "myapex:bar"),
).RunTestWithBp(t, ` ).RunTestWithBp(t, `
apex { apex {
name: "myapex", name: "myapex",
@@ -57,10 +57,23 @@ func TestSystemserverclasspathFragmentContents(t *testing.T) {
], ],
} }
java_library {
name: "bar",
srcs: ["c.java"],
installable: true,
dex_preopt: {
profile: "bar-art-profile",
},
apex_available: [
"myapex",
],
}
systemserverclasspath_fragment { systemserverclasspath_fragment {
name: "mysystemserverclasspathfragment", name: "mysystemserverclasspathfragment",
contents: [ contents: [
"foo", "foo",
"bar",
], ],
apex_available: [ apex_available: [
"myapex", "myapex",
@@ -71,6 +84,8 @@ func TestSystemserverclasspathFragmentContents(t *testing.T) {
ensureExactContents(t, result.TestContext, "myapex", "android_common_myapex_image", []string{ ensureExactContents(t, result.TestContext, "myapex", "android_common_myapex_image", []string{
"etc/classpaths/systemserverclasspath.pb", "etc/classpaths/systemserverclasspath.pb",
"javalib/foo.jar", "javalib/foo.jar",
"javalib/bar.jar",
"javalib/bar.jar.prof",
}) })
java.CheckModuleDependencies(t, result.TestContext, "myapex", "android_common_myapex_image", []string{ java.CheckModuleDependencies(t, result.TestContext, "myapex", "android_common_myapex_image", []string{
@@ -236,7 +251,7 @@ func TestSystemserverclasspathFragmentStandaloneContents(t *testing.T) {
result := android.GroupFixturePreparers( result := android.GroupFixturePreparers(
prepareForTestWithSystemserverclasspathFragment, prepareForTestWithSystemserverclasspathFragment,
prepareForTestWithMyapex, prepareForTestWithMyapex,
dexpreopt.FixtureSetApexStandaloneSystemServerJars("myapex:foo"), dexpreopt.FixtureSetApexStandaloneSystemServerJars("myapex:foo", "myapex:bar"),
).RunTestWithBp(t, ` ).RunTestWithBp(t, `
apex { apex {
name: "myapex", name: "myapex",
@@ -262,10 +277,23 @@ func TestSystemserverclasspathFragmentStandaloneContents(t *testing.T) {
], ],
} }
java_library {
name: "bar",
srcs: ["c.java"],
dex_preopt: {
profile: "bar-art-profile",
},
installable: true,
apex_available: [
"myapex",
],
}
systemserverclasspath_fragment { systemserverclasspath_fragment {
name: "mysystemserverclasspathfragment", name: "mysystemserverclasspathfragment",
standalone_contents: [ standalone_contents: [
"foo", "foo",
"bar",
], ],
apex_available: [ apex_available: [
"myapex", "myapex",
@@ -276,6 +304,8 @@ func TestSystemserverclasspathFragmentStandaloneContents(t *testing.T) {
ensureExactContents(t, result.TestContext, "myapex", "android_common_myapex_image", []string{ ensureExactContents(t, result.TestContext, "myapex", "android_common_myapex_image", []string{
"etc/classpaths/systemserverclasspath.pb", "etc/classpaths/systemserverclasspath.pb",
"javalib/foo.jar", "javalib/foo.jar",
"javalib/bar.jar",
"javalib/bar.jar.prof",
}) })
} }

View File

@@ -101,6 +101,10 @@ func GenerateDexpreoptRule(ctx android.BuilderContext, globalSoong *GlobalSoongC
} }
func dexpreoptDisabled(ctx android.PathContext, global *GlobalConfig, module *ModuleConfig) bool { func dexpreoptDisabled(ctx android.PathContext, global *GlobalConfig, module *ModuleConfig) bool {
if ctx.Config().UnbundledBuild() {
return true
}
if contains(global.DisablePreoptModules, module.Name) { if contains(global.DisablePreoptModules, module.Name) {
return true return true
} }

View File

@@ -27,6 +27,7 @@ type DexpreopterInterface interface {
dexpreoptDisabled(ctx android.BaseModuleContext) bool dexpreoptDisabled(ctx android.BaseModuleContext) bool
DexpreoptBuiltInstalledForApex() []dexpreopterInstall DexpreoptBuiltInstalledForApex() []dexpreopterInstall
AndroidMkEntriesForApex() []android.AndroidMkEntries AndroidMkEntriesForApex() []android.AndroidMkEntries
ProfilePathOnHost() android.Path
} }
type dexpreopterInstall struct { type dexpreopterInstall struct {
@@ -103,6 +104,9 @@ type dexpreopter struct {
// - Dexpreopt post-processing (using dexpreopt artifacts from a prebuilt system image to incrementally // - Dexpreopt post-processing (using dexpreopt artifacts from a prebuilt system image to incrementally
// dexpreopt another partition). // dexpreopt another partition).
configPath android.WritablePath configPath android.WritablePath
// The path to the profile on host.
profilePathOnHost android.Path
} }
type DexpreoptProperties struct { type DexpreoptProperties struct {
@@ -180,9 +184,8 @@ func (d *dexpreopter) dexpreoptDisabled(ctx android.BaseModuleContext) bool {
isApexSystemServerJar := global.AllApexSystemServerJars(ctx).ContainsJar(moduleName(ctx)) isApexSystemServerJar := global.AllApexSystemServerJars(ctx).ContainsJar(moduleName(ctx))
if isApexVariant(ctx) { if isApexVariant(ctx) {
// Don't preopt APEX variant module unless the module is an APEX system server jar and we are // Don't preopt APEX variant module unless the module is an APEX system server jar.
// building the entire system image. if !isApexSystemServerJar {
if !isApexSystemServerJar || ctx.Config().UnbundledBuild() {
return true return true
} }
} else { } else {
@@ -368,21 +371,29 @@ func (d *dexpreopter) dexpreopt(ctx android.ModuleContext, dexJarFile android.Wr
installBase := filepath.Base(install.To) installBase := filepath.Base(install.To)
arch := filepath.Base(installDir) arch := filepath.Base(installDir)
installPath := android.PathForModuleInPartitionInstall(ctx, "", installDir) installPath := android.PathForModuleInPartitionInstall(ctx, "", installDir)
isProfile := strings.HasSuffix(installBase, ".prof")
if isProfile {
d.profilePathOnHost = install.From
}
if isApexSystemServerJar { if isApexSystemServerJar {
// APEX variants of java libraries are hidden from Make, so their dexpreopt // Profiles are handled separately because they are installed into the APEX.
// outputs need special handling. Currently, for APEX variants of java if !isProfile {
// libraries, only those in the system server classpath are handled here. // APEX variants of java libraries are hidden from Make, so their dexpreopt
// Preopting of boot classpath jars in the ART APEX are handled in // outputs need special handling. Currently, for APEX variants of java
// java/dexpreopt_bootjars.go, and other APEX jars are not preopted. // libraries, only those in the system server classpath are handled here.
// The installs will be handled by Make as sub-modules of the java library. // Preopting of boot classpath jars in the ART APEX are handled in
d.builtInstalledForApex = append(d.builtInstalledForApex, dexpreopterInstall{ // java/dexpreopt_bootjars.go, and other APEX jars are not preopted.
name: arch + "-" + installBase, // The installs will be handled by Make as sub-modules of the java library.
moduleName: moduleName(ctx), d.builtInstalledForApex = append(d.builtInstalledForApex, dexpreopterInstall{
outputPathOnHost: install.From, name: arch + "-" + installBase,
installDirOnDevice: installPath, moduleName: moduleName(ctx),
installFileOnDevice: installBase, outputPathOnHost: install.From,
}) installDirOnDevice: installPath,
installFileOnDevice: installBase,
})
}
} else if !d.preventInstall { } else if !d.preventInstall {
ctx.InstallFile(installPath, installBase, install.From) ctx.InstallFile(installPath, installBase, install.From)
} }
@@ -404,3 +415,7 @@ func (d *dexpreopter) AndroidMkEntriesForApex() []android.AndroidMkEntries {
} }
return entries return entries
} }
func (d *dexpreopter) ProfilePathOnHost() android.Path {
return d.profilePathOnHost
}