blob: 5b677dd977b684247f23a3d4e974bb89c5264a04 [file] [log] [blame]
Dan Willemsen218f6562015-07-08 18:13:11 -07001// Copyright 2015 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
Colin Cross635c3b02016-05-18 15:37:25 -070015package android
Dan Willemsen218f6562015-07-08 18:13:11 -070016
17import (
18 "bytes"
Dan Willemsen97750522016-02-09 17:43:51 -080019 "fmt"
Dan Willemsen218f6562015-07-08 18:13:11 -070020 "io"
21 "io/ioutil"
22 "os"
23 "path/filepath"
24 "sort"
Dan Willemsen0fda89f2016-06-01 15:25:32 -070025 "strings"
Dan Willemsen218f6562015-07-08 18:13:11 -070026
Dan Willemsen218f6562015-07-08 18:13:11 -070027 "github.com/google/blueprint"
Colin Cross2465c3d2018-09-28 10:19:18 -070028 "github.com/google/blueprint/bootstrap"
Dan Willemsen218f6562015-07-08 18:13:11 -070029)
30
31func init() {
Colin Cross798bfce2016-10-12 14:28:16 -070032 RegisterSingletonType("androidmk", AndroidMkSingleton)
Dan Willemsen218f6562015-07-08 18:13:11 -070033}
34
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -070035// Deprecated: consider using AndroidMkEntriesProvider instead, especially if you're not going to
36// use the Custom function.
Dan Willemsen218f6562015-07-08 18:13:11 -070037type AndroidMkDataProvider interface {
Colin Crossa18e9cf2017-08-10 17:00:19 -070038 AndroidMk() AndroidMkData
Colin Crossce75d2c2016-10-06 16:12:58 -070039 BaseModuleName() string
Dan Willemsen218f6562015-07-08 18:13:11 -070040}
41
42type AndroidMkData struct {
Sasha Smundakb6d23052019-04-01 18:37:36 -070043 Class string
44 SubName string
45 DistFile OptionalPath
46 OutputFile OptionalPath
47 Disabled bool
48 Include string
49 Required []string
50 Host_required []string
51 Target_required []string
Dan Willemsen218f6562015-07-08 18:13:11 -070052
Colin Cross0f86d182017-08-10 17:07:28 -070053 Custom func(w io.Writer, name, prefix, moduleDir string, data AndroidMkData)
Dan Willemsen218f6562015-07-08 18:13:11 -070054
Colin Cross27a4b052017-08-10 16:32:23 -070055 Extra []AndroidMkExtraFunc
Colin Cross0f86d182017-08-10 17:07:28 -070056
57 preamble bytes.Buffer
Dan Willemsen218f6562015-07-08 18:13:11 -070058}
59
Colin Cross27a4b052017-08-10 16:32:23 -070060type AndroidMkExtraFunc func(w io.Writer, outputFile Path)
61
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -070062// Allows modules to customize their Android*.mk output.
63type AndroidMkEntriesProvider interface {
64 AndroidMkEntries() AndroidMkEntries
65 BaseModuleName() string
66}
67
68type AndroidMkEntries struct {
69 Class string
70 SubName string
71 DistFile OptionalPath
72 OutputFile OptionalPath
73 Disabled bool
74 Include string
75 Required []string
76 Host_required []string
77 Target_required []string
78
79 header bytes.Buffer
80 footer bytes.Buffer
81
82 AddCustomEntries func(name, prefix, moduleDir string, entries *AndroidMkEntries)
83
84 EntryMap map[string][]string
85 entryOrder []string
86}
87
88func (a *AndroidMkEntries) SetString(name, value string) {
89 if _, ok := a.EntryMap[name]; !ok {
90 a.entryOrder = append(a.entryOrder, name)
91 }
92 a.EntryMap[name] = []string{value}
93}
94
95func (a *AndroidMkEntries) SetBoolIfTrue(name string, flag bool) {
96 if flag {
97 if _, ok := a.EntryMap[name]; !ok {
98 a.entryOrder = append(a.entryOrder, name)
99 }
100 a.EntryMap[name] = []string{"true"}
101 }
102}
103
104func (a *AndroidMkEntries) AddStrings(name string, value ...string) {
105 if len(value) == 0 {
106 return
107 }
108 if _, ok := a.EntryMap[name]; !ok {
109 a.entryOrder = append(a.entryOrder, name)
110 }
111 a.EntryMap[name] = append(a.EntryMap[name], value...)
112}
113
114func (a *AndroidMkEntries) fillInEntries(config Config, bpPath string, mod blueprint.Module) {
115 a.EntryMap = make(map[string][]string)
116 amod := mod.(Module).base()
117 name := amod.BaseModuleName()
118
119 if a.Include == "" {
120 a.Include = "$(BUILD_PREBUILT)"
121 }
122 a.Required = append(a.Required, amod.commonProperties.Required...)
123 a.Host_required = append(a.Host_required, amod.commonProperties.Host_required...)
124 a.Target_required = append(a.Target_required, amod.commonProperties.Target_required...)
125
126 // Fill in the header part.
127 if len(amod.commonProperties.Dist.Targets) > 0 {
128 distFile := a.DistFile
129 if !distFile.Valid() {
130 distFile = a.OutputFile
131 }
132 if distFile.Valid() {
133 dest := filepath.Base(distFile.String())
134
135 if amod.commonProperties.Dist.Dest != nil {
136 var err error
137 if dest, err = validateSafePath(*amod.commonProperties.Dist.Dest); err != nil {
138 // This was checked in ModuleBase.GenerateBuildActions
139 panic(err)
140 }
141 }
142
143 if amod.commonProperties.Dist.Suffix != nil {
144 ext := filepath.Ext(dest)
145 suffix := *amod.commonProperties.Dist.Suffix
146 dest = strings.TrimSuffix(dest, ext) + suffix + ext
147 }
148
149 if amod.commonProperties.Dist.Dir != nil {
150 var err error
151 if dest, err = validateSafePath(*amod.commonProperties.Dist.Dir, dest); err != nil {
152 // This was checked in ModuleBase.GenerateBuildActions
153 panic(err)
154 }
155 }
156
157 goals := strings.Join(amod.commonProperties.Dist.Targets, " ")
158 fmt.Fprintln(&a.header, ".PHONY:", goals)
159 fmt.Fprintf(&a.header, "$(call dist-for-goals,%s,%s:%s)\n",
160 goals, distFile.String(), dest)
161 }
162 }
163
164 fmt.Fprintln(&a.header, "\ninclude $(CLEAR_VARS)")
165
166 // Collect make variable assignment entries.
167 a.SetString("LOCAL_PATH", filepath.Dir(bpPath))
168 a.SetString("LOCAL_MODULE", name+a.SubName)
169 a.SetString("LOCAL_MODULE_CLASS", a.Class)
170 a.SetString("LOCAL_PREBUILT_MODULE_FILE", a.OutputFile.String())
171 a.AddStrings("LOCAL_REQUIRED_MODULES", a.Required...)
172 a.AddStrings("LOCAL_HOST_REQUIRED_MODULES", a.Host_required...)
173 a.AddStrings("LOCAL_TARGET_REQUIRED_MODULES", a.Target_required...)
174
175 archStr := amod.Arch().ArchType.String()
176 host := false
177 switch amod.Os().Class {
178 case Host:
179 // Make cannot identify LOCAL_MODULE_HOST_ARCH:= common.
180 if archStr != "common" {
181 a.SetString("LOCAL_MODULE_HOST_ARCH", archStr)
182 }
183 host = true
184 case HostCross:
185 // Make cannot identify LOCAL_MODULE_HOST_CROSS_ARCH:= common.
186 if archStr != "common" {
187 a.SetString("LOCAL_MODULE_HOST_CROSS_ARCH", archStr)
188 }
189 host = true
190 case Device:
191 // Make cannot identify LOCAL_MODULE_TARGET_ARCH:= common.
192 if archStr != "common" {
dimitry1f33e402019-03-26 12:39:31 +0100193 if amod.Target().NativeBridge {
dimitry8d6dde82019-07-11 10:23:53 +0200194 hostArchStr := amod.Target().NativeBridgeHostArchName
dimitry1f33e402019-03-26 12:39:31 +0100195 if hostArchStr != "" {
196 a.SetString("LOCAL_MODULE_TARGET_ARCH", hostArchStr)
197 }
198 } else {
199 a.SetString("LOCAL_MODULE_TARGET_ARCH", archStr)
200 }
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700201 }
202
203 a.AddStrings("LOCAL_INIT_RC", amod.commonProperties.Init_rc...)
204 a.AddStrings("LOCAL_VINTF_FRAGMENTS", amod.commonProperties.Vintf_fragments...)
205 a.SetBoolIfTrue("LOCAL_PROPRIETARY_MODULE", Bool(amod.commonProperties.Proprietary))
206 if Bool(amod.commonProperties.Vendor) || Bool(amod.commonProperties.Soc_specific) {
207 a.SetString("LOCAL_VENDOR_MODULE", "true")
208 }
209 a.SetBoolIfTrue("LOCAL_ODM_MODULE", Bool(amod.commonProperties.Device_specific))
210 a.SetBoolIfTrue("LOCAL_PRODUCT_MODULE", Bool(amod.commonProperties.Product_specific))
Justin Yund5f6c822019-06-25 16:47:17 +0900211 // TODO(b/135957588) product_services_specific is matched to LOCAL_PRODUCT_MODULE
212 // as a workaround. Remove this after clearing all Android.bp
213 a.SetBoolIfTrue("LOCAL_PRODUCT_MODULE", Bool(amod.commonProperties.Product_services_specific))
214 a.SetBoolIfTrue("LOCAL_SYSTEM_EXT_MODULE", Bool(amod.commonProperties.System_ext_specific))
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700215 if amod.commonProperties.Owner != nil {
216 a.SetString("LOCAL_MODULE_OWNER", *amod.commonProperties.Owner)
217 }
218 }
219
220 if amod.noticeFile.Valid() {
221 a.SetString("LOCAL_NOTICE_FILE", amod.noticeFile.String())
222 }
223
224 if host {
225 makeOs := amod.Os().String()
226 if amod.Os() == Linux || amod.Os() == LinuxBionic {
227 makeOs = "linux"
228 }
229 a.SetString("LOCAL_MODULE_HOST_OS", makeOs)
230 a.SetString("LOCAL_IS_HOST_MODULE", "true")
231 }
232
233 prefix := ""
234 if amod.ArchSpecific() {
235 switch amod.Os().Class {
236 case Host:
237 prefix = "HOST_"
238 case HostCross:
239 prefix = "HOST_CROSS_"
240 case Device:
241 prefix = "TARGET_"
242
243 }
244
245 if amod.Arch().ArchType != config.Targets[amod.Os()][0].Arch.ArchType {
246 prefix = "2ND_" + prefix
247 }
248 }
249 blueprintDir := filepath.Dir(bpPath)
250 if a.AddCustomEntries != nil {
251 a.AddCustomEntries(name, prefix, blueprintDir, a)
252 }
253
254 // Write to footer.
255 fmt.Fprintln(&a.footer, "include "+a.Include)
256}
257
258func (a *AndroidMkEntries) write(w io.Writer) {
259 w.Write(a.header.Bytes())
260 for _, name := range a.entryOrder {
261 fmt.Fprintln(w, name+" := "+strings.Join(a.EntryMap[name], " "))
262 }
263 w.Write(a.footer.Bytes())
264}
265
Colin Cross0875c522017-11-28 17:34:01 -0800266func AndroidMkSingleton() Singleton {
Dan Willemsen218f6562015-07-08 18:13:11 -0700267 return &androidMkSingleton{}
268}
269
270type androidMkSingleton struct{}
271
Colin Cross0875c522017-11-28 17:34:01 -0800272func (c *androidMkSingleton) GenerateBuildActions(ctx SingletonContext) {
Colin Crossaabf6792017-11-29 00:27:14 -0800273 if !ctx.Config().EmbeddedInMake() {
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800274 return
275 }
276
Colin Cross2465c3d2018-09-28 10:19:18 -0700277 var androidMkModulesList []blueprint.Module
Colin Cross4f6e4e62016-01-11 12:55:55 -0800278
Colin Cross2465c3d2018-09-28 10:19:18 -0700279 ctx.VisitAllModulesBlueprint(func(module blueprint.Module) {
Colin Cross0875c522017-11-28 17:34:01 -0800280 androidMkModulesList = append(androidMkModulesList, module)
Colin Cross4f6e4e62016-01-11 12:55:55 -0800281 })
Dan Willemsen218f6562015-07-08 18:13:11 -0700282
Colin Cross1ad81422019-01-14 12:47:35 -0800283 sort.SliceStable(androidMkModulesList, func(i, j int) bool {
284 return ctx.ModuleName(androidMkModulesList[i]) < ctx.ModuleName(androidMkModulesList[j])
285 })
Colin Crossd779da42015-12-17 18:00:23 -0800286
Dan Willemsen45133ac2018-03-09 21:22:06 -0800287 transMk := PathForOutput(ctx, "Android"+String(ctx.Config().productVariables.Make_suffix)+".mk")
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700288 if ctx.Failed() {
289 return
290 }
Dan Willemsen218f6562015-07-08 18:13:11 -0700291
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700292 err := translateAndroidMk(ctx, transMk.String(), androidMkModulesList)
Dan Willemsen218f6562015-07-08 18:13:11 -0700293 if err != nil {
294 ctx.Errorf(err.Error())
295 }
296
Colin Cross0875c522017-11-28 17:34:01 -0800297 ctx.Build(pctx, BuildParams{
298 Rule: blueprint.Phony,
299 Output: transMk,
Dan Willemsen218f6562015-07-08 18:13:11 -0700300 })
301}
302
Colin Cross2465c3d2018-09-28 10:19:18 -0700303func translateAndroidMk(ctx SingletonContext, mkFile string, mods []blueprint.Module) error {
Dan Willemsen218f6562015-07-08 18:13:11 -0700304 buf := &bytes.Buffer{}
305
Dan Willemsen97750522016-02-09 17:43:51 -0800306 fmt.Fprintln(buf, "LOCAL_MODULE_MAKEFILE := $(lastword $(MAKEFILE_LIST))")
Dan Willemsen218f6562015-07-08 18:13:11 -0700307
Dan Willemsen70e17fa2016-07-25 16:00:20 -0700308 type_stats := make(map[string]int)
Dan Willemsen218f6562015-07-08 18:13:11 -0700309 for _, mod := range mods {
310 err := translateAndroidMkModule(ctx, buf, mod)
311 if err != nil {
312 os.Remove(mkFile)
313 return err
314 }
Dan Willemsen70e17fa2016-07-25 16:00:20 -0700315
Colin Cross2465c3d2018-09-28 10:19:18 -0700316 if amod, ok := mod.(Module); ok && ctx.PrimaryModule(amod) == amod {
317 type_stats[ctx.ModuleType(amod)] += 1
Dan Willemsen70e17fa2016-07-25 16:00:20 -0700318 }
319 }
320
321 keys := []string{}
322 fmt.Fprintln(buf, "\nSTATS.SOONG_MODULE_TYPE :=")
323 for k := range type_stats {
324 keys = append(keys, k)
325 }
326 sort.Strings(keys)
327 for _, mod_type := range keys {
328 fmt.Fprintln(buf, "STATS.SOONG_MODULE_TYPE +=", mod_type)
329 fmt.Fprintf(buf, "STATS.SOONG_MODULE_TYPE.%s := %d\n", mod_type, type_stats[mod_type])
Dan Willemsen218f6562015-07-08 18:13:11 -0700330 }
331
332 // Don't write to the file if it hasn't changed
333 if _, err := os.Stat(mkFile); !os.IsNotExist(err) {
334 if data, err := ioutil.ReadFile(mkFile); err == nil {
335 matches := buf.Len() == len(data)
336
337 if matches {
338 for i, value := range buf.Bytes() {
339 if value != data[i] {
340 matches = false
341 break
342 }
343 }
344 }
345
346 if matches {
347 return nil
348 }
349 }
350 }
351
352 return ioutil.WriteFile(mkFile, buf.Bytes(), 0666)
353}
354
Colin Cross0875c522017-11-28 17:34:01 -0800355func translateAndroidMkModule(ctx SingletonContext, w io.Writer, mod blueprint.Module) error {
Colin Cross953d3a22018-09-05 16:23:54 -0700356 defer func() {
357 if r := recover(); r != nil {
358 panic(fmt.Errorf("%s in translateAndroidMkModule for module %s variant %s",
359 r, ctx.ModuleName(mod), ctx.ModuleSubDir(mod)))
360 }
361 }()
362
Colin Cross2465c3d2018-09-28 10:19:18 -0700363 switch x := mod.(type) {
364 case AndroidMkDataProvider:
365 return translateAndroidModule(ctx, w, mod, x)
366 case bootstrap.GoBinaryTool:
367 return translateGoBinaryModule(ctx, w, mod, x)
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700368 case AndroidMkEntriesProvider:
369 return translateAndroidMkEntriesModule(ctx, w, mod, x)
Colin Cross2465c3d2018-09-28 10:19:18 -0700370 default:
Dan Willemsen218f6562015-07-08 18:13:11 -0700371 return nil
372 }
Colin Cross2465c3d2018-09-28 10:19:18 -0700373}
374
375func translateGoBinaryModule(ctx SingletonContext, w io.Writer, mod blueprint.Module,
376 goBinary bootstrap.GoBinaryTool) error {
377
378 name := ctx.ModuleName(mod)
379 fmt.Fprintln(w, ".PHONY:", name)
380 fmt.Fprintln(w, name+":", goBinary.InstallPath())
381 fmt.Fprintln(w, "")
382
383 return nil
384}
385
386func translateAndroidModule(ctx SingletonContext, w io.Writer, mod blueprint.Module,
387 provider AndroidMkDataProvider) error {
Dan Willemsen218f6562015-07-08 18:13:11 -0700388
Colin Cross635c3b02016-05-18 15:37:25 -0700389 amod := mod.(Module).base()
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700390 if shouldSkipAndroidMkProcessing(amod) {
Jeff Gaston088e29e2017-11-29 16:47:17 -0800391 return nil
392 }
393
Colin Cross91825d22017-08-10 16:59:47 -0700394 data := provider.AndroidMk()
Colin Cross53499412017-09-07 13:20:25 -0700395 if data.Include == "" {
396 data.Include = "$(BUILD_PREBUILT)"
397 }
398
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700399 // Get the preamble content through AndroidMkEntries logic.
400 entries := AndroidMkEntries{
401 Class: data.Class,
402 SubName: data.SubName,
403 DistFile: data.DistFile,
404 OutputFile: data.OutputFile,
405 Disabled: data.Disabled,
406 Include: data.Include,
407 Required: data.Required,
408 Host_required: data.Host_required,
409 Target_required: data.Target_required,
Dan Willemsen01a405a2016-06-13 17:19:03 -0700410 }
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700411 entries.fillInEntries(ctx.Config(), ctx.BlueprintFile(mod), mod)
412 // preamble doesn't need the footer content.
413 entries.footer = bytes.Buffer{}
414 entries.write(&data.preamble)
Dan Willemsen01a405a2016-06-13 17:19:03 -0700415
Colin Cross0f86d182017-08-10 17:07:28 -0700416 prefix := ""
417 if amod.ArchSpecific() {
418 switch amod.Os().Class {
419 case Host:
420 prefix = "HOST_"
421 case HostCross:
422 prefix = "HOST_CROSS_"
423 case Device:
424 prefix = "TARGET_"
Colin Crossa2344662016-03-24 13:14:12 -0700425
Dan Willemsen218f6562015-07-08 18:13:11 -0700426 }
427
Dan Willemsen0ef639b2018-10-10 17:02:29 -0700428 if amod.Arch().ArchType != ctx.Config().Targets[amod.Os()][0].Arch.ArchType {
Colin Cross0f86d182017-08-10 17:07:28 -0700429 prefix = "2ND_" + prefix
430 }
Dan Willemsen218f6562015-07-08 18:13:11 -0700431 }
432
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700433 name := provider.BaseModuleName()
Colin Cross0f86d182017-08-10 17:07:28 -0700434 blueprintDir := filepath.Dir(ctx.BlueprintFile(mod))
435
436 if data.Custom != nil {
437 data.Custom(w, name, prefix, blueprintDir, data)
438 } else {
439 WriteAndroidMkData(w, data)
440 }
441
442 return nil
443}
444
445func WriteAndroidMkData(w io.Writer, data AndroidMkData) {
446 if data.Disabled {
447 return
448 }
449
450 if !data.OutputFile.Valid() {
451 return
452 }
453
454 w.Write(data.preamble.Bytes())
455
Colin Crossca860ac2016-01-04 14:34:37 -0800456 for _, extra := range data.Extra {
Colin Cross27a4b052017-08-10 16:32:23 -0700457 extra(w, data.OutputFile.Path())
Dan Willemsen97750522016-02-09 17:43:51 -0800458 }
459
Colin Cross53499412017-09-07 13:20:25 -0700460 fmt.Fprintln(w, "include "+data.Include)
Dan Willemsen218f6562015-07-08 18:13:11 -0700461}
Sasha Smundakb6d23052019-04-01 18:37:36 -0700462
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700463func translateAndroidMkEntriesModule(ctx SingletonContext, w io.Writer, mod blueprint.Module,
464 provider AndroidMkEntriesProvider) error {
465 if shouldSkipAndroidMkProcessing(mod.(Module).base()) {
466 return nil
Sasha Smundakb6d23052019-04-01 18:37:36 -0700467 }
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700468
469 entries := provider.AndroidMkEntries()
470 entries.fillInEntries(ctx.Config(), ctx.BlueprintFile(mod), mod)
471
472 entries.write(w)
473
474 return nil
475}
476
477func shouldSkipAndroidMkProcessing(module *ModuleBase) bool {
478 if !module.commonProperties.NamespaceExportedToMake {
479 // TODO(jeffrygaston) do we want to validate that there are no modules being
480 // exported to Kati that depend on this module?
481 return true
Sasha Smundakb6d23052019-04-01 18:37:36 -0700482 }
Jaewoong Jung9aa3ab12019-04-03 15:47:29 -0700483
484 return !module.Enabled() ||
485 module.commonProperties.SkipInstall ||
486 // Make does not understand LinuxBionic
487 module.Os() == LinuxBionic
Sasha Smundakb6d23052019-04-01 18:37:36 -0700488}