blob: 47c5cf531aaf5f9a7413ee1f4707aa41f40d7fef [file] [log] [blame]
Colin Crossce75d2c2016-10-06 16:12:58 -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
Colin Cross74d73e22017-08-02 11:05:49 -070017import (
18 "fmt"
19
20 "github.com/google/blueprint"
21)
Colin Crossce75d2c2016-10-06 16:12:58 -070022
23// This file implements common functionality for handling modules that may exist as prebuilts,
24// source, or both.
25
Nan Zhang2502e122017-03-09 18:43:01 -080026type prebuiltDependencyTag struct {
27 blueprint.BaseDependencyTag
28}
29
30var prebuiltDepTag prebuiltDependencyTag
Colin Crossce75d2c2016-10-06 16:12:58 -070031
Colin Cross74d73e22017-08-02 11:05:49 -070032type PrebuiltProperties struct {
33 // When prefer is set to true the prebuilt will be used instead of any source module with
34 // a matching name.
Nan Zhang0007d812017-11-07 10:57:05 -080035 Prefer *bool `android:"arch_variant"`
Colin Crossce75d2c2016-10-06 16:12:58 -070036
Colin Cross74d73e22017-08-02 11:05:49 -070037 SourceExists bool `blueprint:"mutated"`
38 UsePrebuilt bool `blueprint:"mutated"`
39}
40
41type Prebuilt struct {
42 properties PrebuiltProperties
43 module Module
44 srcs *[]string
Colin Crossce75d2c2016-10-06 16:12:58 -070045}
46
47func (p *Prebuilt) Name(name string) string {
48 return "prebuilt_" + name
49}
50
Colin Cross74d73e22017-08-02 11:05:49 -070051func (p *Prebuilt) SingleSourcePath(ctx ModuleContext) Path {
52 if len(*p.srcs) == 0 {
Colin Crossce75d2c2016-10-06 16:12:58 -070053 ctx.PropertyErrorf("srcs", "missing prebuilt source file")
54 return nil
55 }
56
Colin Cross74d73e22017-08-02 11:05:49 -070057 if len(*p.srcs) > 1 {
Colin Crossce75d2c2016-10-06 16:12:58 -070058 ctx.PropertyErrorf("srcs", "multiple prebuilt source files")
59 return nil
60 }
61
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070062 // Return the singleton source after expanding any filegroup in the
63 // sources.
64 return ctx.ExpandSource((*p.srcs)[0], "")
Colin Cross74d73e22017-08-02 11:05:49 -070065}
66
67func InitPrebuiltModule(module PrebuiltInterface, srcs *[]string) {
68 p := module.Prebuilt()
69 module.AddProperties(&p.properties)
70 p.srcs = srcs
Colin Crossce75d2c2016-10-06 16:12:58 -070071}
72
73type PrebuiltInterface interface {
74 Module
75 Prebuilt() *Prebuilt
Colin Crossce75d2c2016-10-06 16:12:58 -070076}
77
Colin Cross5ea9bcc2017-07-27 15:41:32 -070078func RegisterPrebuiltsPreArchMutators(ctx RegisterMutatorsContext) {
Colin Crosscec81712017-07-13 14:43:27 -070079 ctx.BottomUp("prebuilts", prebuiltMutator).Parallel()
80}
81
Colin Cross5ea9bcc2017-07-27 15:41:32 -070082func RegisterPrebuiltsPostDepsMutators(ctx RegisterMutatorsContext) {
Colin Crosscec81712017-07-13 14:43:27 -070083 ctx.TopDown("prebuilt_select", PrebuiltSelectModuleMutator).Parallel()
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070084 ctx.BottomUp("prebuilt_postdeps", PrebuiltPostDepsMutator).Parallel()
Colin Crosscec81712017-07-13 14:43:27 -070085}
86
Colin Crossce75d2c2016-10-06 16:12:58 -070087// prebuiltMutator ensures that there is always a module with an undecorated name, and marks
88// prebuilt modules that have both a prebuilt and a source module.
89func prebuiltMutator(ctx BottomUpMutatorContext) {
90 if m, ok := ctx.Module().(PrebuiltInterface); ok && m.Prebuilt() != nil {
91 p := m.Prebuilt()
92 name := m.base().BaseModuleName()
93 if ctx.OtherModuleExists(name) {
Nan Zhang2502e122017-03-09 18:43:01 -080094 ctx.AddReverseDependency(ctx.Module(), prebuiltDepTag, name)
Colin Cross74d73e22017-08-02 11:05:49 -070095 p.properties.SourceExists = true
Colin Crossce75d2c2016-10-06 16:12:58 -070096 } else {
97 ctx.Rename(name)
98 }
99 }
100}
101
Colin Crossc3e7fa62017-03-17 13:14:32 -0700102// PrebuiltSelectModuleMutator marks prebuilts that are used, either overriding source modules or
103// because the source module doesn't exist. It also disables installing overridden source modules.
Colin Crossa2f296f2016-11-29 15:16:18 -0800104func PrebuiltSelectModuleMutator(ctx TopDownMutatorContext) {
Colin Crossc3e7fa62017-03-17 13:14:32 -0700105 if m, ok := ctx.Module().(PrebuiltInterface); ok && m.Prebuilt() != nil {
106 p := m.Prebuilt()
Colin Cross74d73e22017-08-02 11:05:49 -0700107 if p.srcs == nil {
108 panic(fmt.Errorf("prebuilt module did not have InitPrebuiltModule called on it"))
109 }
110 if !p.properties.SourceExists {
111 p.properties.UsePrebuilt = p.usePrebuilt(ctx, nil)
Colin Crossc3e7fa62017-03-17 13:14:32 -0700112 }
113 } else if s, ok := ctx.Module().(Module); ok {
Colin Crossee6143c2017-12-30 17:54:27 -0800114 ctx.VisitDirectDepsWithTag(prebuiltDepTag, func(m Module) {
115 p := m.(PrebuiltInterface).Prebuilt()
116 if p.usePrebuilt(ctx, s) {
117 p.properties.UsePrebuilt = true
118 s.SkipInstall()
Colin Crossa2f296f2016-11-29 15:16:18 -0800119 }
120 })
121 }
122}
123
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700124// PrebuiltPostDepsMutator does two operations. It replace dependencies on the
125// source module with dependencies on the prebuilt when both modules exist and
126// the prebuilt should be used. When the prebuilt should not be used, disable
127// installing it. Secondly, it also adds a sourcegroup to any filegroups found
128// in the prebuilt's 'Srcs' property.
129func PrebuiltPostDepsMutator(ctx BottomUpMutatorContext) {
Colin Crossce75d2c2016-10-06 16:12:58 -0700130 if m, ok := ctx.Module().(PrebuiltInterface); ok && m.Prebuilt() != nil {
131 p := m.Prebuilt()
132 name := m.base().BaseModuleName()
Colin Cross74d73e22017-08-02 11:05:49 -0700133 if p.properties.UsePrebuilt {
134 if p.properties.SourceExists {
Colin Cross0f3c72f2016-11-23 15:44:07 -0800135 ctx.ReplaceDependencies(name)
136 }
137 } else {
138 m.SkipInstall()
Colin Crossce75d2c2016-10-06 16:12:58 -0700139 }
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700140 if len(*p.srcs) > 0 {
141 ExtractSourceDeps(ctx, &(*p.srcs)[0])
142 }
Colin Crossce75d2c2016-10-06 16:12:58 -0700143 }
144}
145
Colin Crossa2f296f2016-11-29 15:16:18 -0800146// usePrebuilt returns true if a prebuilt should be used instead of the source module. The prebuilt
147// will be used if it is marked "prefer" or if the source module is disabled.
148func (p *Prebuilt) usePrebuilt(ctx TopDownMutatorContext, source Module) bool {
Colin Cross74d73e22017-08-02 11:05:49 -0700149 if len(*p.srcs) == 0 {
Colin Crossa2f296f2016-11-29 15:16:18 -0800150 return false
Colin Crossce75d2c2016-10-06 16:12:58 -0700151 }
Colin Crossce75d2c2016-10-06 16:12:58 -0700152
Colin Crossa2f296f2016-11-29 15:16:18 -0800153 // TODO: use p.Properties.Name and ctx.ModuleDir to override preference
Nan Zhang0007d812017-11-07 10:57:05 -0800154 if Bool(p.properties.Prefer) {
Colin Crossa2f296f2016-11-29 15:16:18 -0800155 return true
156 }
157
Colin Crossc3e7fa62017-03-17 13:14:32 -0700158 return source == nil || !source.Enabled()
Colin Crossce75d2c2016-10-06 16:12:58 -0700159}