You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
282 lines
8.2 KiB
282 lines
8.2 KiB
4 months ago
|
// 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"
|
||
|
"android/soong/bazel"
|
||
|
)
|
||
|
|
||
|
//
|
||
|
// Objects (for crt*.o)
|
||
|
//
|
||
|
|
||
|
func init() {
|
||
|
android.RegisterModuleType("cc_object", ObjectFactory)
|
||
|
android.RegisterSdkMemberType(ccObjectSdkMemberType)
|
||
|
|
||
|
android.RegisterBp2BuildMutator("cc_object", ObjectBp2Build)
|
||
|
}
|
||
|
|
||
|
var ccObjectSdkMemberType = &librarySdkMemberType{
|
||
|
SdkMemberTypeBase: android.SdkMemberTypeBase{
|
||
|
PropertyName: "native_objects",
|
||
|
SupportsSdk: true,
|
||
|
},
|
||
|
prebuiltModuleType: "cc_prebuilt_object",
|
||
|
linkTypes: nil,
|
||
|
}
|
||
|
|
||
|
type objectLinker struct {
|
||
|
*baseLinker
|
||
|
Properties ObjectLinkerProperties
|
||
|
}
|
||
|
|
||
|
type objectBazelHandler struct {
|
||
|
bazelHandler
|
||
|
|
||
|
module *Module
|
||
|
}
|
||
|
|
||
|
func (handler *objectBazelHandler) generateBazelBuildActions(ctx android.ModuleContext, label string) bool {
|
||
|
bazelCtx := ctx.Config().BazelContext
|
||
|
objPaths, ok := bazelCtx.GetOutputFiles(label, ctx.Arch().ArchType)
|
||
|
if ok {
|
||
|
if len(objPaths) != 1 {
|
||
|
ctx.ModuleErrorf("expected exactly one object file for '%s', but got %s", label, objPaths)
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
handler.module.outputFile = android.OptionalPathForPath(android.PathForBazelOut(ctx, objPaths[0]))
|
||
|
}
|
||
|
return ok
|
||
|
}
|
||
|
|
||
|
type ObjectLinkerProperties struct {
|
||
|
// list of modules that should only provide headers for this module.
|
||
|
Header_libs []string `android:"arch_variant,variant_prepend"`
|
||
|
|
||
|
// names of other cc_object modules to link into this module using partial linking
|
||
|
Objs []string `android:"arch_variant"`
|
||
|
|
||
|
// if set, add an extra objcopy --prefix-symbols= step
|
||
|
Prefix_symbols *string
|
||
|
|
||
|
// if set, the path to a linker script to pass to ld -r when combining multiple object files.
|
||
|
Linker_script *string `android:"path,arch_variant"`
|
||
|
|
||
|
// Indicates that this module is a CRT object. CRT objects will be split
|
||
|
// into a variant per-API level between min_sdk_version and current.
|
||
|
Crt *bool
|
||
|
}
|
||
|
|
||
|
func newObject() *Module {
|
||
|
module := newBaseModule(android.HostAndDeviceSupported, android.MultilibBoth)
|
||
|
module.sanitize = &sanitize{}
|
||
|
module.stl = &stl{}
|
||
|
return module
|
||
|
}
|
||
|
|
||
|
// cc_object runs the compiler without running the linker. It is rarely
|
||
|
// necessary, but sometimes used to generate .s files from .c files to use as
|
||
|
// input to a cc_genrule module.
|
||
|
func ObjectFactory() android.Module {
|
||
|
module := newObject()
|
||
|
module.linker = &objectLinker{
|
||
|
baseLinker: NewBaseLinker(module.sanitize),
|
||
|
}
|
||
|
module.compiler = NewBaseCompiler()
|
||
|
module.bazelHandler = &objectBazelHandler{module: module}
|
||
|
|
||
|
// Clang's address-significance tables are incompatible with ld -r.
|
||
|
module.compiler.appendCflags([]string{"-fno-addrsig"})
|
||
|
|
||
|
module.sdkMemberTypes = []android.SdkMemberType{ccObjectSdkMemberType}
|
||
|
|
||
|
return module.Init()
|
||
|
}
|
||
|
|
||
|
// For bp2build conversion.
|
||
|
type bazelObjectAttributes struct {
|
||
|
Srcs bazel.LabelListAttribute
|
||
|
Hdrs bazel.LabelListAttribute
|
||
|
Deps bazel.LabelListAttribute
|
||
|
Copts bazel.StringListAttribute
|
||
|
Asflags []string
|
||
|
}
|
||
|
|
||
|
type bazelObject struct {
|
||
|
android.BazelTargetModuleBase
|
||
|
bazelObjectAttributes
|
||
|
}
|
||
|
|
||
|
func (m *bazelObject) Name() string {
|
||
|
return m.BaseModuleName()
|
||
|
}
|
||
|
|
||
|
func (m *bazelObject) GenerateAndroidBuildActions(ctx android.ModuleContext) {}
|
||
|
|
||
|
func BazelObjectFactory() android.Module {
|
||
|
module := &bazelObject{}
|
||
|
module.AddProperties(&module.bazelObjectAttributes)
|
||
|
android.InitBazelTargetModule(module)
|
||
|
return module
|
||
|
}
|
||
|
|
||
|
// ObjectBp2Build is the bp2build converter from cc_object modules to the
|
||
|
// Bazel equivalent target, plus any necessary include deps for the cc_object.
|
||
|
func ObjectBp2Build(ctx android.TopDownMutatorContext) {
|
||
|
m, ok := ctx.Module().(*Module)
|
||
|
if !ok || !m.ConvertWithBp2build(ctx) {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// a Module can be something other than a cc_object.
|
||
|
if ctx.ModuleType() != "cc_object" {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if m.compiler == nil {
|
||
|
// a cc_object must have access to the compiler decorator for its props.
|
||
|
ctx.ModuleErrorf("compiler must not be nil for a cc_object module")
|
||
|
}
|
||
|
|
||
|
// Set arch-specific configurable attributes
|
||
|
compilerAttrs := bp2BuildParseCompilerProps(ctx, m)
|
||
|
var asFlags []string
|
||
|
|
||
|
var deps bazel.LabelListAttribute
|
||
|
for _, props := range m.linker.linkerProps() {
|
||
|
if objectLinkerProps, ok := props.(*ObjectLinkerProperties); ok {
|
||
|
deps = bazel.MakeLabelListAttribute(
|
||
|
android.BazelLabelForModuleDeps(ctx, objectLinkerProps.Objs))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
productVariableProps := android.ProductVariableProperties(ctx)
|
||
|
if props, exists := productVariableProps["Asflags"]; exists {
|
||
|
// TODO(b/183595873): consider deduplicating handling of product variable properties
|
||
|
for _, prop := range props {
|
||
|
flags, ok := prop.Property.([]string)
|
||
|
if !ok {
|
||
|
ctx.ModuleErrorf("Could not convert product variable asflag property")
|
||
|
return
|
||
|
}
|
||
|
// TODO(b/183595873) handle other product variable usages -- as selects?
|
||
|
if newFlags, subbed := bazel.TryVariableSubstitutions(flags, prop.ProductConfigVariable); subbed {
|
||
|
asFlags = append(asFlags, newFlags...)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
// TODO(b/183595872) warn/error if we're not handling product variables
|
||
|
|
||
|
attrs := &bazelObjectAttributes{
|
||
|
Srcs: compilerAttrs.srcs,
|
||
|
Deps: deps,
|
||
|
Copts: compilerAttrs.copts,
|
||
|
Asflags: asFlags,
|
||
|
}
|
||
|
|
||
|
props := bazel.BazelTargetModuleProperties{
|
||
|
Rule_class: "cc_object",
|
||
|
Bzl_load_location: "//build/bazel/rules:cc_object.bzl",
|
||
|
}
|
||
|
|
||
|
ctx.CreateBazelTargetModule(BazelObjectFactory, m.Name(), props, attrs)
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) appendLdflags(flags []string) {
|
||
|
panic(fmt.Errorf("appendLdflags on objectLinker not supported"))
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) linkerProps() []interface{} {
|
||
|
return []interface{}{&object.Properties}
|
||
|
}
|
||
|
|
||
|
func (*objectLinker) linkerInit(ctx BaseModuleContext) {}
|
||
|
|
||
|
func (object *objectLinker) linkerDeps(ctx DepsContext, deps Deps) Deps {
|
||
|
deps.HeaderLibs = append(deps.HeaderLibs, object.Properties.Header_libs...)
|
||
|
deps.ObjFiles = append(deps.ObjFiles, object.Properties.Objs...)
|
||
|
return deps
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) linkerFlags(ctx ModuleContext, flags Flags) Flags {
|
||
|
flags.Global.LdFlags = append(flags.Global.LdFlags, ctx.toolchain().ToolchainClangLdflags())
|
||
|
|
||
|
if lds := android.OptionalPathForModuleSrc(ctx, object.Properties.Linker_script); lds.Valid() {
|
||
|
flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-T,"+lds.String())
|
||
|
flags.LdFlagsDeps = append(flags.LdFlagsDeps, lds.Path())
|
||
|
}
|
||
|
return flags
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) link(ctx ModuleContext,
|
||
|
flags Flags, deps PathDeps, objs Objects) android.Path {
|
||
|
|
||
|
objs = objs.Append(deps.Objs)
|
||
|
|
||
|
var outputFile android.Path
|
||
|
builderFlags := flagsToBuilderFlags(flags)
|
||
|
|
||
|
if len(objs.objFiles) == 1 && String(object.Properties.Linker_script) == "" {
|
||
|
outputFile = objs.objFiles[0]
|
||
|
|
||
|
if String(object.Properties.Prefix_symbols) != "" {
|
||
|
output := android.PathForModuleOut(ctx, ctx.ModuleName()+objectExtension)
|
||
|
transformBinaryPrefixSymbols(ctx, String(object.Properties.Prefix_symbols), outputFile,
|
||
|
builderFlags, output)
|
||
|
outputFile = output
|
||
|
}
|
||
|
} else {
|
||
|
output := android.PathForModuleOut(ctx, ctx.ModuleName()+objectExtension)
|
||
|
outputFile = output
|
||
|
|
||
|
if String(object.Properties.Prefix_symbols) != "" {
|
||
|
input := android.PathForModuleOut(ctx, "unprefixed", ctx.ModuleName()+objectExtension)
|
||
|
transformBinaryPrefixSymbols(ctx, String(object.Properties.Prefix_symbols), input,
|
||
|
builderFlags, output)
|
||
|
output = input
|
||
|
}
|
||
|
|
||
|
transformObjsToObj(ctx, objs.objFiles, builderFlags, output, flags.LdFlagsDeps)
|
||
|
}
|
||
|
|
||
|
ctx.CheckbuildFile(outputFile)
|
||
|
return outputFile
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) unstrippedOutputFilePath() android.Path {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) nativeCoverage() bool {
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) coverageOutputFilePath() android.OptionalPath {
|
||
|
return android.OptionalPath{}
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) object() bool {
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
func (object *objectLinker) isCrt() bool {
|
||
|
return Bool(object.Properties.Crt)
|
||
|
}
|