-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcanoutputconfiguration.go
174 lines (146 loc) · 5.52 KB
/
canoutputconfiguration.go
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
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
package xsens
import (
"bytes"
"encoding/binary"
"fmt"
)
type CANOutputConfiguration []CANOutputConfigurationSetting
type CANIDLengthFlag bool
type IDMask = uint32
const (
CANIDLengthFlag11bits CANIDLengthFlag = false
CANIDLengthFlag29bits CANIDLengthFlag = true
)
const (
canOutputCfgSettingSize = 8
// CANDataIdentifier.
canOutputCfgOffsetBlock1 = 0
canOutputCfgLengthBlock1 = 1
canOutputCfgCANDataIDMask = ^byte(1 << 7) // Masks reserved bit in canOutputCfg Block 1 MSB
// CANIDLengthFlag.
canOutputCfgOffsetBlock2 = canOutputCfgOffsetBlock1 + canOutputCfgLengthBlock1
canOutputCfgLengthBlock2 = 1
canOutputCfgIDLengthFlagMask = byte(0b1) // Masks reserved bit in canOutputCfg Block 2 MSB
// IDMask.
canOutputCfgOffsetBlock3 = canOutputCfgOffsetBlock2 + canOutputCfgLengthBlock2
canOutputCfgLengthBlock3 = 4
canOutputCfgIDMaskMask = ^byte(0b111 << 5) // Masks reserved bit in canOutputCfg Block 3 MSB
// OutputFrequency.
canOutputCfgOffsetBlock4 = canOutputCfgOffsetBlock3 + canOutputCfgLengthBlock3
canOutputCfgLengthBlock4 = 2
canOutputCfgOutputFreqMask = byte(0b111) // Masks reserved bit in canOutputCfg Block 4 MSB
)
// CANOutputConfigurationSetting is the output configuration for a single CAN measurement data type.
type CANOutputConfigurationSetting struct {
// DataIdentifier is the data identifier of the data.
CANDataIdentifier CANDataIdentifier
// CANIDLengthFlag specifies whether the CAN address is 11 (CANIDLengthFlag11bits)
// or 29 bits (CANIDLengthFlag29bits).
CANIDLengthFlag CANIDLengthFlag
// IDMask is a uint32 version of CANDataIdentifier.
//
// Only used for reading. DefaultIDMask is used for writing.
IDMask IDMask
// OutputFrequency is the output frequency of the data.
//
// Selecting an Output OutputFrequency of either 0x0000 or 0xFFFF, makes the device select the
// maximum frequency for the given data identifier. The device reports the resulting effective
// frequencies in its response message.
OutputFrequency OutputFrequency
}
// DefaultIDMask returns the default CAN ID Mask.
func (o *CANOutputConfigurationSetting) DefaultIDMask() IDMask {
return uint32(o.CANDataIdentifier)
}
// MarshalBinary returns the wire representation of the CAN output configuration.
func (o *CANOutputConfiguration) MarshalBinary() ([]byte, error) {
buf := make([]byte, len(*o)*canOutputCfgSettingSize)
var b []byte
// Push each setting to the buffer
for i, setting := range *o {
w := buf[i*canOutputCfgSettingSize : (i+1)*canOutputCfgSettingSize-1]
// push CANDataIdentifier
b = extractBytes(w, canOutputCfgOffsetBlock1, canOutputCfgLengthBlock1)
b[0] = uint8(setting.CANDataIdentifier)
b[0] &= canOutputCfgCANDataIDMask // mask MSB
// push IDMaskLen
b = extractBytes(w, canOutputCfgOffsetBlock2, canOutputCfgLengthBlock2)
if setting.CANIDLengthFlag {
b[0] = 1
}
// push IDMask
b = extractBytes(w, canOutputCfgOffsetBlock3, canOutputCfgLengthBlock3)
binary.BigEndian.PutUint32(b, setting.DefaultIDMask())
b[0] &= canOutputCfgIDMaskMask // mask last byte
// push OutputFrequency
b = extractBytes(w, canOutputCfgOffsetBlock4, canOutputCfgLengthBlock4)
binary.BigEndian.PutUint16(b, uint16(setting.OutputFrequency))
b[0] &= canOutputCfgOutputFreqMask // mask last byte
}
return buf, nil
}
// MarshalText returns a text representation of the CAN output configuration.
func (o *CANOutputConfiguration) MarshalText() ([]byte, error) {
var buf bytes.Buffer
canIDLengthMap := map[CANIDLengthFlag]int{
CANIDLengthFlag11bits: 11,
CANIDLengthFlag29bits: 29,
}
for i, setting := range *o {
if _, err := fmt.Fprintf(
&buf,
"%v:\n %v Hz,\n %v bits\n %v mask",
setting.CANDataIdentifier,
setting.OutputFrequency,
canIDLengthMap[setting.CANIDLengthFlag],
setting.IDMask,
); err != nil {
return nil, err
}
if i < len(*o)-1 {
if err := buf.WriteByte('\n'); err != nil {
return nil, err
}
}
}
return buf.Bytes(), nil
}
// UnmarshalBinary sets *o from a wire representation of the CAN output configuration.
func (o *CANOutputConfiguration) UnmarshalBinary(data []byte) error {
settingsCount := len(data) / canOutputCfgSettingSize
if cap(*o) >= settingsCount {
*o = (*o)[:settingsCount]
} else {
*o = append((*o)[:cap(*o)], make([]CANOutputConfigurationSetting, settingsCount-cap(*o))...)
}
for i := 0; i < settingsCount; i++ {
w := data[i*canOutputCfgSettingSize : (i+1)*canOutputCfgSettingSize-1]
// get CANDataIdentifier
b := copyBytes(w, canOutputCfgOffsetBlock1, canOutputCfgLengthBlock1)
d := b[0] & canOutputCfgCANDataIDMask // mask MSB (single byte)
(*o)[i].CANDataIdentifier = CANDataIdentifier(d)
// get IDLengthFlag
b = copyBytes(w, canOutputCfgOffsetBlock2, canOutputCfgLengthBlock2)
d = b[0] & canOutputCfgIDLengthFlagMask // mask MSB (single byte)
(*o)[i].CANIDLengthFlag = d != 0
// get IDMask
b = copyBytes(w, canOutputCfgOffsetBlock3, canOutputCfgLengthBlock3)
b[0] &= canOutputCfgIDMaskMask // mask MSB
(*o)[i].IDMask = binary.BigEndian.Uint32(b)
// get OutputFrequency
b = copyBytes(w, canOutputCfgOffsetBlock4, canOutputCfgLengthBlock4)
b[0] &= canOutputCfgOutputFreqMask // mask MSB
(*o)[i].OutputFrequency = OutputFrequency(binary.BigEndian.Uint16(b))
}
return nil
}
// copyBytes returns a copy of the sub-slice of the byte array.
func copyBytes(w []byte, offset, length int) []byte {
b := make([]byte, length)
copy(b, w[offset:offset+length])
return b
}
// extractBytes returns a sub-slice of the byte array.
func extractBytes(w []byte, offset, length int) []byte {
return w[offset : offset+length]
}