1// Copyright 2018 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 java 16 17import ( 18 "encoding/json" 19 "fmt" 20 21 "android/soong/android" 22) 23 24// This singleton generates android java dependency into to a json file. It does so for each 25// blueprint Android.bp resulting in a java.Module when either make, mm, mma, mmm or mmma is 26// called. Dependency info file is generated in $OUT/module_bp_java_depend.json. 27 28func init() { 29 android.RegisterParallelSingletonType("jdeps_generator", jDepsGeneratorSingleton) 30} 31 32func jDepsGeneratorSingleton() android.Singleton { 33 return &jdepsGeneratorSingleton{} 34} 35 36type jdepsGeneratorSingleton struct { 37 outputPath android.Path 38} 39 40var _ android.SingletonMakeVarsProvider = (*jdepsGeneratorSingleton)(nil) 41 42const ( 43 jdepsJsonFileName = "module_bp_java_deps.json" 44) 45 46func (j *jdepsGeneratorSingleton) GenerateBuildActions(ctx android.SingletonContext) { 47 // (b/204397180) Generate module_bp_java_deps.json by default. 48 moduleInfos := make(map[string]android.IdeInfo) 49 50 ctx.VisitAllModules(func(module android.Module) { 51 if !module.Enabled(ctx) { 52 return 53 } 54 55 // Prevent including both prebuilts and matching source modules when one replaces the other. 56 if !android.IsModulePreferred(module) { 57 return 58 } 59 60 ideInfoProvider, ok := android.OtherModuleProvider(ctx, module, android.IdeInfoProviderKey) 61 if !ok { 62 return 63 } 64 name := ideInfoProvider.BaseModuleName 65 ideModuleNameProvider, ok := module.(android.IDECustomizedModuleName) 66 if ok { 67 name = ideModuleNameProvider.IDECustomizedModuleName() 68 } 69 70 dpInfo := moduleInfos[name] 71 dpInfo = dpInfo.Merge(ideInfoProvider) 72 dpInfo.Paths = []string{ctx.ModuleDir(module)} 73 moduleInfos[name] = dpInfo 74 75 mkProvider, ok := module.(android.AndroidMkDataProvider) 76 if !ok { 77 return 78 } 79 data := mkProvider.AndroidMk() 80 if data.Class != "" { 81 dpInfo.Classes = append(dpInfo.Classes, data.Class) 82 } 83 84 if dep, ok := android.OtherModuleProvider(ctx, module, JavaInfoProvider); ok { 85 dpInfo.Installed_paths = append(dpInfo.Installed_paths, dep.ImplementationJars.Strings()...) 86 } 87 dpInfo.Classes = android.FirstUniqueStrings(dpInfo.Classes) 88 dpInfo.Installed_paths = android.FirstUniqueStrings(dpInfo.Installed_paths) 89 moduleInfos[name] = dpInfo 90 }) 91 92 jfpath := android.PathForOutput(ctx, jdepsJsonFileName) 93 err := createJsonFile(moduleInfos, jfpath) 94 if err != nil { 95 ctx.Errorf(err.Error()) 96 } 97 j.outputPath = jfpath 98 99 // This is necessary to satisfy the dangling rules check as this file is written by Soong rather than a rule. 100 ctx.Build(pctx, android.BuildParams{ 101 Rule: android.Touch, 102 Output: jfpath, 103 }) 104} 105 106func (j *jdepsGeneratorSingleton) MakeVars(ctx android.MakeVarsContext) { 107 if j.outputPath == nil { 108 return 109 } 110 111 ctx.DistForGoal("general-tests", j.outputPath) 112} 113 114func createJsonFile(moduleInfos map[string]android.IdeInfo, jfpath android.WritablePath) error { 115 buf, err := json.MarshalIndent(moduleInfos, "", "\t") 116 if err != nil { 117 return fmt.Errorf("JSON marshal of java deps failed: %s", err) 118 } 119 err = android.WriteFileToOutputDir(jfpath, buf, 0666) 120 if err != nil { 121 return fmt.Errorf("Writing java deps to %s failed: %s", jfpath.String(), err) 122 } 123 return nil 124} 125