eBookReaderSwitch/source/pdf/pdf-nametree.c

243 lines
5.7 KiB
C

#include "mupdf/fitz.h"
#include "mupdf/pdf.h"
static pdf_obj *
pdf_lookup_name_imp(fz_context *ctx, pdf_obj *node, pdf_obj *needle)
{
pdf_obj *kids = pdf_dict_get(ctx, node, PDF_NAME(Kids));
pdf_obj *names = pdf_dict_get(ctx, node, PDF_NAME(Names));
if (pdf_is_array(ctx, kids))
{
int l = 0;
int r = pdf_array_len(ctx, kids) - 1;
while (l <= r)
{
int m = (l + r) >> 1;
pdf_obj *kid = pdf_array_get(ctx, kids, m);
pdf_obj *limits = pdf_dict_get(ctx, kid, PDF_NAME(Limits));
pdf_obj *first = pdf_array_get(ctx, limits, 0);
pdf_obj *last = pdf_array_get(ctx, limits, 1);
if (pdf_objcmp(ctx, needle, first) < 0)
r = m - 1;
else if (pdf_objcmp(ctx, needle, last) > 0)
l = m + 1;
else
{
pdf_obj *obj;
if (pdf_mark_obj(ctx, node))
break;
fz_try(ctx)
obj = pdf_lookup_name_imp(ctx, kid, needle);
fz_always(ctx)
pdf_unmark_obj(ctx, node);
fz_catch(ctx)
fz_rethrow(ctx);
return obj;
}
}
}
if (pdf_is_array(ctx, names))
{
int l = 0;
int r = (pdf_array_len(ctx, names) / 2) - 1;
while (l <= r)
{
int m = (l + r) >> 1;
int c;
pdf_obj *key = pdf_array_get(ctx, names, m * 2);
pdf_obj *val = pdf_array_get(ctx, names, m * 2 + 1);
c = pdf_objcmp(ctx, needle, key);
if (c < 0)
r = m - 1;
else if (c > 0)
l = m + 1;
else
return val;
}
/* Spec says names should be sorted (hence the binary search,
* above), but Acrobat copes with non-sorted. Drop back to a
* simple search if the binary search fails. */
r = pdf_array_len(ctx, names)/2;
for (l = 0; l < r; l++)
if (!pdf_objcmp(ctx, needle, pdf_array_get(ctx, names, l * 2)))
return pdf_array_get(ctx, names, l * 2 + 1);
}
return NULL;
}
pdf_obj *
pdf_lookup_name(fz_context *ctx, pdf_document *doc, pdf_obj *which, pdf_obj *needle)
{
pdf_obj *root = pdf_dict_get(ctx, pdf_trailer(ctx, doc), PDF_NAME(Root));
pdf_obj *names = pdf_dict_get(ctx, root, PDF_NAME(Names));
pdf_obj *tree = pdf_dict_get(ctx, names, which);
return pdf_lookup_name_imp(ctx, tree, needle);
}
pdf_obj *
pdf_lookup_dest(fz_context *ctx, pdf_document *doc, pdf_obj *needle)
{
pdf_obj *root = pdf_dict_get(ctx, pdf_trailer(ctx, doc), PDF_NAME(Root));
pdf_obj *dests = pdf_dict_get(ctx, root, PDF_NAME(Dests));
pdf_obj *names = pdf_dict_get(ctx, root, PDF_NAME(Names));
pdf_obj *dest = NULL;
/* PDF 1.1 has destinations in a dictionary */
if (dests)
{
if (pdf_is_name(ctx, needle))
return pdf_dict_get(ctx, dests, needle);
else
return pdf_dict_gets(ctx, dests, pdf_to_str_buf(ctx, needle));
}
/* PDF 1.2 has destinations in a name tree */
if (names && !dest)
{
pdf_obj *tree = pdf_dict_get(ctx, names, PDF_NAME(Dests));
return pdf_lookup_name_imp(ctx, tree, needle);
}
return NULL;
}
static void
pdf_load_name_tree_imp(fz_context *ctx, pdf_obj *dict, pdf_document *doc, pdf_obj *node)
{
pdf_obj *kids = pdf_dict_get(ctx, node, PDF_NAME(Kids));
pdf_obj *names = pdf_dict_get(ctx, node, PDF_NAME(Names));
int i;
if (kids && !pdf_mark_obj(ctx, node))
{
fz_try(ctx)
{
int len = pdf_array_len(ctx, kids);
for (i = 0; i < len; i++)
pdf_load_name_tree_imp(ctx, dict, doc, pdf_array_get(ctx, kids, i));
}
fz_always(ctx)
pdf_unmark_obj(ctx, node);
fz_catch(ctx)
fz_rethrow(ctx);
}
if (names)
{
int len = pdf_array_len(ctx, names);
for (i = 0; i + 1 < len; i += 2)
{
pdf_obj *key = pdf_array_get(ctx, names, i);
pdf_obj *val = pdf_array_get(ctx, names, i + 1);
if (pdf_is_string(ctx, key))
{
key = pdf_new_name(ctx, pdf_to_text_string(ctx, key));
fz_try(ctx)
pdf_dict_put(ctx, dict, key, val);
fz_always(ctx)
pdf_drop_obj(ctx, key);
fz_catch(ctx)
fz_rethrow(ctx);
}
else if (pdf_is_name(ctx, key))
{
pdf_dict_put(ctx, dict, key, val);
}
}
}
}
pdf_obj *
pdf_load_name_tree(fz_context *ctx, pdf_document *doc, pdf_obj *which)
{
pdf_obj *root = pdf_dict_get(ctx, pdf_trailer(ctx, doc), PDF_NAME(Root));
pdf_obj *names = pdf_dict_get(ctx, root, PDF_NAME(Names));
pdf_obj *tree = pdf_dict_get(ctx, names, which);
if (pdf_is_dict(ctx, tree))
{
pdf_obj *dict = pdf_new_dict(ctx, doc, 100);
pdf_load_name_tree_imp(ctx, dict, doc, tree);
return dict;
}
return NULL;
}
pdf_obj *
pdf_lookup_number(fz_context *ctx, pdf_obj *node, int needle)
{
pdf_obj *kids = pdf_dict_get(ctx, node, PDF_NAME(Kids));
pdf_obj *nums = pdf_dict_get(ctx, node, PDF_NAME(Nums));
if (pdf_is_array(ctx, kids))
{
int l = 0;
int r = pdf_array_len(ctx, kids) - 1;
while (l <= r)
{
int m = (l + r) >> 1;
pdf_obj *kid = pdf_array_get(ctx, kids, m);
pdf_obj *limits = pdf_dict_get(ctx, kid, PDF_NAME(Limits));
int first = pdf_to_int(ctx, pdf_array_get(ctx, limits, 0));
int last = pdf_to_int(ctx, pdf_array_get(ctx, limits, 1));
if (needle < first)
r = m - 1;
else if (needle > last)
l = m + 1;
else
{
pdf_obj *obj;
if (pdf_mark_obj(ctx, node))
break;
fz_try(ctx)
obj = pdf_lookup_number(ctx, kid, needle);
fz_always(ctx)
pdf_unmark_obj(ctx, node);
fz_catch(ctx)
fz_rethrow(ctx);
return obj;
}
}
}
if (pdf_is_array(ctx, nums))
{
pdf_obj *nums = pdf_dict_get(ctx, node, PDF_NAME(Nums));
int l = 0;
int r = (pdf_array_len(ctx, nums) / 2) - 1;
while (l <= r)
{
int m = (l + r) >> 1;
int key = pdf_to_int(ctx, pdf_array_get(ctx, nums, m * 2));
pdf_obj *val = pdf_array_get(ctx, nums, m * 2 + 1);
if (needle < key)
r = m - 1;
else if (needle > key)
l = m + 1;
else
return val;
}
/* Parallel the nametree lookup above by allowing for non-sorted lists. */
r = pdf_array_len(ctx, nums)/2;
for (l = 0; l < r; l++)
if (needle == pdf_to_int(ctx, pdf_array_get(ctx, nums, l * 2)))
return pdf_array_get(ctx, nums, l * 2 + 1);
}
return NULL;
}