| package bp2build |
| |
| import ( |
| "fmt" |
| "reflect" |
| |
| "android/soong/android" |
| "android/soong/bazel" |
| "android/soong/starlark_fmt" |
| ) |
| |
| // Configurability support for bp2build. |
| |
| type selects map[string]reflect.Value |
| |
| func getStringValue(str bazel.StringAttribute) (reflect.Value, []selects) { |
| value := reflect.ValueOf(str.Value) |
| |
| if !str.HasConfigurableValues() { |
| return value, []selects{} |
| } |
| |
| ret := selects{} |
| for _, axis := range str.SortedConfigurationAxes() { |
| configToStrs := str.ConfigurableValues[axis] |
| for config, strs := range configToStrs { |
| selectKey := axis.SelectKey(config) |
| ret[selectKey] = reflect.ValueOf(strs) |
| } |
| } |
| |
| // if there is a select, use the base value as the conditions default value |
| if len(ret) > 0 { |
| if _, ok := ret[bazel.ConditionsDefaultSelectKey]; !ok { |
| ret[bazel.ConditionsDefaultSelectKey] = value |
| value = reflect.Zero(value.Type()) |
| } |
| } |
| |
| return value, []selects{ret} |
| } |
| |
| func getStringListValues(list bazel.StringListAttribute) (reflect.Value, []selects, bool) { |
| value := reflect.ValueOf(list.Value) |
| prepend := list.Prepend |
| if !list.HasConfigurableValues() { |
| return value, []selects{}, prepend |
| } |
| |
| var ret []selects |
| for _, axis := range list.SortedConfigurationAxes() { |
| configToLists := list.ConfigurableValues[axis] |
| archSelects := map[string]reflect.Value{} |
| for config, labels := range configToLists { |
| selectKey := axis.SelectKey(config) |
| archSelects[selectKey] = reflect.ValueOf(labels) |
| } |
| if len(archSelects) > 0 { |
| ret = append(ret, archSelects) |
| } |
| } |
| |
| return value, ret, prepend |
| } |
| |
| func getLabelValue(label bazel.LabelAttribute) (reflect.Value, []selects) { |
| value := reflect.ValueOf(label.Value) |
| if !label.HasConfigurableValues() { |
| return value, []selects{} |
| } |
| |
| ret := selects{} |
| for _, axis := range label.SortedConfigurationAxes() { |
| configToLabels := label.ConfigurableValues[axis] |
| for config, labels := range configToLabels { |
| selectKey := axis.SelectKey(config) |
| ret[selectKey] = reflect.ValueOf(labels) |
| } |
| } |
| |
| // if there is a select, use the base value as the conditions default value |
| if len(ret) > 0 { |
| ret[bazel.ConditionsDefaultSelectKey] = value |
| value = reflect.Zero(value.Type()) |
| } |
| |
| return value, []selects{ret} |
| } |
| |
| func getBoolValue(boolAttr bazel.BoolAttribute) (reflect.Value, []selects) { |
| value := reflect.ValueOf(boolAttr.Value) |
| if !boolAttr.HasConfigurableValues() { |
| return value, []selects{} |
| } |
| |
| ret := selects{} |
| for _, axis := range boolAttr.SortedConfigurationAxes() { |
| configToBools := boolAttr.ConfigurableValues[axis] |
| for config, bools := range configToBools { |
| selectKey := axis.SelectKey(config) |
| ret[selectKey] = reflect.ValueOf(bools) |
| } |
| } |
| // if there is a select, use the base value as the conditions default value |
| if len(ret) > 0 { |
| ret[bazel.ConditionsDefaultSelectKey] = value |
| value = reflect.Zero(value.Type()) |
| } |
| |
| return value, []selects{ret} |
| } |
| func getLabelListValues(list bazel.LabelListAttribute) (reflect.Value, []selects, bool) { |
| value := reflect.ValueOf(list.Value.Includes) |
| prepend := list.Prepend |
| var ret []selects |
| for _, axis := range list.SortedConfigurationAxes() { |
| configToLabels := list.ConfigurableValues[axis] |
| if !configToLabels.HasConfigurableValues() { |
| continue |
| } |
| archSelects := map[string]reflect.Value{} |
| defaultVal := configToLabels[bazel.ConditionsDefaultConfigKey] |
| // Skip empty list values unless ether EmitEmptyList is true, or these values differ from the default. |
| emitEmptyList := list.EmitEmptyList || len(defaultVal.Includes) > 0 |
| for config, labels := range configToLabels { |
| // Omit any entries in the map which match the default value, for brevity. |
| if config != bazel.ConditionsDefaultConfigKey && labels.Equals(defaultVal) { |
| continue |
| } |
| selectKey := axis.SelectKey(config) |
| if use, value := labelListSelectValue(selectKey, labels, emitEmptyList); use { |
| archSelects[selectKey] = value |
| } |
| } |
| if len(archSelects) > 0 { |
| ret = append(ret, archSelects) |
| } |
| } |
| |
| return value, ret, prepend |
| } |
| |
| func labelListSelectValue(selectKey string, list bazel.LabelList, emitEmptyList bool) (bool, reflect.Value) { |
| if selectKey == bazel.ConditionsDefaultSelectKey || emitEmptyList || len(list.Includes) > 0 { |
| return true, reflect.ValueOf(list.Includes) |
| } else if len(list.Excludes) > 0 { |
| // if there is still an excludes -- we need to have an empty list for this select & use the |
| // value in conditions default Includes |
| return true, reflect.ValueOf([]string{}) |
| } |
| return false, reflect.Zero(reflect.TypeOf([]string{})) |
| } |
| |
| var ( |
| emptyBazelList = "[]" |
| bazelNone = "None" |
| ) |
| |
| // prettyPrintAttribute converts an Attribute to its Bazel syntax. May contain |
| // select statements. |
| func prettyPrintAttribute(v bazel.Attribute, indent int) (string, error) { |
| var value reflect.Value |
| // configurableAttrs is the list of individual select statements to be |
| // concatenated together. These select statements should be along different |
| // axes. For example, one element may be |
| // `select({"//color:red": "one", "//color:green": "two"})`, and the second |
| // element may be `select({"//animal:cat": "three", "//animal:dog": "four"}). |
| // These selects should be sorted by axis identifier. |
| var configurableAttrs []selects |
| var prepend bool |
| var defaultSelectValue *string |
| var emitZeroValues bool |
| // If true, print the default attribute value, even if the attribute is zero. |
| shouldPrintDefault := false |
| switch list := v.(type) { |
| case bazel.StringAttribute: |
| if err := list.Collapse(); err != nil { |
| return "", err |
| } |
| value, configurableAttrs = getStringValue(list) |
| defaultSelectValue = &bazelNone |
| case bazel.StringListAttribute: |
| value, configurableAttrs, prepend = getStringListValues(list) |
| defaultSelectValue = &emptyBazelList |
| case bazel.LabelListAttribute: |
| value, configurableAttrs, prepend = getLabelListValues(list) |
| emitZeroValues = list.EmitEmptyList |
| defaultSelectValue = &emptyBazelList |
| if list.ForceSpecifyEmptyList && (!value.IsNil() || list.HasConfigurableValues()) { |
| shouldPrintDefault = true |
| } |
| case bazel.LabelAttribute: |
| if err := list.Collapse(); err != nil { |
| return "", err |
| } |
| value, configurableAttrs = getLabelValue(list) |
| defaultSelectValue = &bazelNone |
| case bazel.BoolAttribute: |
| if err := list.Collapse(); err != nil { |
| return "", err |
| } |
| value, configurableAttrs = getBoolValue(list) |
| defaultSelectValue = &bazelNone |
| default: |
| return "", fmt.Errorf("Not a supported Bazel attribute type: %s", v) |
| } |
| |
| var err error |
| ret := "" |
| if value.Kind() != reflect.Invalid { |
| s, err := prettyPrint(value, indent, false) // never emit zero values for the base value |
| if err != nil { |
| return ret, err |
| } |
| |
| ret += s |
| } |
| // Convenience function to prepend/append selects components to an attribute value. |
| concatenateSelects := func(selectsData selects, defaultValue *string, s string, prepend bool) (string, error) { |
| selectMap, err := prettyPrintSelectMap(selectsData, defaultValue, indent, emitZeroValues) |
| if err != nil { |
| return "", err |
| } |
| var left, right string |
| if prepend { |
| left, right = selectMap, s |
| } else { |
| left, right = s, selectMap |
| } |
| if left != "" && right != "" { |
| left += " + " |
| } |
| left += right |
| |
| return left, nil |
| } |
| |
| for _, configurableAttr := range configurableAttrs { |
| ret, err = concatenateSelects(configurableAttr, defaultSelectValue, ret, prepend) |
| if err != nil { |
| return "", err |
| } |
| } |
| |
| if ret == "" && shouldPrintDefault { |
| return *defaultSelectValue, nil |
| } |
| return ret, nil |
| } |
| |
| // prettyPrintSelectMap converts a map of select keys to reflected Values as a generic way |
| // to construct a select map for any kind of attribute type. |
| func prettyPrintSelectMap(selectMap map[string]reflect.Value, defaultValue *string, indent int, emitZeroValues bool) (string, error) { |
| if selectMap == nil { |
| return "", nil |
| } |
| |
| var selects string |
| for _, selectKey := range android.SortedKeys(selectMap) { |
| if selectKey == bazel.ConditionsDefaultSelectKey { |
| // Handle default condition later. |
| continue |
| } |
| value := selectMap[selectKey] |
| if isZero(value) && !emitZeroValues && isZero(selectMap[bazel.ConditionsDefaultSelectKey]) { |
| // Ignore zero values to not generate empty lists. However, always note zero values if |
| // the default value is non-zero. |
| continue |
| } |
| s, err := prettyPrintSelectEntry(value, selectKey, indent, true) |
| if err != nil { |
| return "", err |
| } |
| // s could still be an empty string, e.g. unset slices of structs with |
| // length of 0. |
| if s != "" { |
| selects += s + ",\n" |
| } |
| } |
| |
| if len(selects) == 0 { |
| // If there is a default value, and there are no selects for this axis, print that without any selects. |
| if val, exists := selectMap[bazel.ConditionsDefaultSelectKey]; exists { |
| return prettyPrint(val, indent, emitZeroValues) |
| } |
| // No conditions (or all values are empty lists), so no need for a map. |
| return "", nil |
| } |
| |
| // Create the map. |
| ret := "select({\n" |
| ret += selects |
| |
| // Handle the default condition |
| s, err := prettyPrintSelectEntry(selectMap[bazel.ConditionsDefaultSelectKey], bazel.ConditionsDefaultSelectKey, indent, emitZeroValues) |
| if err != nil { |
| return "", err |
| } |
| if s != "" { |
| // Print the custom default value. |
| ret += s |
| ret += ",\n" |
| } else if defaultValue != nil { |
| // Print an explicit empty list (the default value) even if the value is |
| // empty, to avoid errors about not finding a configuration that matches. |
| ret += fmt.Sprintf("%s\"%s\": %s,\n", starlark_fmt.Indention(indent+1), bazel.ConditionsDefaultSelectKey, *defaultValue) |
| } |
| |
| ret += starlark_fmt.Indention(indent) |
| ret += "})" |
| |
| return ret, nil |
| } |
| |
| // prettyPrintSelectEntry converts a reflect.Value into an entry in a select map |
| // with a provided key. |
| func prettyPrintSelectEntry(value reflect.Value, key string, indent int, emitZeroValues bool) (string, error) { |
| s := starlark_fmt.Indention(indent + 1) |
| v, err := prettyPrint(value, indent+1, emitZeroValues) |
| if err != nil { |
| return "", err |
| } |
| if v == "" { |
| return "", nil |
| } |
| s += fmt.Sprintf("\"%s\": %s", key, v) |
| return s, nil |
| } |