| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 1 | // Copyright 2017 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 main | 
|  | 16 |  | 
|  | 17 | import ( | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 18 | "context" | 
|  | 19 | "flag" | 
|  | 20 | "fmt" | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 21 | "io/ioutil" | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 22 | "os" | 
|  | 23 | "path/filepath" | 
|  | 24 | "runtime" | 
|  | 25 | "strings" | 
|  | 26 | "sync" | 
| Dan Willemsen | 22de216 | 2017-12-11 14:35:23 -0800 | [diff] [blame] | 27 | "syscall" | 
| Steven Moreland | 552432e | 2017-03-29 19:26:09 -0700 | [diff] [blame] | 28 | "time" | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 29 |  | 
|  | 30 | "android/soong/ui/build" | 
|  | 31 | "android/soong/ui/logger" | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 32 | "android/soong/ui/status" | 
|  | 33 | "android/soong/ui/terminal" | 
| Dan Willemsen | d9f6fa2 | 2016-08-21 15:17:17 -0700 | [diff] [blame] | 34 | "android/soong/ui/tracer" | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 35 | "android/soong/zip" | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 36 | ) | 
|  | 37 |  | 
|  | 38 | // We default to number of cpus / 4, which seems to be the sweet spot for my | 
|  | 39 | // system. I suspect this is mostly due to memory or disk bandwidth though, and | 
|  | 40 | // may depend on the size ofthe source tree, so this probably isn't a great | 
|  | 41 | // default. | 
|  | 42 | func detectNumJobs() int { | 
|  | 43 | if runtime.NumCPU() < 4 { | 
|  | 44 | return 1 | 
|  | 45 | } | 
|  | 46 | return runtime.NumCPU() / 4 | 
|  | 47 | } | 
|  | 48 |  | 
|  | 49 | var numJobs = flag.Int("j", detectNumJobs(), "number of parallel kati jobs") | 
|  | 50 |  | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 51 | var keepArtifacts = flag.Bool("keep", false, "keep archives of artifacts") | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 52 |  | 
|  | 53 | var outDir = flag.String("out", "", "path to store output directories (defaults to tmpdir under $OUT when empty)") | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 54 | var alternateResultDir = flag.Bool("dist", false, "write select results to $DIST_DIR (or <out>/dist when empty)") | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 55 |  | 
|  | 56 | var onlyConfig = flag.Bool("only-config", false, "Only run product config (not Soong or Kati)") | 
|  | 57 | var onlySoong = flag.Bool("only-soong", false, "Only run product config and Soong (not Kati)") | 
|  | 58 |  | 
| Dan Willemsen | 5ed900b | 2017-05-07 11:40:30 -0700 | [diff] [blame] | 59 | var buildVariant = flag.String("variant", "eng", "build variant to use") | 
|  | 60 |  | 
| Dan Willemsen | 9957b9c | 2017-10-06 15:05:05 -0700 | [diff] [blame] | 61 | var skipProducts = flag.String("skip-products", "", "comma-separated list of products to skip (known failures, etc)") | 
| Jeff Gaston | b61e3f7 | 2017-10-25 15:02:45 -0700 | [diff] [blame] | 62 | var includeProducts = flag.String("products", "", "comma-separated list of products to build") | 
| Dan Willemsen | 9957b9c | 2017-10-06 15:05:05 -0700 | [diff] [blame] | 63 |  | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 64 | const errorLeadingLines = 20 | 
|  | 65 | const errorTrailingLines = 20 | 
|  | 66 |  | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 67 | type Product struct { | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 68 | ctx     build.Context | 
|  | 69 | config  build.Config | 
|  | 70 | logFile string | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 71 | action  *status.Action | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 72 | } | 
|  | 73 |  | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 74 | func errMsgFromLog(filename string) string { | 
|  | 75 | if filename == "" { | 
|  | 76 | return "" | 
| Dan Willemsen | a4e43a7 | 2017-05-06 16:58:26 -0700 | [diff] [blame] | 77 | } | 
|  | 78 |  | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 79 | data, err := ioutil.ReadFile(filename) | 
|  | 80 | if err != nil { | 
|  | 81 | return "" | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 82 | } | 
|  | 83 |  | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 84 | lines := strings.Split(strings.TrimSpace(string(data)), "\n") | 
|  | 85 | if len(lines) > errorLeadingLines+errorTrailingLines+1 { | 
|  | 86 | lines[errorLeadingLines] = fmt.Sprintf("... skipping %d lines ...", | 
|  | 87 | len(lines)-errorLeadingLines-errorTrailingLines) | 
| Dan Willemsen | a4e43a7 | 2017-05-06 16:58:26 -0700 | [diff] [blame] | 88 |  | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 89 | lines = append(lines[:errorLeadingLines+1], | 
|  | 90 | lines[len(lines)-errorTrailingLines:]...) | 
| Dan Willemsen | a4e43a7 | 2017-05-06 16:58:26 -0700 | [diff] [blame] | 91 | } | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 92 | var buf strings.Builder | 
|  | 93 | for _, line := range lines { | 
|  | 94 | buf.WriteString("> ") | 
|  | 95 | buf.WriteString(line) | 
|  | 96 | buf.WriteString("\n") | 
| Dan Willemsen | a4e43a7 | 2017-05-06 16:58:26 -0700 | [diff] [blame] | 97 | } | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 98 | return buf.String() | 
| Dan Willemsen | a4e43a7 | 2017-05-06 16:58:26 -0700 | [diff] [blame] | 99 | } | 
|  | 100 |  | 
| Dan Willemsen | 22de216 | 2017-12-11 14:35:23 -0800 | [diff] [blame] | 101 | // TODO(b/70370883): This tool uses a lot of open files -- over the default | 
|  | 102 | // soft limit of 1024 on some systems. So bump up to the hard limit until I fix | 
|  | 103 | // the algorithm. | 
|  | 104 | func setMaxFiles(log logger.Logger) { | 
|  | 105 | var limits syscall.Rlimit | 
|  | 106 |  | 
|  | 107 | err := syscall.Getrlimit(syscall.RLIMIT_NOFILE, &limits) | 
|  | 108 | if err != nil { | 
|  | 109 | log.Println("Failed to get file limit:", err) | 
|  | 110 | return | 
|  | 111 | } | 
|  | 112 |  | 
|  | 113 | log.Verbosef("Current file limits: %d soft, %d hard", limits.Cur, limits.Max) | 
|  | 114 | if limits.Cur == limits.Max { | 
|  | 115 | return | 
|  | 116 | } | 
|  | 117 |  | 
|  | 118 | limits.Cur = limits.Max | 
|  | 119 | err = syscall.Setrlimit(syscall.RLIMIT_NOFILE, &limits) | 
|  | 120 | if err != nil { | 
|  | 121 | log.Println("Failed to increase file limit:", err) | 
|  | 122 | } | 
|  | 123 | } | 
|  | 124 |  | 
| Jeff Gaston | b61e3f7 | 2017-10-25 15:02:45 -0700 | [diff] [blame] | 125 | func inList(str string, list []string) bool { | 
|  | 126 | for _, other := range list { | 
|  | 127 | if str == other { | 
|  | 128 | return true | 
|  | 129 | } | 
|  | 130 | } | 
|  | 131 | return false | 
|  | 132 | } | 
|  | 133 |  | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 134 | func main() { | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 135 | writer := terminal.NewWriter(terminal.StdioImpl{}) | 
|  | 136 | defer writer.Finish() | 
|  | 137 |  | 
| Dan Willemsen | 34218ec | 2018-07-19 22:57:18 -0700 | [diff] [blame] | 138 | log := logger.New(writer) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 139 | defer log.Cleanup() | 
|  | 140 |  | 
|  | 141 | flag.Parse() | 
|  | 142 |  | 
|  | 143 | ctx, cancel := context.WithCancel(context.Background()) | 
|  | 144 | defer cancel() | 
|  | 145 |  | 
| Dan Willemsen | d9f6fa2 | 2016-08-21 15:17:17 -0700 | [diff] [blame] | 146 | trace := tracer.New(log) | 
|  | 147 | defer trace.Close() | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 148 |  | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 149 | stat := &status.Status{} | 
|  | 150 | defer stat.Finish() | 
|  | 151 | stat.AddOutput(terminal.NewStatusOutput(writer, "")) | 
|  | 152 |  | 
| Dan Willemsen | 34218ec | 2018-07-19 22:57:18 -0700 | [diff] [blame] | 153 | var failures failureCount | 
|  | 154 | stat.AddOutput(&failures) | 
|  | 155 |  | 
| Dan Willemsen | d9f6fa2 | 2016-08-21 15:17:17 -0700 | [diff] [blame] | 156 | build.SetupSignals(log, cancel, func() { | 
|  | 157 | trace.Close() | 
|  | 158 | log.Cleanup() | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 159 | stat.Finish() | 
| Dan Willemsen | d9f6fa2 | 2016-08-21 15:17:17 -0700 | [diff] [blame] | 160 | }) | 
|  | 161 |  | 
| Dan Willemsen | 59339a2 | 2018-07-22 21:18:45 -0700 | [diff] [blame] | 162 | buildCtx := build.Context{ContextImpl: &build.ContextImpl{ | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 163 | Context: ctx, | 
|  | 164 | Logger:  log, | 
|  | 165 | Tracer:  trace, | 
|  | 166 | Writer:  writer, | 
|  | 167 | Status:  stat, | 
| Dan Willemsen | d9f6fa2 | 2016-08-21 15:17:17 -0700 | [diff] [blame] | 168 | }} | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 169 |  | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 170 | config := build.NewConfig(buildCtx) | 
|  | 171 | if *outDir == "" { | 
| Steven Moreland | 552432e | 2017-03-29 19:26:09 -0700 | [diff] [blame] | 172 | name := "multiproduct-" + time.Now().Format("20060102150405") | 
|  | 173 |  | 
|  | 174 | *outDir = filepath.Join(config.OutDir(), name) | 
|  | 175 |  | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 176 | // Ensure the empty files exist in the output directory | 
|  | 177 | // containing our output directory too. This is mostly for | 
|  | 178 | // safety, but also triggers the ninja_build file so that our | 
|  | 179 | // build servers know that they can parse the output as if it | 
|  | 180 | // was ninja output. | 
|  | 181 | build.SetupOutDir(buildCtx, config) | 
|  | 182 |  | 
| Steven Moreland | 552432e | 2017-03-29 19:26:09 -0700 | [diff] [blame] | 183 | if err := os.MkdirAll(*outDir, 0777); err != nil { | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 184 | log.Fatalf("Failed to create tempdir: %v", err) | 
|  | 185 | } | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 186 | } | 
|  | 187 | config.Environment().Set("OUT_DIR", *outDir) | 
|  | 188 | log.Println("Output directory:", *outDir) | 
|  | 189 |  | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 190 | logsDir := filepath.Join(config.OutDir(), "logs") | 
|  | 191 | os.MkdirAll(logsDir, 0777) | 
|  | 192 |  | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 193 | build.SetupOutDir(buildCtx, config) | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 194 | if *alternateResultDir { | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 195 | distLogsDir := filepath.Join(config.DistDir(), "logs") | 
|  | 196 | os.MkdirAll(distLogsDir, 0777) | 
|  | 197 | log.SetOutput(filepath.Join(distLogsDir, "soong.log")) | 
|  | 198 | trace.SetOutput(filepath.Join(distLogsDir, "build.trace")) | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 199 | } else { | 
|  | 200 | log.SetOutput(filepath.Join(config.OutDir(), "soong.log")) | 
|  | 201 | trace.SetOutput(filepath.Join(config.OutDir(), "build.trace")) | 
|  | 202 | } | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 203 |  | 
| Dan Willemsen | 22de216 | 2017-12-11 14:35:23 -0800 | [diff] [blame] | 204 | setMaxFiles(log) | 
|  | 205 |  | 
| Dan Willemsen | 04d76ef | 2018-05-15 00:52:29 -0700 | [diff] [blame] | 206 | finder := build.NewSourceFinder(buildCtx, config) | 
|  | 207 | defer finder.Shutdown() | 
|  | 208 |  | 
|  | 209 | build.FindSources(buildCtx, config, finder) | 
|  | 210 |  | 
| Dan Willemsen | b2e6c2e | 2017-07-13 17:24:44 -0700 | [diff] [blame] | 211 | vars, err := build.DumpMakeVars(buildCtx, config, nil, []string{"all_named_products"}) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 212 | if err != nil { | 
|  | 213 | log.Fatal(err) | 
|  | 214 | } | 
| Jeff Gaston | b61e3f7 | 2017-10-25 15:02:45 -0700 | [diff] [blame] | 215 | var productsList []string | 
|  | 216 | allProducts := strings.Fields(vars["all_named_products"]) | 
|  | 217 |  | 
|  | 218 | if *includeProducts != "" { | 
|  | 219 | missingProducts := []string{} | 
|  | 220 | for _, product := range strings.Split(*includeProducts, ",") { | 
|  | 221 | if inList(product, allProducts) { | 
|  | 222 | productsList = append(productsList, product) | 
|  | 223 | } else { | 
|  | 224 | missingProducts = append(missingProducts, product) | 
|  | 225 | } | 
|  | 226 | } | 
|  | 227 | if len(missingProducts) > 0 { | 
|  | 228 | log.Fatalf("Products don't exist: %s\n", missingProducts) | 
|  | 229 | } | 
|  | 230 | } else { | 
|  | 231 | productsList = allProducts | 
|  | 232 | } | 
| Dan Willemsen | 9957b9c | 2017-10-06 15:05:05 -0700 | [diff] [blame] | 233 |  | 
|  | 234 | products := make([]string, 0, len(productsList)) | 
|  | 235 | skipList := strings.Split(*skipProducts, ",") | 
|  | 236 | skipProduct := func(p string) bool { | 
|  | 237 | for _, s := range skipList { | 
|  | 238 | if p == s { | 
|  | 239 | return true | 
|  | 240 | } | 
|  | 241 | } | 
|  | 242 | return false | 
|  | 243 | } | 
|  | 244 | for _, product := range productsList { | 
|  | 245 | if !skipProduct(product) { | 
|  | 246 | products = append(products, product) | 
|  | 247 | } else { | 
|  | 248 | log.Verbose("Skipping: ", product) | 
|  | 249 | } | 
|  | 250 | } | 
|  | 251 |  | 
|  | 252 | log.Verbose("Got product list: ", products) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 253 |  | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 254 | s := buildCtx.Status.StartTool() | 
|  | 255 | s.SetTotalActions(len(products)) | 
| Dan Willemsen | a4e43a7 | 2017-05-06 16:58:26 -0700 | [diff] [blame] | 256 |  | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 257 | var wg sync.WaitGroup | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 258 | productConfigs := make(chan Product, len(products)) | 
|  | 259 |  | 
|  | 260 | // Run the product config for every product in parallel | 
|  | 261 | for _, product := range products { | 
|  | 262 | wg.Add(1) | 
|  | 263 | go func(product string) { | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 264 | var stdLog string | 
|  | 265 |  | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 266 | defer wg.Done() | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 267 |  | 
|  | 268 | action := &status.Action{ | 
|  | 269 | Description: product, | 
|  | 270 | Outputs:     []string{product}, | 
|  | 271 | } | 
|  | 272 | s.StartAction(action) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 273 | defer logger.Recover(func(err error) { | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 274 | s.FinishAction(status.ActionResult{ | 
|  | 275 | Action: action, | 
|  | 276 | Error:  err, | 
|  | 277 | Output: errMsgFromLog(stdLog), | 
|  | 278 | }) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 279 | }) | 
|  | 280 |  | 
|  | 281 | productOutDir := filepath.Join(config.OutDir(), product) | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 282 | productLogDir := filepath.Join(logsDir, product) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 283 |  | 
|  | 284 | if err := os.MkdirAll(productOutDir, 0777); err != nil { | 
|  | 285 | log.Fatalf("Error creating out directory: %v", err) | 
|  | 286 | } | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 287 | if err := os.MkdirAll(productLogDir, 0777); err != nil { | 
|  | 288 | log.Fatalf("Error creating log directory: %v", err) | 
|  | 289 | } | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 290 |  | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 291 | stdLog = filepath.Join(productLogDir, "std.log") | 
|  | 292 | f, err := os.Create(stdLog) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 293 | if err != nil { | 
|  | 294 | log.Fatalf("Error creating std.log: %v", err) | 
|  | 295 | } | 
|  | 296 |  | 
| Dan Willemsen | f196396 | 2017-11-11 15:44:51 -0800 | [diff] [blame] | 297 | productLog := logger.New(f) | 
| Dan Willemsen | f624fb9 | 2017-05-19 16:39:04 -0700 | [diff] [blame] | 298 | productLog.SetOutput(filepath.Join(productLogDir, "soong.log")) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 299 |  | 
| Dan Willemsen | 59339a2 | 2018-07-22 21:18:45 -0700 | [diff] [blame] | 300 | productCtx := build.Context{ContextImpl: &build.ContextImpl{ | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 301 | Context: ctx, | 
|  | 302 | Logger:  productLog, | 
|  | 303 | Tracer:  trace, | 
|  | 304 | Writer:  terminal.NewWriter(terminal.NewCustomStdio(nil, f, f)), | 
|  | 305 | Thread:  trace.NewThread(product), | 
|  | 306 | Status:  &status.Status{}, | 
| Dan Willemsen | d9f6fa2 | 2016-08-21 15:17:17 -0700 | [diff] [blame] | 307 | }} | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 308 | productCtx.Status.AddOutput(terminal.NewStatusOutput(productCtx.Writer, "")) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 309 |  | 
|  | 310 | productConfig := build.NewConfig(productCtx) | 
|  | 311 | productConfig.Environment().Set("OUT_DIR", productOutDir) | 
| Jeff Gaston | 743e29e | 2017-08-17 14:09:23 -0700 | [diff] [blame] | 312 | build.FindSources(productCtx, productConfig, finder) | 
| Dan Willemsen | 5ed900b | 2017-05-07 11:40:30 -0700 | [diff] [blame] | 313 | productConfig.Lunch(productCtx, product, *buildVariant) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 314 |  | 
|  | 315 | build.Build(productCtx, productConfig, build.BuildProductConfig) | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 316 | productConfigs <- Product{productCtx, productConfig, stdLog, action} | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 317 | }(product) | 
|  | 318 | } | 
|  | 319 | go func() { | 
|  | 320 | defer close(productConfigs) | 
|  | 321 | wg.Wait() | 
|  | 322 | }() | 
|  | 323 |  | 
|  | 324 | var wg2 sync.WaitGroup | 
|  | 325 | // Then run up to numJobs worth of Soong and Kati | 
|  | 326 | for i := 0; i < *numJobs; i++ { | 
|  | 327 | wg2.Add(1) | 
|  | 328 | go func() { | 
|  | 329 | defer wg2.Done() | 
|  | 330 | for product := range productConfigs { | 
|  | 331 | func() { | 
|  | 332 | defer logger.Recover(func(err error) { | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 333 | s.FinishAction(status.ActionResult{ | 
|  | 334 | Action: product.action, | 
|  | 335 | Error:  err, | 
|  | 336 | Output: errMsgFromLog(product.logFile), | 
|  | 337 | }) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 338 | }) | 
|  | 339 |  | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 340 | defer func() { | 
|  | 341 | if *keepArtifacts { | 
|  | 342 | args := zip.ZipArgs{ | 
|  | 343 | FileArgs: []zip.FileArg{ | 
|  | 344 | { | 
|  | 345 | GlobDir:             product.config.OutDir(), | 
|  | 346 | SourcePrefixToStrip: product.config.OutDir(), | 
|  | 347 | }, | 
|  | 348 | }, | 
|  | 349 | OutputFilePath:   filepath.Join(config.OutDir(), product.config.TargetProduct()+".zip"), | 
|  | 350 | NumParallelJobs:  runtime.NumCPU(), | 
|  | 351 | CompressionLevel: 5, | 
|  | 352 | } | 
|  | 353 | if err := zip.Run(args); err != nil { | 
|  | 354 | log.Fatalf("Error zipping artifacts: %v", err) | 
|  | 355 | } | 
|  | 356 | } | 
|  | 357 | os.RemoveAll(product.config.OutDir()) | 
|  | 358 | }() | 
|  | 359 |  | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 360 | buildWhat := 0 | 
|  | 361 | if !*onlyConfig { | 
|  | 362 | buildWhat |= build.BuildSoong | 
|  | 363 | if !*onlySoong { | 
|  | 364 | buildWhat |= build.BuildKati | 
|  | 365 | } | 
|  | 366 | } | 
|  | 367 | build.Build(product.ctx, product.config, buildWhat) | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 368 | s.FinishAction(status.ActionResult{ | 
|  | 369 | Action: product.action, | 
|  | 370 | }) | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 371 | }() | 
|  | 372 | } | 
|  | 373 | }() | 
|  | 374 | } | 
| Dan Willemsen | a4e43a7 | 2017-05-06 16:58:26 -0700 | [diff] [blame] | 375 | wg2.Wait() | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 376 |  | 
| Dan Willemsen | e348076 | 2017-11-07 11:23:27 -0800 | [diff] [blame] | 377 | if *alternateResultDir { | 
|  | 378 | args := zip.ZipArgs{ | 
|  | 379 | FileArgs: []zip.FileArg{ | 
|  | 380 | {GlobDir: logsDir, SourcePrefixToStrip: logsDir}, | 
|  | 381 | }, | 
|  | 382 | OutputFilePath:   filepath.Join(config.DistDir(), "logs.zip"), | 
|  | 383 | NumParallelJobs:  runtime.NumCPU(), | 
|  | 384 | CompressionLevel: 5, | 
|  | 385 | } | 
|  | 386 | if err := zip.Run(args); err != nil { | 
|  | 387 | log.Fatalf("Error zipping logs: %v", err) | 
|  | 388 | } | 
|  | 389 | } | 
|  | 390 |  | 
| Dan Willemsen | b82471a | 2018-05-17 16:37:09 -0700 | [diff] [blame] | 391 | s.Finish() | 
| Dan Willemsen | 34218ec | 2018-07-19 22:57:18 -0700 | [diff] [blame] | 392 |  | 
|  | 393 | if failures == 1 { | 
|  | 394 | log.Fatal("1 failure") | 
|  | 395 | } else if failures > 1 { | 
|  | 396 | log.Fatalf("%d failures", failures) | 
|  | 397 | } else { | 
|  | 398 | writer.Print("Success") | 
|  | 399 | } | 
| Dan Willemsen | c2af0be | 2017-01-20 14:10:01 -0800 | [diff] [blame] | 400 | } | 
| Dan Willemsen | 34218ec | 2018-07-19 22:57:18 -0700 | [diff] [blame] | 401 |  | 
|  | 402 | type failureCount int | 
|  | 403 |  | 
|  | 404 | func (f *failureCount) StartAction(action *status.Action, counts status.Counts) {} | 
|  | 405 |  | 
|  | 406 | func (f *failureCount) FinishAction(result status.ActionResult, counts status.Counts) { | 
|  | 407 | if result.Error != nil { | 
|  | 408 | *f += 1 | 
|  | 409 | } | 
|  | 410 | } | 
|  | 411 |  | 
|  | 412 | func (f *failureCount) Message(level status.MsgLevel, message string) { | 
|  | 413 | if level >= status.ErrorLvl { | 
|  | 414 | *f += 1 | 
|  | 415 | } | 
|  | 416 | } | 
|  | 417 |  | 
|  | 418 | func (f *failureCount) Flush() {} |