blob: 9bb279a3b46e735af64dc6726dcf24d01dd3037a [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 Chen07027912021-03-15 06:02:43 -0400115 Srcs bazel.LabelListAttribute
Jingwen Chen63930982021-03-24 10:04:33 -0400116 Hdrs bazel.LabelListAttribute
Jingwen Chen07027912021-03-15 06:02:43 -0400117 Deps bazel.LabelListAttribute
Jingwen Chen5d864492021-02-24 07:20:12 -0500118 Copts bazel.StringListAttribute
Liz Kammera060c452021-03-24 10:14:47 -0400119 Asflags []string
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500120 Local_include_dirs []string
121}
122
123type bazelObject struct {
124 android.BazelTargetModuleBase
125 bazelObjectAttributes
126}
127
128func (m *bazelObject) Name() string {
129 return m.BaseModuleName()
130}
131
132func (m *bazelObject) GenerateAndroidBuildActions(ctx android.ModuleContext) {}
133
134func BazelObjectFactory() android.Module {
135 module := &bazelObject{}
136 module.AddProperties(&module.bazelObjectAttributes)
137 android.InitBazelTargetModule(module)
138 return module
139}
140
141// ObjectBp2Build is the bp2build converter from cc_object modules to the
142// Bazel equivalent target, plus any necessary include deps for the cc_object.
143func ObjectBp2Build(ctx android.TopDownMutatorContext) {
144 m, ok := ctx.Module().(*Module)
Jingwen Chen12b4c272021-03-10 02:05:59 -0500145 if !ok || !m.ConvertWithBp2build(ctx) {
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500146 return
147 }
148
149 // a Module can be something other than a cc_object.
150 if ctx.ModuleType() != "cc_object" {
151 return
152 }
153
154 if m.compiler == nil {
155 // a cc_object must have access to the compiler decorator for its props.
156 ctx.ModuleErrorf("compiler must not be nil for a cc_object module")
157 }
158
Jingwen Chen5d864492021-02-24 07:20:12 -0500159 // Set arch-specific configurable attributes
Jingwen Chen107c0de2021-04-09 10:43:12 +0000160 compilerAttrs := bp2BuildParseCompilerProps(ctx, m)
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500161 var localIncludeDirs []string
Liz Kammera060c452021-03-24 10:14:47 -0400162 var asFlags []string
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500163 for _, props := range m.compiler.compilerProps() {
164 if baseCompilerProps, ok := props.(*BaseCompilerProperties); ok {
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500165 localIncludeDirs = baseCompilerProps.Local_include_dirs
166 break
167 }
168 }
169
Liz Kammera4aa4302021-03-18 16:56:36 -0400170 if c, ok := m.compiler.(*baseCompiler); ok && c.includeBuildDirectory() {
171 localIncludeDirs = append(localIncludeDirs, ".")
172 }
173
Jingwen Chen07027912021-03-15 06:02:43 -0400174 var deps bazel.LabelListAttribute
Jingwen Chendb120242021-02-23 00:46:47 -0500175 for _, props := range m.linker.linkerProps() {
176 if objectLinkerProps, ok := props.(*ObjectLinkerProperties); ok {
Jingwen Chen07027912021-03-15 06:02:43 -0400177 deps = bazel.MakeLabelListAttribute(
178 android.BazelLabelForModuleDeps(ctx, objectLinkerProps.Objs))
Jingwen Chendb120242021-02-23 00:46:47 -0500179 }
180 }
181
Liz Kammera060c452021-03-24 10:14:47 -0400182 productVariableProps := android.ProductVariableProperties(ctx)
183 if props, exists := productVariableProps["Asflags"]; exists {
184 // TODO(b/183595873): consider deduplicating handling of product variable properties
185 for _, prop := range props {
186 flags, ok := prop.Property.([]string)
187 if !ok {
188 ctx.ModuleErrorf("Could not convert product variable asflag property")
189 return
190 }
191 // TODO(b/183595873) handle other product variable usages -- as selects?
192 if newFlags, subbed := bazel.TryVariableSubstitutions(flags, prop.ProductConfigVariable); subbed {
193 asFlags = append(asFlags, newFlags...)
194 }
195 }
196 }
197 // TODO(b/183595872) warn/error if we're not handling product variables
198
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500199 attrs := &bazelObjectAttributes{
Jingwen Chen107c0de2021-04-09 10:43:12 +0000200 Srcs: compilerAttrs.srcs,
201 Hdrs: compilerAttrs.hdrs,
Jingwen Chendb120242021-02-23 00:46:47 -0500202 Deps: deps,
Jingwen Chen107c0de2021-04-09 10:43:12 +0000203 Copts: compilerAttrs.copts,
Liz Kammera060c452021-03-24 10:14:47 -0400204 Asflags: asFlags,
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500205 Local_include_dirs: localIncludeDirs,
206 }
207
Liz Kammerfc46bc12021-02-19 11:06:17 -0500208 props := bazel.BazelTargetModuleProperties{
209 Rule_class: "cc_object",
210 Bzl_load_location: "//build/bazel/rules:cc_object.bzl",
211 }
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500212
Liz Kammerfc46bc12021-02-19 11:06:17 -0500213 ctx.CreateBazelTargetModule(BazelObjectFactory, m.Name(), props, attrs)
Jingwen Chen8c1b97e2021-02-18 03:21:34 -0500214}
215
Colin Cross4d9c2d12016-07-29 12:48:20 -0700216func (object *objectLinker) appendLdflags(flags []string) {
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700217 panic(fmt.Errorf("appendLdflags on objectLinker not supported"))
Colin Cross4d9c2d12016-07-29 12:48:20 -0700218}
219
Colin Cross42742b82016-08-01 13:20:05 -0700220func (object *objectLinker) linkerProps() []interface{} {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700221 return []interface{}{&object.Properties}
222}
223
Colin Cross42742b82016-08-01 13:20:05 -0700224func (*objectLinker) linkerInit(ctx BaseModuleContext) {}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700225
Colin Cross37047f12016-12-13 17:06:13 -0800226func (object *objectLinker) linkerDeps(ctx DepsContext, deps Deps) Deps {
Paul Duffina37832a2019-07-18 12:31:26 +0100227 deps.HeaderLibs = append(deps.HeaderLibs, object.Properties.Header_libs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700228 deps.ObjFiles = append(deps.ObjFiles, object.Properties.Objs...)
229 return deps
230}
231
Pete Bentley74c9bba2019-08-16 20:25:06 +0100232func (object *objectLinker) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Colin Cross4af21ed2019-11-04 09:37:55 -0800233 flags.Global.LdFlags = append(flags.Global.LdFlags, ctx.toolchain().ToolchainClangLdflags())
Colin Cross4d9c2d12016-07-29 12:48:20 -0700234
Pete Bentley74c9bba2019-08-16 20:25:06 +0100235 if lds := android.OptionalPathForModuleSrc(ctx, object.Properties.Linker_script); lds.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800236 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-T,"+lds.String())
Pete Bentley74c9bba2019-08-16 20:25:06 +0100237 flags.LdFlagsDeps = append(flags.LdFlagsDeps, lds.Path())
238 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700239 return flags
240}
241
242func (object *objectLinker) link(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700243 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700244
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700245 objs = objs.Append(deps.Objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700246
247 var outputFile android.Path
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700248 builderFlags := flagsToBuilderFlags(flags)
249
Pete Bentleyab65ba92019-10-18 12:39:56 +0100250 if len(objs.objFiles) == 1 && String(object.Properties.Linker_script) == "" {
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700251 outputFile = objs.objFiles[0]
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700252
Nan Zhang0007d812017-11-07 10:57:05 -0800253 if String(object.Properties.Prefix_symbols) != "" {
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700254 output := android.PathForModuleOut(ctx, ctx.ModuleName()+objectExtension)
Chris Parsonsbf4f55f2020-11-23 17:02:44 -0500255 transformBinaryPrefixSymbols(ctx, String(object.Properties.Prefix_symbols), outputFile,
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700256 builderFlags, output)
257 outputFile = output
258 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700259 } else {
260 output := android.PathForModuleOut(ctx, ctx.ModuleName()+objectExtension)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700261 outputFile = output
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700262
Nan Zhang0007d812017-11-07 10:57:05 -0800263 if String(object.Properties.Prefix_symbols) != "" {
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700264 input := android.PathForModuleOut(ctx, "unprefixed", ctx.ModuleName()+objectExtension)
Chris Parsonsbf4f55f2020-11-23 17:02:44 -0500265 transformBinaryPrefixSymbols(ctx, String(object.Properties.Prefix_symbols), input,
Dan Willemsenefb1dd92017-09-18 22:47:20 -0700266 builderFlags, output)
267 output = input
268 }
269
Chris Parsonsbf4f55f2020-11-23 17:02:44 -0500270 transformObjsToObj(ctx, objs.objFiles, builderFlags, output, flags.LdFlagsDeps)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700271 }
272
273 ctx.CheckbuildFile(outputFile)
274 return outputFile
275}
Jiyong Parkaf6d8952019-01-31 12:21:23 +0900276
277func (object *objectLinker) unstrippedOutputFilePath() android.Path {
278 return nil
279}
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -0700280
281func (object *objectLinker) nativeCoverage() bool {
282 return true
283}
Jiyong Parkee9a98d2019-08-09 14:44:36 +0900284
285func (object *objectLinker) coverageOutputFilePath() android.OptionalPath {
286 return android.OptionalPath{}
287}
Inseob Kim1042d292020-06-01 23:23:05 +0900288
289func (object *objectLinker) object() bool {
290 return true
291}
Dan Albert92fe7402020-07-15 13:33:30 -0700292
293func (object *objectLinker) isCrt() bool {
294 return Bool(object.Properties.Crt)
295}