// Copyright 2015 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 android import ( "github.com/google/blueprint" "github.com/google/blueprint/proptools" ) type defaultsDependencyTag struct { blueprint.BaseDependencyTag } var DefaultsDepTag defaultsDependencyTag type defaultsProperties struct { Defaults []string } type DefaultableModuleBase struct { defaultsProperties defaultsProperties defaultableProperties []interface{} } func (d *DefaultableModuleBase) defaults() *defaultsProperties { return &d.defaultsProperties } func (d *DefaultableModuleBase) setProperties(props []interface{}) { d.defaultableProperties = props } type Defaultable interface { defaults() *defaultsProperties setProperties([]interface{}) applyDefaults(TopDownMutatorContext, []Defaults) } type DefaultableModule interface { Module Defaultable } var _ Defaultable = (*DefaultableModuleBase)(nil) func InitDefaultableModule(module DefaultableModule) { module.(Defaultable).setProperties(module.(Module).GetProperties()) module.AddProperties(module.defaults()) } type DefaultsModuleBase struct { DefaultableModuleBase defaultProperties []interface{} } type Defaults interface { Defaultable isDefaults() bool properties() []interface{} } func (d *DefaultsModuleBase) isDefaults() bool { return true } func (d *DefaultsModuleBase) properties() []interface{} { return d.defaultableProperties } func InitDefaultsModule(module DefaultableModule) { module.AddProperties( &hostAndDeviceProperties{}, &commonProperties{}, &variableProperties{}) InitArchModule(module) InitDefaultableModule(module) module.AddProperties(&module.base().nameProperties) module.base().module = module } var _ Defaults = (*DefaultsModuleBase)(nil) func (defaultable *DefaultableModuleBase) applyDefaults(ctx TopDownMutatorContext, defaultsList []Defaults) { for _, defaults := range defaultsList { for _, prop := range defaultable.defaultableProperties { for _, def := range defaults.properties() { if proptools.TypeEqual(prop, def) { err := proptools.PrependProperties(prop, def, nil) if err != nil { if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok { ctx.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error()) } else { panic(err) } } } } } } } func RegisterDefaultsPreArchMutators(ctx RegisterMutatorsContext) { ctx.BottomUp("defaults_deps", defaultsDepsMutator).Parallel() ctx.TopDown("defaults", defaultsMutator).Parallel() } func defaultsDepsMutator(ctx BottomUpMutatorContext) { if defaultable, ok := ctx.Module().(Defaultable); ok { ctx.AddDependency(ctx.Module(), DefaultsDepTag, defaultable.defaults().Defaults...) } } func defaultsMutator(ctx TopDownMutatorContext) { if defaultable, ok := ctx.Module().(Defaultable); ok && len(defaultable.defaults().Defaults) > 0 { var defaultsList []Defaults seen := make(map[Defaults]bool) ctx.WalkDeps(func(module, parent Module) bool { if ctx.OtherModuleDependencyTag(module) == DefaultsDepTag { if defaults, ok := module.(Defaults); ok { if !seen[defaults] { seen[defaults] = true defaultsList = append(defaultsList, defaults) return len(defaults.defaults().Defaults) > 0 } } else { ctx.PropertyErrorf("defaults", "module %s is not an defaults module", ctx.OtherModuleName(module)) } } return false }) defaultable.applyDefaults(ctx, defaultsList) } }