xref: /aosp_15_r20/external/boringssl/src/util/compare_benchmarks.go (revision 8fb009dc861624b67b6cdb62ea21f0f22d0c584b)
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