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:
15
apex/apex.go
15
apex/apex.go
@@ -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",
|
||||||
|
@@ -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",
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -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
|
||||||
}
|
}
|
||||||
|
@@ -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
|
||||||
|
}
|
||||||
|
Reference in New Issue
Block a user