1*8fb009dcSAndroid Build Coastguard Worker// Copyright (c) 2020 Google Inc. 2*8fb009dcSAndroid Build Coastguard Worker// 3*8fb009dcSAndroid Build Coastguard Worker// Permission to use, copy, modify, and/or distribute this software for any 4*8fb009dcSAndroid Build Coastguard Worker// purpose with or without fee is hereby granted, provided that the above 5*8fb009dcSAndroid Build Coastguard Worker// copyright notice and this permission notice appear in all copies. 6*8fb009dcSAndroid Build Coastguard Worker// 7*8fb009dcSAndroid Build Coastguard Worker// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 8*8fb009dcSAndroid Build Coastguard Worker// WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 9*8fb009dcSAndroid Build Coastguard Worker// MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY 10*8fb009dcSAndroid Build Coastguard Worker// SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 11*8fb009dcSAndroid Build Coastguard Worker// WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION 12*8fb009dcSAndroid Build Coastguard Worker// OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN 13*8fb009dcSAndroid Build Coastguard Worker// CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 14*8fb009dcSAndroid Build Coastguard Worker 15*8fb009dcSAndroid Build Coastguard Worker//go:build ignore 16*8fb009dcSAndroid Build Coastguard Worker 17*8fb009dcSAndroid Build Coastguard Worker// compare_benchmarks takes the JSON-formatted output of bssl speed and 18*8fb009dcSAndroid Build Coastguard Worker// compares it against a baseline output. 19*8fb009dcSAndroid Build Coastguard Workerpackage main 20*8fb009dcSAndroid Build Coastguard Worker 21*8fb009dcSAndroid Build Coastguard Workerimport ( 22*8fb009dcSAndroid Build Coastguard Worker "encoding/json" 23*8fb009dcSAndroid Build Coastguard Worker "flag" 24*8fb009dcSAndroid Build Coastguard Worker "fmt" 25*8fb009dcSAndroid Build Coastguard Worker "os" 26*8fb009dcSAndroid Build Coastguard Worker) 27*8fb009dcSAndroid Build Coastguard Worker 28*8fb009dcSAndroid Build Coastguard Workervar baselineFile = flag.String("baseline", "", "the path to the JSON file containing the base results") 29*8fb009dcSAndroid Build Coastguard Worker 30*8fb009dcSAndroid Build Coastguard Workertype Result struct { 31*8fb009dcSAndroid Build Coastguard Worker Description string `json:"description"` 32*8fb009dcSAndroid Build Coastguard Worker NumCalls int `json:"numCalls"` 33*8fb009dcSAndroid Build Coastguard Worker Microseconds int `json:"microseconds"` 34*8fb009dcSAndroid Build Coastguard Worker BytesPerCall int `json:"bytesPerCall"` 35*8fb009dcSAndroid Build Coastguard Worker} 36*8fb009dcSAndroid Build Coastguard Worker 37*8fb009dcSAndroid Build Coastguard Workerfunc (r *Result) Speed() (float64, string) { 38*8fb009dcSAndroid Build Coastguard Worker callsPerSecond := float64(r.NumCalls) / float64(r.Microseconds) * 1000000 39*8fb009dcSAndroid Build Coastguard Worker if r.BytesPerCall == 0 { 40*8fb009dcSAndroid Build Coastguard Worker return callsPerSecond, "ops/sec" 41*8fb009dcSAndroid Build Coastguard Worker } 42*8fb009dcSAndroid Build Coastguard Worker return callsPerSecond * float64(r.BytesPerCall) / 1000000, "MB/sec" 43*8fb009dcSAndroid Build Coastguard Worker} 44*8fb009dcSAndroid Build Coastguard Worker 45*8fb009dcSAndroid Build Coastguard Workerfunc printResult(result Result, baseline *Result) error { 46*8fb009dcSAndroid Build Coastguard Worker if baseline != nil { 47*8fb009dcSAndroid Build Coastguard Worker if result.Description != baseline.Description { 48*8fb009dcSAndroid Build Coastguard Worker return fmt.Errorf("result did not match baseline: %q vs %q", result.Description, baseline.Description) 49*8fb009dcSAndroid Build Coastguard Worker } 50*8fb009dcSAndroid Build Coastguard Worker 51*8fb009dcSAndroid Build Coastguard Worker if result.BytesPerCall != baseline.BytesPerCall { 52*8fb009dcSAndroid Build Coastguard Worker return fmt.Errorf("result %q bytes per call did not match baseline: %d vs %d", result.Description, result.BytesPerCall, baseline.BytesPerCall) 53*8fb009dcSAndroid Build Coastguard Worker } 54*8fb009dcSAndroid Build Coastguard Worker } 55*8fb009dcSAndroid Build Coastguard Worker 56*8fb009dcSAndroid Build Coastguard Worker newSpeed, unit := result.Speed() 57*8fb009dcSAndroid Build Coastguard Worker fmt.Printf("Did %d %s operations in %dus (%.1f %s)", result.NumCalls, result.Description, result.Microseconds, newSpeed, unit) 58*8fb009dcSAndroid Build Coastguard Worker if baseline != nil { 59*8fb009dcSAndroid Build Coastguard Worker oldSpeed, _ := baseline.Speed() 60*8fb009dcSAndroid Build Coastguard Worker fmt.Printf(" [%+.1f%%]", (newSpeed-oldSpeed)/oldSpeed*100) 61*8fb009dcSAndroid Build Coastguard Worker } 62*8fb009dcSAndroid Build Coastguard Worker fmt.Printf("\n") 63*8fb009dcSAndroid Build Coastguard Worker return nil 64*8fb009dcSAndroid Build Coastguard Worker} 65*8fb009dcSAndroid Build Coastguard Worker 66*8fb009dcSAndroid Build Coastguard Workerfunc readResults(path string) ([]Result, error) { 67*8fb009dcSAndroid Build Coastguard Worker data, err := os.ReadFile(path) 68*8fb009dcSAndroid Build Coastguard Worker if err != nil { 69*8fb009dcSAndroid Build Coastguard Worker return nil, err 70*8fb009dcSAndroid Build Coastguard Worker } 71*8fb009dcSAndroid Build Coastguard Worker var ret []Result 72*8fb009dcSAndroid Build Coastguard Worker if err := json.Unmarshal(data, &ret); err != nil { 73*8fb009dcSAndroid Build Coastguard Worker return nil, err 74*8fb009dcSAndroid Build Coastguard Worker } 75*8fb009dcSAndroid Build Coastguard Worker return ret, nil 76*8fb009dcSAndroid Build Coastguard Worker} 77*8fb009dcSAndroid Build Coastguard Worker 78*8fb009dcSAndroid Build Coastguard Workerfunc main() { 79*8fb009dcSAndroid Build Coastguard Worker flag.Parse() 80*8fb009dcSAndroid Build Coastguard Worker 81*8fb009dcSAndroid Build Coastguard Worker baseline, err := readResults(*baselineFile) 82*8fb009dcSAndroid Build Coastguard Worker if err != nil { 83*8fb009dcSAndroid Build Coastguard Worker fmt.Fprintf(os.Stderr, "Error reading %q: %s\n", *baselineFile, err) 84*8fb009dcSAndroid Build Coastguard Worker os.Exit(1) 85*8fb009dcSAndroid Build Coastguard Worker } 86*8fb009dcSAndroid Build Coastguard Worker 87*8fb009dcSAndroid Build Coastguard Worker fmt.Println(*baselineFile) 88*8fb009dcSAndroid Build Coastguard Worker for _, result := range baseline { 89*8fb009dcSAndroid Build Coastguard Worker if err := printResult(result, nil); err != nil { 90*8fb009dcSAndroid Build Coastguard Worker fmt.Fprintf(os.Stderr, "Error in %q: %s\n", *baselineFile, err) 91*8fb009dcSAndroid Build Coastguard Worker os.Exit(1) 92*8fb009dcSAndroid Build Coastguard Worker } 93*8fb009dcSAndroid Build Coastguard Worker } 94*8fb009dcSAndroid Build Coastguard Worker 95*8fb009dcSAndroid Build Coastguard Worker for _, arg := range flag.Args() { 96*8fb009dcSAndroid Build Coastguard Worker results, err := readResults(arg) 97*8fb009dcSAndroid Build Coastguard Worker if err != nil { 98*8fb009dcSAndroid Build Coastguard Worker fmt.Fprintf(os.Stderr, "Error reading %q: %s\n", arg, err) 99*8fb009dcSAndroid Build Coastguard Worker os.Exit(1) 100*8fb009dcSAndroid Build Coastguard Worker } 101*8fb009dcSAndroid Build Coastguard Worker 102*8fb009dcSAndroid Build Coastguard Worker if len(results) != len(baseline) { 103*8fb009dcSAndroid Build Coastguard Worker fmt.Fprintf(os.Stderr, "Result files %q and %q have different lengths\n", arg, *baselineFile) 104*8fb009dcSAndroid Build Coastguard Worker os.Exit(1) 105*8fb009dcSAndroid Build Coastguard Worker } 106*8fb009dcSAndroid Build Coastguard Worker 107*8fb009dcSAndroid Build Coastguard Worker fmt.Printf("\n%s\n", arg) 108*8fb009dcSAndroid Build Coastguard Worker for i, result := range results { 109*8fb009dcSAndroid Build Coastguard Worker if err := printResult(result, &baseline[i]); err != nil { 110*8fb009dcSAndroid Build Coastguard Worker fmt.Fprintf(os.Stderr, "Error in %q: %s\n", arg, err) 111*8fb009dcSAndroid Build Coastguard Worker os.Exit(1) 112*8fb009dcSAndroid Build Coastguard Worker } 113*8fb009dcSAndroid Build Coastguard Worker } 114*8fb009dcSAndroid Build Coastguard Worker } 115*8fb009dcSAndroid Build Coastguard Worker} 116