aboutsummaryrefslogtreecommitdiffstats
path: root/libparsepcf_get_bitmaps.c
blob: 8265db33e6d6a53124e9780a1e3f882d918f3052 (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
/* See LICENSE file for copyright and license details. */
#include "common.h"


int
libparsepcf_get_bitmaps(const void *file, size_t size,
                        const struct libparsepcf_table *table,
                        struct libparsepcf_bitmaps *meta)
{
	const char *text = file;
	size_t pos, glyph_pad;
	int msb = table->format & LIBPARSEPCF_BYTE;

	(void) size;

	if (table->size < 8)
		goto ebfont;

	pos = table->offset;

	if (table->format != libparsepcf_parse_lsb_uint32__(&text[pos]))
		goto ebfont;
	pos += 4;

	meta->glyph_count = (size_t)PARSE_UINT32(&text[pos], msb);
	pos += 4;

	if (16 > table->size - (pos - table->offset) ||
	    meta->glyph_count > (table->size - (pos - table->offset) - 16) / 4)
		goto ebfont;
	pos += meta->glyph_count * 4;
	glyph_pad = (size_t)(table->format & LIBPARSEPCF_GLYPH_PAD_MASK);
	meta->bitmap_size = (size_t)PARSE_UINT32(&text[pos + 4 * glyph_pad], msb);
	pos += 16;

	meta->bitmap_data = (const void *)&text[pos];
	if (meta->bitmap_size > table->size - (pos - table->offset))
		goto ebfont;

	meta->bit_packing = (size_t)1 << ((table->format & LIBPARSEPCF_SCAN_UNIT_MASK) >> 4);
	meta->row_padding = (size_t)1 << ((table->format & LIBPARSEPCF_GLYPH_PAD_MASK) >> 0);
	meta->msbyte_first = !!(table->format & LIBPARSEPCF_BYTE);
	meta->msbit_first = !!(table->format & LIBPARSEPCF_BIT);

	if (meta->row_padding < meta->bit_packing)
		goto ebfont;

	return 0;

ebfont:
	errno = EBFONT;
	return -1;
}