1// Copyright 2019 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// This executable runs a series of build commands to test and benchmark some critical user journeys. 16package main 17 18import ( 19 "context" 20 "fmt" 21 "os" 22 "path/filepath" 23 "strconv" 24 "strings" 25 "time" 26 27 "android/soong/ui/build" 28 "android/soong/ui/logger" 29 "android/soong/ui/metrics" 30 "android/soong/ui/status" 31 "android/soong/ui/terminal" 32 "android/soong/ui/tracer" 33) 34 35type Test struct { 36 name string 37 args []string 38 before func() error 39 40 results TestResults 41} 42 43type TestResults struct { 44 metrics *metrics.Metrics 45 err error 46} 47 48// Run runs a single build command. It emulates the "m" command line by calling into Soong UI directly. 49func (t *Test) Run(logsDir string) { 50 output := terminal.NewStatusOutput(os.Stdout, "", false, false) 51 52 log := logger.New(output) 53 defer log.Cleanup() 54 55 ctx, cancel := context.WithCancel(context.Background()) 56 defer cancel() 57 58 trace := tracer.New(log) 59 defer trace.Close() 60 61 met := metrics.New() 62 63 stat := &status.Status{} 64 defer stat.Finish() 65 stat.AddOutput(output) 66 stat.AddOutput(trace.StatusTracer()) 67 68 build.SetupSignals(log, cancel, func() { 69 trace.Close() 70 log.Cleanup() 71 stat.Finish() 72 }) 73 74 buildCtx := build.Context{ContextImpl: &build.ContextImpl{ 75 Context: ctx, 76 Logger: log, 77 Metrics: met, 78 Tracer: trace, 79 Writer: output, 80 Status: stat, 81 }} 82 83 defer logger.Recover(func(err error) { 84 t.results.err = err 85 }) 86 87 config := build.NewConfig(buildCtx, t.args...) 88 build.SetupOutDir(buildCtx, config) 89 90 os.MkdirAll(logsDir, 0777) 91 log.SetOutput(filepath.Join(logsDir, "soong.log")) 92 trace.SetOutput(filepath.Join(logsDir, "build.trace")) 93 stat.AddOutput(status.NewVerboseLog(log, filepath.Join(logsDir, "verbose.log"))) 94 stat.AddOutput(status.NewErrorLog(log, filepath.Join(logsDir, "error.log"))) 95 stat.AddOutput(status.NewProtoErrorLog(log, filepath.Join(logsDir, "build_error"))) 96 stat.AddOutput(status.NewCriticalPath(log)) 97 98 defer met.Dump(filepath.Join(logsDir, "soong_metrics")) 99 100 if start, ok := os.LookupEnv("TRACE_BEGIN_SOONG"); ok { 101 if !strings.HasSuffix(start, "N") { 102 if start_time, err := strconv.ParseUint(start, 10, 64); err == nil { 103 log.Verbosef("Took %dms to start up.", 104 time.Since(time.Unix(0, int64(start_time))).Nanoseconds()/time.Millisecond.Nanoseconds()) 105 buildCtx.CompleteTrace(metrics.RunSetupTool, "startup", start_time, uint64(time.Now().UnixNano())) 106 } 107 } 108 109 if executable, err := os.Executable(); err == nil { 110 trace.ImportMicrofactoryLog(filepath.Join(filepath.Dir(executable), "."+filepath.Base(executable)+".trace")) 111 } 112 } 113 114 f := build.NewSourceFinder(buildCtx, config) 115 defer f.Shutdown() 116 build.FindSources(buildCtx, config, f) 117 118 build.Build(buildCtx, config) 119 120 t.results.metrics = met 121} 122 123// Touch the Intent.java file to cause a rebuild of the frameworks to monitor the 124// incremental build speed as mentioned b/152046247. Intent.java file was chosen 125// as it is a key component of the framework and is often modified. 126func touchIntentFile() error { 127 const intentFileName = "frameworks/base/core/java/android/content/Intent.java" 128 currentTime := time.Now().Local() 129 return os.Chtimes(intentFileName, currentTime, currentTime) 130} 131 132func main() { 133 outDir := os.Getenv("OUT_DIR") 134 if outDir == "" { 135 outDir = "out" 136 } 137 138 cujDir := filepath.Join(outDir, "cuj_tests") 139 140 // Use a subdirectory for the out directory for the tests to keep them isolated. 141 os.Setenv("OUT_DIR", filepath.Join(cujDir, "out")) 142 143 // Each of these tests is run in sequence without resetting the output tree. The state of the output tree will 144 // affect each successive test. To maintain the validity of the benchmarks across changes, care must be taken 145 // to avoid changing the state of the tree when a test is run. This is most easily accomplished by adding tests 146 // at the end. 147 tests := []Test{ 148 { 149 // Reset the out directory to get reproducible results. 150 name: "clean", 151 args: []string{"clean"}, 152 }, 153 { 154 // Parse the build files. 155 name: "nothing", 156 args: []string{"nothing"}, 157 }, 158 { 159 // Parse the build files again to monitor issues like globs rerunning. 160 name: "nothing_rebuild", 161 args: []string{"nothing"}, 162 }, 163 { 164 // Parse the build files again, this should always be very short. 165 name: "nothing_rebuild_twice", 166 args: []string{"nothing"}, 167 }, 168 { 169 // Build the framework as a common developer task and one that keeps getting longer. 170 name: "framework", 171 args: []string{"framework"}, 172 }, 173 { 174 // Build the framework again to make sure it doesn't rebuild anything. 175 name: "framework_rebuild", 176 args: []string{"framework"}, 177 }, 178 { 179 // Build the framework again to make sure it doesn't rebuild anything even if it did the second time. 180 name: "framework_rebuild_twice", 181 args: []string{"framework"}, 182 }, 183 { 184 // Scenario major_inc_build (b/152046247): tracking build speed of major incremental build. 185 name: "major_inc_build_droid", 186 args: []string{"droid"}, 187 }, 188 { 189 name: "major_inc_build_framework_minus_apex_after_droid_build", 190 args: []string{"framework-minus-apex"}, 191 before: touchIntentFile, 192 }, 193 { 194 name: "major_inc_build_framework_after_droid_build", 195 args: []string{"framework"}, 196 before: touchIntentFile, 197 }, 198 { 199 name: "major_inc_build_sync_after_droid_build", 200 args: []string{"sync"}, 201 before: touchIntentFile, 202 }, 203 { 204 name: "major_inc_build_droid_rebuild", 205 args: []string{"droid"}, 206 before: touchIntentFile, 207 }, 208 { 209 name: "major_inc_build_update_api_after_droid_rebuild", 210 args: []string{"update-api"}, 211 before: touchIntentFile, 212 }, 213 } 214 215 cujMetrics := metrics.NewCriticalUserJourneysMetrics() 216 defer cujMetrics.Dump(filepath.Join(cujDir, "logs", "cuj_metrics.pb")) 217 218 for i, t := range tests { 219 logsSubDir := fmt.Sprintf("%02d_%s", i, t.name) 220 logsDir := filepath.Join(cujDir, "logs", logsSubDir) 221 if t.before != nil { 222 if err := t.before(); err != nil { 223 fmt.Printf("error running before function on test %q: %v\n", t.name, err) 224 break 225 } 226 } 227 t.Run(logsDir) 228 if t.results.err != nil { 229 fmt.Printf("error running test %q: %s\n", t.name, t.results.err) 230 break 231 } 232 if t.results.metrics != nil { 233 cujMetrics.Add(t.name, t.results.metrics) 234 } 235 } 236} 237