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 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194
|
/* Copyright (C) 2001-2012 Artifex Software, Inc.
All Rights Reserved.
This software is provided AS-IS with no warranty, either express or
implied.
This software is distributed under license and may not be copied,
modified or distributed except as expressly authorized under the terms
of the license contained in the file LICENSE in this distribution.
Refer to licensing information at http://www.artifex.com or contact
Artifex Software, Inc., 7 Mt. Lassen Drive - Suite A-134, San Rafael,
CA 94903, U.S.A., +1(415)492-9861, for further information.
*/
/* Definitions for expandable stacks of refs */
/* Requires iref.h */
#ifndef istack_INCLUDED
# define istack_INCLUDED
#include "isdata.h"
/*
* The 3 principal Ghostscript stacks (operand, execution, and dictionary)
* are implemented as a linked list of blocks.
*
* Since all operators exit cleanly in case of stack under- or overflow,
* we handle all issues related to stack blocks in the top-level error
* recovery code in interp.c. A few situations require special treatment:
* see ostack.h, estack.h, and dstack.h for details.
*/
/*
* Define the structure for a stack block.
* In order to simplify allocation, stack blocks are implemented as
* t_array objects, with the first few elements used for special purposes.
* The actual layout is as follows:
* ref_stack_block structure
* bottom guard if any (see below)
* used elements of block
* unused elements of block
* top guard if any (see below)
* The `next' member of the next higher stack block includes all of this.
* The `used' member only includes the used elements of this block.
* Notes:
* - In the top block, the size of the `used' member may not be correct.
* - In all blocks but the top, we fill the unused elements with nulls.
*/
typedef struct ref_stack_block_s {
ref next; /* t_array, next lower block on stack */
ref used; /* t_array, subinterval of this block */
/* Actual stack starts here */
} ref_stack_block;
#define stack_block_refs (sizeof(ref_stack_block) / sizeof(ref))
/* ------ Procedural interface ------ */
/*
* Initialize a stack. Note that this allocates the stack parameter
* structure iff params is not NULL.
*/
int ref_stack_init(ref_stack_t *pstack, const ref *pblock_array,
uint bot_guard, uint top_guard,
const ref *pguard_value, gs_ref_memory_t *mem,
ref_stack_params_t *params);
/* Set whether a stack is allowed to expand. The initial value is true. */
void ref_stack_allow_expansion(ref_stack_t *pstack, bool expand);
/* Set the error codes for under- and overflow. The initial values are -1. */
void ref_stack_set_error_codes(ref_stack_t *pstack, int underflow_error,
int overflow_error);
/*
* Set the maximum number of elements allowed on a stack.
* Note that the value is a long, not a uint or a ulong.
*/
int ref_stack_set_max_count(ref_stack_t *pstack, long nmax);
/*
* Set the margin between the limit and the top of the stack.
* Note that this may require allocating a block.
*/
int ref_stack_set_margin(ref_stack_t *pstack, uint margin);
/* Return the number of elements on a stack. */
uint ref_stack_count(const ref_stack_t *pstack);
#define ref_stack_count_inline(pstk)\
((pstk)->p + 1 - (pstk)->bot + (pstk)->extension_used)
/* Return the maximum number of elements allowed on a stack. */
#define ref_stack_max_count(pstk) (uint)((pstk)->max_stack.value.intval)
/*
* Return a pointer to a given element from the stack, counting from
* 0 as the top element. If the index is out of range, return 0.
* Note that the index is a long, not a uint or a ulong.
*/
ref *ref_stack_index(const ref_stack_t *pstack, long index);
/*
* Count the number of elements down to and including the first mark.
* If no mark is found, return 0.
*/
uint ref_stack_counttomark(const ref_stack_t *pstack);
/*
* Do the store check for storing 'count' elements of a stack, starting
* 'skip' elements below the top, into an array. Return 0 or e_invalidaccess.
*/
int ref_stack_store_check(const ref_stack_t *pstack, ref *parray,
uint count, uint skip);
/*
* Store the top 'count' elements of a stack, starting 'skip' elements below
* the top, into an array, with or without store/undo checking. age=-1 for
* no check, 0 for old, 1 for new. May return e_rangecheck or
* e_invalidaccess.
*/
#ifndef gs_dual_memory_DEFINED
# define gs_dual_memory_DEFINED
typedef struct gs_dual_memory_s gs_dual_memory_t;
#endif
int ref_stack_store(const ref_stack_t *pstack, ref *parray, uint count,
uint skip, int age, bool check,
gs_dual_memory_t *idmem, client_name_t cname);
/*
* Pop the top N elements off a stack.
* The number must not exceed the number of elements in use.
*/
void ref_stack_pop(ref_stack_t *pstack, uint count);
#define ref_stack_clear(pstk) ref_stack_pop(pstk, ref_stack_count(pstk))
#define ref_stack_pop_to(pstk, depth)\
ref_stack_pop(pstk, ref_stack_count(pstk) - (depth))
/* Pop the top block off a stack. May return underflow_error. */
int ref_stack_pop_block(ref_stack_t *pstack);
/*
* Extend a stack to recover from an overflow condition.
* Uses the requested value to decide what to do.
* May return overflow_error or e_VMerror.
*/
int ref_stack_extend(ref_stack_t *pstack, uint request);
/*
* Push N empty slots onto a stack. These slots are not initialized:
* the caller must immediately fill them. May return overflow_error
* (if max_stack would be exceeded, or the stack has no allocator)
* or e_VMerror.
*/
int ref_stack_push(ref_stack_t *pstack, uint count);
/*
* Enumerate the blocks of a stack from top to bottom, as follows:
ref_stack_enum_t rsenum;
ref_stack_enum_begin(&rsenum, pstack);
do {
... process rsenum.size refs starting at rsenum.ptr ...
} while (ref_stack_enum_next(&rsenum));
*/
typedef struct ref_stack_enum_s {
ref_stack_block *block;
ref *ptr;
uint size;
} ref_stack_enum_t;
void ref_stack_enum_begin(ref_stack_enum_t *prse, const ref_stack_t *pstack);
bool ref_stack_enum_next(ref_stack_enum_t *prse);
/* Clean up a stack for garbage collection. */
void ref_stack_cleanup(ref_stack_t *pstack);
/*
* Free the entire contents of a stack, including the bottom block. The
* client must still free the ref_stack_t object. Note that after calling
* ref_stack_release, the stack is no longer usable.
*/
void ref_stack_release(ref_stack_t *pstack);
/*
* Release a stack and then free the stack object.
*/
void ref_stack_free(ref_stack_t *pstack);
#endif /* istack_INCLUDED */
|