blob: 2440ca13f3b8ea59d819c97decbda45c78a4ef36 [file] [log] [blame]
Jingwen Chen30f5aaa2020-11-19 05:38:02 -05001// Copyright 2020 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 bazel
16
Rupert Shuttleworth2e4219b2021-03-12 11:04:21 +000017import (
18 "fmt"
Liz Kammera060c452021-03-24 10:14:47 -040019 "regexp"
Rupert Shuttleworth2e4219b2021-03-12 11:04:21 +000020 "sort"
21)
Jingwen Chen5d864492021-02-24 07:20:12 -050022
Jingwen Chen73850672020-12-14 08:25:34 -050023// BazelTargetModuleProperties contain properties and metadata used for
24// Blueprint to BUILD file conversion.
25type BazelTargetModuleProperties struct {
26 // The Bazel rule class for this target.
Liz Kammerfc46bc12021-02-19 11:06:17 -050027 Rule_class string `blueprint:"mutated"`
Jingwen Chen40067de2021-01-26 21:58:43 -050028
29 // The target label for the bzl file containing the definition of the rule class.
Liz Kammerfc46bc12021-02-19 11:06:17 -050030 Bzl_load_location string `blueprint:"mutated"`
Jingwen Chen73850672020-12-14 08:25:34 -050031}
Liz Kammer356f7d42021-01-26 09:18:53 -050032
Jingwen Chenfb4692a2021-02-07 10:05:16 -050033const BazelTargetModuleNamePrefix = "__bp2build__"
34
Liz Kammera060c452021-03-24 10:14:47 -040035var productVariableSubstitutionPattern = regexp.MustCompile("%(d|s)")
36
Liz Kammer356f7d42021-01-26 09:18:53 -050037// Label is used to represent a Bazel compatible Label. Also stores the original bp text to support
38// string replacement.
39type Label struct {
40 Bp_text string
41 Label string
42}
43
44// LabelList is used to represent a list of Bazel labels.
45type LabelList struct {
46 Includes []Label
47 Excludes []Label
48}
49
50// Append appends the fields of other labelList to the corresponding fields of ll.
51func (ll *LabelList) Append(other LabelList) {
52 if len(ll.Includes) > 0 || len(other.Includes) > 0 {
53 ll.Includes = append(ll.Includes, other.Includes...)
54 }
55 if len(ll.Excludes) > 0 || len(other.Excludes) > 0 {
56 ll.Excludes = append(other.Excludes, other.Excludes...)
57 }
58}
Jingwen Chen5d864492021-02-24 07:20:12 -050059
Rupert Shuttleworth2e4219b2021-03-12 11:04:21 +000060func UniqueBazelLabels(originalLabels []Label) []Label {
61 uniqueLabelsSet := make(map[Label]bool)
62 for _, l := range originalLabels {
63 uniqueLabelsSet[l] = true
64 }
65 var uniqueLabels []Label
66 for l, _ := range uniqueLabelsSet {
67 uniqueLabels = append(uniqueLabels, l)
68 }
69 sort.SliceStable(uniqueLabels, func(i, j int) bool {
70 return uniqueLabels[i].Label < uniqueLabels[j].Label
71 })
72 return uniqueLabels
73}
74
75func UniqueBazelLabelList(originalLabelList LabelList) LabelList {
76 var uniqueLabelList LabelList
77 uniqueLabelList.Includes = UniqueBazelLabels(originalLabelList.Includes)
78 uniqueLabelList.Excludes = UniqueBazelLabels(originalLabelList.Excludes)
79 return uniqueLabelList
80}
81
Jingwen Chen07027912021-03-15 06:02:43 -040082const (
Jingwen Chen91220d72021-03-24 02:18:33 -040083 // ArchType names in arch.go
Jingwen Chen07027912021-03-15 06:02:43 -040084 ARCH_ARM = "arm"
85 ARCH_ARM64 = "arm64"
Jingwen Chen91220d72021-03-24 02:18:33 -040086 ARCH_X86 = "x86"
87 ARCH_X86_64 = "x86_64"
88
89 // OsType names in arch.go
90 OS_ANDROID = "android"
91 OS_DARWIN = "darwin"
92 OS_FUCHSIA = "fuchsia"
93 OS_LINUX = "linux_glibc"
94 OS_LINUX_BIONIC = "linux_bionic"
95 OS_WINDOWS = "windows"
Jingwen Chen07027912021-03-15 06:02:43 -040096)
97
98var (
Jingwen Chenc1c26502021-04-05 10:35:13 +000099 // These are the list of OSes and architectures with a Bazel config_setting
100 // and constraint value equivalent. These exist in arch.go, but the android
101 // package depends on the bazel package, so a cyclic dependency prevents
102 // using those variables here.
Jingwen Chen91220d72021-03-24 02:18:33 -0400103
104 // A map of architectures to the Bazel label of the constraint_value
105 // for the @platforms//cpu:cpu constraint_setting
106 PlatformArchMap = map[string]string{
107 ARCH_ARM: "//build/bazel/platforms/arch:arm",
108 ARCH_ARM64: "//build/bazel/platforms/arch:arm64",
109 ARCH_X86: "//build/bazel/platforms/arch:x86",
110 ARCH_X86_64: "//build/bazel/platforms/arch:x86_64",
111 }
112
113 // A map of target operating systems to the Bazel label of the
114 // constraint_value for the @platforms//os:os constraint_setting
115 PlatformOsMap = map[string]string{
116 OS_ANDROID: "//build/bazel/platforms/os:android",
117 OS_DARWIN: "//build/bazel/platforms/os:darwin",
118 OS_FUCHSIA: "//build/bazel/platforms/os:fuchsia",
119 OS_LINUX: "//build/bazel/platforms/os:linux",
120 OS_LINUX_BIONIC: "//build/bazel/platforms/os:linux_bionic",
121 OS_WINDOWS: "//build/bazel/platforms/os:windows",
122 }
Jingwen Chen07027912021-03-15 06:02:43 -0400123)
124
Jingwen Chenc1c26502021-04-05 10:35:13 +0000125type Attribute interface {
126 HasConfigurableValues() bool
127}
128
Jingwen Chen07027912021-03-15 06:02:43 -0400129// Arch-specific label_list typed Bazel attribute values. This should correspond
130// to the types of architectures supported for compilation in arch.go.
131type labelListArchValues struct {
132 X86 LabelList
133 X86_64 LabelList
134 Arm LabelList
135 Arm64 LabelList
Jingwen Chen91220d72021-03-24 02:18:33 -0400136 Common LabelList
137}
138
139type labelListOsValues struct {
140 Android LabelList
141 Darwin LabelList
142 Fuchsia LabelList
143 Linux LabelList
144 LinuxBionic LabelList
145 Windows LabelList
Jingwen Chen07027912021-03-15 06:02:43 -0400146}
147
148// LabelListAttribute is used to represent a list of Bazel labels as an
149// attribute.
150type LabelListAttribute struct {
151 // The non-arch specific attribute label list Value. Required.
152 Value LabelList
153
154 // The arch-specific attribute label list values. Optional. If used, these
155 // are generated in a select statement and appended to the non-arch specific
156 // label list Value.
157 ArchValues labelListArchValues
Jingwen Chen91220d72021-03-24 02:18:33 -0400158
159 // The os-specific attribute label list values. Optional. If used, these
160 // are generated in a select statement and appended to the non-os specific
161 // label list Value.
162 OsValues labelListOsValues
Jingwen Chen07027912021-03-15 06:02:43 -0400163}
164
165// MakeLabelListAttribute initializes a LabelListAttribute with the non-arch specific value.
166func MakeLabelListAttribute(value LabelList) LabelListAttribute {
167 return LabelListAttribute{Value: UniqueBazelLabelList(value)}
168}
169
170// HasArchSpecificValues returns true if the attribute contains
171// architecture-specific label_list values.
Jingwen Chenc1c26502021-04-05 10:35:13 +0000172func (attrs LabelListAttribute) HasConfigurableValues() bool {
173 for arch := range PlatformArchMap {
Jingwen Chen91220d72021-03-24 02:18:33 -0400174 if len(attrs.GetValueForArch(arch).Includes) > 0 {
175 return true
176 }
177 }
178
Jingwen Chenc1c26502021-04-05 10:35:13 +0000179 for os := range PlatformOsMap {
Jingwen Chen91220d72021-03-24 02:18:33 -0400180 if len(attrs.GetValueForOS(os).Includes) > 0 {
Jingwen Chen07027912021-03-15 06:02:43 -0400181 return true
182 }
183 }
184 return false
185}
186
Jingwen Chen91220d72021-03-24 02:18:33 -0400187func (attrs *LabelListAttribute) archValuePtrs() map[string]*LabelList {
188 return map[string]*LabelList{
189 ARCH_X86: &attrs.ArchValues.X86,
190 ARCH_X86_64: &attrs.ArchValues.X86_64,
191 ARCH_ARM: &attrs.ArchValues.Arm,
192 ARCH_ARM64: &attrs.ArchValues.Arm64,
193 }
194}
195
Jingwen Chen07027912021-03-15 06:02:43 -0400196// GetValueForArch returns the label_list attribute value for an architecture.
197func (attrs *LabelListAttribute) GetValueForArch(arch string) LabelList {
Jingwen Chen91220d72021-03-24 02:18:33 -0400198 var v *LabelList
199 if v = attrs.archValuePtrs()[arch]; v == nil {
Jingwen Chen07027912021-03-15 06:02:43 -0400200 panic(fmt.Errorf("Unknown arch: %s", arch))
201 }
Jingwen Chen91220d72021-03-24 02:18:33 -0400202 return *v
Jingwen Chen07027912021-03-15 06:02:43 -0400203}
204
205// SetValueForArch sets the label_list attribute value for an architecture.
206func (attrs *LabelListAttribute) SetValueForArch(arch string, value LabelList) {
Jingwen Chen91220d72021-03-24 02:18:33 -0400207 var v *LabelList
208 if v = attrs.archValuePtrs()[arch]; v == nil {
Jingwen Chen07027912021-03-15 06:02:43 -0400209 panic(fmt.Errorf("Unknown arch: %s", arch))
210 }
Jingwen Chen91220d72021-03-24 02:18:33 -0400211 *v = value
212}
213
214func (attrs *LabelListAttribute) osValuePtrs() map[string]*LabelList {
215 return map[string]*LabelList{
216 OS_ANDROID: &attrs.OsValues.Android,
217 OS_DARWIN: &attrs.OsValues.Darwin,
218 OS_FUCHSIA: &attrs.OsValues.Fuchsia,
219 OS_LINUX: &attrs.OsValues.Linux,
220 OS_LINUX_BIONIC: &attrs.OsValues.LinuxBionic,
221 OS_WINDOWS: &attrs.OsValues.Windows,
222 }
223}
224
225// GetValueForOS returns the label_list attribute value for an OS target.
226func (attrs *LabelListAttribute) GetValueForOS(os string) LabelList {
227 var v *LabelList
228 if v = attrs.osValuePtrs()[os]; v == nil {
229 panic(fmt.Errorf("Unknown os: %s", os))
230 }
231 return *v
232}
233
234// SetValueForArch sets the label_list attribute value for an OS target.
235func (attrs *LabelListAttribute) SetValueForOS(os string, value LabelList) {
236 var v *LabelList
237 if v = attrs.osValuePtrs()[os]; v == nil {
238 panic(fmt.Errorf("Unknown os: %s", os))
239 }
240 *v = value
Jingwen Chen07027912021-03-15 06:02:43 -0400241}
242
Jingwen Chen5d864492021-02-24 07:20:12 -0500243// StringListAttribute corresponds to the string_list Bazel attribute type with
244// support for additional metadata, like configurations.
245type StringListAttribute struct {
246 // The base value of the string list attribute.
247 Value []string
248
Jingwen Chenc1c26502021-04-05 10:35:13 +0000249 // The arch-specific attribute string list values. Optional. If used, these
250 // are generated in a select statement and appended to the non-arch specific
251 // label list Value.
Jingwen Chen5d864492021-02-24 07:20:12 -0500252 ArchValues stringListArchValues
Jingwen Chenc1c26502021-04-05 10:35:13 +0000253
254 // The os-specific attribute string list values. Optional. If used, these
255 // are generated in a select statement and appended to the non-os specific
256 // label list Value.
257 OsValues stringListOsValues
Jingwen Chen5d864492021-02-24 07:20:12 -0500258}
259
260// Arch-specific string_list typed Bazel attribute values. This should correspond
261// to the types of architectures supported for compilation in arch.go.
262type stringListArchValues struct {
Jingwen Chen07027912021-03-15 06:02:43 -0400263 X86 []string
264 X86_64 []string
265 Arm []string
266 Arm64 []string
Jingwen Chen91220d72021-03-24 02:18:33 -0400267 Common []string
Jingwen Chen5d864492021-02-24 07:20:12 -0500268}
269
Jingwen Chenc1c26502021-04-05 10:35:13 +0000270type stringListOsValues struct {
271 Android []string
272 Darwin []string
273 Fuchsia []string
274 Linux []string
275 LinuxBionic []string
276 Windows []string
277}
278
Jingwen Chen91220d72021-03-24 02:18:33 -0400279// HasConfigurableValues returns true if the attribute contains
Jingwen Chen5d864492021-02-24 07:20:12 -0500280// architecture-specific string_list values.
Jingwen Chenc1c26502021-04-05 10:35:13 +0000281func (attrs StringListAttribute) HasConfigurableValues() bool {
282 for arch := range PlatformArchMap {
Jingwen Chen5d864492021-02-24 07:20:12 -0500283 if len(attrs.GetValueForArch(arch)) > 0 {
284 return true
285 }
286 }
Jingwen Chenc1c26502021-04-05 10:35:13 +0000287
288 for os := range PlatformOsMap {
289 if len(attrs.GetValueForOS(os)) > 0 {
290 return true
291 }
292 }
Jingwen Chen5d864492021-02-24 07:20:12 -0500293 return false
294}
295
Jingwen Chen91220d72021-03-24 02:18:33 -0400296func (attrs *StringListAttribute) archValuePtrs() map[string]*[]string {
297 return map[string]*[]string{
298 ARCH_X86: &attrs.ArchValues.X86,
299 ARCH_X86_64: &attrs.ArchValues.X86_64,
300 ARCH_ARM: &attrs.ArchValues.Arm,
301 ARCH_ARM64: &attrs.ArchValues.Arm64,
302 }
303}
304
Jingwen Chen5d864492021-02-24 07:20:12 -0500305// GetValueForArch returns the string_list attribute value for an architecture.
306func (attrs *StringListAttribute) GetValueForArch(arch string) []string {
Jingwen Chen91220d72021-03-24 02:18:33 -0400307 var v *[]string
308 if v = attrs.archValuePtrs()[arch]; v == nil {
Jingwen Chen5d864492021-02-24 07:20:12 -0500309 panic(fmt.Errorf("Unknown arch: %s", arch))
310 }
Jingwen Chen91220d72021-03-24 02:18:33 -0400311 return *v
Jingwen Chen5d864492021-02-24 07:20:12 -0500312}
313
314// SetValueForArch sets the string_list attribute value for an architecture.
315func (attrs *StringListAttribute) SetValueForArch(arch string, value []string) {
Jingwen Chen91220d72021-03-24 02:18:33 -0400316 var v *[]string
317 if v = attrs.archValuePtrs()[arch]; v == nil {
Jingwen Chen5d864492021-02-24 07:20:12 -0500318 panic(fmt.Errorf("Unknown arch: %s", arch))
319 }
Jingwen Chen91220d72021-03-24 02:18:33 -0400320 *v = value
Jingwen Chen5d864492021-02-24 07:20:12 -0500321}
Liz Kammera060c452021-03-24 10:14:47 -0400322
Jingwen Chenc1c26502021-04-05 10:35:13 +0000323func (attrs *StringListAttribute) osValuePtrs() map[string]*[]string {
324 return map[string]*[]string{
325 OS_ANDROID: &attrs.OsValues.Android,
326 OS_DARWIN: &attrs.OsValues.Darwin,
327 OS_FUCHSIA: &attrs.OsValues.Fuchsia,
328 OS_LINUX: &attrs.OsValues.Linux,
329 OS_LINUX_BIONIC: &attrs.OsValues.LinuxBionic,
330 OS_WINDOWS: &attrs.OsValues.Windows,
331 }
332}
333
334// GetValueForOS returns the string_list attribute value for an OS target.
335func (attrs *StringListAttribute) GetValueForOS(os string) []string {
336 var v *[]string
337 if v = attrs.osValuePtrs()[os]; v == nil {
338 panic(fmt.Errorf("Unknown os: %s", os))
339 }
340 return *v
341}
342
343// SetValueForArch sets the string_list attribute value for an OS target.
344func (attrs *StringListAttribute) SetValueForOS(os string, value []string) {
345 var v *[]string
346 if v = attrs.osValuePtrs()[os]; v == nil {
347 panic(fmt.Errorf("Unknown os: %s", os))
348 }
349 *v = value
350}
351
Liz Kammera060c452021-03-24 10:14:47 -0400352// TryVariableSubstitution, replace string substitution formatting within each string in slice with
353// Starlark string.format compatible tag for productVariable.
354func TryVariableSubstitutions(slice []string, productVariable string) ([]string, bool) {
355 ret := make([]string, 0, len(slice))
356 changesMade := false
357 for _, s := range slice {
358 newS, changed := TryVariableSubstitution(s, productVariable)
359 ret = append(ret, newS)
360 changesMade = changesMade || changed
361 }
362 return ret, changesMade
363}
364
365// TryVariableSubstitution, replace string substitution formatting within s with Starlark
366// string.format compatible tag for productVariable.
367func TryVariableSubstitution(s string, productVariable string) (string, bool) {
368 sub := productVariableSubstitutionPattern.ReplaceAllString(s, "{"+productVariable+"}")
369 return sub, s != sub
370}