% pagetree.w % % Copyright 2006-2010 Taco Hoekwater % This file is part of LuaTeX. % LuaTeX is free software; you can redistribute it and/or modify it under % the terms of the GNU General Public License as published by the Free % Software Foundation; either version 2 of the License, or (at your % option) any later version. % LuaTeX 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. % You should have received a copy of the GNU General Public License along % with LuaTeX; if not, see . @ @c #include "ptexlib.h" static const char _svn_version[] = "$Id: pagetree.w 3887 2010-09-14 21:32:23Z hhenkel $ " "$URL: http://foundry.supelec.fr/svn/luatex/tags/beta-0.70.1/source/texk/web2c/luatexdir/pdf/pagetree.w $"; @* Page diversions. @ @c #ifdef DEBUG # define PAGES_TREE_KIDSMAX 3 #else # define PAGES_TREE_KIDSMAX 6 #endif static struct avl_table *divert_list_tree = NULL; typedef struct pages_entry_ { int objnum; /* object number of this /Pages object */ int number_of_pages; /* total number of all pages below */ int number_of_kids; /* number of direct kid objects */ int kids[PAGES_TREE_KIDSMAX]; /* array of kid object numbers */ struct pages_entry_ *next; } pages_entry; typedef struct divert_list_entry_ { int divnum; pages_entry *first; pages_entry *last; } divert_list_entry; static int comp_divert_list_entry(const void *pa, const void *pb, void *p) { (void) p; if (((const divert_list_entry *) pa)->divnum > ((const divert_list_entry *) pb)->divnum) return 1; if (((const divert_list_entry *) pa)->divnum < ((const divert_list_entry *) pb)->divnum) return -1; return 0; } @ @c static pages_entry *new_pages_entry(PDF pdf) { pages_entry *p; int i; p = xtalloc(1, pages_entry); p->number_of_pages = p->number_of_kids = 0; for (i = 0; i < PAGES_TREE_KIDSMAX; i++) p->kids[i] = 0; p->next = NULL; p->objnum = pdf_create_obj(pdf, obj_type_pages, 0); return p; } @ @c static divert_list_entry *new_divert_list_entry(void) { divert_list_entry *d; d = xtalloc(1, divert_list_entry); d->first = d->last = NULL; return d; } @ @c static void ensure_list_tree(void) { if (divert_list_tree == NULL) { divert_list_tree = avl_create(comp_divert_list_entry, NULL, &avl_xallocator); assert(divert_list_tree != NULL); } } @ @c static divert_list_entry *get_divert_list(int divnum) { divert_list_entry *d, tmp; void **aa; tmp.divnum = divnum; d = (divert_list_entry *) avl_find(divert_list_tree, &tmp); if (d == NULL) { d = new_divert_list_entry(); d->divnum = divnum; aa = avl_probe(divert_list_tree, d); assert(aa != NULL); } return d; } @ |pdf_do_page_divert()| returns the current /Parent object number @c int pdf_do_page_divert(PDF pdf, int objnum, int divnum) { divert_list_entry *d; pages_entry *p; #ifdef DEBUG pages_entry *q; struct avl_traverser t; int i; #endif /* initialize the tree */ ensure_list_tree(); /* make sure we have a list for this diversion */ d = get_divert_list(divnum); if (d->first == NULL || d->last->number_of_kids == PAGES_TREE_KIDSMAX) { /* append a new |pages_entry| */ p = new_pages_entry(pdf); if (d->first == NULL) d->first = p; else d->last->next = p; d->last = p; } p = d->last; p->kids[p->number_of_kids++] = objnum; p->number_of_pages++; #ifdef DEBUG printf("\n"); avl_t_init(&t, divert_list_tree); for (d = avl_t_first(&t, divert_list_tree); d != NULL; d = avl_t_next(&t)) { printf("===== D-LIST %d: ", d->divnum); for (q = d->first; q != NULL; q = q->next) { printf("P=%d NK=%d (", q->objnum, q->number_of_kids); for (i = 0; i < q->number_of_kids; i++) printf("%d ", q->kids[i]); printf(") "); } printf("\n"); } printf("\n"); #endif return p->objnum; } @ @c static void movelist(divert_list_entry * d, divert_list_entry * dto) { if (d != NULL && d->first != NULL && d->divnum != dto->divnum) { /* no undivert of empty list or into self */ if (dto->first == NULL) dto->first = d->first; else dto->last->next = d->first; dto->last = d->last; d->first = d->last = NULL; /* one could as well remove this |divert_list_entry| */ } } @ undivert from diversion |divnum| into diversion |curdivnum| @c void pdf_do_page_undivert(int divnum, int curdivnum) { divert_list_entry *d, *dto, tmp; struct avl_traverser t; #ifdef DEBUG pages_entry *p; int i; #endif /* initialize the tree */ ensure_list_tree(); /* find the diversion |curdivnum| list where diversion |divnum| should go */ dto = get_divert_list(curdivnum); if (divnum == 0) { /* 0 = special case: undivert {\it all\/} lists */ avl_t_init(&t, divert_list_tree); for (d = avl_t_first(&t, divert_list_tree); d != NULL; d = avl_t_next(&t)) movelist(d, dto); } else { tmp.divnum = divnum; d = (divert_list_entry *) avl_find(divert_list_tree, &tmp); movelist(d, dto); } #ifdef DEBUG printf("\n"); avl_t_init(&t, divert_list_tree); for (d = avl_t_first(&t, divert_list_tree); d != NULL; d = avl_t_next(&t)) { printf("===== U-LIST %d: ", d->divnum); for (p = d->first; p != NULL; p = p->next) { printf("P=%d NK=%d (", p->objnum, p->number_of_kids); for (i = 0; i < p->number_of_kids; i++) printf("%d ", p->kids[i]); printf(") "); } printf("\n"); } printf("\n"); #endif } @ write a /Pages object @c #define pdf_pages_attr equiv(pdf_pages_attr_loc) static void write_pages(PDF pdf, pages_entry * p, int parent) { int i; assert(p != NULL); pdf_begin_dict(pdf, p->objnum, 1); pdf_printf(pdf, "/Type /Pages\n"); if (parent == 0) { /* it's root */ if (pdf_pages_attr != null) pdf_print_toks_ln(pdf, pdf_pages_attr); print_pdf_table_string(pdf, "pagesattributes"); } else pdf_printf(pdf, "/Parent %d 0 R\n", parent); pdf_printf(pdf, "/Count %d\n/Kids [", (int) p->number_of_pages); for (i = 0; i < p->number_of_kids; i++) pdf_printf(pdf, "%d 0 R ", (int) p->kids[i]); pdf_remove_last_space(pdf); pdf_printf(pdf, "]\n"); pdf_end_dict(pdf); } @ loop over all /Pages objects, output them, create their parents, recursing bottom up, return the /Pages root object number @c static int output_pages_list(PDF pdf, pages_entry * pe) { pages_entry *p, *q, *r; assert(pe != NULL); if (pe->next == NULL) { /* everything fits into one |pages_entry| */ write_pages(pdf, pe, 0); /* --> /Pages root found */ return pe->objnum; } q = r = new_pages_entry(pdf); /* one level higher needed */ for (p = pe; p != NULL; p = p->next) { if (q->number_of_kids == PAGES_TREE_KIDSMAX) { q->next = new_pages_entry(pdf); q = q->next; } q->kids[q->number_of_kids++] = p->objnum; q->number_of_pages += p->number_of_pages; write_pages(pdf, p, q->objnum); } return output_pages_list(pdf, r); /* recurse through next higher level */ } @ @c int output_pages_tree(PDF pdf) { divert_list_entry *d; pdf_do_page_undivert(0, 0); /* concatenate all diversions into diversion 0 */ d = get_divert_list(0); /* get diversion 0 */ return output_pages_list(pdf, d->first); }