// 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" "android/soong/android" ) func getNdkStlFamily(m LinkableInterface) string { family, _ := getNdkStlFamilyAndLinkType(m) return family } func deduplicateStlInput(stl string) string { switch stl { case "c++_shared": return "libc++" case "c++_static": return "libc++_static" } return stl } func getNdkStlFamilyAndLinkType(m LinkableInterface) (string, string) { stl := m.SelectedStl() switch stl { case "ndk_libc++_shared", "libc++": return "libc++", "shared" case "ndk_libc++_static", "libc++_static": return "libc++", "static" case "ndk_system": return "system", "shared" case "": return "none", "none" default: panic(fmt.Errorf("stl: %q is not a valid STL", stl)) } } type StlProperties struct { // Select the STL library to use. Possible values are "libc++", // "libc++_static", "libstdc++", or "none". Leave blank to select the // default. Stl *string `android:"arch_variant"` SelectedStl string `blueprint:"mutated"` } type stl struct { Properties StlProperties } func (stl *stl) props() []interface{} { return []interface{}{&stl.Properties} } func (stl *stl) begin(ctx BaseModuleContext) { stl.Properties.SelectedStl = func() string { s := "" if stl.Properties.Stl != nil { s = *stl.Properties.Stl } else if ctx.header() { s = "none" } if s == "none" { return "" } s = deduplicateStlInput(s) if ctx.useSdk() && ctx.Device() { switch s { case "", "system": return "ndk_system" case "libc++": return "ndk_libc++_shared" case "libc++_static": return "ndk_libc++_static" default: ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", s) return "" } } else if ctx.Windows() { switch s { case "libc++", "libc++_static", "": // Only use static libc++ for Windows. return "libc++_static" default: ctx.ModuleErrorf("stl: %q is not a supported STL for windows", s) return "" } } else { switch s { case "libc++", "libc++_static": return s case "", "system": if ctx.static() { return "libc++_static" } else { return "libc++" } default: ctx.ModuleErrorf("stl: %q is not a supported STL", s) return "" } } }() } func staticUnwinder(ctx android.BaseModuleContext) string { vndkVersion := ctx.Module().(*Module).VndkVersion() // Modules using R vndk use different unwinder if vndkVersion == "30" { if ctx.Arch().ArchType == android.Arm { return "libunwind_llvm" } else { return "libgcc_stripped" } } return "libunwind" } func (stl *stl) deps(ctx BaseModuleContext, deps Deps) Deps { switch stl.Properties.SelectedStl { case "libstdc++": // Nothing case "libc++", "libc++_static": if stl.Properties.SelectedStl == "libc++" { deps.SharedLibs = append(deps.SharedLibs, stl.Properties.SelectedStl) } else { deps.StaticLibs = append(deps.StaticLibs, stl.Properties.SelectedStl) } if ctx.Device() && !ctx.useSdk() { // __cxa_demangle is not a part of libc++.so on the device since // it's large and most processes don't need it. Statically link // libc++demangle into every process so that users still have it if // needed, but the linker won't include this unless it is actually // called. // http://b/138245375 deps.StaticLibs = append(deps.StaticLibs, "libc++demangle") } if ctx.toolchain().Bionic() { if ctx.staticBinary() { deps.StaticLibs = append(deps.StaticLibs, "libm", "libc", staticUnwinder(ctx)) } else { deps.StaticUnwinderIfLegacy = true } } case "": // None or error. if ctx.toolchain().Bionic() && ctx.Module().Name() == "libc++" { deps.StaticUnwinderIfLegacy = true } case "ndk_system": // TODO: Make a system STL prebuilt for the NDK. // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have // its own includes. The includes are handled in CCBase.Flags(). deps.SharedLibs = append([]string{"libstdc++"}, deps.SharedLibs...) deps.HeaderLibs = append([]string{"ndk_system"}, deps.HeaderLibs...) case "ndk_libc++_shared", "ndk_libc++_static": if stl.Properties.SelectedStl == "ndk_libc++_shared" { deps.SharedLibs = append(deps.SharedLibs, stl.Properties.SelectedStl) } else { deps.StaticLibs = append(deps.StaticLibs, stl.Properties.SelectedStl, "ndk_libc++abi") } deps.StaticLibs = append(deps.StaticLibs, "ndk_libunwind") default: panic(fmt.Errorf("Unknown stl: %q", stl.Properties.SelectedStl)) } return deps } func (stl *stl) flags(ctx ModuleContext, flags Flags) Flags { switch stl.Properties.SelectedStl { case "libc++", "libc++_static": if ctx.Darwin() { // libc++'s headers are annotated with availability macros that // indicate which version of Mac OS was the first to ship with a // libc++ feature available in its *system's* libc++.dylib. We do // not use the system's library, but rather ship our own. As such, // these availability attributes are meaningless for us but cause // build breaks when we try to use code that would not be available // in the system's dylib. flags.Local.CppFlags = append(flags.Local.CppFlags, "-D_LIBCPP_DISABLE_AVAILABILITY") } if !ctx.toolchain().Bionic() { flags.Local.CppFlags = append(flags.Local.CppFlags, "-nostdinc++") flags.extraLibFlags = append(flags.extraLibFlags, "-nostdlib++") if ctx.Windows() { flags.Local.CppFlags = append(flags.Local.CppFlags, // These macros can also be defined by libc++'s __config // or __config_site headers so define them the same way // (i.e. to nothing). Disable visibility annotations since // we're using static libc++. "-D_LIBCPP_DISABLE_VISIBILITY_ANNOTATIONS=", "-D_LIBCXXABI_DISABLE_VISIBILITY_ANNOTATIONS=", // Use Win32 threads in libc++. "-D_LIBCPP_HAS_THREAD_API_WIN32=") } } case "libstdc++": // Nothing case "ndk_system": // Nothing: The exports of ndk_system will be added automatically to the local cflags case "ndk_libc++_shared", "ndk_libc++_static": if ctx.Arch().ArchType == android.Arm { // Make sure the _Unwind_XXX symbols are not re-exported. flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,--exclude-libs,libunwind.a") } case "": // None or error. if !ctx.toolchain().Bionic() { flags.Local.CppFlags = append(flags.Local.CppFlags, "-nostdinc++") flags.extraLibFlags = append(flags.extraLibFlags, "-nostdlib++") } default: panic(fmt.Errorf("Unknown stl: %q", stl.Properties.SelectedStl)) } return flags }