aboutsummaryrefslogtreecommitdiff
path: root/internal/zlibx/decompress.go
blob: 126c1fcb33a56db198d18d59ef106d308300d6e2 (about) (plain) (blame)
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
package zlibx

import (
	"encoding/binary"
	"io"

	"codeberg.org/lindenii/furgit/internal/adler32"
	"codeberg.org/lindenii/furgit/internal/bufpool"
	"codeberg.org/lindenii/furgit/internal/flatex"
)

func Decompress(src []byte) (bufpool.Buffer, error) {
	out, _, err := DecompressSized(src, 0)
	return out, err
}

func DecompressSized(src []byte, sizeHint int) (buf bufpool.Buffer, consumed int, err error) {
	if len(src) < 6 {
		return bufpool.Buffer{}, 0, io.ErrUnexpectedEOF
	}

	cmf := src[0]
	flg := src[1]
	if (cmf&0x0f != zlibDeflate) || (cmf>>4 > zlibMaxWindow) || (binary.BigEndian.Uint16(src[:2])%31 != 0) {
		return bufpool.Buffer{}, 0, ErrHeader
	}

	offset := 2
	if flg&0x20 != 0 {
		return bufpool.Buffer{}, 0, ErrHeader
	}

	if len(src[offset:]) < 4 {
		return bufpool.Buffer{}, 0, io.ErrUnexpectedEOF
	}

	deflateData := src[offset:]
	out, consumed, err := flatex.DecompressSized(deflateData, sizeHint)
	if err != nil {
		return bufpool.Buffer{}, 0, err
	}

	checksumPos := offset + consumed
	if checksumPos+4 > len(src) {
		out.Release()
		return bufpool.Buffer{}, 0, io.ErrUnexpectedEOF
	}
	expected := binary.BigEndian.Uint32(src[checksumPos : checksumPos+4])
	if expected != adler32.Checksum(out.Bytes()) {
		out.Release()
		return bufpool.Buffer{}, 0, ErrChecksum
	}
	return out, checksumPos + 4, nil
}