Move sanitizer.libraries.txt logic into Soong

Currently sanitizer.libraries.txt module is defined from Makefile, while
all logics to create the list of modules is implmented within the Soong.
This change moves sanitizer.libraries.txt module definition into Soong,
so it can be generated without sharing list of modules over Make
variable.

Bug: 339131599
Test: AOSP CF build succeeded, with same list of modules in
/system/etc/sanitizer.libraries.txt

Change-Id: I987684877aa8dae221a03227d784f2a8ca4f5cc4
This commit is contained in:
Kiyoung Kim
2024-05-07 14:47:35 +09:00
parent c15b0234e9
commit 0d8908c2e4
5 changed files with 180 additions and 78 deletions

View File

@@ -209,58 +209,58 @@ func addPrefix(list []string, prefix string) []string {
return list return list
} }
func LibclangRuntimeLibrary(t Toolchain, library string) string { func LibclangRuntimeLibrary(library string) string {
return "libclang_rt." + library return "libclang_rt." + library
} }
func BuiltinsRuntimeLibrary(t Toolchain) string { func BuiltinsRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "builtins") return LibclangRuntimeLibrary("builtins")
} }
func AddressSanitizerRuntimeLibrary(t Toolchain) string { func AddressSanitizerRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "asan") return LibclangRuntimeLibrary("asan")
} }
func AddressSanitizerStaticRuntimeLibrary(t Toolchain) string { func AddressSanitizerStaticRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "asan.static") return LibclangRuntimeLibrary("asan.static")
} }
func AddressSanitizerCXXStaticRuntimeLibrary(t Toolchain) string { func AddressSanitizerCXXStaticRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "asan_cxx.static") return LibclangRuntimeLibrary("asan_cxx.static")
} }
func HWAddressSanitizerRuntimeLibrary(t Toolchain) string { func HWAddressSanitizerRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "hwasan") return LibclangRuntimeLibrary("hwasan")
} }
func HWAddressSanitizerStaticLibrary(t Toolchain) string { func HWAddressSanitizerStaticLibrary() string {
return LibclangRuntimeLibrary(t, "hwasan_static") return LibclangRuntimeLibrary("hwasan_static")
} }
func UndefinedBehaviorSanitizerRuntimeLibrary(t Toolchain) string { func UndefinedBehaviorSanitizerRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "ubsan_standalone") return LibclangRuntimeLibrary("ubsan_standalone")
} }
func UndefinedBehaviorSanitizerMinimalRuntimeLibrary(t Toolchain) string { func UndefinedBehaviorSanitizerMinimalRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "ubsan_minimal") return LibclangRuntimeLibrary("ubsan_minimal")
} }
func ThreadSanitizerRuntimeLibrary(t Toolchain) string { func ThreadSanitizerRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "tsan") return LibclangRuntimeLibrary("tsan")
} }
func ScudoRuntimeLibrary(t Toolchain) string { func ScudoRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "scudo") return LibclangRuntimeLibrary("scudo")
} }
func ScudoMinimalRuntimeLibrary(t Toolchain) string { func ScudoMinimalRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "scudo_minimal") return LibclangRuntimeLibrary("scudo_minimal")
} }
func LibFuzzerRuntimeLibrary(t Toolchain) string { func LibFuzzerRuntimeLibrary() string {
return LibclangRuntimeLibrary(t, "fuzzer") return LibclangRuntimeLibrary("fuzzer")
} }
func LibFuzzerRuntimeInterceptors(t Toolchain) string { func LibFuzzerRuntimeInterceptors() string {
return LibclangRuntimeLibrary(t, "fuzzer_interceptors") return LibclangRuntimeLibrary("fuzzer_interceptors")
} }

View File

