aboutsummaryrefslogtreecommitdiff
path: root/compiler_wrapper/rusage_flag.go
diff options
context:
space:
mode:
authorTobias Bosch <tbosch@google.com>2019-07-10 06:16:04 -0700
committerTobias Bosch <tbosch@google.com>2019-07-11 08:28:44 +0000
commit9d60930e882d0e39b48e6dfab0bffa12f6f544ee (patch)
treed391ca536e9371ee9c7431f0f195a88c42ae47c8 /compiler_wrapper/rusage_flag.go
parent9332d21c19199f99886b1476cab6f4dd89e82a72 (diff)
downloadtoolchain-utils-9d60930e882d0e39b48e6dfab0bffa12f6f544ee.tar.gz
Support resource usage logging
BUG=chromium:773875 TEST=unit test Change-Id: Ifc0a7311b439e767dbb83bee350c210abbff54e4 Reviewed-on: https://chromium-review.googlesource.com/c/chromiumos/third_party/toolchain-utils/+/1695802 Reviewed-by: Tobias Bosch <tbosch@google.com> Tested-by: Tobias Bosch <tbosch@google.com>
Diffstat (limited to 'compiler_wrapper/rusage_flag.go')
-rw-r--r--compiler_wrapper/rusage_flag.go68
1 files changed, 68 insertions, 0 deletions
diff --git a/compiler_wrapper/rusage_flag.go b/compiler_wrapper/rusage_flag.go
new file mode 100644
index 00000000..b103bb38
--- /dev/null
+++ b/compiler_wrapper/rusage_flag.go
@@ -0,0 +1,68 @@
+package main
+
+import (
+ "fmt"
+ "os"
+ "path/filepath"
+ "strings"
+ "syscall"
+ "time"
+)
+
+func getRusageLogFilename(env env) string {
+ return env.getenv("GETRUSAGE")
+}
+
+func logRusage(env env, logFileName string, compilerCmd *command) (exitCode int, err error) {
+ rusageBefore := syscall.Rusage{}
+ if err := syscall.Getrusage(syscall.RUSAGE_CHILDREN, &rusageBefore); err != nil {
+ return 0, err
+ }
+ compilerCmdWithoutRusage := &command{
+ path: compilerCmd.path,
+ args: compilerCmd.args,
+ envUpdates: append(compilerCmd.envUpdates, "GETRUSAGE="),
+ }
+ startTime := time.Now()
+ exitCode, err = wrapSubprocessErrorWithSourceLoc(compilerCmdWithoutRusage,
+ env.run(compilerCmdWithoutRusage, env.stdout(), env.stderr()))
+ if err != nil {
+ return 0, err
+ }
+ elapsedRealTime := time.Since(startTime)
+ rusageAfter := syscall.Rusage{}
+ if err := syscall.Getrusage(syscall.RUSAGE_CHILDREN, &rusageAfter); err != nil {
+ return 0, err
+ }
+ elapsedSysTime := time.Duration(rusageAfter.Stime.Nano()-rusageBefore.Stime.Nano()) * time.Nanosecond
+ elapsedUserTime := time.Duration(rusageAfter.Utime.Nano()-rusageBefore.Utime.Nano()) * time.Nanosecond
+ // Note: We assume that the compiler takes more heap than any other
+ // subcommands that we might have executed before.
+ maxMemUsed := rusageAfter.Maxrss
+ absCompilerPath := compilerCmd.path
+ if !filepath.IsAbs(absCompilerPath) {
+ absCompilerPath = filepath.Join(env.getwd(), absCompilerPath)
+ }
+
+ if err := os.MkdirAll(filepath.Dir(logFileName), 0777); err != nil {
+ return 0, wrapErrorwithSourceLocf(err, "error creating rusage log directory %s", logFileName)
+ }
+ // Note: using file mode 0666 so that a root-created log is writable by others.
+ logFile, err := os.OpenFile(logFileName, os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0666)
+ if err != nil {
+ return 0, wrapErrorwithSourceLocf(err, "error creating rusage logfile %s", logFileName)
+ }
+ timeUnit := float64(time.Second)
+ if _, err := fmt.Fprintf(logFile, "%.5f : %.5f : %.5f : %d : %s : %s\n",
+ float64(elapsedRealTime)/timeUnit, float64(elapsedUserTime)/timeUnit, float64(elapsedSysTime)/timeUnit,
+ maxMemUsed, absCompilerPath,
+ strings.Join(append([]string{filepath.Base(absCompilerPath)}, compilerCmd.args...), " ")); err != nil {
+ _ = logFile.Close()
+ return 0, wrapErrorwithSourceLocf(err, "error writing rusage logfile %s", logFileName)
+ }
+ if err := logFile.Close(); err != nil {
+ return 0, wrapErrorwithSourceLocf(err, "error closing rusage logfile %s", logFileName)
+ }
+
+ return exitCode, nil
+}