| // |
| // Copyright (C) 2021 The Android Open Source Project |
| // |
| // 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 rustprebuilts |
| |
| import ( |
| "path" |
| "path/filepath" |
| "strings" |
| |
| "github.com/google/blueprint/proptools" |
| |
| "android/soong/android" |
| "android/soong/rust" |
| "android/soong/rust/config" |
| ) |
| |
| // This module is used to generate the rust host stdlib prebuilts |
| // When RUST_PREBUILTS_VERSION is set, the library will generated |
| // from the given Rust version. |
| func init() { |
| android.RegisterModuleType("rust_stdlib_prebuilt_host", |
| rustHostPrebuiltSysrootLibraryFactory) |
| android.RegisterModuleType("rust_stdlib_prebuilt_filegroup_host", |
| rustToolchainFilegroupFactory) |
| } |
| |
| func getRustPrebuiltVersion(ctx android.LoadHookContext) string { |
| return ctx.Config().GetenvWithDefault("RUST_PREBUILTS_VERSION", config.RustDefaultVersion) |
| } |
| |
| func getRustLibDir(ctx android.LoadHookContext) string { |
| rustDir := getRustPrebuiltVersion(ctx) |
| return path.Join(rustDir, "lib", "rustlib") |
| } |
| |
| // getPrebuilt returns the module relative Rust library path and the suffix hash. |
| func getPrebuilt(ctx android.LoadHookContext, dir, lib, extension string) (string, string) { |
| globPath := path.Join(ctx.ModuleDir(), dir, lib) + "-*" + extension |
| libMatches := ctx.Glob(globPath, nil) |
| |
| if len(libMatches) != 1 { |
| ctx.ModuleErrorf("Unexpected number of matches for prebuilt libraries at path %q, found %d matches", globPath, len(libMatches)) |
| return "", "" |
| } |
| |
| // Collect the suffix by trimming the extension from the Base, then removing the library name and hyphen. |
| suffix := strings.TrimSuffix(libMatches[0].Base(), extension)[len(lib)+1:] |
| |
| // Get the relative path from the match by trimming out the module directory. |
| relPath := strings.TrimPrefix(libMatches[0].String(), ctx.ModuleDir()+"/") |
| |
| return relPath, suffix |
| } |
| |
| type targetProps struct { |
| Suffix *string |
| Dylib struct { |
| Srcs []string |
| } |
| Rlib struct { |
| Srcs []string |
| } |
| Link_dirs []string |
| Enabled *bool |
| } |
| |
| type props struct { |
| Enabled *bool |
| Target struct { |
| Linux_glibc_x86_64 targetProps |
| Linux_glibc_x86 targetProps |
| Linux_musl_x86_64 targetProps |
| Linux_musl_x86 targetProps |
| Darwin_x86_64 targetProps |
| } |
| } |
| |
| func (target *targetProps) addPrebuiltToTarget(ctx android.LoadHookContext, libName, rustDir, platform, arch string, rlib, solib bool) { |
| dir := path.Join(platform, rustDir, arch, "lib") |
| target.Link_dirs = []string{dir} |
| target.Enabled = proptools.BoolPtr(true) |
| if rlib { |
| rlib, suffix := getPrebuilt(ctx, dir, libName, ".rlib") |
| target.Rlib.Srcs = []string{rlib} |
| target.Suffix = proptools.StringPtr(suffix) |
| } |
| if solib { |
| // The suffixes are the same between the dylib and the rlib, |
| // so it's okay if we overwrite the rlib suffix |
| var soSuffix string |
| if strings.Contains(platform, "darwin") { |
| soSuffix = ".dylib" |
| } else { |
| soSuffix = ".so" |
| } |
| dylib, suffix := getPrebuilt(ctx, dir, libName, soSuffix) |
| target.Dylib.Srcs = []string{dylib} |
| target.Suffix = proptools.StringPtr(suffix) |
| } |
| } |
| |
| func constructLibProps(rlib, solib bool) func(ctx android.LoadHookContext) { |
| return func(ctx android.LoadHookContext) { |
| rustDir := getRustLibDir(ctx) |
| name := android.RemoveOptionalPrebuiltPrefix(ctx.ModuleName()) |
| name = strings.Replace(name, ".rust_sysroot", "", -1) |
| |
| p := props{} |
| p.Enabled = proptools.BoolPtr(false) |
| |
| if ctx.Config().BuildOS == android.Linux { |
| p.Target.Linux_glibc_x86_64.addPrebuiltToTarget(ctx, name, rustDir, "linux-x86", "x86_64-unknown-linux-gnu", rlib, solib) |
| p.Target.Linux_glibc_x86.addPrebuiltToTarget(ctx, name, rustDir, "linux-x86", "i686-unknown-linux-gnu", rlib, solib) |
| } else if ctx.Config().BuildOS == android.LinuxMusl { |
| p.Target.Linux_musl_x86_64.addPrebuiltToTarget(ctx, name, rustDir, "linux-musl-x86", "x86_64-unknown-linux-musl", rlib, solib) |
| p.Target.Linux_musl_x86.addPrebuiltToTarget(ctx, name, rustDir, "linux-musl-x86", "i686-unknown-linux-musl", rlib, solib) |
| } else if ctx.Config().BuildOS == android.Darwin { |
| p.Target.Darwin_x86_64.addPrebuiltToTarget(ctx, name, rustDir, "darwin-x86", "x86_64-apple-darwin", rlib, solib) |
| } |
| |
| ctx.AppendProperties(&p) |
| } |
| } |
| |
| func rustHostPrebuiltSysrootLibraryFactory() android.Module { |
| module, _ := rust.NewPrebuiltLibrary(android.HostSupportedNoCross) |
| android.AddLoadHook(module, constructLibProps( /*rlib=*/ true /*solib=*/, true)) |
| return module.Init() |
| } |
| |
| type toolchainFilegroupProperties struct { |
| // path to toolchain files, relative to the top of the toolchain source |
| Toolchain_srcs []string |
| } |
| |
| func rustToolchainFilegroupFactory() android.Module { |
| module := android.FileGroupFactory() |
| module.AddProperties(&toolchainFilegroupProperties{}) |
| android.AddLoadHook(module, func(ctx android.LoadHookContext) { |
| var toolchainProps *toolchainFilegroupProperties |
| for _, p := range ctx.Module().GetProperties() { |
| toolchainProperties, ok := p.(*toolchainFilegroupProperties) |
| if ok { |
| toolchainProps = toolchainProperties |
| } |
| } |
| |
| var archTriple string |
| if ctx.Config().BuildOS == android.Linux { |
| if ctx.Config().BuildArch == android.X86_64 { |
| archTriple = "x86_64-unknown-linux-gnu" |
| } else { |
| archTriple = "i686-unknown-linux-gnu" |
| } |
| } else if ctx.Config().BuildOS == android.LinuxMusl { |
| if ctx.Config().BuildArch == android.X86_64 { |
| archTriple = "x86_64-unknown-linux-musl" |
| } else { |
| archTriple = "i686-unknown-linux-musl" |
| } |
| |
| } else if ctx.Config().BuildOS == android.Darwin { |
| if ctx.Config().BuildArch == android.X86_64 { |
| archTriple = "x86_64-apple-darwin" |
| } |
| } |
| |
| prefix := filepath.Join(config.HostPrebuiltTag(ctx.Config()), rust.GetRustPrebuiltVersion(ctx), "lib", "rustlib", archTriple) |
| srcs := make([]string, 0, len(toolchainProps.Toolchain_srcs)) |
| for _, s := range toolchainProps.Toolchain_srcs { |
| srcs = append(srcs, path.Join(prefix, s)) |
| } |
| |
| props := struct { |
| Srcs []string |
| }{ |
| Srcs: srcs, |
| } |
| ctx.AppendProperties(&props) |
| }) |
| return module |
| } |