/* See LICENSE file for copyright and license details. */
#include "common.h"
static uint16_t
post_parse16(const char *data)
{
uint16_t b1 = (uint16_t)((const uint8_t *)data)[0];
uint16_t b2 = (uint16_t)((const uint8_t *)data)[1];
b1 = (uint16_t)(b1 << 8);
b2 = (uint16_t)(b2 << 0);
return (uint16_t)(b1 | b2);
}
static void
post_sign16(union {uint16_t u; int16_t s;} *info)
{
info->s = ((info->u >> 15) ? -(int16_t)(~info->u + 1) : (int16_t)info->u);
}
int
libparsesfnt_parse_post(
const char *data, size_t size,
struct libparsesfnt_post *infop,
const struct libparsesfnt_tabdir_entry *tag)
{
int ret;
ret = PARSE(LIBPARSESFNT_POST__, tag->offset, 0, 0, 0, 1, tag);
if (!ret) {
if (infop->format >> 16 == 2) {
if (2 > tag->length - tag->offset)
goto ebfont;
infop->subtable.v2.number_of_glyphs = post_parse16(&data[tag->offset + 2]);
}
}
return ret;
ebfont:
errno = EBFONT;
return -1;
}
static int
post_indices(
const char *data, size_t size,
uint16_t *indexp,
const struct libparsesfnt_tabdir_entry *tag,
size_t first, size_t count,
size_t offset)
{
if (offset > tag->length || first > (tag->length - offset) / 2)
goto ebfont;
offset += first * 2;
if (count > (tag->length - offset) / 2)
goto ebfont;
if (tag->offset > size || offset > size - tag->offset)
goto ebfont;
offset += tag->offset;
for (; count--; indexp++, offset += 2)
*indexp = post_parse16(&data[offset]);
return 0;
ebfont:
errno = EBFONT;
return -1;
}
int
libparsesfnt_parse_post_format_2_0_indices(
const char *data, size_t size,
uint16_t *indexp,
const struct libparsesfnt_tabdir_entry *tag,
size_t first, size_t count)
{
return post_indices(data, size, indexp, tag, first, count, 34);
}
int
libparsesfnt_parse_post_format_2_0_name(
const char *data, size_t size,
char namep[256],
const struct libparsesfnt_tabdir_entry *tag, const struct libparsesfnt_post *post,
size_t *offsetp /* start at 0 */)
{
size_t len;
if (!*offsetp) {
if ((size_t)post->subtable.v2.number_of_glyphs > ((size_t)tag->length - 34) / 2)
goto ebfont;
*offsetp = 34 + (size_t)post->subtable.v2.number_of_glyphs / 2;
if ((size_t)tag->length > size - (size_t)tag->offset)
goto ebfont;
}
if (*offsetp >= (size_t)tag->length)
goto ebfont;
len = (size_t)*(const uint8_t *)&data[*offsetp];
*offsetp += 1;
if (len > (size_t)tag->length - *offsetp)
goto ebfont;
memcpy(namep, &data[*offsetp], len);
namep[len] = 0;
*offsetp += len;
return 0;
ebfont:
errno = EBFONT;
return -1;
}
int
libparsesfnt_parse_post_format_2_5_offsets(
const char *data, size_t size,
int16_t *offsetp,
const struct libparsesfnt_tabdir_entry *tag,
size_t first, size_t count)
{
if (post_indices(data, size, (uint16_t *)offsetp, tag, first, count, 34))
return -1;
for (; count--; offsetp++)
post_sign16((void *)offsetp);
return 0;
}
int
libparsesfnt_parse_post_format_4_0_indices(
const char *data, size_t size,
uint16_t *indexp,
const struct libparsesfnt_tabdir_entry *tag,
size_t first, size_t count)
{
return post_indices(data, size, indexp, tag, first, count, 32);
}