@@ -128,13 +128,13 @@ func (fuzzBin *fuzzBinary) linkerDeps(ctx DepsContext, deps Deps) Deps {
if ctx.Config().Getenv("FUZZ_FRAMEWORK") == "AFL" { if ctx.Config().Getenv("FUZZ_FRAMEWORK") == "AFL" {
deps.HeaderLibs = append(deps.HeaderLibs, "libafl_headers") deps.HeaderLibs = append(deps.HeaderLibs, "libafl_headers")
} else { } else {
deps.StaticLibs = append(deps.StaticLibs, config.LibFuzzerRuntimeLibrary(ctx.toolchain())) deps.StaticLibs = append(deps.StaticLibs, config.LibFuzzerRuntimeLibrary())
// Fuzzers built with HWASAN should use the interceptors for better // Fuzzers built with HWASAN should use the interceptors for better
// mutation based on signals in strcmp, memcpy, etc. This is only needed for // mutation based on signals in strcmp, memcpy, etc. This is only needed for
// fuzz targets, not generic HWASAN-ified binaries or libraries. // fuzz targets, not generic HWASAN-ified binaries or libraries.
if module, ok := ctx.Module().(*Module); ok { if module, ok := ctx.Module().(*Module); ok {
if module.IsSanitizerEnabled(Hwasan) { if module.IsSanitizerEnabled(Hwasan) {
deps.StaticLibs = append(deps.StaticLibs, config.LibFuzzerRuntimeInterceptors(ctx.toolchain())) deps.StaticLibs = append(deps.StaticLibs, config.LibFuzzerRuntimeInterceptors())
} }
} }
} }

View File

@@ -431,7 +431,7 @@ func (linker *baseLinker) linkerDeps(ctx DepsContext, deps Deps) Deps {
if ctx.toolchain().Bionic() { if ctx.toolchain().Bionic() {
// libclang_rt.builtins has to be last on the command line // libclang_rt.builtins has to be last on the command line
if linker.Properties.libCrt() && !ctx.header() { if linker.Properties.libCrt() && !ctx.header() {
deps.UnexportedStaticLibs = append(deps.UnexportedStaticLibs, config.BuiltinsRuntimeLibrary(ctx.toolchain())) deps.UnexportedStaticLibs = append(deps.UnexportedStaticLibs, config.BuiltinsRuntimeLibrary())
} }
if inList("libdl", deps.SharedLibs) { if inList("libdl", deps.SharedLibs) {
@@ -454,7 +454,7 @@ func (linker *baseLinker) linkerDeps(ctx DepsContext, deps Deps) Deps {
} }
} else if ctx.toolchain().Musl() { } else if ctx.toolchain().Musl() {
if linker.Properties.libCrt() && !ctx.header() { if linker.Properties.libCrt() && !ctx.header() {
deps.UnexportedStaticLibs = append(deps.UnexportedStaticLibs, config.BuiltinsRuntimeLibrary(ctx.toolchain())) deps.UnexportedStaticLibs = append(deps.UnexportedStaticLibs, config.BuiltinsRuntimeLibrary())
} }
} }

View File

@@ -28,6 +28,16 @@ var (
modulesWarningsAllowedKey = android.NewOnceKey("ModulesWarningsAllowed") modulesWarningsAllowedKey = android.NewOnceKey("ModulesWarningsAllowed")
modulesUsingWnoErrorKey = android.NewOnceKey("ModulesUsingWnoError") modulesUsingWnoErrorKey = android.NewOnceKey("ModulesUsingWnoError")
modulesMissingProfileFileKey = android.NewOnceKey("ModulesMissingProfileFile") modulesMissingProfileFileKey = android.NewOnceKey("ModulesMissingProfileFile")
sanitizerVariables = map[string]string{
"ADDRESS_SANITIZER_RUNTIME_LIBRARY": config.AddressSanitizerRuntimeLibrary(),
"HWADDRESS_SANITIZER_RUNTIME_LIBRARY": config.HWAddressSanitizerRuntimeLibrary(),
"HWADDRESS_SANITIZER_STATIC_LIBRARY": config.HWAddressSanitizerStaticLibrary(),
"UBSAN_RUNTIME_LIBRARY": config.UndefinedBehaviorSanitizerRuntimeLibrary(),
"UBSAN_MINIMAL_RUNTIME_LIBRARY": config.UndefinedBehaviorSanitizerMinimalRuntimeLibrary(),
"TSAN_RUNTIME_LIBRARY": config.ThreadSanitizerRuntimeLibrary(),
"SCUDO_RUNTIME_LIBRARY": config.ScudoRuntimeLibrary(),
"SCUDO_MINIMAL_RUNTIME_LIBRARY": config.ScudoMinimalRuntimeLibrary(),
}
) )
func init() { func init() {
@@ -261,43 +271,9 @@ func makeVarsToolchain(ctx android.MakeVarsContext, secondPrefix string,
}, " ")) }, " "))
if target.Os.Class == android.Device { if target.Os.Class == android.Device {
sanitizerVariables := map[string]string{
"ADDRESS_SANITIZER_RUNTIME_LIBRARY": config.AddressSanitizerRuntimeLibrary(toolchain),
"HWADDRESS_SANITIZER_RUNTIME_LIBRARY": config.HWAddressSanitizerRuntimeLibrary(toolchain),
"HWADDRESS_SANITIZER_STATIC_LIBRARY": config.HWAddressSanitizerStaticLibrary(toolchain),
"UBSAN_RUNTIME_LIBRARY": config.UndefinedBehaviorSanitizerRuntimeLibrary(toolchain),
"UBSAN_MINIMAL_RUNTIME_LIBRARY": config.UndefinedBehaviorSanitizerMinimalRuntimeLibrary(toolchain),
"TSAN_RUNTIME_LIBRARY": config.ThreadSanitizerRuntimeLibrary(toolchain),
"SCUDO_RUNTIME_LIBRARY": config.ScudoRuntimeLibrary(toolchain),
"SCUDO_MINIMAL_RUNTIME_LIBRARY": config.ScudoMinimalRuntimeLibrary(toolchain),
}
for variable, value := range sanitizerVariables { for variable, value := range sanitizerVariables {
ctx.Strict(secondPrefix+variable, value) ctx.Strict(secondPrefix+variable, value)
} }
sanitizerLibs := android.SortedStringValues(sanitizerVariables)
var sanitizerLibStems []string
ctx.VisitAllModules(func(m android.Module) {
if !m.Enabled(ctx) {
return
}
ccModule, _ := m.(*Module)
if ccModule == nil || ccModule.library == nil || !ccModule.library.shared() {
return
}
if android.InList(strings.TrimPrefix(ctx.ModuleName(m), "prebuilt_"), sanitizerLibs) &&
m.Target().Os == target.Os && m.Target().Arch.ArchType == target.Arch.ArchType {
outputFile := ccModule.outputFile
if outputFile.Valid() {
sanitizerLibStems = append(sanitizerLibStems, outputFile.Path().Base())
}
}
})
sanitizerLibStems = android.SortedUniqueStrings(sanitizerLibStems)
ctx.Strict(secondPrefix+"SANITIZER_STEMS", strings.Join(sanitizerLibStems, " "))
} }
// This is used by external/gentoo/... // This is used by external/gentoo/...

