Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 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 | |
| 15 | package cc |
| 16 | |
| 17 | import ( |
| 18 | "encoding/json" |
| 19 | "log" |
| 20 | "os" |
| 21 | "path/filepath" |
| 22 | "strings" |
| 23 | |
| 24 | "android/soong/android" |
| 25 | ) |
| 26 | |
| 27 | // This singleton generates a compile_commands.json file. It does so for each |
| 28 | // blueprint Android.bp resulting in a cc.Module when either make, mm, mma, mmm |
| 29 | // or mmma is called. It will only create a single compile_commands.json file |
Changyeon Jo | 9627925 | 2019-12-19 23:26:16 +0000 | [diff] [blame] | 30 | // at ${OUT_DIR}/soong/development/ide/compdb/compile_commands.json. It will also symlink it |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 31 | // to ${SOONG_LINK_COMPDB_TO} if set. In general this should be created by running |
| 32 | // make SOONG_GEN_COMPDB=1 nothing to get all targets. |
| 33 | |
| 34 | func init() { |
LaMont Jones | 0c10e4d | 2023-05-16 00:58:37 +0000 | [diff] [blame] | 35 | android.RegisterParallelSingletonType("compdb_generator", compDBGeneratorSingleton) |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 36 | } |
| 37 | |
| 38 | func compDBGeneratorSingleton() android.Singleton { |
| 39 | return &compdbGeneratorSingleton{} |
| 40 | } |
| 41 | |
| 42 | type compdbGeneratorSingleton struct{} |
| 43 | |
| 44 | const ( |
| 45 | compdbFilename = "compile_commands.json" |
Changyeon Jo | 9627925 | 2019-12-19 23:26:16 +0000 | [diff] [blame] | 46 | compdbOutputProjectsDirectory = "development/ide/compdb" |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 47 | |
| 48 | // Environment variables used to modify behavior of this singleton. |
| 49 | envVariableGenerateCompdb = "SOONG_GEN_COMPDB" |
| 50 | envVariableGenerateCompdbDebugInfo = "SOONG_GEN_COMPDB_DEBUG" |
| 51 | envVariableCompdbLink = "SOONG_LINK_COMPDB_TO" |
| 52 | ) |
| 53 | |
| 54 | // A compdb entry. The compile_commands.json file is a list of these. |
| 55 | type compDbEntry struct { |
| 56 | Directory string `json:"directory"` |
| 57 | Arguments []string `json:"arguments"` |
| 58 | File string `json:"file"` |
| 59 | Output string `json:"output,omitempty"` |
| 60 | } |
| 61 | |
| 62 | func (c *compdbGeneratorSingleton) GenerateBuildActions(ctx android.SingletonContext) { |
| 63 | if !ctx.Config().IsEnvTrue(envVariableGenerateCompdb) { |
| 64 | return |
| 65 | } |
| 66 | |
| 67 | // Instruct the generator to indent the json file for easier debugging. |
| 68 | outputCompdbDebugInfo := ctx.Config().IsEnvTrue(envVariableGenerateCompdbDebugInfo) |
| 69 | |
| 70 | // We only want one entry per file. We don't care what module/isa it's from |
| 71 | m := make(map[string]compDbEntry) |
| 72 | ctx.VisitAllModules(func(module android.Module) { |
| 73 | if ccModule, ok := module.(*Module); ok { |
| 74 | if compiledModule, ok := ccModule.compiler.(CompiledInterface); ok { |
| 75 | generateCompdbProject(compiledModule, ctx, ccModule, m) |
| 76 | } |
| 77 | } |
| 78 | }) |
| 79 | |
| 80 | // Create the output file. |
Changyeon Jo | 9627925 | 2019-12-19 23:26:16 +0000 | [diff] [blame] | 81 | dir := android.PathForOutput(ctx, compdbOutputProjectsDirectory) |
Colin Cross | 988414c | 2020-01-11 01:11:46 +0000 | [diff] [blame] | 82 | os.MkdirAll(filepath.Join(android.AbsSrcDirForExistingUseCases(), dir.String()), 0777) |
Changyeon Jo | 9627925 | 2019-12-19 23:26:16 +0000 | [diff] [blame] | 83 | compDBFile := dir.Join(ctx, compdbFilename) |
Colin Cross | 988414c | 2020-01-11 01:11:46 +0000 | [diff] [blame] | 84 | f, err := os.Create(filepath.Join(android.AbsSrcDirForExistingUseCases(), compDBFile.String())) |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 85 | if err != nil { |
Changyeon Jo | 9627925 | 2019-12-19 23:26:16 +0000 | [diff] [blame] | 86 | log.Fatalf("Could not create file %s: %s", compDBFile, err) |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 87 | } |
| 88 | defer f.Close() |
| 89 | |
| 90 | v := make([]compDbEntry, 0, len(m)) |
| 91 | |
| 92 | for _, value := range m { |
| 93 | v = append(v, value) |
| 94 | } |
| 95 | var dat []byte |
| 96 | if outputCompdbDebugInfo { |
| 97 | dat, err = json.MarshalIndent(v, "", " ") |
| 98 | } else { |
| 99 | dat, err = json.Marshal(v) |
| 100 | } |
| 101 | if err != nil { |
| 102 | log.Fatalf("Failed to marshal: %s", err) |
| 103 | } |
| 104 | f.Write(dat) |
| 105 | |
Colin Cross | 988414c | 2020-01-11 01:11:46 +0000 | [diff] [blame] | 106 | if finalLinkDir := ctx.Config().Getenv(envVariableCompdbLink); finalLinkDir != "" { |
| 107 | finalLinkPath := filepath.Join(finalLinkDir, compdbFilename) |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 108 | os.Remove(finalLinkPath) |
Changyeon Jo | 9627925 | 2019-12-19 23:26:16 +0000 | [diff] [blame] | 109 | if err := os.Symlink(compDBFile.String(), finalLinkPath); err != nil { |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 110 | log.Fatalf("Unable to symlink %s to %s: %s", compDBFile, finalLinkPath, err) |
| 111 | } |
| 112 | } |
| 113 | } |
| 114 | |
| 115 | func expandAllVars(ctx android.SingletonContext, args []string) []string { |
| 116 | var out []string |
| 117 | for _, arg := range args { |
| 118 | if arg != "" { |
| 119 | if val, err := evalAndSplitVariable(ctx, arg); err == nil { |
| 120 | out = append(out, val...) |
| 121 | } else { |
| 122 | out = append(out, arg) |
| 123 | } |
| 124 | } |
| 125 | } |
| 126 | return out |
| 127 | } |
| 128 | |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 129 | func getArguments(src android.Path, ctx android.SingletonContext, ccModule *Module, ccPath string, cxxPath string) []string { |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 130 | var args []string |
| 131 | isCpp := false |
| 132 | isAsm := false |
| 133 | // TODO It would be better to ask soong for the types here. |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 134 | var clangPath string |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 135 | switch src.Ext() { |
| 136 | case ".S", ".s", ".asm": |
| 137 | isAsm = true |
| 138 | isCpp = false |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 139 | clangPath = ccPath |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 140 | case ".c": |
| 141 | isAsm = false |
| 142 | isCpp = false |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 143 | clangPath = ccPath |
Colin Cross | d34ab7c | 2019-06-27 14:46:10 -0700 | [diff] [blame] | 144 | case ".cpp", ".cc", ".cxx", ".mm": |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 145 | isAsm = false |
| 146 | isCpp = true |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 147 | clangPath = cxxPath |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 148 | default: |
| 149 | log.Print("Unknown file extension " + src.Ext() + " on file " + src.String()) |
| 150 | isAsm = true |
| 151 | isCpp = false |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 152 | clangPath = ccPath |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 153 | } |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 154 | args = append(args, clangPath) |
Colin Cross | 4af21ed | 2019-11-04 09:37:55 -0800 | [diff] [blame] | 155 | args = append(args, expandAllVars(ctx, ccModule.flags.Global.CommonFlags)...) |
| 156 | args = append(args, expandAllVars(ctx, ccModule.flags.Local.CommonFlags)...) |
| 157 | args = append(args, expandAllVars(ctx, ccModule.flags.Global.CFlags)...) |
| 158 | args = append(args, expandAllVars(ctx, ccModule.flags.Local.CFlags)...) |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 159 | if isCpp { |
Colin Cross | 4af21ed | 2019-11-04 09:37:55 -0800 | [diff] [blame] | 160 | args = append(args, expandAllVars(ctx, ccModule.flags.Global.CppFlags)...) |
| 161 | args = append(args, expandAllVars(ctx, ccModule.flags.Local.CppFlags)...) |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 162 | } else if !isAsm { |
Colin Cross | 4af21ed | 2019-11-04 09:37:55 -0800 | [diff] [blame] | 163 | args = append(args, expandAllVars(ctx, ccModule.flags.Global.ConlyFlags)...) |
| 164 | args = append(args, expandAllVars(ctx, ccModule.flags.Local.ConlyFlags)...) |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 165 | } |
| 166 | args = append(args, expandAllVars(ctx, ccModule.flags.SystemIncludeFlags)...) |
| 167 | args = append(args, src.String()) |
| 168 | return args |
| 169 | } |
| 170 | |
| 171 | func generateCompdbProject(compiledModule CompiledInterface, ctx android.SingletonContext, ccModule *Module, builds map[string]compDbEntry) { |
| 172 | srcs := compiledModule.Srcs() |
| 173 | if len(srcs) == 0 { |
| 174 | return |
| 175 | } |
| 176 | |
Colin Cross | 988414c | 2020-01-11 01:11:46 +0000 | [diff] [blame] | 177 | pathToCC, err := ctx.Eval(pctx, "${config.ClangBin}") |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 178 | ccPath := "/bin/false" |
| 179 | cxxPath := "/bin/false" |
| 180 | if err == nil { |
Colin Cross | 988414c | 2020-01-11 01:11:46 +0000 | [diff] [blame] | 181 | ccPath = filepath.Join(pathToCC, "clang") |
| 182 | cxxPath = filepath.Join(pathToCC, "clang++") |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 183 | } |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 184 | for _, src := range srcs { |
| 185 | if _, ok := builds[src.String()]; !ok { |
| 186 | builds[src.String()] = compDbEntry{ |
Colin Cross | 988414c | 2020-01-11 01:11:46 +0000 | [diff] [blame] | 187 | Directory: android.AbsSrcDirForExistingUseCases(), |
Alex Light | be96aea | 2018-11-07 11:35:47 -0800 | [diff] [blame] | 188 | Arguments: getArguments(src, ctx, ccModule, ccPath, cxxPath), |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 189 | File: src.String(), |
| 190 | } |
| 191 | } |
| 192 | } |
| 193 | } |
| 194 | |
| 195 | func evalAndSplitVariable(ctx android.SingletonContext, str string) ([]string, error) { |
| 196 | evaluated, err := ctx.Eval(pctx, str) |
| 197 | if err == nil { |
Alex Light | 0c7cc1c | 2018-10-02 11:19:46 -0700 | [diff] [blame] | 198 | return strings.Fields(evaluated), nil |
Alex Light | ec868fc | 2018-04-17 16:50:48 -0700 | [diff] [blame] | 199 | } |
| 200 | return []string{""}, err |
| 201 | } |