blob: cd711617b797cfae3b835d109ada3c4c4277df6c [file] [log] [blame]
Colin Cross4d9c2d12016-07-29 12:48:20 -07001// Copyright 2016 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package cc
16
17import (
18 "fmt"
19
Colin Cross4d9c2d12016-07-29 12:48:20 -070020 "android/soong/android"
Jingwen Chen8c1b97e2021-02-18 03:21:34 -050021 "android/soong/bazel"
Colin Cross4d9c2d12016-07-29 12:48:20 -070022)
23
24//
25// Objects (for crt*.o)
26//
27
28func init() {
Colin Crosse40b4ea2018-10-02 22:25:58 -070029 android.RegisterModuleType("cc_object", ObjectFactory)
Martin Stjernholmcd07bce2020-03-10 22:37:59 +000030 android.RegisterSdkMemberType(ccObjectSdkMemberType)
Jingwen Chen8c1b97e2021-02-18 03:21:34 -050031
32 android.RegisterBp2BuildMutator("cc_object", ObjectBp2Build)
Martin Stjernholmcd07bce2020-03-10 22:37:59 +000033}
34
35var ccObjectSdkMemberType = &librarySdkMemberType{
36 SdkMemberTypeBase: android.SdkMemberTypeBase{
37 PropertyName: "native_objects",
38 SupportsSdk: true,
39 },
40 prebuiltModuleType: "cc_prebuilt_object",
41 linkTypes: nil,
Colin Cross4d9c2d12016-07-29 12:48:20 -070042}
43
44type objectLinker struct {
Colin Crossb916a382016-07-29 17:28:03 -070045 *baseLinker
Colin Cross4d9c2d12016-07-29 12:48:20 -070046 Properties ObjectLinkerProperties
47}
48
Chris Parsons8d6e4332021-02-22 16:13:50 -050049type objectBazelHandler struct {
50 bazelHandler
51
52 module *Module
53}
54
55func (handler *objectBazelHandler) generateBazelBuildActions(ctx android.ModuleContext, label string) bool {
Liz Kammer8206d4f2021-03-03 16:40:52 -050056 bazelCtx := ctx.Config().BazelContext
57 objPaths, ok := bazelCtx.GetOutputFiles(label, ctx.Arch().ArchType)
58 if ok {
59 if len(objPaths) != 1 {
60 ctx.ModuleErrorf("expected exactly one object file for '%s', but got %s", label, objPaths)
61 return false
62 }
63
64 handler.module.outputFile = android.OptionalPathForPath(android.PathForBazelOut(ctx, objPaths[0]))
65 }
66 return ok
Chris Parsons8d6e4332021-02-22 16:13:50 -050067}
68
Pete Bentley74c9bba2019-08-16 20:25:06 +010069type ObjectLinkerProperties struct {
70 // list of modules that should only provide headers for this module.
71 Header_libs []string `android:"arch_variant,variant_prepend"`
72
73 // names of other cc_object modules to link into this module using partial linking
74 Objs []string `android:"arch_variant"`
75
76 // if set, add an extra objcopy --prefix-symbols= step
77 Prefix_symbols *string
78
79 // if set, the path to a linker script to pass to ld -r when combining multiple object files.
80 Linker_script *string `android:"path,arch_variant"`
Dan Albert92fe7402020-07-15 13:33:30 -070081
82 // Indicates that this module is a CRT object. CRT objects will be split
83 // into a variant per-API level between min_sdk_version and current.
84 Crt *bool
Pete Bentley74c9bba2019-08-16 20:25:06 +010085}
86
Martin Stjernholm0b92ac82020-03-11 21:45:49 +000087func newObject() *Module {
88 module := newBaseModule(android.HostAndDeviceSupported, android.MultilibBoth)
89 module.sanitize = &sanitize{}
90 module.stl = &stl{}
91 return module
92}
93
Patrice Arrudabaff0ce2019-03-26 10:39:49 -070094// cc_object runs the compiler without running the linker. It is rarely
95// necessary, but sometimes used to generate .s files from .c files to use as
96// input to a cc_genrule module.
Colin Crosse40b4ea2018-10-02 22:25:58 -070097func ObjectFactory() android.Module {
Martin Stjernholm0b92ac82020-03-11 21:45:49 +000098 module := newObject()
Colin Crossb916a382016-07-29 17:28:03 -070099 module.linker = &objectLinker{
Peter Collingbourne1c648b82019-09-26 12:24:45 -0700100 baseLinker: NewBaseLinker(module.sanitize),
Colin Crossb916a382016-07-29 17:28:03 -0700101 }
102 module.compiler = NewBaseCompiler()
Chris Parsons8d6e4332021-02-22 16:13:50 -0500103 module.bazelHandler = &objectBazelHandler{module: module}
Peter Collingbourne486e42c2018-10-25 10:53:44 -0700104
105 // Clang's address-significance tables are incompatible with ld -r.
106 module.compiler.appendCflags([]string{"-fno-addrsig"})
107
Martin Stjernholmcd07bce2020-03-10 22:37:59 +0000108 module.sdkMemberTypes = []android.SdkMemberType{ccObjectSdkMemberType}
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500109
Colin Cross4d9c2d12016-07-29 12:48:20 -0700110 return module.Init()
111}
112
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500113// For bp2build conversion.
114type bazelObjectAttributes struct {
Jingwen Chened9c17d2021-04-13 07:14:55 +0000115 Srcs bazel.LabelListAttribute
116 Hdrs bazel.LabelListAttribute
117 Deps bazel.LabelListAttribute
118 Copts bazel.StringListAttribute
Liz Kammer6fd7b3f2021-05-06 13:54:29 -0400119 Asflags bazel.StringListAttribute
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500120}
121
122type bazelObject struct {
123 android.BazelTargetModuleBase
124 bazelObjectAttributes
125}
126
127func (m *bazelObject) Name() string {
128 return m.BaseModuleName()
129}
130
131func (m *bazelObject) GenerateAndroidBuildActions(ctx android.ModuleContext) {}
132
133func BazelObjectFactory() android.Module {
134 module := &bazelObject{}
135 module.AddProperties(&module.bazelObjectAttributes)
136 android.InitBazelTargetModule(module)
137 return module
138}
139
140// ObjectBp2Build is the bp2build converter from cc_object modules to the
141// Bazel equivalent target, plus any necessary include deps for the cc_object.
142func ObjectBp2Build(ctx android.TopDownMutatorContext) {
143 m, ok := ctx.Module().(*Module)
Jingwen Chen12b4c272021-03-10 02:05:59 -0500144 if !ok || !m.ConvertWithBp2build(ctx) {
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500145 return
146 }
147
148 // a Module can be something other than a cc_object.
149 if ctx.ModuleType() != "cc_object" {
150 return
151 }
152
153 if m.compiler == nil {
154 // a cc_object must have access to the compiler decorator for its props.
155 ctx.ModuleErrorf("compiler must not be nil for a cc_object module")
156 }
157
Jingwen Chen5d864492021-02-24 07:20:12 -0500158 // Set arch-specific configurable attributes
Jingwen Chen107c0de2021-04-09 10:43:12 +0000159 compilerAttrs := bp2BuildParseCompilerProps(ctx, m)
Liz Kammer6fd7b3f2021-05-06 13:54:29 -0400160 var asFlags bazel.StringListAttribute
Liz Kammera4aa4302021-03-18 16:56:36 -0400161
Jingwen Chen07027912021-03-15 06:02:43 -0400162 var deps bazel.LabelListAttribute
Jingwen Chendb120242021-02-23 00:46:47 -0500163 for _, props := range m.linker.linkerProps() {
164 if objectLinkerProps, ok := props.(*ObjectLinkerProperties); ok {
Jingwen Chen07027912021-03-15 06:02:43 -0400165 deps = bazel.MakeLabelListAttribute(
166 android.BazelLabelForModuleDeps(ctx, objectLinkerProps.Objs))
Jingwen Chendb120242021-02-23 00:46:47 -0500167 }
168 }
169
Liz Kammera060c452021-03-24 10:14:47 -0400170 productVariableProps := android.ProductVariableProperties(ctx)
171 if props, exists := productVariableProps["Asflags"]; exists {
172 // TODO(b/183595873): consider deduplicating handling of product variable properties
173 for _, prop := range props {
174 flags, ok := prop.Property.([]string)
175 if !ok {
176 ctx.ModuleErrorf("Could not convert product variable asflag property")
177 return
178 }
Liz Kammer6fd7b3f2021-05-06 13:54:29 -0400179 newFlags, _ := bazel.TryVariableSubstitutions(flags, prop.ProductConfigVariable)
180 asFlags.ProductValues = append(asFlags.ProductValues, bazel.ProductVariableValues{
181 ProductVariable: prop.ProductConfigVariable,
182 Values: newFlags,
183 })
Liz Kammera060c452021-03-24 10:14:47 -0400184 }
185 }
186 // TODO(b/183595872) warn/error if we're not handling product variables
187
Chris Parsons990c4f42021-05-25 12:10:58 -0400188 // Don't split cc_object srcs across languages. Doing so would add complexity,
189 // and this isn't typically done for cc_object.
190 srcs := compilerAttrs.srcs
191 srcs.Append(compilerAttrs.cSrcs)
192 srcs.Append(compilerAttrs.asSrcs)
193
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500194 attrs := &bazelObjectAttributes{
Chris Parsons990c4f42021-05-25 12:10:58 -0400195 Srcs: srcs,
Jingwen Chened9c17d2021-04-13 07:14:55 +0000196 Deps: deps,
197 Copts: compilerAttrs.copts,
198 Asflags: asFlags,
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500199 }
200
Liz Kammerfc46bc12021-02-19 11:06:17 -0500201 props := bazel.BazelTargetModuleProperties{
202 Rule_class: "cc_object",
203 Bzl_load_location: "//build/bazel/rules:cc_object.bzl",
204 }
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500205
Liz Kammerfc46bc12021-02-19 11:06:17 -0500206 ctx.CreateBazelTargetModule(BazelObjectFactory, m.Name(), props, attrs)
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500207}
208
Colin Cross4d9c2d12016-07-29 12:48:20 -0700209func (object *objectLinker) appendLdflags(flags []string) {
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700210 panic(fmt.Errorf("appendLdflags on objectLinker not supported"))
Colin Cross4d9c2d12016-07-29 12:48:20 -0700211}
212
Colin Cross42742b82016-08-01 13:20:05 -0700213func (object *objectLinker) linkerProps() []interface{} {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700214 return []interface{}{&object.Properties}
215}
216
Colin Cross42742b82016-08-01 13:20:05 -0700217func (*objectLinker) linkerInit(ctx BaseModuleContext) {}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700218
Colin Cross37047f12016-12-13 17:06:13 -0800219func (object *objectLinker) linkerDeps(ctx DepsContext, deps Deps) Deps {
Paul Duffina37832a2019-07-18 12:31:26 +0100220 deps.HeaderLibs = append(deps.HeaderLibs, object.Properties.Header_libs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700221 deps.ObjFiles = append(deps.ObjFiles, object.Properties.Objs...)
222 return deps
223}
224
Pete Bentley74c9bba2019-08-16 20:25:06 +0100225func (object *objectLinker) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Colin Cross4af21ed2019-11-04 09:37:55 -0800226 flags.Global.LdFlags = append(flags.Global.LdFlags, ctx.toolchain().ToolchainClangLdflags())
Colin Cross4d9c2d12016-07-29 12:48:20 -0700227
Pete Bentley74c9bba2019-08-16 20:25:06 +0100228 if lds := android.OptionalPathForModuleSrc(ctx, object.Properties.Linker_script); lds.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800229 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-T,"+lds.String())
Pete Bentley74c9bba2019-08-16 20:25:06 +0100230 flags.LdFlagsDeps = append(flags.LdFlagsDeps, lds.Path())
231 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700232 return flags
233}
234
235func (object *objectLinker) link(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700236 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700237
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700238 objs = objs.Append(deps.Objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700239
240 var outputFile android.Path
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700241 builderFlags := flagsToBuilderFlags(flags)
242
Pete Bentleyab65ba92019-10-18 12:39:56 +0100243 if len(objs.objFiles) == 1 && String(object.Properties.Linker_script) == "" {
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700244 outputFile = objs.objFiles[0]
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700245
Nan Zhang0007d812017-11-07 10:57:05 -0800246 if String(object.Properties.Prefix_symbols) != "" {
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700247 output := android.PathForModuleOut(ctx, ctx.ModuleName()+objectExtension)
Chris Parsonsbf4f55f2020-11-23 17:02:44 -0500248 transformBinaryPrefixSymbols(ctx, String(object.Properties.Prefix_symbols), outputFile,
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700249 builderFlags, output)
250 outputFile = output
251 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700252 } else {
253 output := android.PathForModuleOut(ctx, ctx.ModuleName()+objectExtension)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700254 outputFile = output
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700255
Nan Zhang0007d812017-11-07 10:57:05 -0800256 if String(object.Properties.Prefix_symbols) != "" {
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700257 input := android.PathForModuleOut(ctx, "unprefixed", ctx.ModuleName()+objectExtension)
Chris Parsonsbf4f55f2020-11-23 17:02:44 -0500258 transformBinaryPrefixSymbols(ctx, String(object.Properties.Prefix_symbols), input,
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700259 builderFlags, output)
260 output = input
261 }
262
Chris Parsonsbf4f55f2020-11-23 17:02:44 -0500263 transformObjsToObj(ctx, objs.objFiles, builderFlags, output, flags.LdFlagsDeps)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700264 }
265
266 ctx.CheckbuildFile(outputFile)
267 return outputFile
268}
Jiyong Parkaf6d8952019-01-31 12:21:23 +0900269
270func (object *objectLinker) unstrippedOutputFilePath() android.Path {
271 return nil
272}
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -0700273
274func (object *objectLinker) nativeCoverage() bool {
275 return true
276}
Jiyong Parkee9a98d2019-08-09 14:44:36 +0900277
278func (object *objectLinker) coverageOutputFilePath() android.OptionalPath {
279 return android.OptionalPath{}
280}
Inseob Kim1042d292020-06-01 23:23:05 +0900281
282func (object *objectLinker) object() bool {
283 return true
284}
Dan Albert92fe7402020-07-15 13:33:30 -0700285
286func (object *objectLinker) isCrt() bool {
287 return Bool(object.Properties.Crt)
288}