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
|
// Copyright (C) 2022 Luke Shumaker <lukeshu@lukeshu.com>
//
// SPDX-License-Identifier: GPL-2.0-or-later
package binstruct
import (
"errors"
"fmt"
"reflect"
)
type StaticSizer interface {
BinaryStaticSize() int
}
func StaticSize(obj any) int {
sz, err := staticSize(reflect.TypeOf(obj))
if err != nil {
panic(err)
}
return sz
}
var (
staticSizerType = reflect.TypeOf((*StaticSizer)(nil)).Elem()
marshalerType = reflect.TypeOf((*Marshaler)(nil)).Elem()
unmarshalerType = reflect.TypeOf((*Unmarshaler)(nil)).Elem()
)
func staticSize(typ reflect.Type) (int, error) {
if typ.Implements(staticSizerType) {
return reflect.New(typ).Elem().Interface().(StaticSizer).BinaryStaticSize(), nil
}
if typ.Implements(marshalerType) || typ.Implements(unmarshalerType) {
// If you implement binstruct.Marshaler or binstruct.Unmarshaler,
// then you must implement if you wish to be statically sized.
return 0, &InvalidTypeError{
Type: typ,
Err: errors.New("does not implement binstruct.StaticSizer but does implement binstruct.Marshaler or binstruct.Unmarshaler"),
}
}
switch typ.Kind() {
case reflect.Uint8, reflect.Int8:
return 1, nil
case reflect.Uint16, reflect.Int16:
return 2, nil
case reflect.Uint32, reflect.Int32:
return 4, nil
case reflect.Uint64, reflect.Int64:
return 8, nil
case reflect.Ptr:
return staticSize(typ.Elem())
case reflect.Array:
elemSize, err := staticSize(typ.Elem())
if err != nil {
return 0, err
}
return elemSize * typ.Len(), nil
case reflect.Struct:
return getStructHandler(typ).Size, nil
default:
return 0, &InvalidTypeError{
Type: typ,
Err: fmt.Errorf("does not implement binfmt.StaticSizer and kind=%v is not a supported statically-sized kind",
typ.Kind()),
}
}
}
|