mattermost-community-enterp.../vendor/github.com/bodgit/sevenzip/internal/bcj2/reader.go
Claude ec1f89217a Merge: Complete Mattermost Server with Community Enterprise
Full Mattermost server source with integrated Community Enterprise features.
Includes vendor directory for offline/air-gapped builds.

Structure:
- enterprise-impl/: Enterprise feature implementations
- enterprise-community/: Init files that register implementations
- enterprise/: Bridge imports (community_imports.go)
- vendor/: All dependencies for offline builds

Build (online):
  go build ./cmd/mattermost

Build (offline/air-gapped):
  go build -mod=vendor ./cmd/mattermost

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-17 23:59:07 +09:00

233 lines
4.2 KiB
Go

// Package bcj2 implements the BCJ2 filter for x86 binaries.
package bcj2
import (
"bytes"
"encoding/binary"
"errors"
"fmt"
"io"
"github.com/bodgit/sevenzip/internal/util"
)
type readCloser struct {
main util.ReadCloser
call io.ReadCloser
jump io.ReadCloser
rd util.ReadCloser
nrange uint
code uint
sd [256 + 2]uint
previous byte
written uint32
buf *bytes.Buffer
}
const (
numMoveBits = 5
numbitModelTotalBits = 11
bitModelTotal uint = 1 << numbitModelTotalBits
numTopBits = 24
topValue uint = 1 << numTopBits
)
var (
errAlreadyClosed = errors.New("bcj2: already closed")
errNeedFourReaders = errors.New("bcj2: need exactly four readers")
)
func isJcc(b0, b1 byte) bool {
return b0 == 0x0f && (b1&0xf0) == 0x80
}
func isJ(b0, b1 byte) bool {
return (b1&0xfe) == 0xe8 || isJcc(b0, b1)
}
func index(b0, b1 byte) int {
switch b1 {
case 0xe8:
return int(b0)
case 0xe9:
return 256
default:
return 257
}
}
// NewReader returns a new BCJ2 io.ReadCloser.
func NewReader(_ []byte, _ uint64, readers []io.ReadCloser) (io.ReadCloser, error) {
if len(readers) != 4 {
return nil, errNeedFourReaders
}
rc := &readCloser{
main: util.ByteReadCloser(readers[0]),
call: readers[1],
jump: readers[2],
rd: util.ByteReadCloser(readers[3]),
nrange: 0xffffffff,
buf: new(bytes.Buffer),
}
rc.buf.Grow(1 << 16)
b := make([]byte, 5)
if _, err := io.ReadFull(rc.rd, b); err != nil {
if !errors.Is(err, io.EOF) {
err = fmt.Errorf("bcj2: error reading initial state: %w", err)
}
return nil, err
}
for _, x := range b {
rc.code = (rc.code << 8) | uint(x)
}
for i := range rc.sd {
rc.sd[i] = bitModelTotal >> 1
}
return rc, nil
}
func (rc *readCloser) Close() error {
if rc.main == nil || rc.call == nil || rc.jump == nil || rc.rd == nil {
return errAlreadyClosed
}
if err := errors.Join(rc.main.Close(), rc.call.Close(), rc.jump.Close(), rc.rd.Close()); err != nil {
return fmt.Errorf("bcj2: error closing: %w", err)
}
rc.main, rc.call, rc.jump, rc.rd = nil, nil, nil, nil
return nil
}
func (rc *readCloser) Read(p []byte) (int, error) {
if rc.main == nil || rc.call == nil || rc.jump == nil || rc.rd == nil {
return 0, errAlreadyClosed
}
if err := rc.read(); err != nil && !errors.Is(err, io.EOF) {
return 0, err
}
n, err := rc.buf.Read(p)
if err != nil && !errors.Is(err, io.EOF) {
err = fmt.Errorf("bcj2: error reading: %w", err)
}
return n, err
}
func (rc *readCloser) update() error {
if rc.nrange < topValue {
b, err := rc.rd.ReadByte()
if err != nil && !errors.Is(err, io.EOF) {
return fmt.Errorf("bcj2: error reading byte: %w", err)
}
rc.code = (rc.code << 8) | uint(b)
rc.nrange <<= 8
}
return nil
}
func (rc *readCloser) decode(i int) (bool, error) {
newBound := (rc.nrange >> numbitModelTotalBits) * rc.sd[i]
if rc.code < newBound {
rc.nrange = newBound
rc.sd[i] += (bitModelTotal - rc.sd[i]) >> numMoveBits
if err := rc.update(); err != nil {
return false, err
}
return false, nil
}
rc.nrange -= newBound
rc.code -= newBound
rc.sd[i] -= rc.sd[i] >> numMoveBits
if err := rc.update(); err != nil {
return false, err
}
return true, nil
}
//nolint:cyclop,funlen
func (rc *readCloser) read() error {
var (
b byte
err error
)
for {
if b, err = rc.main.ReadByte(); err != nil {
if !errors.Is(err, io.EOF) {
err = fmt.Errorf("bcj2: error reading byte: %w", err)
}
return err
}
rc.written++
_ = rc.buf.WriteByte(b)
if isJ(rc.previous, b) {
break
}
rc.previous = b
if rc.buf.Len() == rc.buf.Cap() {
return nil
}
}
bit, err := rc.decode(index(rc.previous, b))
if err != nil {
return err
}
//nolint:nestif
if bit {
var r io.Reader
if b == 0xe8 {
r = rc.call
} else {
r = rc.jump
}
var dest uint32
if err = binary.Read(r, binary.BigEndian, &dest); err != nil {
if !errors.Is(err, io.EOF) {
err = fmt.Errorf("bcj2: error reading uint32: %w", err)
}
return err
}
dest -= rc.written + 4
_ = binary.Write(rc.buf, binary.LittleEndian, dest)
rc.previous = byte(dest >> 24)
rc.written += 4
} else {
rc.previous = b
}
return nil
}