-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathuint32buffer.go
304 lines (245 loc) · 10.4 KB
/
uint32buffer.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
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
/*
* Copyright (c) 2013 Zhen, LLC. http://zhen.io. All rights reserved.
* Use of this source code is governed by the Apache 2.0 license.
*
*/
// Uint32Buffer is a Go implementation of the Java IntBuffer
// http://docs.oracle.com/javase/7/docs/api/java/nio/IntBuffer.html
// Please refer to the above link for method descriptions.
//
// This is an INCOMPLETE implementation. It's only implemented enough so it can act as a view buffer
// for ByteBuffer.
package bytebuffer
import (
"encoding/binary"
"errors"
"fmt"
"log"
)
type Uint32Buffer struct {
buf *ByteBuffer
// A buffer's capacity is the number of elements it contains. The capacity of a buffer is never
// negative and never changes.
size int
// A buffer's position is the index of the next element to be read or written. A buffer's position is
// never negative and is never greater than its limit.
pos int
// A buffer's mark is the index to which its position will be reset when the reset method is invoked.
// The mark is not always defined, but when it is defined it is never negative and is never greater
// than the position. If the mark is defined then it is discarded when the position or the limit is
// adjusted to a value smaller than the mark. If the mark is not defined then invoking the reset
// method causes an InvalidMark error.
mark int
// A buffer's limit is the index of the first element that should not be read or written. A buffer's
// limit is never negative and is never greater than its capacity.
limit int
// A read-only buffer does not allow its content to be changed, but its mark, position, and limit values
// are mutable. Whether or not a buffer is read-only may be determined by invoking its isReadOnly method.
readOnly bool
}
// Order retrieves this buffer's byte order.
// The byte order is used when reading or writing multibyte values, and when creating buffers that are
// views of this byte buffer. The order of a newly-created byte buffer is always binary.BigEndian.
func (this *Uint32Buffer) Order() binary.ByteOrder {
return this.buf.Order()
}
// SetOrder modifies this buffer's byte order.
func (this *Uint32Buffer) SetOrder(bo binary.ByteOrder) {
this.buf.SetOrder(bo)
}
// Reset resets this buffer's position to the previously-marked position.
// Invoking this method neither changes nor discards the mark's value.
func (this *Uint32Buffer) Reset() error {
if this.mark == -1 {
return errors.New("Uint32Buffer/Reset: Invalid Mark (has not been set)")
}
this.pos = this.mark
return nil
}
// Clears this buffer. The position is set to zero, the limit is set to the capacity, and the mark is discarded.
func (this *Uint32Buffer) Clear() error {
if this.readOnly {
return errors.New("Uint32Buffer/Resize: Cannot clear a read-only buffer")
}
this.pos = 0
this.mark = -1
this.limit = this.size
this.buf.Clear()
return nil
}
// Flips this buffer. The limit is set to the current position and then the position is set to zero.
// If the mark is defined then it is discarded.
func (this *Uint32Buffer) Flip() error {
this.limit = this.pos
this.pos = 0
return nil
}
// Rewinds this buffer. The position is set to zero and the mark is discarded.
func (this *Uint32Buffer) Rewind() error {
this.pos = 0
this.mark = -1
return nil
}
// Returns this buffer's capacity.
func (this *Uint32Buffer) Capacity() int {
return this.size
}
// Position returns this buffer's position.
func (this *Uint32Buffer) Position() int {
return this.pos
}
// SetPosition sets this buffer's position.
func (this *Uint32Buffer) SetPosition(pos int) error {
if pos > this.limit {
return errors.New("Uint32Buffer/SetPosition: Position must not be greater than buffer limit")
}
if pos < 0 {
return errors.New("Uint32Buffer/SetPosition: Position must be a non-negative number")
}
this.pos = pos
return nil
}
// Mark returns this buffer's mark.
func (this *Uint32Buffer) Mark() int {
return this.mark
}
// Limit returns this buffer's position.
func (this *Uint32Buffer) Limit() int {
return this.limit
}
// SetLimit sets this buffer's limit. If the position is larger than the new limit then it is set to the
// new limit. If the mark is defined and larger than the new limit then it is discarded.
func (this *Uint32Buffer) SetLimit(limit int) error {
if this.limit > this.size {
return errors.New("Uint32Buffer/SetLimit: Limit must not be greater than buffer size")
}
if this.limit < 0 {
return errors.New("Uint32Buffer/SetLimit: Limit must be a non-negative number")
}
this.limit = limit
if this.pos > this.limit {
this.pos = this.limit
}
if this.mark > this.limit {
this.mark = -1
}
return nil
}
// Remaining returns the number of elements between the current position and the capacity
func (this *Uint32Buffer) Remaining() int {
return this.limit - this.pos
}
// HasRemaining tells whether there are any elements between the current position and the limit.
func (this *Uint32Buffer) HasRemaining() bool {
return this.limit-this.pos != 0
}
// IsReadOnly tells whether or not this buffer is read-only.
func (this *Uint32Buffer) IsReadOnly() bool {
return this.readOnly
}
// GetUint32 is a relative get method for reading a uint32 value.
// Reads the next four bytes at this buffer's current position, composing them into a uint32 value
// according to the current byte order, and then increments the position by two.
func (this *Uint32Buffer) Get() (uint32, error) {
//fmt.Printf("uint32buffer/Get: remaining = %d\n", this.Remaining())
if !this.HasRemaining() {
return 0, errors.New("Uint32Buffer/Get: Insufficient remaining buffer for Uint32")
}
result, err := this.buf.GetUint32()
if err == nil {
this.pos += 1
}
return result, err
}
// GetUint16At is an absolute get method for reading a uint32 value.
// Reads four bytes at the given index, composing them into a uint32 value according to the current byte order.
func (this *Uint32Buffer) GetAt(index int) (uint32, error) {
if index < 0 || index+1 > this.limit {
return 0, errors.New("Uint32Buffer/GetAt: Index must be non-negative and not larger than the buffer limit.")
}
return this.buf.GetUint32At(index * 4)
}
// GetUint32s is a relative bulk get method.
//
// This method transfers ints from this buffer into the given destination array. If there are fewer
// ints remaining in the buffer than are required to satisfy the request, that is, if
// length > remaining(), then no ints are transferred and a BufferUnderflowException is thrown.
//
// Otherwise, this method copies length ints from this buffer into the given array, starting at the
// current position of this buffer and at the given offset in the array. The position of this buffer
// is then incremented by length.
func (this *Uint32Buffer) GetUint32s(dst []uint32, offset, length int) error {
//fmt.Printf("uint32buffer/GetUint32s: length = %d, remaining = %d, offset = %d\n", length, this.Remaining(), offset)
if offset < 0 || offset > cap(dst) {
return errors.New("Uint32Buffer/GetUint32s: Offset must be non-negative and no larger than length of dst")
}
if length < 0 || length > cap(dst)-offset {
//fmt.Printf("Uint32Buffer/GetUint32s: cap(dst)-offset = %d\n", cap(dst) - offset)
//fmt.Printf("Uint32Buffer/GetUint32s: buf = %v\n", this.buf)
return errors.New("Uint32Buffer/GetUint32s: Length must be non-negative and no larger than length of dst - offset ")
}
if length > this.Remaining() {
return errors.New("Uint32Buffer/GetUint32s: Insufficient uint32s to get. Length is greater than remaining bytes.")
}
for i := offset; i < length+offset; i++ {
var err error
//fmt.Printf("Uint32Buffer/GetUint32s: i = %d\n", i)
dst[i], err = this.Get()
if err != nil {
return errors.New("Uint32Buffer/GetUint32s: " + err.Error())
}
}
return nil
}
// GetUint32 is a relative get method for reading a uint32 value.
// Reads the next four bytes at this buffer's current position, composing them into a uint32 value
// according to the current byte order, and then increments the position by two.
func (this *Uint32Buffer) Put(value uint32) error {
if !this.HasRemaining() {
return fmt.Errorf("Uint32Buffer/Put: Insufficient remaining space (%t) for putting uint32", this.HasRemaining())
}
if err := this.buf.PutUint32(value); err != nil {
return errors.New("uint32buffer/Put: " + err.Error())
}
this.pos += 1
return nil
}
// PutUint32At is an absolute put method for writing a uint32 value
// Writes four bytes containing the given uint32 value, in the current byte order, into this buffer at the given index.
func (this *Uint32Buffer) PutAt(index int, value uint32) error {
if index < 0 || index+1 > this.limit {
return errors.New("Uint32Buffer/PutAt: Index must be non-negative and not larger than the buffer limit.")
}
return this.buf.PutUint32At(index*4, value)
}
// PutUint32s is a Relative bulk put method
// This method transfers ints into this buffer from the given source array. If there are more ints
// to be copied from the array than remain in this buffer, that is, if length > remaining(), then no
// ints are transferred and a BufferOverflowException is thrown.
//
// Otherwise, this method copies length ints from the given array into this buffer, starting at the g
// iven offset in the array and at the current position of this buffer. The position of this buffer
// is then incremented by length.
func (this *Uint32Buffer) PutUint32s(dst []uint32, offset, length int) error {
if offset < 0 || offset > cap(dst) {
return fmt.Errorf("Uint32Buffer/PutUint32s: Offset (%d) must be non-negative and no larger than length of dst", offset)
}
if length < 0 || length > cap(dst)-offset {
return fmt.Errorf("Uint32Buffer/PutUint32s: Length (%d) must be non-negative and no larger than length of dst - offset (%d)", length, cap(dst)-offset)
}
if length > this.Remaining() {
return fmt.Errorf("Uint32Buffer/PutUint32s: Insufficient buffer size. Length (%d) is greater than remaining buffer (%d).", length, this.Remaining())
}
log.Printf("Uint32Buffer/PutUint32s: before remaining = %d, length = %d, offset = %d, pos = %d\n", this.Remaining(), length, offset, this.pos)
for i := offset; i < length+offset; i++ {
if err := this.Put(dst[i]); err != nil {
return errors.New("Uint32Buffer/PutUint32s: " + err.Error())
}
log.Printf("Uint32Buffer/PutUint32s: this.pos = %d\n", this.pos)
}
log.Printf("Uint32Buffer/PutUint32s: after remaining = %d\n", this.Remaining())
return nil
}
func (this *Uint32Buffer) String() string {
return fmt.Sprintf("Uint32Buffer/String: Capacity = %d, limit = %d, mark = %d, position = %d\n", this.Capacity(), this.Limit(), this.Mark(), this.Position())
}