pdf_name_tree.c

Fri, 07 Mar 2003 10:16:08 +0000

author
eric
date
Fri, 07 Mar 2003 10:16:08 +0000
changeset 80
96c64a7566df
child 83
bbc7dd27aa5b
permissions
-rw-r--r--

added name trees and number trees.

     1 /*
     2  * t2p: Create a PDF file from the contents of one or more TIFF
     3  *      bilevel image files.  The images in the resulting PDF file
     4  *      will be compressed using ITU-T T.6 (G4) fax encoding.
     5  *
     6  * PDF routines
     7  * $Id: pdf_name_tree.c,v 1.1 2003/03/07 02:16:08 eric Exp $
     8  * Copyright 2003 Eric Smith <eric@brouhaha.com>
     9  *
    10  * This program is free software; you can redistribute it and/or modify
    11  * it under the terms of the GNU General Public License version 2 as
    12  * published by the Free Software Foundation.  Note that permission is
    13  * not granted to redistribute this program under the terms of any
    14  * other version of the General Public License.
    15  *
    16  * This program is distributed in the hope that it will be useful,
    17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    19  * GNU General Public License for more details.
    20  *
    21  * You should have received a copy of the GNU General Public License
    22  * along with this program; if not, write to the Free Software
    23  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111 USA
    24  */
    27 #include <stdbool.h>
    28 #include <stdint.h>
    29 #include <stdio.h>
    30 #include <stdlib.h>
    31 #include <string.h>
    34 #include "bitblt.h"
    35 #include "pdf.h"
    36 #include "pdf_util.h"
    37 #include "pdf_prim.h"
    38 #include "pdf_private.h"
    39 #include "pdf_name_tree.h"
    42 #define MAX_NAME_TREE_NODE_ENTRIES 16
    45 struct pdf_name_tree_node
    46 {
    47   struct pdf_obj *dict;    /* indirect reference */
    49   struct pdf_name_tree_node *parent;  /* NULL for root */
    50   bool leaf;
    52   int count;               /* how many kids or names/numbers are
    53 			      attached to this node */
    55   struct pdf_name_tree_node *kids [MAX_NAME_TREE_NODE_ENTRIES];  /* non-leaf only */
    57   struct pdf_obj *min_key;
    58   struct pdf_obj *max_key;
    60   /* following fields valid in leaf nodes only: */
    62   struct pdf_obj *keys [MAX_NAME_TREE_NODE_ENTRIES];
    63   struct pdf_obj *values [MAX_NAME_TREE_NODE_ENTRIES];
    64 };
    67 struct pdf_name_tree *pdf_new_name_tree (pdf_file_handle pdf_file,
    68 					 bool number_tree)
    69 {
    70   struct pdf_name_tree *tree;
    71   struct pdf_name_tree_node *root;
    73   root = pdf_calloc (1, sizeof (struct pdf_name_tree_node));
    74   tree = pdf_calloc (1, sizeof (struct pdf_name_tree));
    76   tree->pdf_file = pdf_file;
    77   tree->root = root;
    78   tree->number_tree = number_tree;
    80   root->parent = NULL;
    81   root->leaf = 1;
    83   return (tree);
    84 }
    87 static void pdf_split_name_tree_node (struct pdf_name_tree *tree,
    88 				      struct pdf_name_tree_node *node)
    89 {
    90   struct pdf_name_tree_node *new_node;
    92   if (node == tree->root)
    93     {
    94       /* create new root above current root */
    95       struct pdf_name_tree_node *new_root_node;
    97       new_root_node = pdf_calloc (1, sizeof (struct pdf_name_tree_node));
    99       new_root_node->parent = NULL;
   100       new_root_node->leaf = 0;
   102       new_root_node->count = 1;
   103       new_root_node->kids [0] = node;
   105       new_root_node->min_key = node->min_key;
   106       new_root_node->max_key = node->max_key;
   108       node->parent = new_root_node;
   109       tree->root = new_root_node;
   110     }
   112   new_node = pdf_calloc (1, sizeof (struct pdf_name_tree_node));
   113   new_node->parent = node->parent;
   114   new_node->leaf = node->leaf;
   115 }
   118 static void pdf_add_tree_element (struct pdf_name_tree *tree,
   119 				  struct pdf_obj *key,
   120 				  struct pdf_obj *val)
   121 {
   122   struct pdf_name_tree_node *node;
   124   /* find node which should contain element */
   125   node = tree->root;
   127   /* if node is full, split, recursing to root if necessary */
   128   if (node->count == MAX_NAME_TREE_NODE_ENTRIES)
   129     {
   130       pdf_split_name_tree_node (tree, node);
   131       pdf_add_tree_element (tree, key, val);
   132       return;
   133     }
   134 }
   137 void pdf_add_name_tree_element (struct pdf_name_tree *tree,
   138 				char *key,
   139 				struct pdf_obj *val)
   140 {
   141   struct pdf_obj *key_obj = pdf_new_string (key);
   142   pdf_add_tree_element (tree, key_obj, val);
   143 }
   146 void pdf_add_number_tree_element (struct pdf_name_tree *tree,
   147 				  long key,
   148 				  struct pdf_obj *val)
   149 {
   150   struct pdf_obj *key_obj = pdf_new_integer (key);
   151   pdf_add_tree_element (tree, key_obj, val);
   152 }
   155 static void pdf_finalize_name_tree_node (struct pdf_name_tree *tree,
   156 					 struct pdf_name_tree_node *node)
   157 {
   158   int i;
   160   node->dict = pdf_new_ind_ref (tree->pdf_file, pdf_new_obj (PT_DICTIONARY));
   162   if (node->leaf)
   163     {
   164       /* write Names or Nums array */
   165       struct pdf_obj *names = pdf_new_obj (PT_ARRAY);
   166       for (i = 0; i < node->count; i++)
   167 	{
   168 	  pdf_add_array_elem (names, node->keys [i]);
   169 	  pdf_add_array_elem (names, node->values [i]);
   170 	}
   171       pdf_set_dict_entry (node->dict,
   172 			  tree->number_tree ? "Nums" : "Names",
   173 			  names);
   174     }
   175   else
   176     {
   177       /* finalize the children first so that their dict ind ref is
   178 	 available */
   179       for (i = 0; i < node->count; i++)
   180 	pdf_finalize_name_tree_node (tree, node->kids [i]);
   182       /* write Kids array */
   183       struct pdf_obj *kids = pdf_new_obj (PT_ARRAY);
   184       for (i = 0; i < node->count; i++)
   185 	pdf_add_array_elem (kids, node->kids [i]->dict);
   186       pdf_set_dict_entry (node->dict, "Kids", kids);
   187     }
   189   if (! node->parent)
   190     {
   191       /* write Limits array */
   192       struct pdf_obj *limits = pdf_new_obj (PT_ARRAY);
   193       pdf_add_array_elem (limits, node->min_key);
   194       pdf_add_array_elem (limits, node->max_key);
   195       pdf_set_dict_entry (node->dict, "Limits", limits);
   196     }
   197 }
   200 void pdf_finalize_name_tree (struct pdf_name_tree *tree)
   201 {
   202   pdf_finalize_name_tree_node (tree, tree->root);
   203 }