| // Copyright 2016 Google Inc. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package cc |
| |
| import ( |
| "fmt" |
| "sort" |
| "strings" |
| "sync" |
| |
| "android/soong/android" |
| "android/soong/cc/config" |
| ) |
| |
| var ( |
| modulesWarningsAllowedKey = android.NewOnceKey("ModulesWarningsAllowed") |
| modulesUsingWnoErrorKey = android.NewOnceKey("ModulesUsingWnoError") |
| 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() { |
| android.RegisterMakeVarsProvider(pctx, makeVarsProvider) |
| } |
| |
| func getNamedMapForConfig(config android.Config, key android.OnceKey) *sync.Map { |
| return config.Once(key, func() interface{} { |
| return &sync.Map{} |
| }).(*sync.Map) |
| } |
| |
| func makeStringOfKeys(ctx android.MakeVarsContext, key android.OnceKey) string { |
| set := getNamedMapForConfig(ctx.Config(), key) |
| keys := []string{} |
| set.Range(func(key interface{}, value interface{}) bool { |
| keys = append(keys, key.(string)) |
| return true |
| }) |
| sort.Strings(keys) |
| return strings.Join(keys, " ") |
| } |
| |
| func makeStringOfWarningAllowedProjects() string { |
| allProjects := append([]string{}, config.WarningAllowedProjects...) |
| sort.Strings(allProjects) |
| // Makefile rules use pattern "path/%" to match module paths. |
| if len(allProjects) > 0 { |
| return strings.Join(allProjects, "% ") + "%" |
| } else { |
| return "" |
| } |
| } |
| |
| type notOnHostContext struct { |
| } |
| |
| func (c *notOnHostContext) Host() bool { |
| return false |
| } |
| |
| func makeVarsProvider(ctx android.MakeVarsContext) { |
| ctx.Strict("LLVM_RELEASE_VERSION", "${config.ClangShortVersion}") |
| ctx.Strict("LLVM_PREBUILTS_VERSION", "${config.ClangVersion}") |
| ctx.Strict("LLVM_PREBUILTS_BASE", "${config.ClangBase}") |
| ctx.Strict("LLVM_PREBUILTS_PATH", "${config.ClangBin}") |
| ctx.Strict("CLANG", "${config.ClangBin}/clang") |
| ctx.Strict("CLANG_CXX", "${config.ClangBin}/clang++") |
| ctx.Strict("LLVM_AS", "${config.ClangBin}/llvm-as") |
| ctx.Strict("LLVM_LINK", "${config.ClangBin}/llvm-link") |
| ctx.Strict("LLVM_OBJCOPY", "${config.ClangBin}/llvm-objcopy") |
| ctx.Strict("LLVM_STRIP", "${config.ClangBin}/llvm-strip") |
| ctx.Strict("PATH_TO_CLANG_TIDY", "${config.ClangBin}/clang-tidy") |
| ctx.StrictSorted("CLANG_CONFIG_UNKNOWN_CFLAGS", strings.Join(config.ClangUnknownCflags, " ")) |
| |
| ctx.Strict("RS_LLVM_PREBUILTS_VERSION", "${config.RSClangVersion}") |
| ctx.Strict("RS_LLVM_PREBUILTS_BASE", "${config.RSClangBase}") |
| ctx.Strict("RS_LLVM_PREBUILTS_PATH", "${config.RSLLVMPrebuiltsPath}") |
| ctx.Strict("RS_LLVM_INCLUDES", "${config.RSIncludePath}") |
| ctx.Strict("RS_CLANG", "${config.RSLLVMPrebuiltsPath}/clang") |
| ctx.Strict("RS_LLVM_AS", "${config.RSLLVMPrebuiltsPath}/llvm-as") |
| ctx.Strict("RS_LLVM_LINK", "${config.RSLLVMPrebuiltsPath}/llvm-link") |
| |
| ctx.Strict("CLANG_EXTERNAL_CFLAGS", "${config.ExternalCflags}") |
| ctx.Strict("GLOBAL_CLANG_CFLAGS_NO_OVERRIDE", "${config.NoOverrideGlobalCflags}") |
| ctx.Strict("GLOBAL_CLANG_CFLAGS_64_NO_OVERRIDE", "${config.NoOverride64GlobalCflags}") |
| ctx.Strict("GLOBAL_CLANG_CPPFLAGS_NO_OVERRIDE", "") |
| ctx.Strict("GLOBAL_CLANG_EXTERNAL_CFLAGS_NO_OVERRIDE", "${config.NoOverrideExternalGlobalCflags}") |
| |
| // Filter vendor_public_library that are exported to make |
| exportedVendorPublicLibraries := []string{} |
| ctx.VisitAllModules(func(module android.Module) { |
| if ccModule, ok := module.(*Module); ok { |
| baseName := ccModule.BaseModuleName() |
| if ccModule.IsVendorPublicLibrary() && module.ExportedToMake() { |
| if !inList(baseName, exportedVendorPublicLibraries) { |
| exportedVendorPublicLibraries = append(exportedVendorPublicLibraries, baseName) |
| } |
| } |
| } |
| }) |
| sort.Strings(exportedVendorPublicLibraries) |
| ctx.Strict("VENDOR_PUBLIC_LIBRARIES", strings.Join(exportedVendorPublicLibraries, " ")) |
| |
| sort.Strings(lsdumpPaths) |
| ctx.Strict("LSDUMP_PATHS", strings.Join(lsdumpPaths, " ")) |
| |
| ctx.Strict("ANDROID_WARNING_ALLOWED_PROJECTS", makeStringOfWarningAllowedProjects()) |
| ctx.Strict("SOONG_MODULES_WARNINGS_ALLOWED", makeStringOfKeys(ctx, modulesWarningsAllowedKey)) |
| ctx.Strict("SOONG_MODULES_USING_WNO_ERROR", makeStringOfKeys(ctx, modulesUsingWnoErrorKey)) |
| ctx.Strict("SOONG_MODULES_MISSING_PGO_PROFILE_FILE", makeStringOfKeys(ctx, modulesMissingProfileFileKey)) |
| |
| ctx.Strict("CLANG_COVERAGE_CONFIG_CFLAGS", strings.Join(clangCoverageCFlags, " ")) |
| ctx.Strict("CLANG_COVERAGE_CONFIG_COMMFLAGS", strings.Join(clangCoverageCommonFlags, " ")) |
| ctx.Strict("CLANG_COVERAGE_HOST_LDFLAGS", strings.Join(clangCoverageHostLdFlags, " ")) |
| ctx.Strict("CLANG_COVERAGE_INSTR_PROFILE", profileInstrFlag) |
| ctx.Strict("CLANG_COVERAGE_CONTINUOUS_FLAGS", strings.Join(clangContinuousCoverageFlags, " ")) |
| ctx.Strict("CLANG_COVERAGE_HWASAN_FLAGS", strings.Join(clangCoverageHWASanFlags, " ")) |
| |
| ctx.Strict("ADDRESS_SANITIZER_CONFIG_EXTRA_CFLAGS", strings.Join(asanCflags, " ")) |
| ctx.Strict("ADDRESS_SANITIZER_CONFIG_EXTRA_LDFLAGS", strings.Join(asanLdflags, " ")) |
| |
| ctx.Strict("HWADDRESS_SANITIZER_CONFIG_EXTRA_CFLAGS", strings.Join(hwasanCflags, " ")) |
| ctx.Strict("HWADDRESS_SANITIZER_GLOBAL_OPTIONS", strings.Join(hwasanGlobalOptions, ",")) |
| |
| ctx.Strict("CFI_EXTRA_CFLAGS", strings.Join(cfiCflags, " ")) |
| ctx.Strict("CFI_EXTRA_ASFLAGS", strings.Join(cfiAsflags, " ")) |
| ctx.Strict("CFI_EXTRA_LDFLAGS", strings.Join(cfiLdflags, " ")) |
| |
| ctx.Strict("INTEGER_OVERFLOW_EXTRA_CFLAGS", strings.Join(intOverflowCflags, " ")) |
| |
| ctx.Strict("DEFAULT_C_STD_VERSION", config.CStdVersion) |
| ctx.Strict("DEFAULT_CPP_STD_VERSION", config.CppStdVersion) |
| ctx.Strict("EXPERIMENTAL_C_STD_VERSION", config.ExperimentalCStdVersion) |
| ctx.Strict("EXPERIMENTAL_CPP_STD_VERSION", config.ExperimentalCppStdVersion) |
| |
| ctx.Strict("DEFAULT_GLOBAL_TIDY_CHECKS", "${config.TidyDefaultGlobalChecks}") |
| ctx.Strict("DEFAULT_LOCAL_TIDY_CHECKS", joinLocalTidyChecks(config.DefaultLocalTidyChecks)) |
| ctx.Strict("DEFAULT_TIDY_HEADER_DIRS", "${config.TidyDefaultHeaderDirs}") |
| ctx.Strict("WITH_TIDY_FLAGS", "${config.TidyWithTidyFlags}") |
| |
| ctx.Strict("AIDL_CPP", "${aidlCmd}") |
| ctx.Strict("ALLOWED_MANUAL_INTERFACE_PATHS", strings.Join(allowedManualInterfacePaths, " ")) |
| |
| ctx.Strict("RS_GLOBAL_INCLUDES", "${config.RsGlobalIncludes}") |
| |
| ctx.Strict("SOONG_STRIP_PATH", "${stripPath}") |
| ctx.Strict("XZ", "${xzCmd}") |
| ctx.Strict("CREATE_MINIDEBUGINFO", "${createMiniDebugInfo}") |
| |
| includeFlags, err := ctx.Eval("${config.CommonGlobalIncludes}") |
| if err != nil { |
| panic(err) |
| } |
| includes, systemIncludes := splitSystemIncludes(ctx, includeFlags) |
| ctx.StrictRaw("SRC_HEADERS", strings.Join(includes, " ")) |
| ctx.StrictRaw("SRC_SYSTEM_HEADERS", strings.Join(systemIncludes, " ")) |
| |
| ndkKnownLibs := *getNDKKnownLibs(ctx.Config()) |
| sort.Strings(ndkKnownLibs) |
| ctx.Strict("NDK_KNOWN_LIBS", strings.Join(ndkKnownLibs, " ")) |
| |
| hostTargets := ctx.Config().Targets[ctx.Config().BuildOS] |
| makeVarsToolchain(ctx, "", hostTargets[0]) |
| if len(hostTargets) > 1 { |
| makeVarsToolchain(ctx, "2ND_", hostTargets[1]) |
| } |
| |
| deviceTargets := ctx.Config().Targets[android.Android] |
| makeVarsToolchain(ctx, "", deviceTargets[0]) |
| if len(deviceTargets) > 1 { |
| makeVarsToolchain(ctx, "2ND_", deviceTargets[1]) |
| } |
| |
| makeLlndkVars(ctx) |
| } |
| |
| func makeVarsToolchain(ctx android.MakeVarsContext, secondPrefix string, |
| target android.Target) { |
| var typePrefix string |
| switch target.Os.Class { |
| case android.Host: |
| typePrefix = "HOST_" |
| case android.Device: |
| typePrefix = "TARGET_" |
| } |
| makePrefix := secondPrefix + typePrefix |
| |
| toolchain := config.FindToolchain(target.Os, target.Arch) |
| |
| var productExtraCflags string |
| var productExtraLdflags string |
| |
| hod := "Host" |
| if target.Os.Class == android.Device { |
| hod = "Device" |
| } |
| |
| if target.Os.Class == android.Host && ctx.Config().HostStaticBinaries() { |
| productExtraLdflags += "-static" |
| } |
| |
| includeFlags, err := ctx.Eval(toolchain.IncludeFlags()) |
| if err != nil { |
| panic(err) |
| } |
| includes, systemIncludes := splitSystemIncludes(ctx, includeFlags) |
| ctx.StrictRaw(makePrefix+"C_INCLUDES", strings.Join(includes, " ")) |
| ctx.StrictRaw(makePrefix+"C_SYSTEM_INCLUDES", strings.Join(systemIncludes, " ")) |
| |
| if target.Arch.ArchType == android.Arm { |
| flags, err := toolchain.InstructionSetFlags("arm") |
| if err != nil { |
| panic(err) |
| } |
| ctx.Strict(makePrefix+"arm_CFLAGS", flags) |
| |
| flags, err = toolchain.InstructionSetFlags("thumb") |
| if err != nil { |
| panic(err) |
| } |
| ctx.Strict(makePrefix+"thumb_CFLAGS", flags) |
| } |
| |
| clangPrefix := secondPrefix + "CLANG_" + typePrefix |
| |
| ctx.Strict(clangPrefix+"TRIPLE", toolchain.ClangTriple()) |
| ctx.Strict(clangPrefix+"GLOBAL_CFLAGS", strings.Join([]string{ |
| toolchain.Cflags(), |
| "${config.CommonGlobalCflags}", |
| fmt.Sprintf("${config.%sGlobalCflags}", hod), |
| toolchain.ToolchainCflags(), |
| productExtraCflags, |
| }, " ")) |
| ctx.Strict(clangPrefix+"GLOBAL_CPPFLAGS", strings.Join([]string{ |
| "${config.CommonGlobalCppflags}", |
| fmt.Sprintf("${config.%sGlobalCppflags}", hod), |
| toolchain.Cppflags(), |
| }, " ")) |
| ctx.Strict(clangPrefix+"GLOBAL_LDFLAGS", strings.Join([]string{ |
| fmt.Sprintf("${config.%sGlobalLdflags}", hod), |
| toolchain.Ldflags(), |
| toolchain.ToolchainLdflags(), |
| productExtraLdflags, |
| }, " ")) |
| ctx.Strict(clangPrefix+"GLOBAL_LLDFLAGS", strings.Join([]string{ |
| fmt.Sprintf("${config.%sGlobalLldflags}", hod), |
| toolchain.Lldflags(), |
| toolchain.ToolchainLdflags(), |
| productExtraLdflags, |
| }, " ")) |
| |
| if target.Os.Class == android.Device { |
| for variable, value := range sanitizerVariables { |
| ctx.Strict(secondPrefix+variable, value) |
| } |
| } |
| |
| // This is used by external/gentoo/... |
| ctx.Strict("CLANG_CONFIG_"+target.Arch.ArchType.Name+"_"+typePrefix+"TRIPLE", |
| toolchain.ClangTriple()) |
| |
| if target.Os == android.Darwin { |
| ctx.Strict(makePrefix+"AR", "${config.MacArPath}") |
| ctx.Strict(makePrefix+"NM", "${config.MacToolPath}/nm") |
| ctx.Strict(makePrefix+"OTOOL", "${config.MacToolPath}/otool") |
| ctx.Strict(makePrefix+"STRIP", "${config.MacStripPath}") |
| } else { |
| ctx.Strict(makePrefix+"AR", "${config.ClangBin}/llvm-ar") |
| ctx.Strict(makePrefix+"READELF", "${config.ClangBin}/llvm-readelf") |
| ctx.Strict(makePrefix+"NM", "${config.ClangBin}/llvm-nm") |
| ctx.Strict(makePrefix+"STRIP", "${config.ClangBin}/llvm-strip") |
| } |
| |
| if target.Os.Class == android.Device { |
| ctx.Strict(makePrefix+"OBJCOPY", "${config.ClangBin}/llvm-objcopy") |
| ctx.Strict(makePrefix+"LD", "${config.ClangBin}/lld") |
| ctx.Strict(makePrefix+"NDK_TRIPLE", config.NDKTriple(toolchain)) |
| ctx.Strict(makePrefix+"TOOLS_PREFIX", "${config.ClangBin}/llvm-") |
| } |
| |
| if target.Os.Class == android.Host { |
| ctx.Strict(makePrefix+"AVAILABLE_LIBRARIES", strings.Join(toolchain.AvailableLibraries(), " ")) |
| } |
| |
| ctx.Strict(makePrefix+"SHLIB_SUFFIX", toolchain.ShlibSuffix()) |
| ctx.Strict(makePrefix+"EXECUTABLE_SUFFIX", toolchain.ExecutableSuffix()) |
| } |
| |
| func splitSystemIncludes(ctx android.MakeVarsContext, val string) (includes, systemIncludes []string) { |
| flags, err := ctx.Eval(val) |
| if err != nil { |
| panic(err) |
| } |
| |
| extract := func(flags string, dirs []string, prefix string) (string, []string, bool) { |
| if strings.HasPrefix(flags, prefix) { |
| flags = strings.TrimPrefix(flags, prefix) |
| flags = strings.TrimLeft(flags, " ") |
| s := strings.SplitN(flags, " ", 2) |
| dirs = append(dirs, s[0]) |
| if len(s) > 1 { |
| return strings.TrimLeft(s[1], " "), dirs, true |
| } |
| return "", dirs, true |
| } else { |
| return flags, dirs, false |
| } |
| } |
| |
| flags = strings.TrimLeft(flags, " ") |
| for flags != "" { |
| found := false |
| flags, includes, found = extract(flags, includes, "-I") |
| if !found { |
| flags, systemIncludes, found = extract(flags, systemIncludes, "-isystem ") |
| } |
| if !found { |
| panic(fmt.Errorf("Unexpected flag in %q", flags)) |
| } |
| } |
| |
| return includes, systemIncludes |
| } |
| |
| func joinLocalTidyChecks(checks []config.PathBasedTidyCheck) string { |
| rets := make([]string, len(checks)) |
| for i, check := range config.DefaultLocalTidyChecks { |
| rets[i] = check.PathPrefix + ":" + check.Checks |
| } |
| return strings.Join(rets, " ") |
| } |