aboutsummaryrefslogtreecommitdiff
path: root/src/liblzma/common/vli_encoder.c
blob: 1ecdb0d25a1bcccbcc4fd9e6ffcc2fd2402bd92d (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
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
///////////////////////////////////////////////////////////////////////////////
//
/// \file       vli_encoder.c
/// \brief      Encodes variable-length integers
//
//  Copyright (C) 2007 Lasse Collin
//
//  This library is free software; you can redistribute it and/or
//  modify it under the terms of the GNU Lesser General Public
//  License as published by the Free Software Foundation; either
//  version 2.1 of the License, or (at your option) any later version.
//
//  This library is distributed in the hope that it will be useful,
//  but WITHOUT ANY WARRANTY; without even the implied warranty of
//  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
//  Lesser General Public License for more details.
//
///////////////////////////////////////////////////////////////////////////////

#include "common.h"


extern LZMA_API lzma_ret
lzma_vli_encode(lzma_vli vli, size_t *restrict vli_pos, size_t vli_size,
		uint8_t *restrict out, size_t *restrict out_pos,
		size_t out_size)
{
	if (vli > LZMA_VLI_VALUE_MAX || *vli_pos >= 9 || vli_size > 9
			|| (vli != 0 && (vli >> (7 * *vli_pos)) == 0))
		return LZMA_PROG_ERROR;

	if (*out_pos >= out_size)
		return LZMA_BUF_ERROR;

	if (*vli_pos == 0) {
		*vli_pos = 1;

		if (vli <= 0x7F && *vli_pos >= vli_size) {
			// Single-byte integer
			out[(*out_pos)++] = vli;
			return LZMA_STREAM_END;
		}

		// First byte of a multibyte integer
		out[(*out_pos)++] = (vli & 0x7F) | 0x80;
	}

	while (*out_pos < out_size) {
		const lzma_vli b = vli >> (7 * *vli_pos);
		++*vli_pos;

		if (b <= 0x7F && *vli_pos >= vli_size) {
			// Last byte of a multibyte integer
			out[(*out_pos)++] = (b & 0xFF) | 0x80;
			return LZMA_STREAM_END;
		}

		// Middle byte of a multibyte integer
		out[(*out_pos)++] = b & 0x7F;
	}

	// vli is not yet completely written out.
	return LZMA_OK;
}


extern LZMA_API size_t
lzma_vli_size(lzma_vli vli)
{
	if (vli > LZMA_VLI_VALUE_MAX)
		return 0;

	size_t i = 0;
	do {
		vli >>= 7;
		++i;
	} while (vli != 0);

	assert(i <= 9);
	return i;
}