summaryrefslogtreecommitdiffhomepage
path: root/pkg/binary
diff options
context:
space:
mode:
Diffstat (limited to 'pkg/binary')
-rw-r--r--pkg/binary/BUILD16
-rw-r--r--pkg/binary/binary.go266
-rw-r--r--pkg/binary/binary_test.go266
3 files changed, 548 insertions, 0 deletions
diff --git a/pkg/binary/BUILD b/pkg/binary/BUILD
new file mode 100644
index 000000000..7ca2fda90
--- /dev/null
+++ b/pkg/binary/BUILD
@@ -0,0 +1,16 @@
+load("//tools:defs.bzl", "go_library", "go_test")
+
+package(licenses = ["notice"])
+
+go_library(
+ name = "binary",
+ srcs = ["binary.go"],
+ visibility = ["//:sandbox"],
+)
+
+go_test(
+ name = "binary_test",
+ size = "small",
+ srcs = ["binary_test.go"],
+ library = ":binary",
+)
diff --git a/pkg/binary/binary.go b/pkg/binary/binary.go
new file mode 100644
index 000000000..25065aef9
--- /dev/null
+++ b/pkg/binary/binary.go
@@ -0,0 +1,266 @@
+// Copyright 2018 The gVisor 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
+//
+// http://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.
+
+// Package binary translates between select fixed-sized types and a binary
+// representation.
+package binary
+
+import (
+ "encoding/binary"
+ "fmt"
+ "io"
+ "reflect"
+)
+
+// LittleEndian is the same as encoding/binary.LittleEndian.
+//
+// It is included here as a convenience.
+var LittleEndian = binary.LittleEndian
+
+// BigEndian is the same as encoding/binary.BigEndian.
+//
+// It is included here as a convenience.
+var BigEndian = binary.BigEndian
+
+// AppendUint16 appends the binary representation of a uint16 to buf.
+func AppendUint16(buf []byte, order binary.ByteOrder, num uint16) []byte {
+ buf = append(buf, make([]byte, 2)...)
+ order.PutUint16(buf[len(buf)-2:], num)
+ return buf
+}
+
+// AppendUint32 appends the binary representation of a uint32 to buf.
+func AppendUint32(buf []byte, order binary.ByteOrder, num uint32) []byte {
+ buf = append(buf, make([]byte, 4)...)
+ order.PutUint32(buf[len(buf)-4:], num)
+ return buf
+}
+
+// AppendUint64 appends the binary representation of a uint64 to buf.
+func AppendUint64(buf []byte, order binary.ByteOrder, num uint64) []byte {
+ buf = append(buf, make([]byte, 8)...)
+ order.PutUint64(buf[len(buf)-8:], num)
+ return buf
+}
+
+// Marshal appends a binary representation of data to buf.
+//
+// data must only contain fixed-length signed and unsigned ints, arrays,
+// slices, structs and compositions of said types. data may be a pointer,
+// but cannot contain pointers.
+func Marshal(buf []byte, order binary.ByteOrder, data interface{}) []byte {
+ return marshal(buf, order, reflect.Indirect(reflect.ValueOf(data)))
+}
+
+func marshal(buf []byte, order binary.ByteOrder, data reflect.Value) []byte {
+ switch data.Kind() {
+ case reflect.Int8:
+ buf = append(buf, byte(int8(data.Int())))
+ case reflect.Int16:
+ buf = AppendUint16(buf, order, uint16(int16(data.Int())))
+ case reflect.Int32:
+ buf = AppendUint32(buf, order, uint32(int32(data.Int())))
+ case reflect.Int64:
+ buf = AppendUint64(buf, order, uint64(data.Int()))
+
+ case reflect.Uint8:
+ buf = append(buf, byte(data.Uint()))
+ case reflect.Uint16:
+ buf = AppendUint16(buf, order, uint16(data.Uint()))
+ case reflect.Uint32:
+ buf = AppendUint32(buf, order, uint32(data.Uint()))
+ case reflect.Uint64:
+ buf = AppendUint64(buf, order, data.Uint())
+
+ case reflect.Array, reflect.Slice:
+ for i, l := 0, data.Len(); i < l; i++ {
+ buf = marshal(buf, order, data.Index(i))
+ }
+
+ case reflect.Struct:
+ for i, l := 0, data.NumField(); i < l; i++ {
+ buf = marshal(buf, order, data.Field(i))
+ }
+
+ default:
+ panic("invalid type: " + data.Type().String())
+ }
+ return buf
+}
+
+// Unmarshal unpacks buf into data.
+//
+// data must be a slice or a pointer and buf must have a length of exactly
+// Size(data). data must only contain fixed-length signed and unsigned ints,
+// arrays, slices, structs and compositions of said types.
+func Unmarshal(buf []byte, order binary.ByteOrder, data interface{}) {
+ value := reflect.ValueOf(data)
+ switch value.Kind() {
+ case reflect.Ptr:
+ value = value.Elem()
+ case reflect.Slice:
+ default:
+ panic("invalid type: " + value.Type().String())
+ }
+ buf = unmarshal(buf, order, value)
+ if len(buf) != 0 {
+ panic(fmt.Sprintf("buffer too long by %d bytes", len(buf)))
+ }
+}
+
+func unmarshal(buf []byte, order binary.ByteOrder, data reflect.Value) []byte {
+ switch data.Kind() {
+ case reflect.Int8:
+ data.SetInt(int64(int8(buf[0])))
+ buf = buf[1:]
+ case reflect.Int16:
+ data.SetInt(int64(int16(order.Uint16(buf))))
+ buf = buf[2:]
+ case reflect.Int32:
+ data.SetInt(int64(int32(order.Uint32(buf))))
+ buf = buf[4:]
+ case reflect.Int64:
+ data.SetInt(int64(order.Uint64(buf)))
+ buf = buf[8:]
+
+ case reflect.Uint8:
+ data.SetUint(uint64(buf[0]))
+ buf = buf[1:]
+ case reflect.Uint16:
+ data.SetUint(uint64(order.Uint16(buf)))
+ buf = buf[2:]
+ case reflect.Uint32:
+ data.SetUint(uint64(order.Uint32(buf)))
+ buf = buf[4:]
+ case reflect.Uint64:
+ data.SetUint(order.Uint64(buf))
+ buf = buf[8:]
+
+ case reflect.Array, reflect.Slice:
+ for i, l := 0, data.Len(); i < l; i++ {
+ buf = unmarshal(buf, order, data.Index(i))
+ }
+
+ case reflect.Struct:
+ for i, l := 0, data.NumField(); i < l; i++ {
+ if field := data.Field(i); field.CanSet() {
+ buf = unmarshal(buf, order, field)
+ } else {
+ buf = buf[sizeof(field):]
+ }
+ }
+
+ default:
+ panic("invalid type: " + data.Type().String())
+ }
+ return buf
+}
+
+// Size calculates the buffer sized needed by Marshal or Unmarshal.
+//
+// Size only support the types supported by Marshal.
+func Size(v interface{}) uintptr {
+ return sizeof(reflect.Indirect(reflect.ValueOf(v)))
+}
+
+func sizeof(data reflect.Value) uintptr {
+ switch data.Kind() {
+ case reflect.Int8, reflect.Uint8:
+ return 1
+ case reflect.Int16, reflect.Uint16:
+ return 2
+ case reflect.Int32, reflect.Uint32:
+ return 4
+ case reflect.Int64, reflect.Uint64:
+ return 8
+
+ case reflect.Array, reflect.Slice:
+ var size uintptr
+ for i, l := 0, data.Len(); i < l; i++ {
+ size += sizeof(data.Index(i))
+ }
+ return size
+
+ case reflect.Struct:
+ var size uintptr
+ for i, l := 0, data.NumField(); i < l; i++ {
+ size += sizeof(data.Field(i))
+ }
+ return size
+
+ default:
+ panic("invalid type: " + data.Type().String())
+ }
+}
+
+// ReadUint16 reads a uint16 from r.
+func ReadUint16(r io.Reader, order binary.ByteOrder) (uint16, error) {
+ buf := make([]byte, 2)
+ if _, err := io.ReadFull(r, buf); err != nil {
+ return 0, err
+ }
+ return order.Uint16(buf), nil
+}
+
+// ReadUint32 reads a uint32 from r.
+func ReadUint32(r io.Reader, order binary.ByteOrder) (uint32, error) {
+ buf := make([]byte, 4)
+ if _, err := io.ReadFull(r, buf); err != nil {
+ return 0, err
+ }
+ return order.Uint32(buf), nil
+}
+
+// ReadUint64 reads a uint64 from r.
+func ReadUint64(r io.Reader, order binary.ByteOrder) (uint64, error) {
+ buf := make([]byte, 8)
+ if _, err := io.ReadFull(r, buf); err != nil {
+ return 0, err
+ }
+ return order.Uint64(buf), nil
+}
+
+// WriteUint16 writes a uint16 to w.
+func WriteUint16(w io.Writer, order binary.ByteOrder, num uint16) error {
+ buf := make([]byte, 2)
+ order.PutUint16(buf, num)
+ _, err := w.Write(buf)
+ return err
+}
+
+// WriteUint32 writes a uint32 to w.
+func WriteUint32(w io.Writer, order binary.ByteOrder, num uint32) error {
+ buf := make([]byte, 4)
+ order.PutUint32(buf, num)
+ _, err := w.Write(buf)
+ return err
+}
+
+// WriteUint64 writes a uint64 to w.
+func WriteUint64(w io.Writer, order binary.ByteOrder, num uint64) error {
+ buf := make([]byte, 8)
+ order.PutUint64(buf, num)
+ _, err := w.Write(buf)
+ return err
+}
+
+// AlignUp rounds a length up to an alignment. align must be a power of 2.
+func AlignUp(length int, align uint) int {
+ return (length + int(align) - 1) & ^(int(align) - 1)
+}
+
+// AlignDown rounds a length down to an alignment. align must be a power of 2.
+func AlignDown(length int, align uint) int {
+ return length & ^(int(align) - 1)
+}
diff --git a/pkg/binary/binary_test.go b/pkg/binary/binary_test.go
new file mode 100644
index 000000000..4d609a438
--- /dev/null
+++ b/pkg/binary/binary_test.go
@@ -0,0 +1,266 @@
+// Copyright 2018 The gVisor 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
+//
+// http://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.
+
+package binary
+
+import (
+ "bytes"
+ "encoding/binary"
+ "errors"
+ "fmt"
+ "io"
+ "reflect"
+ "strings"
+ "testing"
+)
+
+func newInt32(i int32) *int32 {
+ return &i
+}
+
+func TestSize(t *testing.T) {
+ if got, want := Size(uint32(10)), uintptr(4); got != want {
+ t.Errorf("Got = %d, want = %d", got, want)
+ }
+}
+
+func TestPanic(t *testing.T) {
+ tests := []struct {
+ name string
+ f func([]byte, binary.ByteOrder, interface{})
+ data interface{}
+ want string
+ }{
+ {"Unmarshal int", Unmarshal, 5, "invalid type: int"},
+ {"Unmarshal []int", Unmarshal, []int{5}, "invalid type: int"},
+ {"Marshal int", func(_ []byte, bo binary.ByteOrder, d interface{}) { Marshal(nil, bo, d) }, 5, "invalid type: int"},
+ {"Marshal int[]", func(_ []byte, bo binary.ByteOrder, d interface{}) { Marshal(nil, bo, d) }, []int{5}, "invalid type: int"},
+ {"Unmarshal short buffer", Unmarshal, newInt32(5), "runtime error: index out of range"},
+ {"Unmarshal long buffer", func(_ []byte, bo binary.ByteOrder, d interface{}) { Unmarshal(make([]byte, 50), bo, d) }, newInt32(5), "buffer too long by 46 bytes"},
+ {"marshal int", func(_ []byte, bo binary.ByteOrder, d interface{}) { marshal(nil, bo, reflect.ValueOf(d)) }, 5, "invalid type: int"},
+ {"Size int", func(_ []byte, _ binary.ByteOrder, d interface{}) { Size(d) }, 5, "invalid type: int"},
+ }
+
+ for _, test := range tests {
+ t.Run(test.name, func(t *testing.T) {
+ defer func() {
+ r := recover()
+ if got := fmt.Sprint(r); !strings.HasPrefix(got, test.want) {
+ t.Errorf("Got recover() = %q, want prefix = %q", got, test.want)
+ }
+ }()
+
+ test.f(nil, LittleEndian, test.data)
+ })
+ }
+}
+
+type inner struct {
+ Field int32
+}
+
+type outer struct {
+ Int8 int8
+ Int16 int16
+ Int32 int32
+ Int64 int64
+ Uint8 uint8
+ Uint16 uint16
+ Uint32 uint32
+ Uint64 uint64
+
+ Slice []int32
+ Array [5]int32
+ Struct inner
+}
+
+func TestMarshalUnmarshal(t *testing.T) {
+ want := outer{
+ 1, 2, 3, 4, 5, 6, 7, 8,
+ []int32{9, 10, 11},
+ [5]int32{12, 13, 14, 15, 16},
+ inner{17},
+ }
+ buf := Marshal(nil, LittleEndian, want)
+ got := outer{Slice: []int32{0, 0, 0}}
+ Unmarshal(buf, LittleEndian, &got)
+ if !reflect.DeepEqual(&got, &want) {
+ t.Errorf("Got = %#v, want = %#v", got, want)
+ }
+}
+
+type outerBenchmark struct {
+ Int8 int8
+ Int16 int16
+ Int32 int32
+ Int64 int64
+ Uint8 uint8
+ Uint16 uint16
+ Uint32 uint32
+ Uint64 uint64
+
+ Array [5]int32
+ Struct inner
+}
+
+func BenchmarkMarshalUnmarshal(b *testing.B) {
+ b.ReportAllocs()
+
+ in := outerBenchmark{
+ 1, 2, 3, 4, 5, 6, 7, 8,
+ [5]int32{9, 10, 11, 12, 13},
+ inner{14},
+ }
+ buf := make([]byte, Size(&in))
+ out := outerBenchmark{}
+
+ for i := 0; i < b.N; i++ {
+ buf := Marshal(buf[:0], LittleEndian, &in)
+ Unmarshal(buf, LittleEndian, &out)
+ }
+}
+
+func BenchmarkReadWrite(b *testing.B) {
+ b.ReportAllocs()
+
+ in := outerBenchmark{
+ 1, 2, 3, 4, 5, 6, 7, 8,
+ [5]int32{9, 10, 11, 12, 13},
+ inner{14},
+ }
+ buf := bytes.NewBuffer(make([]byte, binary.Size(&in)))
+ out := outerBenchmark{}
+
+ for i := 0; i < b.N; i++ {
+ buf.Reset()
+ if err := binary.Write(buf, LittleEndian, &in); err != nil {
+ b.Error("Write:", err)
+ }
+ if err := binary.Read(buf, LittleEndian, &out); err != nil {
+ b.Error("Read:", err)
+ }
+ }
+}
+
+type outerPadding struct {
+ _ int8
+ _ int16
+ _ int32
+ _ int64
+ _ uint8
+ _ uint16
+ _ uint32
+ _ uint64
+
+ _ []int32
+ _ [5]int32
+ _ inner
+}
+
+func TestMarshalUnmarshalPadding(t *testing.T) {
+ var want outerPadding
+ buf := Marshal(nil, LittleEndian, want)
+ var got outerPadding
+ Unmarshal(buf, LittleEndian, &got)
+ if !reflect.DeepEqual(&got, &want) {
+ t.Errorf("Got = %#v, want = %#v", got, want)
+ }
+}
+
+// Numbers with bits in every byte that distinguishable in big and little endian.
+const (
+ want16 = 64<<8 | 128
+ want32 = 16<<24 | 32<<16 | want16
+ want64 = 1<<56 | 2<<48 | 4<<40 | 8<<32 | want32
+)
+
+func TestReadWriteUint16(t *testing.T) {
+ const want = uint16(want16)
+ var buf bytes.Buffer
+ if err := WriteUint16(&buf, LittleEndian, want); err != nil {
+ t.Error("WriteUint16:", err)
+ }
+ got, err := ReadUint16(&buf, LittleEndian)
+ if err != nil {
+ t.Error("ReadUint16:", err)
+ }
+ if got != want {
+ t.Errorf("got = %d, want = %d", got, want)
+ }
+}
+
+func TestReadWriteUint32(t *testing.T) {
+ const want = uint32(want32)
+ var buf bytes.Buffer
+ if err := WriteUint32(&buf, LittleEndian, want); err != nil {
+ t.Error("WriteUint32:", err)
+ }
+ got, err := ReadUint32(&buf, LittleEndian)
+ if err != nil {
+ t.Error("ReadUint32:", err)
+ }
+ if got != want {
+ t.Errorf("got = %d, want = %d", got, want)
+ }
+}
+
+func TestReadWriteUint64(t *testing.T) {
+ const want = uint64(want64)
+ var buf bytes.Buffer
+ if err := WriteUint64(&buf, LittleEndian, want); err != nil {
+ t.Error("WriteUint64:", err)
+ }
+ got, err := ReadUint64(&buf, LittleEndian)
+ if err != nil {
+ t.Error("ReadUint64:", err)
+ }
+ if got != want {
+ t.Errorf("got = %d, want = %d", got, want)
+ }
+}
+
+type readWriter struct {
+ err error
+}
+
+func (rw *readWriter) Write([]byte) (int, error) {
+ return 0, rw.err
+}
+
+func (rw *readWriter) Read([]byte) (int, error) {
+ return 0, rw.err
+}
+
+func TestReadWriteError(t *testing.T) {
+ tests := []struct {
+ name string
+ f func(rw io.ReadWriter) error
+ }{
+ {"WriteUint16", func(rw io.ReadWriter) error { return WriteUint16(rw, LittleEndian, 0) }},
+ {"ReadUint16", func(rw io.ReadWriter) error { _, err := ReadUint16(rw, LittleEndian); return err }},
+ {"WriteUint32", func(rw io.ReadWriter) error { return WriteUint32(rw, LittleEndian, 0) }},
+ {"ReadUint32", func(rw io.ReadWriter) error { _, err := ReadUint32(rw, LittleEndian); return err }},
+ {"WriteUint64", func(rw io.ReadWriter) error { return WriteUint64(rw, LittleEndian, 0) }},
+ {"ReadUint64", func(rw io.ReadWriter) error { _, err := ReadUint64(rw, LittleEndian); return err }},
+ }
+
+ for _, test := range tests {
+ t.Run(test.name, func(t *testing.T) {
+ want := errors.New("want")
+ if got := test.f(&readWriter{want}); got != want {
+ t.Errorf("got = %v, want = %v", got, want)
+ }
+ })
+ }
+}