blob: ee8d9c2cdec9c03c5dc35767ee9270d5a600b3a2 [file] [log] [blame]
//
// 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
}