blob: c932ffad21ee5bc879705a679763af7e7fef48bf [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"
Colin Cross068e0fe2016-12-13 15:23:47 -080021)
22
23func init() {
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070024 RegisterModuleType("filegroup", FileGroupFactory)
Jingwen Chen32b4ece2021-01-21 03:20:18 -050025}
26
Paul Duffin35816122021-02-24 01:49:52 +000027var PrepareForTestWithFilegroup = FixtureRegisterWithContext(func(ctx RegistrationContext) {
28 ctx.RegisterModuleType("filegroup", FileGroupFactory)
29})
30
Jingwen Chen32b4ece2021-01-21 03:20:18 -050031// https://docs.bazel.build/versions/master/be/general.html#filegroup
32type bazelFilegroupAttributes struct {
Jingwen Chen07027912021-03-15 06:02:43 -040033 Srcs bazel.LabelListAttribute
Jingwen Chen32b4ece2021-01-21 03:20:18 -050034}
35
Liz Kammerbe46fcc2021-11-01 15:32:43 -040036// ConvertWithBp2build performs bp2build conversion of filegroup
37func (fg *fileGroup) ConvertWithBp2build(ctx TopDownMutatorContext) {
Jingwen Chen07027912021-03-15 06:02:43 -040038 srcs := bazel.MakeLabelListAttribute(
39 BazelLabelForModuleSrcExcludes(ctx, fg.properties.Srcs, fg.properties.Exclude_srcs))
Jingwen Chen5146ac02021-09-02 11:44:42 +000040
41 // For Bazel compatibility, don't generate the filegroup if there is only 1
42 // source file, and that the source file is named the same as the module
43 // itself. In Bazel, eponymous filegroups like this would be an error.
44 //
45 // Instead, dependents on this single-file filegroup can just depend
46 // on the file target, instead of rule target, directly.
47 //
48 // You may ask: what if a filegroup has multiple files, and one of them
49 // shares the name? The answer: we haven't seen that in the wild, and
50 // should lock Soong itself down to prevent the behavior. For now,
51 // we raise an error if bp2build sees this problem.
52 for _, f := range srcs.Value.Includes {
53 if f.Label == fg.Name() {
54 if len(srcs.Value.Includes) > 1 {
55 ctx.ModuleErrorf("filegroup '%s' cannot contain a file with the same name", fg.Name())
56 }
57 return
58 }
59 }
60
Jingwen Chen1fd14692021-02-05 03:01:50 -050061 attrs := &bazelFilegroupAttributes{
Jingwen Chen07027912021-03-15 06:02:43 -040062 Srcs: srcs,
Jingwen Chen1fd14692021-02-05 03:01:50 -050063 }
64
Jingwen Chen14a8bda2021-06-02 11:10:02 +000065 props := bazel.BazelTargetModuleProperties{
66 Rule_class: "filegroup",
67 Bzl_load_location: "//build/bazel/rules:filegroup.bzl",
68 }
Jingwen Chen1fd14692021-02-05 03:01:50 -050069
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux447f6c92021-08-31 20:30:36 +000070 ctx.CreateBazelTargetModule(props, CommonAttributes{Name: fg.Name()}, attrs)
Colin Cross068e0fe2016-12-13 15:23:47 -080071}
72
73type fileGroupProperties struct {
74 // srcs lists files that will be included in this filegroup
Colin Cross27b922f2019-03-04 22:35:41 -080075 Srcs []string `android:"path"`
Colin Cross068e0fe2016-12-13 15:23:47 -080076
Colin Cross27b922f2019-03-04 22:35:41 -080077 Exclude_srcs []string `android:"path"`
Colin Crossfaeb7aa2017-02-01 14:12:44 -080078
79 // The base path to the files. May be used by other modules to determine which portion
80 // of the path to use. For example, when a filegroup is used as data in a cc_test rule,
81 // the base path is stripped off the path and the remaining path is used as the
82 // installation directory.
Nan Zhangea568a42017-11-08 21:20:04 -080083 Path *string
Colin Crossd91d7ac2017-09-12 22:52:12 -070084
85 // Create a make variable with the specified name that contains the list of files in the
86 // filegroup, relative to the root of the source tree.
Nan Zhangea568a42017-11-08 21:20:04 -080087 Export_to_make_var *string
Colin Cross068e0fe2016-12-13 15:23:47 -080088}
89
90type fileGroup struct {
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070091 ModuleBase
Liz Kammerea6666f2021-02-17 10:17:28 -050092 BazelModuleBase
Colin Cross068e0fe2016-12-13 15:23:47 -080093 properties fileGroupProperties
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070094 srcs Paths
Colin Cross068e0fe2016-12-13 15:23:47 -080095}
96
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -070097var _ SourceFileProducer = (*fileGroup)(nil)
Colin Cross068e0fe2016-12-13 15:23:47 -080098
Patrice Arruda8958a942019-03-12 10:06:00 -070099// filegroup contains a list of files that are referenced by other modules
100// properties (such as "srcs") using the syntax ":<name>". filegroup are
101// also be used to export files across package boundaries.
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700102func FileGroupFactory() Module {
Colin Cross068e0fe2016-12-13 15:23:47 -0800103 module := &fileGroup{}
Colin Cross36242852017-06-23 15:06:31 -0700104 module.AddProperties(&module.properties)
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700105 InitAndroidModule(module)
Liz Kammerea6666f2021-02-17 10:17:28 -0500106 InitBazelModule(module)
Colin Cross36242852017-06-23 15:06:31 -0700107 return module
Colin Cross068e0fe2016-12-13 15:23:47 -0800108}
109
Jingwen Chen8f222742021-10-07 12:02:23 +0000110func (fg *fileGroup) maybeGenerateBazelBuildActions(ctx ModuleContext) {
Liz Kammer5bde22f2021-04-19 14:04:14 -0400111 if !fg.MixedBuildsEnabled(ctx) {
Jingwen Chen8f222742021-10-07 12:02:23 +0000112 return
113 }
114
115 archVariant := ctx.Arch().ArchType
Chris Parsons787fb362021-10-14 18:43:51 -0400116 osVariant := ctx.Os()
Jingwen Chen8f222742021-10-07 12:02:23 +0000117 if len(fg.Srcs()) == 1 && fg.Srcs()[0].Base() == fg.Name() {
118 // This will be a regular file target, not filegroup, in Bazel.
119 // See FilegroupBp2Build for more information.
120 archVariant = Common
Chris Parsons787fb362021-10-14 18:43:51 -0400121 osVariant = CommonOS
Liz Kammer5bde22f2021-04-19 14:04:14 -0400122 }
Colin Cross2fafa3e2019-03-05 12:39:51 -0800123
Liz Kammer5bde22f2021-04-19 14:04:14 -0400124 bazelCtx := ctx.Config().BazelContext
Chris Parsons787fb362021-10-14 18:43:51 -0400125 filePaths, ok := bazelCtx.GetOutputFiles(fg.GetBazelLabel(ctx, fg), configKey{archVariant, osVariant})
Liz Kammer5bde22f2021-04-19 14:04:14 -0400126 if !ok {
Jingwen Chen8f222742021-10-07 12:02:23 +0000127 return
Liz Kammer5bde22f2021-04-19 14:04:14 -0400128 }
129
130 bazelOuts := make(Paths, 0, len(filePaths))
131 for _, p := range filePaths {
132 src := PathForBazelOut(ctx, p)
133 bazelOuts = append(bazelOuts, src)
134 }
135
136 fg.srcs = bazelOuts
Liz Kammer5bde22f2021-04-19 14:04:14 -0400137}
138
139func (fg *fileGroup) GenerateAndroidBuildActions(ctx ModuleContext) {
Liz Kammer5bde22f2021-04-19 14:04:14 -0400140 fg.srcs = PathsForModuleSrcExcludes(ctx, fg.properties.Srcs, fg.properties.Exclude_srcs)
Colin Cross2fafa3e2019-03-05 12:39:51 -0800141 if fg.properties.Path != nil {
142 fg.srcs = PathsWithModuleSrcSubDir(ctx, fg.srcs, String(fg.properties.Path))
143 }
Jingwen Chen8f222742021-10-07 12:02:23 +0000144
145 fg.maybeGenerateBazelBuildActions(ctx)
Colin Cross068e0fe2016-12-13 15:23:47 -0800146}
147
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700148func (fg *fileGroup) Srcs() Paths {
149 return append(Paths{}, fg.srcs...)
Colin Cross068e0fe2016-12-13 15:23:47 -0800150}
Colin Crossd91d7ac2017-09-12 22:52:12 -0700151
Dan Willemsen6a6478d2020-07-17 19:28:53 -0700152func (fg *fileGroup) MakeVars(ctx MakeVarsModuleContext) {
153 if makeVar := String(fg.properties.Export_to_make_var); makeVar != "" {
154 ctx.StrictRaw(makeVar, strings.Join(fg.srcs.Strings(), " "))
Colin Crossd91d7ac2017-09-12 22:52:12 -0700155 }
156}