420 lines
12 KiB
Go
420 lines
12 KiB
Go
// Copyright (C) 2019 The Android Open Source Project
|
|
//
|
|
// 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 sdk
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"strings"
|
|
|
|
"android/soong/android"
|
|
)
|
|
|
|
type bpPropertySet struct {
|
|
properties map[string]interface{}
|
|
tags map[string]android.BpPropertyTag
|
|
comments map[string]string
|
|
order []string
|
|
}
|
|
|
|
var _ android.BpPropertySet = (*bpPropertySet)(nil)
|
|
|
|
func (s *bpPropertySet) init() {
|
|
s.properties = make(map[string]interface{})
|
|
s.tags = make(map[string]android.BpPropertyTag)
|
|
}
|
|
|
|
// Converts the given value, which is assumed to be a struct, to a
|
|
// bpPropertySet.
|
|
func convertToPropertySet(value reflect.Value) *bpPropertySet {
|
|
res := newPropertySet()
|
|
structType := value.Type()
|
|
|
|
for i := 0; i < structType.NumField(); i++ {
|
|
field := structType.Field(i)
|
|
fieldVal := value.Field(i)
|
|
|
|
switch fieldVal.Type().Kind() {
|
|
case reflect.Ptr:
|
|
if fieldVal.IsNil() {
|
|
continue // nil pointer means the property isn't set.
|
|
}
|
|
fieldVal = fieldVal.Elem()
|
|
case reflect.Slice:
|
|
if fieldVal.IsNil() {
|
|
continue // Ignore a nil slice (but not one with length zero).
|
|
}
|
|
}
|
|
|
|
if fieldVal.Type().Kind() == reflect.Struct {
|
|
fieldVal = fieldVal.Addr() // Avoid struct copy below.
|
|
}
|
|
res.AddProperty(strings.ToLower(field.Name), fieldVal.Interface())
|
|
}
|
|
|
|
return res
|
|
}
|
|
|
|
// Converts the given value to something that can be set in a property.
|
|
func coercePropertyValue(value interface{}) interface{} {
|
|
val := reflect.ValueOf(value)
|
|
switch val.Kind() {
|
|
case reflect.Struct:
|
|
// convertToPropertySet requires an addressable struct, and this is probably
|
|
// a mistake.
|
|
panic(fmt.Sprintf("Value is a struct, not a pointer to one: %v", value))
|
|
case reflect.Ptr:
|
|
if _, ok := value.(*bpPropertySet); !ok {
|
|
derefValue := reflect.Indirect(val)
|
|
if derefValue.Kind() != reflect.Struct {
|
|
panic(fmt.Sprintf("A pointer must be to a struct, got: %v", value))
|
|
}
|
|
return convertToPropertySet(derefValue)
|
|
}
|
|
}
|
|
return value
|
|
}
|
|
|
|
// Merges the fields of the given property set into s.
|
|
func (s *bpPropertySet) mergePropertySet(propSet *bpPropertySet) {
|
|
for _, name := range propSet.order {
|
|
if tag, ok := propSet.tags[name]; ok {
|
|
s.AddPropertyWithTag(name, propSet.properties[name], tag)
|
|
} else {
|
|
s.AddProperty(name, propSet.properties[name])
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *bpPropertySet) AddProperty(name string, value interface{}) {
|
|
value = coercePropertyValue(value)
|
|
|
|
if propSetValue, ok := value.(*bpPropertySet); ok {
|
|
if curValue, ok := s.properties[name]; ok {
|
|
if curSet, ok := curValue.(*bpPropertySet); ok {
|
|
curSet.mergePropertySet(propSetValue)
|
|
return
|
|
}
|
|
// If the current value isn't a property set we got conflicting types.
|
|
// Continue down to the check below to complain about it.
|
|
}
|
|
}
|
|
|
|
if s.properties[name] != nil {
|
|
panic(fmt.Sprintf("Property %q already exists in property set", name))
|
|
}
|
|
|
|
s.properties[name] = value
|
|
s.order = append(s.order, name)
|
|
}
|
|
|
|
func (s *bpPropertySet) AddPropertyWithTag(name string, value interface{}, tag android.BpPropertyTag) {
|
|
s.AddProperty(name, value)
|
|
s.tags[name] = tag
|
|
}
|
|
|
|
func (s *bpPropertySet) AddPropertySet(name string) android.BpPropertySet {
|
|
s.AddProperty(name, newPropertySet())
|
|
return s.properties[name].(android.BpPropertySet)
|
|
}
|
|
|
|
func (s *bpPropertySet) getValue(name string) interface{} {
|
|
return s.properties[name]
|
|
}
|
|
|
|
func (s *bpPropertySet) getOptionalValue(name string) (interface{}, bool) {
|
|
value, ok := s.properties[name]
|
|
return value, ok
|
|
}
|
|
|
|
func (s *bpPropertySet) getTag(name string) interface{} {
|
|
return s.tags[name]
|
|
}
|
|
|
|
func (s *bpPropertySet) AddCommentForProperty(name, text string) {
|
|
if s.comments == nil {
|
|
s.comments = map[string]string{}
|
|
}
|
|
s.comments[name] = strings.TrimSpace(text)
|
|
}
|
|
|
|
func (s *bpPropertySet) transformContents(transformer bpPropertyTransformer) {
|
|
var newOrder []string
|
|
for _, name := range s.order {
|
|
value := s.properties[name]
|
|
tag := s.tags[name]
|
|
var newValue interface{}
|
|
var newTag android.BpPropertyTag
|
|
if propertySet, ok := value.(*bpPropertySet); ok {
|
|
var newPropertySet *bpPropertySet
|
|
newPropertySet, newTag = transformPropertySet(transformer, name, propertySet, tag)
|
|
if newPropertySet == nil {
|
|
newValue = nil
|
|
} else {
|
|
newValue = newPropertySet
|
|
}
|
|
} else {
|
|
newValue, newTag = transformer.transformProperty(name, value, tag)
|
|
}
|
|
|
|
if newValue == nil {
|
|
// Delete the property from the map and exclude it from the new order.
|
|
delete(s.properties, name)
|
|
} else {
|
|
// Update the property in the map and add the name to the new order list.
|
|
s.properties[name] = newValue
|
|
s.tags[name] = newTag
|
|
newOrder = append(newOrder, name)
|
|
}
|
|
}
|
|
s.order = newOrder
|
|
}
|
|
|
|
func transformPropertySet(transformer bpPropertyTransformer, name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag) {
|
|
newPropertySet, newTag := transformer.transformPropertySetBeforeContents(name, propertySet, tag)
|
|
if newPropertySet != nil {
|
|
newPropertySet.transformContents(transformer)
|
|
|
|
newPropertySet, newTag = transformer.transformPropertySetAfterContents(name, newPropertySet, newTag)
|
|
}
|
|
return newPropertySet, newTag
|
|
}
|
|
|
|
func (s *bpPropertySet) setProperty(name string, value interface{}) {
|
|
if s.properties[name] == nil {
|
|
s.AddProperty(name, value)
|
|
} else {
|
|
s.properties[name] = value
|
|
s.tags[name] = nil
|
|
}
|
|
}
|
|
|
|
func (s *bpPropertySet) removeProperty(name string) {
|
|
delete(s.properties, name)
|
|
delete(s.tags, name)
|
|
_, s.order = android.RemoveFromList(name, s.order)
|
|
}
|
|
|
|
func (s *bpPropertySet) insertAfter(position string, name string, value interface{}) {
|
|
if s.properties[name] != nil {
|
|
panic("Property %q already exists in property set")
|
|
}
|
|
|
|
// Add the name to the end of the order, to ensure it has necessary capacity
|
|
// and to handle the case when the position does not exist.
|
|
s.order = append(s.order, name)
|
|
|
|
// Search through the order for the item that matches supplied position. If
|
|
// found then insert the name of the new property after it.
|
|
for i, v := range s.order {
|
|
if v == position {
|
|
// Copy the items after the one where the new property should be inserted.
|
|
copy(s.order[i+2:], s.order[i+1:])
|
|
// Insert the item in the list.
|
|
s.order[i+1] = name
|
|
}
|
|
}
|
|
|
|
s.properties[name] = value
|
|
}
|
|
|
|
type bpModule struct {
|
|
*bpPropertySet
|
|
moduleType string
|
|
}
|
|
|
|
func (m *bpModule) ModuleType() string {
|
|
return m.moduleType
|
|
}
|
|
|
|
func (m *bpModule) Name() string {
|
|
name, hasName := m.getOptionalValue("name")
|
|
if hasName {
|
|
return name.(string)
|
|
} else {
|
|
return ""
|
|
}
|
|
}
|
|
|
|
var _ android.BpModule = (*bpModule)(nil)
|
|
|
|
type bpPropertyTransformer interface {
|
|
// Transform the property set, returning the new property set/tag to insert back into the
|
|
// parent property set (or module if this is the top level property set).
|
|
//
|
|
// This will be called before transforming the properties in the supplied set.
|
|
//
|
|
// The name will be "" for the top level property set.
|
|
//
|
|
// Returning (nil, ...) will cause the property set to be removed.
|
|
transformPropertySetBeforeContents(name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag)
|
|
|
|
// Transform the property set, returning the new property set/tag to insert back into the
|
|
// parent property set (or module if this is the top level property set).
|
|
//
|
|
// This will be called after transforming the properties in the supplied set.
|
|
//
|
|
// The name will be "" for the top level property set.
|
|
//
|
|
// Returning (nil, ...) will cause the property set to be removed.
|
|
transformPropertySetAfterContents(name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag)
|
|
|
|
// Transform a property, return the new value/tag to insert back into the property set.
|
|
//
|
|
// Returning (nil, ...) will cause the property to be removed.
|
|
transformProperty(name string, value interface{}, tag android.BpPropertyTag) (interface{}, android.BpPropertyTag)
|
|
}
|
|
|
|
// Interface for transforming bpModule objects.
|
|
type bpTransformer interface {
|
|
// Transform the module, returning the result.
|
|
//
|
|
// The method can either create a new module and return that, or modify the supplied module
|
|
// in place and return that.
|
|
//
|
|
// After this returns the transformer is applied to the contents of the returned module.
|
|
transformModule(module *bpModule) *bpModule
|
|
|
|
bpPropertyTransformer
|
|
}
|
|
|
|
type identityTransformation struct{}
|
|
|
|
var _ bpTransformer = (*identityTransformation)(nil)
|
|
|
|
func (t identityTransformation) transformModule(module *bpModule) *bpModule {
|
|
return module
|
|
}
|
|
|
|
func (t identityTransformation) transformPropertySetBeforeContents(name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag) {
|
|
return propertySet, tag
|
|
}
|
|
|
|
func (t identityTransformation) transformPropertySetAfterContents(name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag) {
|
|
return propertySet, tag
|
|
}
|
|
|
|
func (t identityTransformation) transformProperty(name string, value interface{}, tag android.BpPropertyTag) (interface{}, android.BpPropertyTag) {
|
|
return value, tag
|
|
}
|
|
|
|
func (m *bpModule) deepCopy() *bpModule {
|
|
return m.transform(deepCopyTransformer)
|
|
}
|
|
|
|
func (m *bpModule) transform(transformer bpTransformer) *bpModule {
|
|
transformedModule := transformer.transformModule(m)
|
|
// Copy the contents of the returned property set into the module and then transform that.
|
|
transformedModule.bpPropertySet, _ = transformPropertySet(transformer, "", transformedModule.bpPropertySet, nil)
|
|
return transformedModule
|
|
}
|
|
|
|
type deepCopyTransformation struct {
|
|
identityTransformation
|
|
}
|
|
|
|
func (t deepCopyTransformation) transformModule(module *bpModule) *bpModule {
|
|
// Take a shallow copy of the module. Any mutable property values will be copied by the
|
|
// transformer.
|
|
moduleCopy := *module
|
|
return &moduleCopy
|
|
}
|
|
|
|
func (t deepCopyTransformation) transformPropertySetBeforeContents(name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag) {
|
|
// Create a shallow copy of the properties map. Any mutable property values will be copied by the
|
|
// transformer.
|
|
propertiesCopy := make(map[string]interface{})
|
|
for propertyName, value := range propertySet.properties {
|
|
propertiesCopy[propertyName] = value
|
|
}
|
|
|
|
// Ditto for tags map.
|
|
tagsCopy := make(map[string]android.BpPropertyTag)
|
|
for propertyName, propertyTag := range propertySet.tags {
|
|
tagsCopy[propertyName] = propertyTag
|
|
}
|
|
|
|
// Create a new property set.
|
|
return &bpPropertySet{
|
|
properties: propertiesCopy,
|
|
tags: tagsCopy,
|
|
order: append([]string(nil), propertySet.order...),
|
|
}, tag
|
|
}
|
|
|
|
func (t deepCopyTransformation) transformProperty(name string, value interface{}, tag android.BpPropertyTag) (interface{}, android.BpPropertyTag) {
|
|
// Copy string slice, otherwise return value.
|
|
if values, ok := value.([]string); ok {
|
|
valuesCopy := make([]string, len(values))
|
|
copy(valuesCopy, values)
|
|
return valuesCopy, tag
|
|
}
|
|
return value, tag
|
|
}
|
|
|
|
var deepCopyTransformer bpTransformer = deepCopyTransformation{}
|
|
|
|
// A .bp file
|
|
type bpFile struct {
|
|
modules map[string]*bpModule
|
|
order []*bpModule
|
|
}
|
|
|
|
// Add a module.
|
|
//
|
|
// The module must have had its "name" property set to a string value that
|
|
// is unique within this file.
|
|
func (f *bpFile) AddModule(module android.BpModule) {
|
|
m := module.(*bpModule)
|
|
moduleType := module.ModuleType()
|
|
name := m.Name()
|
|
hasName := true
|
|
if name == "" {
|
|
// Use a prefixed module type as the name instead just in case this is something like a package
|
|
// of namespace module which does not require a name.
|
|
name = "#" + moduleType
|
|
hasName = false
|
|
}
|
|
|
|
if f.modules[name] != nil {
|
|
if hasName {
|
|
panic(fmt.Sprintf("Module %q already exists in bp file", name))
|
|
} else {
|
|
panic(fmt.Sprintf("Unnamed module type %q already exists in bp file", moduleType))
|
|
}
|
|
}
|
|
|
|
f.modules[name] = m
|
|
f.order = append(f.order, m)
|
|
}
|
|
|
|
func (f *bpFile) newModule(moduleType string) *bpModule {
|
|
return newModule(moduleType)
|
|
}
|
|
|
|
func newModule(moduleType string) *bpModule {
|
|
module := &bpModule{
|
|
moduleType: moduleType,
|
|
bpPropertySet: newPropertySet(),
|
|
}
|
|
return module
|
|
}
|
|
|
|
func newPropertySet() *bpPropertySet {
|
|
set := &bpPropertySet{}
|
|
set.init()
|
|
return set
|
|
}
|