blob: 50356d1c391ce821baf68c681322ccda6e7f95e0 [file] [log] [blame]
Colin Cross068e0fe2016-12-13 15:23:47 -08001// 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
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070015package android
Colin Cross068e0fe2016-12-13 15:23:47 -080016
17import (
Colin Crossd91d7ac2017-09-12 22:52:12 -070018 "strings"
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux0d990452021-08-11 16:46:13 +000019
20 "android/soong/bazel"
Sam Delmericoc7681022022-02-04 21:01:20 +000021
22 "github.com/google/blueprint"
Colin Cross068e0fe2016-12-13 15:23:47 -080023)
24
25func init() {
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070026 RegisterModuleType("filegroup", FileGroupFactory)
Jingwen Chen32b4ece2021-01-21 03:20:18 -050027}
28
Paul Duffin35816122021-02-24 01:49:52 +000029var PrepareForTestWithFilegroup = FixtureRegisterWithContext(func(ctx RegistrationContext) {
30 ctx.RegisterModuleType("filegroup", FileGroupFactory)
31})
32
Sam Delmericoc7681022022-02-04 21:01:20 +000033// IsFilegroup checks that a module is a filegroup type
34func IsFilegroup(ctx bazel.OtherModuleContext, m blueprint.Module) bool {
35 return ctx.OtherModuleType(m) == "filegroup"
36}
37
Jingwen Chen32b4ece2021-01-21 03:20:18 -050038// https://docs.bazel.build/versions/master/be/general.html#filegroup
39type bazelFilegroupAttributes struct {
Jingwen Chen07027912021-03-15 06:02:43 -040040 Srcs bazel.LabelListAttribute
Jingwen Chen32b4ece2021-01-21 03:20:18 -050041}
42
Liz Kammerbe46fcc2021-11-01 15:32:43 -040043// ConvertWithBp2build performs bp2build conversion of filegroup
44func (fg *fileGroup) ConvertWithBp2build(ctx TopDownMutatorContext) {
Jingwen Chen07027912021-03-15 06:02:43 -040045 srcs := bazel.MakeLabelListAttribute(
46 BazelLabelForModuleSrcExcludes(ctx, fg.properties.Srcs, fg.properties.Exclude_srcs))
Jingwen Chen5146ac02021-09-02 11:44:42 +000047
48 // For Bazel compatibility, don't generate the filegroup if there is only 1
49 // source file, and that the source file is named the same as the module
50 // itself. In Bazel, eponymous filegroups like this would be an error.
51 //
52 // Instead, dependents on this single-file filegroup can just depend
53 // on the file target, instead of rule target, directly.
54 //
55 // You may ask: what if a filegroup has multiple files, and one of them
56 // shares the name? The answer: we haven't seen that in the wild, and
57 // should lock Soong itself down to prevent the behavior. For now,
58 // we raise an error if bp2build sees this problem.
59 for _, f := range srcs.Value.Includes {
60 if f.Label == fg.Name() {
61 if len(srcs.Value.Includes) > 1 {
62 ctx.ModuleErrorf("filegroup '%s' cannot contain a file with the same name", fg.Name())
63 }
64 return
65 }
66 }
67
Jingwen Chen1fd14692021-02-05 03:01:50 -050068 attrs := &bazelFilegroupAttributes{
Jingwen Chen07027912021-03-15 06:02:43 -040069 Srcs: srcs,
Jingwen Chen1fd14692021-02-05 03:01:50 -050070 }
71
Jingwen Chen14a8bda2021-06-02 11:10:02 +000072 props := bazel.BazelTargetModuleProperties{
73 Rule_class: "filegroup",
74 Bzl_load_location: "//build/bazel/rules:filegroup.bzl",
75 }
Jingwen Chen1fd14692021-02-05 03:01:50 -050076
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux447f6c92021-08-31 20:30:36 +000077 ctx.CreateBazelTargetModule(props, CommonAttributes{Name: fg.Name()}, attrs)
Colin Cross068e0fe2016-12-13 15:23:47 -080078}
79
80type fileGroupProperties struct {
81 // srcs lists files that will be included in this filegroup
Colin Cross27b922f2019-03-04 22:35:41 -080082 Srcs []string `android:"path"`
Colin Cross068e0fe2016-12-13 15:23:47 -080083
Colin Cross27b922f2019-03-04 22:35:41 -080084 Exclude_srcs []string `android:"path"`
Colin Crossfaeb7aa2017-02-01 14:12:44 -080085
86 // The base path to the files. May be used by other modules to determine which portion
87 // of the path to use. For example, when a filegroup is used as data in a cc_test rule,
88 // the base path is stripped off the path and the remaining path is used as the
89 // installation directory.
Nan Zhangea568a42017-11-08 21:20:04 -080090 Path *string
Colin Crossd91d7ac2017-09-12 22:52:12 -070091
92 // Create a make variable with the specified name that contains the list of files in the
93 // filegroup, relative to the root of the source tree.
Nan Zhangea568a42017-11-08 21:20:04 -080094 Export_to_make_var *string
Colin Cross068e0fe2016-12-13 15:23:47 -080095}
96
97type fileGroup struct {
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070098 ModuleBase
Liz Kammerea6666f2021-02-17 10:17:28 -050099 BazelModuleBase
Colin Cross068e0fe2016-12-13 15:23:47 -0800100 properties fileGroupProperties
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700101 srcs Paths
Colin Cross068e0fe2016-12-13 15:23:47 -0800102}
103
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700104var _ SourceFileProducer = (*fileGroup)(nil)
Colin Cross068e0fe2016-12-13 15:23:47 -0800105
Patrice Arruda8958a942019-03-12 10:06:00 -0700106// filegroup contains a list of files that are referenced by other modules
107// properties (such as "srcs") using the syntax ":<name>". filegroup are
108// also be used to export files across package boundaries.
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700109func FileGroupFactory() Module {
Colin Cross068e0fe2016-12-13 15:23:47 -0800110 module := &fileGroup{}
Colin Cross36242852017-06-23 15:06:31 -0700111 module.AddProperties(&module.properties)
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700112 InitAndroidModule(module)
Liz Kammerea6666f2021-02-17 10:17:28 -0500113 InitBazelModule(module)
Colin Cross36242852017-06-23 15:06:31 -0700114 return module
Colin Cross068e0fe2016-12-13 15:23:47 -0800115}
116
Jingwen Chen8f222742021-10-07 12:02:23 +0000117func (fg *fileGroup) maybeGenerateBazelBuildActions(ctx ModuleContext) {
Liz Kammer5bde22f2021-04-19 14:04:14 -0400118 if !fg.MixedBuildsEnabled(ctx) {
Jingwen Chen8f222742021-10-07 12:02:23 +0000119 return
120 }
121
Liz Kammer0940b892022-03-18 15:55:04 -0400122 archVariant := ctx.Arch().String()
Chris Parsons787fb362021-10-14 18:43:51 -0400123 osVariant := ctx.Os()
Jingwen Chen8f222742021-10-07 12:02:23 +0000124 if len(fg.Srcs()) == 1 && fg.Srcs()[0].Base() == fg.Name() {
125 // This will be a regular file target, not filegroup, in Bazel.
126 // See FilegroupBp2Build for more information.
Liz Kammer0940b892022-03-18 15:55:04 -0400127 archVariant = Common.String()
Chris Parsons787fb362021-10-14 18:43:51 -0400128 osVariant = CommonOS
Liz Kammer5bde22f2021-04-19 14:04:14 -0400129 }
Colin Cross2fafa3e2019-03-05 12:39:51 -0800130
Liz Kammer5bde22f2021-04-19 14:04:14 -0400131 bazelCtx := ctx.Config().BazelContext
Chris Parsons787fb362021-10-14 18:43:51 -0400132 filePaths, ok := bazelCtx.GetOutputFiles(fg.GetBazelLabel(ctx, fg), configKey{archVariant, osVariant})
Liz Kammer5bde22f2021-04-19 14:04:14 -0400133 if !ok {
Jingwen Chen8f222742021-10-07 12:02:23 +0000134 return
Liz Kammer5bde22f2021-04-19 14:04:14 -0400135 }
136
137 bazelOuts := make(Paths, 0, len(filePaths))
138 for _, p := range filePaths {
139 src := PathForBazelOut(ctx, p)
140 bazelOuts = append(bazelOuts, src)
141 }
142
143 fg.srcs = bazelOuts
Liz Kammer5bde22f2021-04-19 14:04:14 -0400144}
145
146func (fg *fileGroup) GenerateAndroidBuildActions(ctx ModuleContext) {
Liz Kammer5bde22f2021-04-19 14:04:14 -0400147 fg.srcs = PathsForModuleSrcExcludes(ctx, fg.properties.Srcs, fg.properties.Exclude_srcs)
Colin Cross2fafa3e2019-03-05 12:39:51 -0800148 if fg.properties.Path != nil {
149 fg.srcs = PathsWithModuleSrcSubDir(ctx, fg.srcs, String(fg.properties.Path))
150 }
Jingwen Chen8f222742021-10-07 12:02:23 +0000151
152 fg.maybeGenerateBazelBuildActions(ctx)
Colin Cross068e0fe2016-12-13 15:23:47 -0800153}
154
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700155func (fg *fileGroup) Srcs() Paths {
156 return append(Paths{}, fg.srcs...)
Colin Cross068e0fe2016-12-13 15:23:47 -0800157}
Colin Crossd91d7ac2017-09-12 22:52:12 -0700158
Dan Willemsen6a6478d2020-07-17 19:28:53 -0700159func (fg *fileGroup) MakeVars(ctx MakeVarsModuleContext) {
160 if makeVar := String(fg.properties.Export_to_make_var); makeVar != "" {
161 ctx.StrictRaw(makeVar, strings.Join(fg.srcs.Strings(), " "))
Colin Crossd91d7ac2017-09-12 22:52:12 -0700162 }
163}