250 lines
7.7 KiB
Go
250 lines
7.7 KiB
Go
// Copyright 2020 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 bazel
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"sort"
|
|
)
|
|
|
|
// BazelTargetModuleProperties contain properties and metadata used for
|
|
// Blueprint to BUILD file conversion.
|
|
type BazelTargetModuleProperties struct {
|
|
// The Bazel rule class for this target.
|
|
Rule_class string `blueprint:"mutated"`
|
|
|
|
// The target label for the bzl file containing the definition of the rule class.
|
|
Bzl_load_location string `blueprint:"mutated"`
|
|
}
|
|
|
|
const BazelTargetModuleNamePrefix = "__bp2build__"
|
|
|
|
var productVariableSubstitutionPattern = regexp.MustCompile("%(d|s)")
|
|
|
|
// Label is used to represent a Bazel compatible Label. Also stores the original bp text to support
|
|
// string replacement.
|
|
type Label struct {
|
|
Bp_text string
|
|
Label string
|
|
}
|
|
|
|
// LabelList is used to represent a list of Bazel labels.
|
|
type LabelList struct {
|
|
Includes []Label
|
|
Excludes []Label
|
|
}
|
|
|
|
// Append appends the fields of other labelList to the corresponding fields of ll.
|
|
func (ll *LabelList) Append(other LabelList) {
|
|
if len(ll.Includes) > 0 || len(other.Includes) > 0 {
|
|
ll.Includes = append(ll.Includes, other.Includes...)
|
|
}
|
|
if len(ll.Excludes) > 0 || len(other.Excludes) > 0 {
|
|
ll.Excludes = append(other.Excludes, other.Excludes...)
|
|
}
|
|
}
|
|
|
|
func UniqueBazelLabels(originalLabels []Label) []Label {
|
|
uniqueLabelsSet := make(map[Label]bool)
|
|
for _, l := range originalLabels {
|
|
uniqueLabelsSet[l] = true
|
|
}
|
|
var uniqueLabels []Label
|
|
for l, _ := range uniqueLabelsSet {
|
|
uniqueLabels = append(uniqueLabels, l)
|
|
}
|
|
sort.SliceStable(uniqueLabels, func(i, j int) bool {
|
|
return uniqueLabels[i].Label < uniqueLabels[j].Label
|
|
})
|
|
return uniqueLabels
|
|
}
|
|
|
|
func UniqueBazelLabelList(originalLabelList LabelList) LabelList {
|
|
var uniqueLabelList LabelList
|
|
uniqueLabelList.Includes = UniqueBazelLabels(originalLabelList.Includes)
|
|
uniqueLabelList.Excludes = UniqueBazelLabels(originalLabelList.Excludes)
|
|
return uniqueLabelList
|
|
}
|
|
|
|
const (
|
|
ARCH_X86 = "x86"
|
|
ARCH_X86_64 = "x86_64"
|
|
ARCH_ARM = "arm"
|
|
ARCH_ARM64 = "arm64"
|
|
)
|
|
|
|
var (
|
|
// This is the list of architectures with a Bazel config_setting and
|
|
// constraint value equivalent. is actually android.ArchTypeList, but the
|
|
// android package depends on the bazel package, so a cyclic dependency
|
|
// prevents using that here.
|
|
selectableArchs = []string{ARCH_X86, ARCH_X86_64, ARCH_ARM, ARCH_ARM64}
|
|
)
|
|
|
|
// Arch-specific label_list typed Bazel attribute values. This should correspond
|
|
// to the types of architectures supported for compilation in arch.go.
|
|
type labelListArchValues struct {
|
|
X86 LabelList
|
|
X86_64 LabelList
|
|
Arm LabelList
|
|
Arm64 LabelList
|
|
// TODO(b/181299724): this is currently missing the "common" arch, which
|
|
// doesn't have an equivalent platform() definition yet.
|
|
}
|
|
|
|
// LabelListAttribute is used to represent a list of Bazel labels as an
|
|
// attribute.
|
|
type LabelListAttribute struct {
|
|
// The non-arch specific attribute label list Value. Required.
|
|
Value LabelList
|
|
|
|
// The arch-specific attribute label list values. Optional. If used, these
|
|
// are generated in a select statement and appended to the non-arch specific
|
|
// label list Value.
|
|
ArchValues labelListArchValues
|
|
}
|
|
|
|
// MakeLabelListAttribute initializes a LabelListAttribute with the non-arch specific value.
|
|
func MakeLabelListAttribute(value LabelList) LabelListAttribute {
|
|
return LabelListAttribute{Value: UniqueBazelLabelList(value)}
|
|
}
|
|
|
|
// HasArchSpecificValues returns true if the attribute contains
|
|
// architecture-specific label_list values.
|
|
func (attrs *LabelListAttribute) HasArchSpecificValues() bool {
|
|
for _, arch := range selectableArchs {
|
|
if len(attrs.GetValueForArch(arch).Includes) > 0 || len(attrs.GetValueForArch(arch).Excludes) > 0 {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// GetValueForArch returns the label_list attribute value for an architecture.
|
|
func (attrs *LabelListAttribute) GetValueForArch(arch string) LabelList {
|
|
switch arch {
|
|
case ARCH_X86:
|
|
return attrs.ArchValues.X86
|
|
case ARCH_X86_64:
|
|
return attrs.ArchValues.X86_64
|
|
case ARCH_ARM:
|
|
return attrs.ArchValues.Arm
|
|
case ARCH_ARM64:
|
|
return attrs.ArchValues.Arm64
|
|
default:
|
|
panic(fmt.Errorf("Unknown arch: %s", arch))
|
|
}
|
|
}
|
|
|
|
// SetValueForArch sets the label_list attribute value for an architecture.
|
|
func (attrs *LabelListAttribute) SetValueForArch(arch string, value LabelList) {
|
|
switch arch {
|
|
case "x86":
|
|
attrs.ArchValues.X86 = value
|
|
case "x86_64":
|
|
attrs.ArchValues.X86_64 = value
|
|
case "arm":
|
|
attrs.ArchValues.Arm = value
|
|
case "arm64":
|
|
attrs.ArchValues.Arm64 = value
|
|
default:
|
|
panic(fmt.Errorf("Unknown arch: %s", arch))
|
|
}
|
|
}
|
|
|
|
// StringListAttribute corresponds to the string_list Bazel attribute type with
|
|
// support for additional metadata, like configurations.
|
|
type StringListAttribute struct {
|
|
// The base value of the string list attribute.
|
|
Value []string
|
|
|
|
// Optional additive set of list values to the base value.
|
|
ArchValues stringListArchValues
|
|
}
|
|
|
|
// Arch-specific string_list typed Bazel attribute values. This should correspond
|
|
// to the types of architectures supported for compilation in arch.go.
|
|
type stringListArchValues struct {
|
|
X86 []string
|
|
X86_64 []string
|
|
Arm []string
|
|
Arm64 []string
|
|
// TODO(b/181299724): this is currently missing the "common" arch, which
|
|
// doesn't have an equivalent platform() definition yet.
|
|
}
|
|
|
|
// HasArchSpecificValues returns true if the attribute contains
|
|
// architecture-specific string_list values.
|
|
func (attrs *StringListAttribute) HasArchSpecificValues() bool {
|
|
for _, arch := range selectableArchs {
|
|
if len(attrs.GetValueForArch(arch)) > 0 {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// GetValueForArch returns the string_list attribute value for an architecture.
|
|
func (attrs *StringListAttribute) GetValueForArch(arch string) []string {
|
|
switch arch {
|
|
case ARCH_X86:
|
|
return attrs.ArchValues.X86
|
|
case ARCH_X86_64:
|
|
return attrs.ArchValues.X86_64
|
|
case ARCH_ARM:
|
|
return attrs.ArchValues.Arm
|
|
case ARCH_ARM64:
|
|
return attrs.ArchValues.Arm64
|
|
default:
|
|
panic(fmt.Errorf("Unknown arch: %s", arch))
|
|
}
|
|
}
|
|
|
|
// SetValueForArch sets the string_list attribute value for an architecture.
|
|
func (attrs *StringListAttribute) SetValueForArch(arch string, value []string) {
|
|
switch arch {
|
|
case ARCH_X86:
|
|
attrs.ArchValues.X86 = value
|
|
case ARCH_X86_64:
|
|
attrs.ArchValues.X86_64 = value
|
|
case ARCH_ARM:
|
|
attrs.ArchValues.Arm = value
|
|
case ARCH_ARM64:
|
|
attrs.ArchValues.Arm64 = value
|
|
default:
|
|
panic(fmt.Errorf("Unknown arch: %s", arch))
|
|
}
|
|
}
|
|
|
|
// TryVariableSubstitution, replace string substitution formatting within each string in slice with
|
|
// Starlark string.format compatible tag for productVariable.
|
|
func TryVariableSubstitutions(slice []string, productVariable string) ([]string, bool) {
|
|
ret := make([]string, 0, len(slice))
|
|
changesMade := false
|
|
for _, s := range slice {
|
|
newS, changed := TryVariableSubstitution(s, productVariable)
|
|
ret = append(ret, newS)
|
|
changesMade = changesMade || changed
|
|
}
|
|
return ret, changesMade
|
|
}
|
|
|
|
// TryVariableSubstitution, replace string substitution formatting within s with Starlark
|
|
// string.format compatible tag for productVariable.
|
|
func TryVariableSubstitution(s string, productVariable string) (string, bool) {
|
|
sub := productVariableSubstitutionPattern.ReplaceAllString(s, "{"+productVariable+"}")
|
|
return sub, s != sub
|
|
}
|