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 { // 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 }