Add script/bench-go-deflate
diff --git a/script/bench-go-deflate/main.go b/script/bench-go-deflate/main.go
new file mode 100644
index 0000000..5d69180
--- /dev/null
+++ b/script/bench-go-deflate/main.go
@@ -0,0 +1,122 @@
+// Copyright 2019 The Wuffs Authors.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// https://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+// +build ignore
+
+package main
+
+// This program exercises the Go standard library's Deflate decoder.
+//
+// Wuffs' C code doesn't depend on Go per se, but this program gives some
+// performance data for specific Go Deflate implementations. The equivalent
+// Wuffs benchmarks (on the same test files) are run via:
+//
+// wuffs bench std/deflate
+
+import (
+ "bytes"
+ "compress/flate"
+ "fmt"
+ "io"
+ "io/ioutil"
+ "os"
+ "runtime"
+ "time"
+)
+
+const (
+ iterscale = 10
+ reps = 5
+)
+
+type testCase = struct {
+ benchname string
+ src []byte
+ itersUnscaled uint32
+}
+
+// The various magic constants below are copied from test/c/std/deflate.c
+var testCases = []testCase{{
+ benchname: "go_deflate_decode_1k",
+ src: mustLoad("test/data/romeo.txt.gz")[20:550],
+ itersUnscaled: 2000,
+}, {
+ benchname: "go_deflate_decode_10k",
+ src: mustLoad("test/data/midsummer.txt.gz")[24:5166],
+ itersUnscaled: 300,
+}, {
+ benchname: "go_deflate_decode_100k",
+ src: mustLoad("test/data/pi.txt.gz")[17:48335],
+ itersUnscaled: 30,
+}}
+
+func mustLoad(filename string) []byte {
+ src, err := ioutil.ReadFile("../../" + filename)
+ if err != nil {
+ panic(err.Error())
+ }
+ return src
+}
+
+func main() {
+ if err := main1(); err != nil {
+ os.Stderr.WriteString(err.Error() + "\n")
+ os.Exit(1)
+ }
+}
+
+func main1() error {
+ fmt.Printf("# Go %s\n", runtime.Version())
+ fmt.Printf("#\n")
+ fmt.Printf("# The output format, including the \"Benchmark\" prefixes, is compatible with the\n")
+ fmt.Printf("# https://godoc.org/golang.org/x/perf/cmd/benchstat tool. To install it, first\n")
+ fmt.Printf("# install Go, then run \"go get golang.org/x/perf/cmd/benchstat\".\n")
+
+ for i := -1; i < reps; i++ {
+ for _, tc := range testCases {
+ runtime.GC()
+
+ start := time.Now()
+
+ iters := uint64(tc.itersUnscaled) * iterscale
+ numBytes, err := decode(tc.src)
+ if err != nil {
+ return err
+ }
+ for j := uint64(1); j < iters; j++ {
+ decode(tc.src)
+ }
+
+ elapsedNanos := time.Since(start)
+
+ kbPerS := numBytes * uint64(iters) * 1000000 / uint64(elapsedNanos)
+
+ if i < 0 {
+ continue // Warm up rep.
+ }
+
+ fmt.Printf("Benchmark%-30s %8d %12d ns/op %8d.%03d MB/s\n",
+ tc.benchname, iters, uint64(elapsedNanos)/iters, kbPerS/1000, kbPerS%1000)
+ }
+ }
+
+ return nil
+}
+
+func decode(src []byte) (numBytes uint64, retErr error) {
+ r := flate.NewReader(bytes.NewReader(src))
+ defer r.Close()
+ n, err := io.Copy(ioutil.Discard, r)
+ return uint64(n), err
+}