// 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 ( "path/filepath" "strings" "android/soong/android" "android/soong/bazel" ) func init() { android.RegisterModuleType("ndk_prebuilt_static_stl", NdkPrebuiltStaticStlFactory) android.RegisterModuleType("ndk_prebuilt_shared_stl", NdkPrebuiltSharedStlFactory) } // NDK prebuilt libraries. // // These differ from regular prebuilts in that they aren't stripped and usually aren't installed // either (with the exception of the shared STLs, which are installed to the app's directory rather // than to the system image). type ndkPrebuiltStlLinker struct { *libraryDecorator } func (ndk *ndkPrebuiltStlLinker) linkerProps() []interface{} { return append(ndk.libraryDecorator.linkerProps(), &ndk.Properties, &ndk.flagExporter.Properties) } func (*ndkPrebuiltStlLinker) linkerDeps(ctx DepsContext, deps Deps) Deps { // NDK libraries can't have any dependencies return deps } func (*ndkPrebuiltStlLinker) availableFor(what string) bool { // ndk prebuilt objects are available to everywhere return true } // ndk_prebuilt_shared_stl exports a precompiled ndk shared standard template // library (stl) library for linking operation. The soong's module name format // is ndk_.so where the library is located under // ./prebuilts/ndk/current/sources/cxx-stl/llvm-libc++/libs/$(HOST_ARCH)/.so. func NdkPrebuiltSharedStlFactory() android.Module { module, library := NewLibrary(android.DeviceSupported) library.BuildOnlyShared() module.compiler = nil module.linker = &ndkPrebuiltStlLinker{ libraryDecorator: library, } module.installer = nil module.Properties.Sdk_version = StringPtr("minimum") module.Properties.AlwaysSdk = true module.stl.Properties.Stl = StringPtr("none") module.bazelable = true return module.Init() } // ndk_prebuilt_static_stl exports a precompiled ndk static standard template // library (stl) library for linking operation. The soong's module name format // is ndk_.a where the library is located under // ./prebuilts/ndk/current/sources/cxx-stl/llvm-libc++/libs/$(HOST_ARCH)/.a. func NdkPrebuiltStaticStlFactory() android.Module { module, library := NewLibrary(android.DeviceSupported) library.BuildOnlyStatic() module.compiler = nil module.linker = &ndkPrebuiltStlLinker{ libraryDecorator: library, } module.installer = nil module.Properties.Sdk_version = StringPtr("minimum") module.Properties.HideFromMake = true module.Properties.AlwaysSdk = true module.Properties.Sdk_version = StringPtr("current") module.stl.Properties.Stl = StringPtr("none") module.bazelable = true return module.Init() } const ( libDir = "current/sources/cxx-stl/llvm-libc++/libs" ) func getNdkStlLibDir(ctx android.ModuleContext) android.SourcePath { return android.PathForSource(ctx, ctx.ModuleDir(), libDir).Join(ctx, ctx.Arch().Abi[0]) } func (ndk *ndkPrebuiltStlLinker) link(ctx ModuleContext, flags Flags, deps PathDeps, objs Objects) android.Path { // A null build step, but it sets up the output path. if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") { ctx.ModuleErrorf("NDK prebuilt libraries must have an ndk_lib prefixed name") } ndk.libraryDecorator.flagExporter.exportIncludesAsSystem(ctx) libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_") libExt := flags.Toolchain.ShlibSuffix() if ndk.static() { libExt = staticLibraryExtension } libDir := getNdkStlLibDir(ctx) lib := libDir.Join(ctx, libName+libExt) ndk.libraryDecorator.flagExporter.setProvider(ctx) if ndk.static() { depSet := android.NewDepSetBuilder[android.Path](android.TOPOLOGICAL).Direct(lib).Build() ctx.SetProvider(StaticLibraryInfoProvider, StaticLibraryInfo{ StaticLibrary: lib, TransitiveStaticLibrariesForOrdering: depSet, }) } else { ctx.SetProvider(SharedLibraryInfoProvider, SharedLibraryInfo{ SharedLibrary: lib, Target: ctx.Target(), }) } return lib } var ( archToAbiDirMap = map[string]string{ "android_arm": "armeabi-v7a", "android_arm64": "arm64-v8a", "android_riscv64": "riscv64", "android_x86": "x86", "android_x86_64": "x86_64", } ) // stlSrcBp2build returns a bazel label for the checked-in .so/.a file // It contains a select statement for each ABI func stlSrcBp2build(ctx android.Bp2buildMutatorContext, c *Module) bazel.LabelAttribute { libName := strings.TrimPrefix(c.Name(), "ndk_") libExt := ".so" // TODO - b/201079053: Support windows if ctx.ModuleType() == "ndk_prebuilt_static_stl" { libExt = ".a" } src := bazel.LabelAttribute{} for arch, abiDir := range archToAbiDirMap { srcPath := filepath.Join(libDir, abiDir, libName+libExt) src.SetSelectValue( bazel.OsArchConfigurationAxis, arch, android.BazelLabelForModuleSrcSingle(ctx, srcPath), ) } return src } // stlIncludesBp2build returns the includes exported by the STL func stlIncludesBp2build(c *Module) bazel.StringListAttribute { linker, _ := c.linker.(*ndkPrebuiltStlLinker) includeDirs := append( []string{}, linker.libraryDecorator.flagExporter.Properties.Export_include_dirs..., ) includeDirs = append( includeDirs, linker.libraryDecorator.flagExporter.Properties.Export_system_include_dirs..., ) return bazel.MakeStringListAttribute(android.FirstUniqueStrings(includeDirs)) } func ndkPrebuiltStlBp2build(ctx android.Bp2buildMutatorContext, c *Module) { if ctx.ModuleType() == "ndk_prebuilt_static_stl" { ndkPrebuiltStaticStlBp2build(ctx, c) } else { ndkPrebuiltSharedStlBp2build(ctx, c) } } func ndkPrebuiltStaticStlBp2build(ctx android.Bp2buildMutatorContext, c *Module) { props := bazel.BazelTargetModuleProperties{ Rule_class: "cc_prebuilt_library_static", Bzl_load_location: "//build/bazel/rules/cc:cc_prebuilt_library_static.bzl", } attrs := &bazelPrebuiltLibraryStaticAttributes{ Static_library: stlSrcBp2build(ctx, c), Export_system_includes: stlIncludesBp2build(c), // The exports are always as system } // TODO: min_sdk_version ctx.CreateBazelTargetModule(props, android.CommonAttributes{Name: c.Name()}, attrs) } func ndkPrebuiltSharedStlBp2build(ctx android.Bp2buildMutatorContext, c *Module) { props := bazel.BazelTargetModuleProperties{ Rule_class: "cc_prebuilt_library_shared", Bzl_load_location: "//build/bazel/rules/cc:cc_prebuilt_library_shared.bzl", } attrs := &bazelPrebuiltLibrarySharedAttributes{ Shared_library: stlSrcBp2build(ctx, c), Export_system_includes: stlIncludesBp2build(c), // The exports are always as system } // TODO: min_sdk_version ctx.CreateBazelTargetModule(props, android.CommonAttributes{Name: c.Name()}, attrs) }