View File

@@ -25,6 +25,7 @@ import (
"android/soong/android" "android/soong/android"
"android/soong/cc/config" "android/soong/cc/config"
"android/soong/etc"
) )
var ( var (
@@ -408,6 +409,8 @@ func init() {
android.RegisterMakeVarsProvider(pctx, cfiMakeVarsProvider) android.RegisterMakeVarsProvider(pctx, cfiMakeVarsProvider)
android.RegisterMakeVarsProvider(pctx, hwasanMakeVarsProvider) android.RegisterMakeVarsProvider(pctx, hwasanMakeVarsProvider)
android.RegisterMakeVarsProvider(pctx, memtagStackMakeVarsProvider) android.RegisterMakeVarsProvider(pctx, memtagStackMakeVarsProvider)
RegisterSanitizerLibrariesTxtType(android.InitRegistrationContext)
} }
func (sanitize *sanitize) props() []interface{} { func (sanitize *sanitize) props() []interface{} {
@@ -1316,7 +1319,7 @@ func (s *sanitizerSplitMutator) Mutate(mctx android.BottomUpMutatorContext, vari
} else if s.sanitizer == cfi { } else if s.sanitizer == cfi {
cfiStaticLibs(mctx.Config()).add(c, c.Module().Name()) cfiStaticLibs(mctx.Config()).add(c, c.Module().Name())
} else if s.sanitizer == Memtag_stack { } else if s.sanitizer == Memtag_stack {
memtagStackStaticLibs(mctx.Config()).add(c, c.Module().Name()); memtagStackStaticLibs(mctx.Config()).add(c, c.Module().Name())
} }
} }
} else if c.IsSanitizerEnabled(s.sanitizer) { } else if c.IsSanitizerEnabled(s.sanitizer) {
@@ -1522,25 +1525,25 @@ func sanitizerRuntimeMutator(mctx android.BottomUpMutatorContext) {
if Bool(sanProps.Address) { if Bool(sanProps.Address) {
if toolchain.Musl() || (c.staticBinary() && toolchain.Bionic()) { if toolchain.Musl() || (c.staticBinary() && toolchain.Bionic()) {
// Use a static runtime for musl to match what clang does for glibc. // Use a static runtime for musl to match what clang does for glibc.
addStaticDeps(config.AddressSanitizerStaticRuntimeLibrary(toolchain), false) addStaticDeps(config.AddressSanitizerStaticRuntimeLibrary(), false)
addStaticDeps(config.AddressSanitizerCXXStaticRuntimeLibrary(toolchain), false) addStaticDeps(config.AddressSanitizerCXXStaticRuntimeLibrary(), false)
} else { } else {
runtimeSharedLibrary = config.AddressSanitizerRuntimeLibrary(toolchain) runtimeSharedLibrary = config.AddressSanitizerRuntimeLibrary()
} }
} else if Bool(sanProps.Hwaddress) { } else if Bool(sanProps.Hwaddress) {
if c.staticBinary() { if c.staticBinary() {
addStaticDeps(config.HWAddressSanitizerStaticLibrary(toolchain), true) addStaticDeps(config.HWAddressSanitizerStaticLibrary(), true)
addStaticDeps("libdl", false) addStaticDeps("libdl", false)
} else { } else {
runtimeSharedLibrary = config.HWAddressSanitizerRuntimeLibrary(toolchain) runtimeSharedLibrary = config.HWAddressSanitizerRuntimeLibrary()
} }
} else if Bool(sanProps.Thread) { } else if Bool(sanProps.Thread) {
runtimeSharedLibrary = config.ThreadSanitizerRuntimeLibrary(toolchain) runtimeSharedLibrary = config.ThreadSanitizerRuntimeLibrary()
} else if Bool(sanProps.Scudo) { } else if Bool(sanProps.Scudo) {
if len(diagSanitizers) == 0 && !c.sanitize.Properties.UbsanRuntimeDep { if len(diagSanitizers) == 0 && !c.sanitize.Properties.UbsanRuntimeDep {
runtimeSharedLibrary = config.ScudoMinimalRuntimeLibrary(toolchain) runtimeSharedLibrary = config.ScudoMinimalRuntimeLibrary()
} else { } else {
runtimeSharedLibrary = config.ScudoRuntimeLibrary(toolchain) runtimeSharedLibrary = config.ScudoRuntimeLibrary()
} }
} else if len(diagSanitizers) > 0 || c.sanitize.Properties.UbsanRuntimeDep || } else if len(diagSanitizers) > 0 || c.sanitize.Properties.UbsanRuntimeDep ||
Bool(sanProps.Fuzzer) || Bool(sanProps.Fuzzer) ||
@@ -1553,17 +1556,17 @@ func sanitizerRuntimeMutator(mctx android.BottomUpMutatorContext) {
// Also manually add a static runtime for musl to match what clang does for glibc. // Also manually add a static runtime for musl to match what clang does for glibc.
// Otherwise dlopening libraries that depend on libclang_rt.ubsan_standalone.so fails with: // Otherwise dlopening libraries that depend on libclang_rt.ubsan_standalone.so fails with:
// Error relocating ...: initial-exec TLS resolves to dynamic definition // Error relocating ...: initial-exec TLS resolves to dynamic definition
addStaticDeps(config.UndefinedBehaviorSanitizerRuntimeLibrary(toolchain)+".static", true) addStaticDeps(config.UndefinedBehaviorSanitizerRuntimeLibrary()+".static", true)
} else { } else {
runtimeSharedLibrary = config.UndefinedBehaviorSanitizerRuntimeLibrary(toolchain) runtimeSharedLibrary = config.UndefinedBehaviorSanitizerRuntimeLibrary()
} }
} }
if enableMinimalRuntime(c.sanitize) || c.sanitize.Properties.MinimalRuntimeDep { if enableMinimalRuntime(c.sanitize) || c.sanitize.Properties.MinimalRuntimeDep {
addStaticDeps(config.UndefinedBehaviorSanitizerMinimalRuntimeLibrary(toolchain), true) addStaticDeps(config.UndefinedBehaviorSanitizerMinimalRuntimeLibrary(), true)
} }
if c.sanitize.Properties.BuiltinsDep { if c.sanitize.Properties.BuiltinsDep {
addStaticDeps(config.BuiltinsRuntimeLibrary(toolchain), true) addStaticDeps(config.BuiltinsRuntimeLibrary(), true)
} }
if runtimeSharedLibrary != "" && (toolchain.Bionic() || toolchain.Musl()) { if runtimeSharedLibrary != "" && (toolchain.Bionic() || toolchain.Musl()) {
@@ -1787,3 +1790,126 @@ func hwasanMakeVarsProvider(ctx android.MakeVarsContext) {
func memtagStackMakeVarsProvider(ctx android.MakeVarsContext) { func memtagStackMakeVarsProvider(ctx android.MakeVarsContext) {
memtagStackStaticLibs(ctx.Config()).exportToMake(ctx) memtagStackStaticLibs(ctx.Config()).exportToMake(ctx)
} }
type sanitizerLibrariesTxtModule struct {
android.ModuleBase
outputFile android.OutputPath
}
var _ etc.PrebuiltEtcModule = (*sanitizerLibrariesTxtModule)(nil)
var _ android.OutputFileProducer = (*sanitizerLibrariesTxtModule)(nil)
func RegisterSanitizerLibrariesTxtType(ctx android.RegistrationContext) {
ctx.RegisterModuleType("sanitizer_libraries_txt", sanitizerLibrariesTxtFactory)
}
func sanitizerLibrariesTxtFactory() android.Module {
m := &sanitizerLibrariesTxtModule{}
android.InitAndroidArchModule(m, android.DeviceSupported, android.MultilibCommon)
return m
}
type sanitizerLibraryDependencyTag struct {
blueprint.BaseDependencyTag
}
func (t sanitizerLibraryDependencyTag) AllowDisabledModuleDependency(target android.Module) bool {
return true
}
var _ android.AllowDisabledModuleDependency = (*sanitizerLibraryDependencyTag)(nil)
func (txt *sanitizerLibrariesTxtModule) DepsMutator(actx android.BottomUpMutatorContext) {
targets := actx.Config().Targets[android.Android]
depTag := sanitizerLibraryDependencyTag{}
for _, target := range targets {
variation := append(target.Variations(),
blueprint.Variation{Mutator: "image", Variation: ""},
blueprint.Variation{Mutator: "sdk", Variation: ""},
blueprint.Variation{Mutator: "link", Variation: "shared"},
)
for _, lib := range android.SortedStringValues(sanitizerVariables) {
if actx.OtherModuleFarDependencyVariantExists(variation, lib) {
actx.AddFarVariationDependencies(variation, depTag, lib)
}
prebuiltLibName := "prebuilt_" + lib
if actx.OtherModuleFarDependencyVariantExists(variation, prebuiltLibName) {
actx.AddFarVariationDependencies(variation, depTag, prebuiltLibName)
}
}
}
}
func (txt *sanitizerLibrariesTxtModule) getSanitizerLibs(ctx android.ModuleContext) string {
var sanitizerLibStems []string
ctx.VisitDirectDepsIf(func(m android.Module) bool {
if !m.Enabled(ctx) {
return false
}
ccModule, _ := m.(*Module)
if ccModule == nil || ccModule.library == nil || !ccModule.library.shared() {
return false
}
targets := ctx.Config().Targets[android.Android]
for _, target := range targets {
if m.Target().Os == target.Os && m.Target().Arch.ArchType == target.Arch.ArchType {
return true
}
}
return false
}, func(m android.Module) {
ccModule, _ := m.(*Module)
outputFile := ccModule.outputFile
if outputFile.Valid() {
sanitizerLibStems = append(sanitizerLibStems, outputFile.Path().Base())
}
})
sanitizerLibStems = android.SortedUniqueStrings(sanitizerLibStems)
return strings.Join(sanitizerLibStems, "\n")
}
func (txt *sanitizerLibrariesTxtModule) GenerateAndroidBuildActions(ctx android.ModuleContext) {
filename := txt.Name()
txt.outputFile = android.PathForModuleOut(ctx, filename).OutputPath
android.WriteFileRule(ctx, txt.outputFile, txt.getSanitizerLibs(ctx))
installPath := android.PathForModuleInstall(ctx, "etc")
ctx.InstallFile(installPath, filename, txt.outputFile)
}
func (txt *sanitizerLibrariesTxtModule) AndroidMkEntries() []android.AndroidMkEntries {
return []android.AndroidMkEntries{{
Class: "ETC",
OutputFile: android.OptionalPathForPath(txt.outputFile),
}}
}
// PrebuiltEtcModule interface
func (txt *sanitizerLibrariesTxtModule) OutputFile() android.OutputPath {
return txt.outputFile
}
// PrebuiltEtcModule interface
func (txt *sanitizerLibrariesTxtModule) BaseDir() string {
return "etc"
}
// PrebuiltEtcModule interface
func (txt *sanitizerLibrariesTxtModule) SubDir() string {
return ""
}
func (txt *sanitizerLibrariesTxtModule) OutputFiles(tag string) (android.Paths, error) {
return android.Paths{txt.outputFile}, nil
}