|  | // Copyright 2009 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package binary | 
|  |  | 
|  | import ( | 
|  | "bytes" | 
|  | "fmt" | 
|  | "io" | 
|  | "io/ioutil" | 
|  | "math" | 
|  | "reflect" | 
|  | "strings" | 
|  | "sync" | 
|  | "testing" | 
|  | ) | 
|  |  | 
|  | type Struct struct { | 
|  | Int8       int8 | 
|  | Int16      int16 | 
|  | Int32      int32 | 
|  | Int64      int64 | 
|  | Uint8      uint8 | 
|  | Uint16     uint16 | 
|  | Uint32     uint32 | 
|  | Uint64     uint64 | 
|  | Float32    float32 | 
|  | Float64    float64 | 
|  | Complex64  complex64 | 
|  | Complex128 complex128 | 
|  | Array      [4]uint8 | 
|  | Bool       bool | 
|  | BoolArray  [4]bool | 
|  | } | 
|  |  | 
|  | type T struct { | 
|  | Int     int | 
|  | Uint    uint | 
|  | Uintptr uintptr | 
|  | Array   [4]int | 
|  | } | 
|  |  | 
|  | var s = Struct{ | 
|  | 0x01, | 
|  | 0x0203, | 
|  | 0x04050607, | 
|  | 0x08090a0b0c0d0e0f, | 
|  | 0x10, | 
|  | 0x1112, | 
|  | 0x13141516, | 
|  | 0x1718191a1b1c1d1e, | 
|  |  | 
|  | math.Float32frombits(0x1f202122), | 
|  | math.Float64frombits(0x232425262728292a), | 
|  | complex( | 
|  | math.Float32frombits(0x2b2c2d2e), | 
|  | math.Float32frombits(0x2f303132), | 
|  | ), | 
|  | complex( | 
|  | math.Float64frombits(0x333435363738393a), | 
|  | math.Float64frombits(0x3b3c3d3e3f404142), | 
|  | ), | 
|  |  | 
|  | [4]uint8{0x43, 0x44, 0x45, 0x46}, | 
|  |  | 
|  | true, | 
|  | [4]bool{true, false, true, false}, | 
|  | } | 
|  |  | 
|  | var big = []byte{ | 
|  | 1, | 
|  | 2, 3, | 
|  | 4, 5, 6, 7, | 
|  | 8, 9, 10, 11, 12, 13, 14, 15, | 
|  | 16, | 
|  | 17, 18, | 
|  | 19, 20, 21, 22, | 
|  | 23, 24, 25, 26, 27, 28, 29, 30, | 
|  |  | 
|  | 31, 32, 33, 34, | 
|  | 35, 36, 37, 38, 39, 40, 41, 42, | 
|  | 43, 44, 45, 46, 47, 48, 49, 50, | 
|  | 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, | 
|  |  | 
|  | 67, 68, 69, 70, | 
|  |  | 
|  | 1, | 
|  | 1, 0, 1, 0, | 
|  | } | 
|  |  | 
|  | var little = []byte{ | 
|  | 1, | 
|  | 3, 2, | 
|  | 7, 6, 5, 4, | 
|  | 15, 14, 13, 12, 11, 10, 9, 8, | 
|  | 16, | 
|  | 18, 17, | 
|  | 22, 21, 20, 19, | 
|  | 30, 29, 28, 27, 26, 25, 24, 23, | 
|  |  | 
|  | 34, 33, 32, 31, | 
|  | 42, 41, 40, 39, 38, 37, 36, 35, | 
|  | 46, 45, 44, 43, 50, 49, 48, 47, | 
|  | 58, 57, 56, 55, 54, 53, 52, 51, 66, 65, 64, 63, 62, 61, 60, 59, | 
|  |  | 
|  | 67, 68, 69, 70, | 
|  |  | 
|  | 1, | 
|  | 1, 0, 1, 0, | 
|  | } | 
|  |  | 
|  | var src = []byte{1, 2, 3, 4, 5, 6, 7, 8} | 
|  | var res = []int32{0x01020304, 0x05060708} | 
|  | var putbuf = []byte{0, 0, 0, 0, 0, 0, 0, 0} | 
|  |  | 
|  | func checkResult(t *testing.T, dir string, order ByteOrder, err error, have, want interface{}) { | 
|  | if err != nil { | 
|  | t.Errorf("%v %v: %v", dir, order, err) | 
|  | return | 
|  | } | 
|  | if !reflect.DeepEqual(have, want) { | 
|  | t.Errorf("%v %v:\n\thave %+v\n\twant %+v", dir, order, have, want) | 
|  | } | 
|  | } | 
|  |  | 
|  | func testRead(t *testing.T, order ByteOrder, b []byte, s1 interface{}) { | 
|  | var s2 Struct | 
|  | err := Read(bytes.NewReader(b), order, &s2) | 
|  | checkResult(t, "Read", order, err, s2, s1) | 
|  | } | 
|  |  | 
|  | func testWrite(t *testing.T, order ByteOrder, b []byte, s1 interface{}) { | 
|  | buf := new(bytes.Buffer) | 
|  | err := Write(buf, order, s1) | 
|  | checkResult(t, "Write", order, err, buf.Bytes(), b) | 
|  | } | 
|  |  | 
|  | func TestLittleEndianRead(t *testing.T)     { testRead(t, LittleEndian, little, s) } | 
|  | func TestLittleEndianWrite(t *testing.T)    { testWrite(t, LittleEndian, little, s) } | 
|  | func TestLittleEndianPtrWrite(t *testing.T) { testWrite(t, LittleEndian, little, &s) } | 
|  |  | 
|  | func TestBigEndianRead(t *testing.T)     { testRead(t, BigEndian, big, s) } | 
|  | func TestBigEndianWrite(t *testing.T)    { testWrite(t, BigEndian, big, s) } | 
|  | func TestBigEndianPtrWrite(t *testing.T) { testWrite(t, BigEndian, big, &s) } | 
|  |  | 
|  | func TestReadSlice(t *testing.T) { | 
|  | slice := make([]int32, 2) | 
|  | err := Read(bytes.NewReader(src), BigEndian, slice) | 
|  | checkResult(t, "ReadSlice", BigEndian, err, slice, res) | 
|  | } | 
|  |  | 
|  | func TestWriteSlice(t *testing.T) { | 
|  | buf := new(bytes.Buffer) | 
|  | err := Write(buf, BigEndian, res) | 
|  | checkResult(t, "WriteSlice", BigEndian, err, buf.Bytes(), src) | 
|  | } | 
|  |  | 
|  | func TestReadBool(t *testing.T) { | 
|  | var res bool | 
|  | var err error | 
|  | err = Read(bytes.NewReader([]byte{0}), BigEndian, &res) | 
|  | checkResult(t, "ReadBool", BigEndian, err, res, false) | 
|  | res = false | 
|  | err = Read(bytes.NewReader([]byte{1}), BigEndian, &res) | 
|  | checkResult(t, "ReadBool", BigEndian, err, res, true) | 
|  | res = false | 
|  | err = Read(bytes.NewReader([]byte{2}), BigEndian, &res) | 
|  | checkResult(t, "ReadBool", BigEndian, err, res, true) | 
|  | } | 
|  |  | 
|  | func TestReadBoolSlice(t *testing.T) { | 
|  | slice := make([]bool, 4) | 
|  | err := Read(bytes.NewReader([]byte{0, 1, 2, 255}), BigEndian, slice) | 
|  | checkResult(t, "ReadBoolSlice", BigEndian, err, slice, []bool{false, true, true, true}) | 
|  | } | 
|  |  | 
|  | // Addresses of arrays are easier to manipulate with reflection than are slices. | 
|  | var intArrays = []interface{}{ | 
|  | &[100]int8{}, | 
|  | &[100]int16{}, | 
|  | &[100]int32{}, | 
|  | &[100]int64{}, | 
|  | &[100]uint8{}, | 
|  | &[100]uint16{}, | 
|  | &[100]uint32{}, | 
|  | &[100]uint64{}, | 
|  | } | 
|  |  | 
|  | func TestSliceRoundTrip(t *testing.T) { | 
|  | buf := new(bytes.Buffer) | 
|  | for _, array := range intArrays { | 
|  | src := reflect.ValueOf(array).Elem() | 
|  | unsigned := false | 
|  | switch src.Index(0).Kind() { | 
|  | case reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: | 
|  | unsigned = true | 
|  | } | 
|  | for i := 0; i < src.Len(); i++ { | 
|  | if unsigned { | 
|  | src.Index(i).SetUint(uint64(i * 0x07654321)) | 
|  | } else { | 
|  | src.Index(i).SetInt(int64(i * 0x07654321)) | 
|  | } | 
|  | } | 
|  | buf.Reset() | 
|  | srcSlice := src.Slice(0, src.Len()) | 
|  | err := Write(buf, BigEndian, srcSlice.Interface()) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | dst := reflect.New(src.Type()).Elem() | 
|  | dstSlice := dst.Slice(0, dst.Len()) | 
|  | err = Read(buf, BigEndian, dstSlice.Interface()) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | if !reflect.DeepEqual(src.Interface(), dst.Interface()) { | 
|  | t.Fatal(src) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestWriteT(t *testing.T) { | 
|  | buf := new(bytes.Buffer) | 
|  | ts := T{} | 
|  | if err := Write(buf, BigEndian, ts); err == nil { | 
|  | t.Errorf("WriteT: have err == nil, want non-nil") | 
|  | } | 
|  |  | 
|  | tv := reflect.Indirect(reflect.ValueOf(ts)) | 
|  | for i, n := 0, tv.NumField(); i < n; i++ { | 
|  | typ := tv.Field(i).Type().String() | 
|  | if typ == "[4]int" { | 
|  | typ = "int" // the problem is int, not the [4] | 
|  | } | 
|  | if err := Write(buf, BigEndian, tv.Field(i).Interface()); err == nil { | 
|  | t.Errorf("WriteT.%v: have err == nil, want non-nil", tv.Field(i).Type()) | 
|  | } else if !strings.Contains(err.Error(), typ) { | 
|  | t.Errorf("WriteT: have err == %q, want it to mention %s", err, typ) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | type BlankFields struct { | 
|  | A uint32 | 
|  | _ int32 | 
|  | B float64 | 
|  | _ [4]int16 | 
|  | C byte | 
|  | _ [7]byte | 
|  | _ struct { | 
|  | f [8]float32 | 
|  | } | 
|  | } | 
|  |  | 
|  | type BlankFieldsProbe struct { | 
|  | A  uint32 | 
|  | P0 int32 | 
|  | B  float64 | 
|  | P1 [4]int16 | 
|  | C  byte | 
|  | P2 [7]byte | 
|  | P3 struct { | 
|  | F [8]float32 | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestBlankFields(t *testing.T) { | 
|  | buf := new(bytes.Buffer) | 
|  | b1 := BlankFields{A: 1234567890, B: 2.718281828, C: 42} | 
|  | if err := Write(buf, LittleEndian, &b1); err != nil { | 
|  | t.Error(err) | 
|  | } | 
|  |  | 
|  | // zero values must have been written for blank fields | 
|  | var p BlankFieldsProbe | 
|  | if err := Read(buf, LittleEndian, &p); err != nil { | 
|  | t.Error(err) | 
|  | } | 
|  |  | 
|  | // quick test: only check first value of slices | 
|  | if p.P0 != 0 || p.P1[0] != 0 || p.P2[0] != 0 || p.P3.F[0] != 0 { | 
|  | t.Errorf("non-zero values for originally blank fields: %#v", p) | 
|  | } | 
|  |  | 
|  | // write p and see if we can probe only some fields | 
|  | if err := Write(buf, LittleEndian, &p); err != nil { | 
|  | t.Error(err) | 
|  | } | 
|  |  | 
|  | // read should ignore blank fields in b2 | 
|  | var b2 BlankFields | 
|  | if err := Read(buf, LittleEndian, &b2); err != nil { | 
|  | t.Error(err) | 
|  | } | 
|  | if b1.A != b2.A || b1.B != b2.B || b1.C != b2.C { | 
|  | t.Errorf("%#v != %#v", b1, b2) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestSizeStructCache(t *testing.T) { | 
|  | // Reset the cache, otherwise multiple test runs fail. | 
|  | structSize = sync.Map{} | 
|  |  | 
|  | count := func() int { | 
|  | var i int | 
|  | structSize.Range(func(_, _ interface{}) bool { | 
|  | i++ | 
|  | return true | 
|  | }) | 
|  | return i | 
|  | } | 
|  |  | 
|  | var total int | 
|  | added := func() int { | 
|  | delta := count() - total | 
|  | total += delta | 
|  | return delta | 
|  | } | 
|  |  | 
|  | type foo struct { | 
|  | A uint32 | 
|  | } | 
|  |  | 
|  | type bar struct { | 
|  | A Struct | 
|  | B foo | 
|  | C Struct | 
|  | } | 
|  |  | 
|  | testcases := []struct { | 
|  | val  interface{} | 
|  | want int | 
|  | }{ | 
|  | {new(foo), 1}, | 
|  | {new(bar), 1}, | 
|  | {new(bar), 0}, | 
|  | {new(struct{ A Struct }), 1}, | 
|  | {new(struct{ A Struct }), 0}, | 
|  | } | 
|  |  | 
|  | for _, tc := range testcases { | 
|  | if Size(tc.val) == -1 { | 
|  | t.Fatalf("Can't get the size of %T", tc.val) | 
|  | } | 
|  |  | 
|  | if n := added(); n != tc.want { | 
|  | t.Errorf("Sizing %T added %d entries to the cache, want %d", tc.val, n, tc.want) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // An attempt to read into a struct with an unexported field will | 
|  | // panic. This is probably not the best choice, but at this point | 
|  | // anything else would be an API change. | 
|  |  | 
|  | type Unexported struct { | 
|  | a int32 | 
|  | } | 
|  |  | 
|  | func TestUnexportedRead(t *testing.T) { | 
|  | var buf bytes.Buffer | 
|  | u1 := Unexported{a: 1} | 
|  | if err := Write(&buf, LittleEndian, &u1); err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | defer func() { | 
|  | if recover() == nil { | 
|  | t.Fatal("did not panic") | 
|  | } | 
|  | }() | 
|  | var u2 Unexported | 
|  | Read(&buf, LittleEndian, &u2) | 
|  | } | 
|  |  | 
|  | func TestReadErrorMsg(t *testing.T) { | 
|  | var buf bytes.Buffer | 
|  | read := func(data interface{}) { | 
|  | err := Read(&buf, LittleEndian, data) | 
|  | want := "binary.Read: invalid type " + reflect.TypeOf(data).String() | 
|  | if err == nil { | 
|  | t.Errorf("%T: got no error; want %q", data, want) | 
|  | return | 
|  | } | 
|  | if got := err.Error(); got != want { | 
|  | t.Errorf("%T: got %q; want %q", data, got, want) | 
|  | } | 
|  | } | 
|  | read(0) | 
|  | s := new(struct{}) | 
|  | read(&s) | 
|  | p := &s | 
|  | read(&p) | 
|  | } | 
|  |  | 
|  | func TestReadTruncated(t *testing.T) { | 
|  | const data = "0123456789abcdef" | 
|  |  | 
|  | var b1 = make([]int32, 4) | 
|  | var b2 struct { | 
|  | A, B, C, D byte | 
|  | E          int32 | 
|  | F          float64 | 
|  | } | 
|  |  | 
|  | for i := 0; i <= len(data); i++ { | 
|  | var errWant error | 
|  | switch i { | 
|  | case 0: | 
|  | errWant = io.EOF | 
|  | case len(data): | 
|  | errWant = nil | 
|  | default: | 
|  | errWant = io.ErrUnexpectedEOF | 
|  | } | 
|  |  | 
|  | if err := Read(strings.NewReader(data[:i]), LittleEndian, &b1); err != errWant { | 
|  | t.Errorf("Read(%d) with slice: got %v, want %v", i, err, errWant) | 
|  | } | 
|  | if err := Read(strings.NewReader(data[:i]), LittleEndian, &b2); err != errWant { | 
|  | t.Errorf("Read(%d) with struct: got %v, want %v", i, err, errWant) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func testUint64SmallSliceLengthPanics() (panicked bool) { | 
|  | defer func() { | 
|  | panicked = recover() != nil | 
|  | }() | 
|  | b := [8]byte{1, 2, 3, 4, 5, 6, 7, 8} | 
|  | LittleEndian.Uint64(b[:4]) | 
|  | return false | 
|  | } | 
|  |  | 
|  | func testPutUint64SmallSliceLengthPanics() (panicked bool) { | 
|  | defer func() { | 
|  | panicked = recover() != nil | 
|  | }() | 
|  | b := [8]byte{} | 
|  | LittleEndian.PutUint64(b[:4], 0x0102030405060708) | 
|  | return false | 
|  | } | 
|  |  | 
|  | func TestEarlyBoundsChecks(t *testing.T) { | 
|  | if testUint64SmallSliceLengthPanics() != true { | 
|  | t.Errorf("binary.LittleEndian.Uint64 expected to panic for small slices, but didn't") | 
|  | } | 
|  | if testPutUint64SmallSliceLengthPanics() != true { | 
|  | t.Errorf("binary.LittleEndian.PutUint64 expected to panic for small slices, but didn't") | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestReadInvalidDestination(t *testing.T) { | 
|  | testReadInvalidDestination(t, BigEndian) | 
|  | testReadInvalidDestination(t, LittleEndian) | 
|  | } | 
|  |  | 
|  | func testReadInvalidDestination(t *testing.T, order ByteOrder) { | 
|  | destinations := []interface{}{ | 
|  | int8(0), | 
|  | int16(0), | 
|  | int32(0), | 
|  | int64(0), | 
|  |  | 
|  | uint8(0), | 
|  | uint16(0), | 
|  | uint32(0), | 
|  | uint64(0), | 
|  |  | 
|  | bool(false), | 
|  | } | 
|  |  | 
|  | for _, dst := range destinations { | 
|  | err := Read(bytes.NewReader([]byte{1, 2, 3, 4, 5, 6, 7, 8}), order, dst) | 
|  | want := fmt.Sprintf("binary.Read: invalid type %T", dst) | 
|  | if err == nil || err.Error() != want { | 
|  | t.Fatalf("for type %T: got %q; want %q", dst, err, want) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | type byteSliceReader struct { | 
|  | remain []byte | 
|  | } | 
|  |  | 
|  | func (br *byteSliceReader) Read(p []byte) (int, error) { | 
|  | n := copy(p, br.remain) | 
|  | br.remain = br.remain[n:] | 
|  | return n, nil | 
|  | } | 
|  |  | 
|  | func BenchmarkReadSlice1000Int32s(b *testing.B) { | 
|  | bsr := &byteSliceReader{} | 
|  | slice := make([]int32, 1000) | 
|  | buf := make([]byte, len(slice)*4) | 
|  | b.SetBytes(int64(len(buf))) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | bsr.remain = buf | 
|  | Read(bsr, BigEndian, slice) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkReadStruct(b *testing.B) { | 
|  | bsr := &byteSliceReader{} | 
|  | var buf bytes.Buffer | 
|  | Write(&buf, BigEndian, &s) | 
|  | b.SetBytes(int64(dataSize(reflect.ValueOf(s)))) | 
|  | t := s | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | bsr.remain = buf.Bytes() | 
|  | Read(bsr, BigEndian, &t) | 
|  | } | 
|  | b.StopTimer() | 
|  | if b.N > 0 && !reflect.DeepEqual(s, t) { | 
|  | b.Fatalf("struct doesn't match:\ngot  %v;\nwant %v", t, s) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkWriteStruct(b *testing.B) { | 
|  | b.SetBytes(int64(Size(&s))) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | Write(ioutil.Discard, BigEndian, &s) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkReadInts(b *testing.B) { | 
|  | var ls Struct | 
|  | bsr := &byteSliceReader{} | 
|  | var r io.Reader = bsr | 
|  | b.SetBytes(2 * (1 + 2 + 4 + 8)) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | bsr.remain = big | 
|  | Read(r, BigEndian, &ls.Int8) | 
|  | Read(r, BigEndian, &ls.Int16) | 
|  | Read(r, BigEndian, &ls.Int32) | 
|  | Read(r, BigEndian, &ls.Int64) | 
|  | Read(r, BigEndian, &ls.Uint8) | 
|  | Read(r, BigEndian, &ls.Uint16) | 
|  | Read(r, BigEndian, &ls.Uint32) | 
|  | Read(r, BigEndian, &ls.Uint64) | 
|  | } | 
|  | b.StopTimer() | 
|  | want := s | 
|  | want.Float32 = 0 | 
|  | want.Float64 = 0 | 
|  | want.Complex64 = 0 | 
|  | want.Complex128 = 0 | 
|  | want.Array = [4]uint8{0, 0, 0, 0} | 
|  | want.Bool = false | 
|  | want.BoolArray = [4]bool{false, false, false, false} | 
|  | if b.N > 0 && !reflect.DeepEqual(ls, want) { | 
|  | b.Fatalf("struct doesn't match:\ngot  %v;\nwant %v", ls, want) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkWriteInts(b *testing.B) { | 
|  | buf := new(bytes.Buffer) | 
|  | var w io.Writer = buf | 
|  | b.SetBytes(2 * (1 + 2 + 4 + 8)) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | buf.Reset() | 
|  | Write(w, BigEndian, s.Int8) | 
|  | Write(w, BigEndian, s.Int16) | 
|  | Write(w, BigEndian, s.Int32) | 
|  | Write(w, BigEndian, s.Int64) | 
|  | Write(w, BigEndian, s.Uint8) | 
|  | Write(w, BigEndian, s.Uint16) | 
|  | Write(w, BigEndian, s.Uint32) | 
|  | Write(w, BigEndian, s.Uint64) | 
|  | } | 
|  | b.StopTimer() | 
|  | if b.N > 0 && !bytes.Equal(buf.Bytes(), big[:30]) { | 
|  | b.Fatalf("first half doesn't match: %x %x", buf.Bytes(), big[:30]) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkWriteSlice1000Int32s(b *testing.B) { | 
|  | slice := make([]int32, 1000) | 
|  | buf := new(bytes.Buffer) | 
|  | var w io.Writer = buf | 
|  | b.SetBytes(4 * 1000) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | buf.Reset() | 
|  | Write(w, BigEndian, slice) | 
|  | } | 
|  | b.StopTimer() | 
|  | } | 
|  |  | 
|  | func BenchmarkPutUint16(b *testing.B) { | 
|  | b.SetBytes(2) | 
|  | for i := 0; i < b.N; i++ { | 
|  | BigEndian.PutUint16(putbuf[:], uint16(i)) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkPutUint32(b *testing.B) { | 
|  | b.SetBytes(4) | 
|  | for i := 0; i < b.N; i++ { | 
|  | BigEndian.PutUint32(putbuf[:], uint32(i)) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkPutUint64(b *testing.B) { | 
|  | b.SetBytes(8) | 
|  | for i := 0; i < b.N; i++ { | 
|  | BigEndian.PutUint64(putbuf[:], uint64(i)) | 
|  | } | 
|  | } | 
|  | func BenchmarkLittleEndianPutUint16(b *testing.B) { | 
|  | b.SetBytes(2) | 
|  | for i := 0; i < b.N; i++ { | 
|  | LittleEndian.PutUint16(putbuf[:], uint16(i)) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkLittleEndianPutUint32(b *testing.B) { | 
|  | b.SetBytes(4) | 
|  | for i := 0; i < b.N; i++ { | 
|  | LittleEndian.PutUint32(putbuf[:], uint32(i)) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkLittleEndianPutUint64(b *testing.B) { | 
|  | b.SetBytes(8) | 
|  | for i := 0; i < b.N; i++ { | 
|  | LittleEndian.PutUint64(putbuf[:], uint64(i)) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkReadFloats(b *testing.B) { | 
|  | var ls Struct | 
|  | bsr := &byteSliceReader{} | 
|  | var r io.Reader = bsr | 
|  | b.SetBytes(4 + 8) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | bsr.remain = big[30:] | 
|  | Read(r, BigEndian, &ls.Float32) | 
|  | Read(r, BigEndian, &ls.Float64) | 
|  | } | 
|  | b.StopTimer() | 
|  | want := s | 
|  | want.Int8 = 0 | 
|  | want.Int16 = 0 | 
|  | want.Int32 = 0 | 
|  | want.Int64 = 0 | 
|  | want.Uint8 = 0 | 
|  | want.Uint16 = 0 | 
|  | want.Uint32 = 0 | 
|  | want.Uint64 = 0 | 
|  | want.Complex64 = 0 | 
|  | want.Complex128 = 0 | 
|  | want.Array = [4]uint8{0, 0, 0, 0} | 
|  | want.Bool = false | 
|  | want.BoolArray = [4]bool{false, false, false, false} | 
|  | if b.N > 0 && !reflect.DeepEqual(ls, want) { | 
|  | b.Fatalf("struct doesn't match:\ngot  %v;\nwant %v", ls, want) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkWriteFloats(b *testing.B) { | 
|  | buf := new(bytes.Buffer) | 
|  | var w io.Writer = buf | 
|  | b.SetBytes(4 + 8) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | buf.Reset() | 
|  | Write(w, BigEndian, s.Float32) | 
|  | Write(w, BigEndian, s.Float64) | 
|  | } | 
|  | b.StopTimer() | 
|  | if b.N > 0 && !bytes.Equal(buf.Bytes(), big[30:30+4+8]) { | 
|  | b.Fatalf("first half doesn't match: %x %x", buf.Bytes(), big[30:30+4+8]) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkReadSlice1000Float32s(b *testing.B) { | 
|  | bsr := &byteSliceReader{} | 
|  | slice := make([]float32, 1000) | 
|  | buf := make([]byte, len(slice)*4) | 
|  | b.SetBytes(int64(len(buf))) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | bsr.remain = buf | 
|  | Read(bsr, BigEndian, slice) | 
|  | } | 
|  | } | 
|  |  | 
|  | func BenchmarkWriteSlice1000Float32s(b *testing.B) { | 
|  | slice := make([]float32, 1000) | 
|  | buf := new(bytes.Buffer) | 
|  | var w io.Writer = buf | 
|  | b.SetBytes(4 * 1000) | 
|  | b.ResetTimer() | 
|  | for i := 0; i < b.N; i++ { | 
|  | buf.Reset() | 
|  | Write(w, BigEndian, slice) | 
|  | } | 
|  | b.StopTimer() | 
|  | } |