blob: 9eaa1ac85904add4544892a7cd5fe76bc352cbde [file] [log] [blame]
Colin Cross178a5092016-09-13 13:42:32 -07001// Copyright 2016 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package android
16
17import (
Colin Cross31a738b2019-12-30 18:45:15 -080018 "reflect"
19
Colin Cross178a5092016-09-13 13:42:32 -070020 "github.com/google/blueprint"
Colin Cross31a738b2019-12-30 18:45:15 -080021 "github.com/google/blueprint/proptools"
Colin Cross178a5092016-09-13 13:42:32 -070022)
23
24// This file implements hooks that external module types can use to inject logic into existing
25// module types. Each hook takes an interface as a parameter so that new methods can be added
26// to the interface without breaking existing module types.
27
28// Load hooks are run after the module's properties have been filled from the blueprint file, but
29// before the module has been split into architecture variants, and before defaults modules have
30// been applied.
31type LoadHookContext interface {
Colin Cross1184b642019-12-30 18:43:07 -080032 EarlyModuleContext
33
Colin Cross178a5092016-09-13 13:42:32 -070034 AppendProperties(...interface{})
35 PrependProperties(...interface{})
Colin Crosse003c4a2019-09-25 12:58:36 -070036 CreateModule(ModuleFactory, ...interface{}) Module
Colin Cross9d34f352019-11-22 16:03:51 -080037
38 registerScopedModuleType(name string, factory blueprint.ModuleFactory)
39 moduleFactories() map[string]blueprint.ModuleFactory
Colin Cross178a5092016-09-13 13:42:32 -070040}
41
Paul Duffinafa9fa12020-04-29 16:47:28 +010042// Add a hook that will be called once the module has been loaded, i.e. its
43// properties have been initialized from the Android.bp file.
44//
45// Consider using SetDefaultableHook to register a hook for any module that implements
46// DefaultableModule as the hook is called after any defaults have been applied to the
47// module which could reduce duplication and make it easier to use.
Colin Cross178a5092016-09-13 13:42:32 -070048func AddLoadHook(m blueprint.Module, hook func(LoadHookContext)) {
Colin Cross31a738b2019-12-30 18:45:15 -080049 blueprint.AddLoadHook(m, func(ctx blueprint.LoadHookContext) {
50 actx := &loadHookContext{
51 earlyModuleContext: m.(Module).base().earlyModuleContextFactory(ctx),
52 bp: ctx,
53 }
54 hook(actx)
55 })
Colin Cross178a5092016-09-13 13:42:32 -070056}
57
Colin Cross31a738b2019-12-30 18:45:15 -080058type loadHookContext struct {
59 earlyModuleContext
60 bp blueprint.LoadHookContext
61 module Module
62}
63
Colin Cross9d34f352019-11-22 16:03:51 -080064func (l *loadHookContext) moduleFactories() map[string]blueprint.ModuleFactory {
65 return l.bp.ModuleFactories()
66}
67
Ustaef3676c2021-11-23 12:31:55 -050068func (l *loadHookContext) appendPrependHelper(props []interface{},
69 extendFn func([]interface{}, interface{}, proptools.ExtendPropertyFilterFunc) error) {
Colin Cross31a738b2019-12-30 18:45:15 -080070 for _, p := range props {
Ustaef3676c2021-11-23 12:31:55 -050071 err := extendFn(l.Module().base().customizableProperties, p, nil)
Colin Cross31a738b2019-12-30 18:45:15 -080072 if err != nil {
73 if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok {
74 l.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error())
75 } else {
76 panic(err)
Colin Cross178a5092016-09-13 13:42:32 -070077 }
78 }
79 }
80}
Ustaef3676c2021-11-23 12:31:55 -050081func (l *loadHookContext) AppendProperties(props ...interface{}) {
82 l.appendPrependHelper(props, proptools.AppendMatchingProperties)
83}
Colin Cross178a5092016-09-13 13:42:32 -070084
Colin Cross31a738b2019-12-30 18:45:15 -080085func (l *loadHookContext) PrependProperties(props ...interface{}) {
Ustaef3676c2021-11-23 12:31:55 -050086 l.appendPrependHelper(props, proptools.PrependMatchingProperties)
Colin Cross31a738b2019-12-30 18:45:15 -080087}
88
89func (l *loadHookContext) CreateModule(factory ModuleFactory, props ...interface{}) Module {
90 inherited := []interface{}{&l.Module().base().commonProperties}
91 module := l.bp.CreateModule(ModuleFactoryAdaptor(factory), append(inherited, props...)...).(Module)
92
93 if l.Module().base().variableProperties != nil && module.base().variableProperties != nil {
94 src := l.Module().base().variableProperties
95 dst := []interface{}{
96 module.base().variableProperties,
97 // Put an empty copy of the src properties into dst so that properties in src that are not in dst
98 // don't cause a "failed to find property to extend" error.
Colin Cross43e789d2020-01-28 09:46:50 -080099 proptools.CloneEmptyProperties(reflect.ValueOf(src)).Interface(),
Colin Cross31a738b2019-12-30 18:45:15 -0800100 }
101 err := proptools.AppendMatchingProperties(dst, src, nil)
102 if err != nil {
103 panic(err)
104 }
105 }
106
107 return module
108}
109
Colin Cross9d34f352019-11-22 16:03:51 -0800110func (l *loadHookContext) registerScopedModuleType(name string, factory blueprint.ModuleFactory) {
111 l.bp.RegisterScopedModuleType(name, factory)
112}
113
Colin Cross178a5092016-09-13 13:42:32 -0700114type InstallHookContext interface {
115 ModuleContext
David Srbecky07656412020-06-04 01:26:16 +0100116 SrcPath() Path
Colin Cross70dda7e2019-10-01 22:05:35 -0700117 Path() InstallPath
Colin Cross178a5092016-09-13 13:42:32 -0700118 Symlink() bool
119}
120
121// Install hooks are run after a module creates a rule to install a file or symlink.
122// The installed path is available from InstallHookContext.Path(), and
123// InstallHookContext.Symlink() will be true if it was a symlink.
124func AddInstallHook(m blueprint.Module, hook func(InstallHookContext)) {
125 h := &m.(Module).base().hooks
126 h.install = append(h.install, hook)
127}
128
129type installHookContext struct {
130 ModuleContext
David Srbecky07656412020-06-04 01:26:16 +0100131 srcPath Path
Colin Cross70dda7e2019-10-01 22:05:35 -0700132 path InstallPath
Colin Cross178a5092016-09-13 13:42:32 -0700133 symlink bool
134}
135
David Srbecky07656412020-06-04 01:26:16 +0100136var _ InstallHookContext = &installHookContext{}
137
138func (x *installHookContext) SrcPath() Path {
139 return x.srcPath
140}
141
Colin Cross70dda7e2019-10-01 22:05:35 -0700142func (x *installHookContext) Path() InstallPath {
Colin Cross178a5092016-09-13 13:42:32 -0700143 return x.path
144}
145
146func (x *installHookContext) Symlink() bool {
147 return x.symlink
148}
149
David Srbecky07656412020-06-04 01:26:16 +0100150func (x *hooks) runInstallHooks(ctx ModuleContext, srcPath Path, path InstallPath, symlink bool) {
Colin Cross178a5092016-09-13 13:42:32 -0700151 if len(x.install) > 0 {
152 mctx := &installHookContext{
153 ModuleContext: ctx,
David Srbecky07656412020-06-04 01:26:16 +0100154 srcPath: srcPath,
Colin Cross178a5092016-09-13 13:42:32 -0700155 path: path,
156 symlink: symlink,
157 }
158 for _, x := range x.install {
159 x(mctx)
160 if mctx.Failed() {
161 return
162 }
163 }
164 }
165}
166
167type hooks struct {
Colin Cross178a5092016-09-13 13:42:32 -0700168 install []func(InstallHookContext)
169}