package roaring import ( "encoding/binary" "io" ) type byteInput interface { // next returns a slice containing the next n bytes from the buffer, // advancing the buffer as if the bytes had been returned by Read. next(n int) ([]byte, error) // readUInt32 reads uint32 with LittleEndian order readUInt32() (uint32, error) // readUInt16 reads uint16 with LittleEndian order readUInt16() (uint16, error) // getReadBytes returns read bytes getReadBytes() int64 // skipBytes skips exactly n bytes skipBytes(n int) error } func newByteInputFromReader(reader io.Reader) byteInput { return &byteInputAdapter{ r: reader, readBytes: 0, } } func newByteInput(buf []byte) byteInput { return &byteBuffer{ buf: buf, off: 0, } } type byteBuffer struct { buf []byte off int } // next returns a slice containing the next n bytes from the reader // If there are fewer bytes than the given n, io.ErrUnexpectedEOF will be returned func (b *byteBuffer) next(n int) ([]byte, error) { m := len(b.buf) - b.off if n > m { return nil, io.ErrUnexpectedEOF } data := b.buf[b.off : b.off+n] b.off += n return data, nil } // readUInt32 reads uint32 with LittleEndian order func (b *byteBuffer) readUInt32() (uint32, error) { if len(b.buf)-b.off < 4 { return 0, io.ErrUnexpectedEOF } v := binary.LittleEndian.Uint32(b.buf[b.off:]) b.off += 4 return v, nil } // readUInt16 reads uint16 with LittleEndian order func (b *byteBuffer) readUInt16() (uint16, error) { if len(b.buf)-b.off < 2 { return 0, io.ErrUnexpectedEOF } v := binary.LittleEndian.Uint16(b.buf[b.off:]) b.off += 2 return v, nil } // getReadBytes returns read bytes func (b *byteBuffer) getReadBytes() int64 { return int64(b.off) } // skipBytes skips exactly n bytes func (b *byteBuffer) skipBytes(n int) error { m := len(b.buf) - b.off if n > m { return io.ErrUnexpectedEOF } b.off += n return nil } // reset resets the given buffer with a new byte slice func (b *byteBuffer) reset(buf []byte) { b.buf = buf b.off = 0 } type byteInputAdapter struct { r io.Reader readBytes int } // next returns a slice containing the next n bytes from the buffer, // advancing the buffer as if the bytes had been returned by Read. func (b *byteInputAdapter) next(n int) ([]byte, error) { buf := make([]byte, n) m, err := io.ReadAtLeast(b.r, buf, n) b.readBytes += m if err != nil { return nil, err } return buf, nil } // readUInt32 reads uint32 with LittleEndian order func (b *byteInputAdapter) readUInt32() (uint32, error) { buf, err := b.next(4) if err != nil { return 0, err } return binary.LittleEndian.Uint32(buf), nil } // readUInt16 reads uint16 with LittleEndian order func (b *byteInputAdapter) readUInt16() (uint16, error) { buf, err := b.next(2) if err != nil { return 0, err } return binary.LittleEndian.Uint16(buf), nil } // getReadBytes returns read bytes func (b *byteInputAdapter) getReadBytes() int64 { return int64(b.readBytes) } // skipBytes skips exactly n bytes func (b *byteInputAdapter) skipBytes(n int) error { _, err := b.next(n) return err } // reset resets the given buffer with a new stream func (b *byteInputAdapter) reset(stream io.Reader) { b.r = stream b.readBytes = 0 }