home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
PC World Komputer 1996 February
/
PCWK0296.iso
/
sharewar
/
dos
/
program
/
gs300sr1
/
gs300sr1.exe
/
IPARAM.C
< prev
next >
Wrap
C/C++ Source or Header
|
1994-07-27
|
17KB
|
554 lines
/* Copyright (C) 1993, 1994 Aladdin Enterprises. All rights reserved.
This file is part of Aladdin Ghostscript.
Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND. No author
or distributor accepts any responsibility for the consequences of using it,
or for whether it serves any particular purpose or works at all, unless he
or she says so in writing. Refer to the Aladdin Ghostscript Free Public
License (the "License") for full details.
Every copy of Aladdin Ghostscript must include a copy of the License,
normally in a plain ASCII text file named PUBLIC. The License grants you
the right to copy, modify and redistribute Aladdin Ghostscript, but only
under certain conditions described in the License. Among other things, the
License requires that the copyright notice and this notice be preserved on
all copies.
*/
/* iparam.c */
/* Interpreter implementations of parameter dictionaries */
#include "memory_.h"
#include "string_.h"
#include "ghost.h"
#include "errors.h"
#include "opcheck.h"
#include "ialloc.h"
#include "idict.h"
#include "imemory.h" /* for iutil.h */
#include "iname.h"
#include "istack.h"
#include "iparam.h"
#include "iutil.h" /* for num_params */
#include "store.h"
#define iplist ((iparam_list *)plist)
/* Generic routines for getting parameters and converting to refs. */
private param_proc_xmit_null(ref_param_write_null);
private param_proc_xmit_bool(ref_param_write_bool);
private param_proc_xmit_int(ref_param_write_int);
private param_proc_xmit_long(ref_param_write_long);
private param_proc_xmit_float(ref_param_write_float);
private param_proc_xmit_string(ref_param_write_string);
private param_proc_xmit_name(ref_param_write_name);
private param_proc_xmit_int_array(ref_param_write_int_array);
private param_proc_xmit_float_array(ref_param_write_float_array);
private const gs_param_list_procs ref_write_procs = {
ref_param_write_null,
ref_param_write_bool,
ref_param_write_int,
ref_param_write_long,
ref_param_write_float,
ref_param_write_string,
ref_param_write_name,
ref_param_write_int_array,
ref_param_write_float_array
};
private bool near ref_param_wanted(P2(const iparam_list *, gs_param_name));
private int near ref_param_write(P3(iparam_list *, gs_param_name, ref *));
private int
ref_param_write_null(gs_param_list *plist, gs_param_name pkey)
{ ref value;
make_null(&value);
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_bool(gs_param_list *plist, gs_param_name pkey, bool *pvalue)
{ ref value;
make_bool(&value, *pvalue);
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_int(gs_param_list *plist, gs_param_name pkey, int *pvalue)
{ ref value;
make_int(&value, *pvalue);
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_long(gs_param_list *plist, gs_param_name pkey, long *pvalue)
{ ref value;
make_int(&value, *pvalue);
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_float(gs_param_list *plist, gs_param_name pkey, float *pvalue)
{ ref value;
make_real(&value, *pvalue);
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_string(gs_param_list *plist, gs_param_name pkey,
gs_param_string *pvalue)
{ ref value;
const byte *pdata = pvalue->data;
uint n = pvalue->size;
if ( !ref_param_wanted(iplist, pkey) )
return 0;
if ( pvalue->persistent )
make_const_string(&value, a_readonly | a_foreign, n, pdata);
else
{ byte *pstr = ialloc_bytes(n, "ref_write_string_param");
if ( pstr == 0 )
return_error(e_VMerror);
memcpy(pstr, pdata, n);
make_string(&value, a_all, n, pstr);
}
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_name(gs_param_list *plist, gs_param_name pkey,
gs_param_string *pvalue)
{ ref value;
int code;
if ( !ref_param_wanted(iplist, pkey) )
return 0;
code = name_ref(pvalue->data, pvalue->size, &value,
(pvalue->persistent ? 0 : 1));
if ( code < 0 )
return code;
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_int_array(gs_param_list *plist, gs_param_name pkey,
gs_param_int_array *pvalue)
{ ref value;
const int *pdata = pvalue->data;
uint n = pvalue->size;
ref *pe;
int code;
if ( !ref_param_wanted(iplist, pkey) )
return 0;
code = ialloc_ref_array(&value, a_all, n,
"ref_write_int_array_param");
if ( code < 0 )
return code;
for ( pe = value.value.refs; n > 0; n--, pe++, pdata++ )
make_int(pe, *pdata);
return ref_param_write(iplist, pkey, &value);
}
private int
ref_param_write_float_array(gs_param_list *plist, gs_param_name pkey,
gs_param_float_array *pvalue)
{ ref value;
const float *pdata = pvalue->data;
uint n = pvalue->size;
int code;
ref *pe;
if ( !ref_param_wanted(iplist, pkey) )
return 0;
code = ialloc_ref_array(&value, a_all, n,
"ref_write_float_array_param");
if ( code < 0 )
return code;
for ( pe = value.value.refs; n > 0; n--, pe++, pdata++ )
make_real(pe, *pdata);
return ref_param_write(iplist, pkey, &value);
}
/* Check whether a given parameter is wanted. */
private bool near
ref_param_wanted(const iparam_list *plist, gs_param_name pkey)
{ ref *ignore_value;
return !(r_has_type(&plist->u.w.wanted, t_dictionary) &&
dict_find_string(&plist->u.w.wanted, pkey, &ignore_value) <= 0);
}
/* Generic routine for writing a ref parameter. */
private int near
ref_param_write(iparam_list *plist, gs_param_name pkey, ref *pvalue)
{ ref nref;
int code;
if ( !ref_param_wanted(plist, pkey) )
return 0;
code = name_ref((const byte *)pkey, strlen(pkey), &nref, 0);
if ( code < 0 )
return code;
return (*plist->u.w.write)(plist, &nref, pvalue);
}
/* Initialize for writing parameters. */
private void near
ref_param_write_init(iparam_list *plist, const ref *pwanted)
{ if ( pwanted == 0 )
make_null(&plist->u.w.wanted);
else
plist->u.w.wanted = *pwanted;
plist->results = 0;
}
/* Implementation for getting parameters to a stack. */
private int
stack_param_write(iparam_list *plist, const ref *pkey, ref *pvalue)
{
#define splist ((stack_param_list *)plist)
ref_stack *pstack = splist->pstack;
s_ptr p = pstack->p;
if ( pstack->top - p < 2 )
{ int code = ref_stack_push(pstack, 2);
if ( code < 0 )
return code;
*ref_stack_index(pstack, 1) = *pkey;
p = pstack->p;
}
else
{ pstack->p = p += 2;
p[-1] = *pkey;
}
*p = *pvalue;
splist->count++;
#undef splist
return 0;
}
int
stack_param_list_write(stack_param_list *plist, ref_stack *pstack,
const ref *pwanted)
{ plist->procs = &ref_write_procs;
plist->u.w.write = stack_param_write;
ref_param_write_init((iparam_list *)plist, pwanted);
plist->pstack = pstack;
/* plist->skip not used */
plist->count = 0;
return 0;
}
/* Implementation for getting parameters to a dictionary. */
private int
dict_param_write(iparam_list *plist, const ref *pkey, ref *pvalue)
{ return dict_put(&((dict_param_list *)plist)->dict, pkey, pvalue);
}
int
dict_param_list_write(dict_param_list *plist, ref *pdict,
const ref *pwanted)
{ check_dict_write(*pdict);
plist->procs = &ref_write_procs;
plist->u.w.write = dict_param_write;
ref_param_write_init((iparam_list *)plist, pwanted);
plist->dict = *pdict;
return 0;
}
/* Generic routine for putting parameters converted from refs. */
private param_proc_xmit_null(ref_param_read_null);
private param_proc_xmit_bool(ref_param_read_bool);
private param_proc_xmit_int(ref_param_read_int);
private param_proc_xmit_long(ref_param_read_long);
private param_proc_xmit_float(ref_param_read_float);
private param_proc_xmit_string(ref_param_read_string);
private param_proc_xmit_int_array(ref_param_read_int_array);
private param_proc_xmit_float_array(ref_param_read_float_array);
private param_proc_get_policy(ref_param_read_get_policy);
private param_proc_signal_error(ref_param_read_signal_error);
private const gs_param_list_procs ref_read_procs = {
ref_param_read_null,
ref_param_read_bool,
ref_param_read_int,
ref_param_read_long,
ref_param_read_float,
ref_param_read_string,
ref_param_read_string, /* name = string */
ref_param_read_int_array,
ref_param_read_float_array,
ref_param_read_get_policy,
ref_param_read_signal_error
};
private int near ref_param_read(P3(iparam_list *, gs_param_name,
iparam_loc *));
#define iparam_return_error(loc, code)\
return_error(*(loc).presult = code)
#define iparam_check_type(loc, typ)\
if ( !r_has_type((loc).pvalue, typ) )\
iparam_return_error(loc, e_typecheck)
#define iparam_check_read(loc)\
if ( !r_has_attr((loc).pvalue, a_read) )\
iparam_return_error(loc, e_invalidaccess)
private int
ref_param_read_null(gs_param_list *plist, gs_param_name pkey)
{ iparam_loc loc;
int code = ref_param_read(iplist, pkey, &loc);
if ( code != 0 )
return code;
iparam_check_type(loc, t_null);
return 0;
}
private int
ref_param_read_bool(gs_param_list *plist, gs_param_name pkey, bool *pvalue)
{ iparam_loc loc;
int code = ref_param_read(iplist, pkey, &loc);
if ( code != 0 )
return code;
iparam_check_type(loc, t_boolean);
*pvalue = loc.pvalue->value.boolval;
return 0;
}
private int
ref_param_read_int(gs_param_list *plist, gs_param_name pkey, int *pvalue)
{ iparam_loc loc;
int code = ref_param_read(iplist, pkey, &loc);
if ( code != 0 )
return code;
iparam_check_type(loc, t_integer);
#if arch_sizeof_int < arch_sizeof_long
if ( loc.pvalue->value.intval != (int)loc.pvalue->value.intval )
return_error(e_rangecheck);
#endif
*pvalue = (int)loc.pvalue->value.intval;
return 0;
}
private int
ref_param_read_long(gs_param_list *plist, gs_param_name pkey, long *pvalue)
{ iparam_loc loc;
int code = ref_param_read(iplist, pkey, &loc);
if ( code != 0 )
return code;
iparam_check_type(loc, t_integer);
*pvalue = loc.pvalue->value.intval;
return 0;
}
private int
ref_param_read_float(gs_param_list *plist, gs_param_name pkey, float *pvalue)
{ iparam_loc loc;
int code = ref_param_read(iplist, pkey, &loc);
if ( code != 0 )
return code;
switch ( r_type(loc.pvalue) )
{
case t_integer:
*pvalue = loc.pvalue->value.intval;
break;
case t_real:
*pvalue = loc.pvalue->value.realval;
break;
default:
iparam_return_error(loc, e_typecheck);
}
return 0;
}
private int
ref_param_read_string(gs_param_list *plist, gs_param_name pkey,
gs_param_string *pvalue)
{ iparam_loc loc;
ref nref;
int code = ref_param_read(iplist, pkey, &loc);
if ( code != 0 )
return code;
switch ( r_type(loc.pvalue) )
{
case t_name:
name_string_ref(loc.pvalue, &nref);
loc.pvalue = &nref;
pvalue->persistent = true;
goto s;
case t_string:
iparam_check_read(loc);
pvalue->persistent = false;
s: pvalue->data = loc.pvalue->value.const_bytes;
pvalue->size = r_size(loc.pvalue);
break;
default:
iparam_return_error(loc, e_typecheck);
}
return 0;
}
private int
ref_param_read_int_array(gs_param_list *plist, gs_param_name pkey,
gs_param_int_array *pvalue)
{ iparam_loc loc;
int code = ref_param_read(iplist, pkey, &loc);
int *piv;
uint size;
uint i;
if ( code != 0 )
return code;
if ( !r_is_array(loc.pvalue) )
iparam_return_error(loc, e_typecheck);
iparam_check_read(loc);
size = r_size(loc.pvalue);
piv = (int *)ialloc_byte_array(size, sizeof(int),
"ref_read_int_array_param");
if ( piv == 0 )
return_error(e_VMerror);
for ( i = 0; i < size; i++ )
{ const ref *pe = loc.pvalue->value.const_refs + i;
if ( !r_has_type(pe, t_integer) )
iparam_return_error(loc, e_typecheck);
#if arch_sizeof_int < arch_sizeof_long
if ( pe->value.intval != (int)pe->value.intval )
iparam_return_error(loc, e_rangecheck);
#endif
piv[i] = (int)pe->value.intval;
}
pvalue->data = piv;
pvalue->size = size;
pvalue->persistent = true;
return 0;
}
private int
ref_param_read_float_array(gs_param_list *plist, gs_param_name pkey,
gs_param_float_array *pvalue)
{ iparam_loc loc;
int code = ref_param_read(iplist, pkey, &loc);
float *pfv;
uint size;
if ( code != 0 )
return code;
if ( !r_is_array(loc.pvalue) )
return_error(e_typecheck);
iparam_check_read(loc);
size = r_size(loc.pvalue);
pfv = (float *)ialloc_byte_array(size, sizeof(float),
"ref_read_float_array_param");
if ( pfv == 0 )
return_error(e_VMerror);
code = num_params(loc.pvalue->value.const_refs + size - 1, size, pfv);
if ( code < 0 )
{ ifree_object(pfv, "ref_read_float_array_param");
return (*loc.presult = code);
}
pvalue->data = pfv;
pvalue->size = size;
pvalue->persistent = true;
return 0;
}
private int
ref_param_read_get_policy(gs_param_list *plist, gs_param_name pkey)
{ ref *pvalue;
if ( !(r_has_type(&iplist->u.r.policies, t_dictionary) &&
dict_find_string(&iplist->u.r.policies, pkey, &pvalue) <= 0 &&
r_has_type(pvalue, t_integer))
)
return gs_policy_ignore;
return (int)pvalue->value.intval;
}
private int
ref_param_read_signal_error(gs_param_list *plist, gs_param_name pkey, int code)
{ iparam_loc loc;
ref_param_read(iplist, pkey, &loc); /* can't fail */
*loc.presult = code;
switch ( ref_param_read_get_policy(plist, pkey) )
{
case gs_policy_ignore:
return 0;
case gs_policy_consult_user:
return_error(e_configurationerror);
default:
return code;
}
}
/* Generic routine for reading a ref parameter. */
private int near
ref_param_read(iparam_list *plist, gs_param_name pkey, iparam_loc *ploc)
{ ref nref;
int code = name_ref((const byte *)pkey, strlen(pkey), &nref, 0);
if ( code < 0 )
return code;
return (*plist->u.r.read)(iplist, &nref, ploc);
}
/* Initialize for reading parameters. */
private int
ref_param_read_init(iparam_list *plist, uint count, const ref *ppolicies)
{ if ( ppolicies == 0 )
make_null(&plist->u.r.policies);
else
plist->u.r.policies = *ppolicies;
plist->count = count;
plist->results =
(int *)ialloc_byte_array(count, sizeof(int), "ref_param_read_init");
if ( plist->results == 0 )
return_error(e_VMerror);
memset(plist->results, 0, count * sizeof(int));
return 0;
}
/* Implementation for putting parameters from an array. */
private int
array_param_read(iparam_list *plist, const ref *pkey, iparam_loc *ploc)
{ ref *bot = ((array_param_list *)plist)->bot;
ref *ptr = bot;
ref *top = ((array_param_list *)plist)->top;
for ( ; ptr < top; ptr += 2 )
{ if ( r_has_type(ptr, t_name) && name_eq(ptr, pkey) )
{ ploc->pvalue = ptr + 1;
ploc->presult = &plist->results[ptr - bot];
*ploc->presult = 1;
return 0;
}
}
return 1;
}
int
array_param_list_read(array_param_list *plist, ref *bot, uint count,
const ref *ppolicies)
{ if ( count & 1 )
return_error(e_rangecheck);
plist->procs = &ref_read_procs;
plist->u.r.read = array_param_read;
plist->bot = bot;
plist->top = bot + count;
return ref_param_read_init(iplist, count, ppolicies);
}
/* Implementation for putting parameters from a stack. */
private int
stack_param_read(iparam_list *plist, const ref *pkey, iparam_loc *ploc)
{
#define splist ((stack_param_list *)plist)
ref_stack *pstack = splist->pstack;
/* This implementation is slow, but it probably doesn't matter. */
uint index = splist->skip + 1;
uint count = splist->count;
for ( ; count; count--, index += 2 )
{ const ref *p = ref_stack_index(pstack, index);
if ( r_has_type(p, t_name) && name_eq(p, pkey) )
{ ploc->pvalue = ref_stack_index(pstack, index - 1);
ploc->presult = &plist->results[count - 1];
*ploc->presult = 1;
return 0;
}
}
#undef splist
return 1;
}
int
stack_param_list_read(stack_param_list *plist, ref_stack *pstack, uint skip,
const ref *ppolicies)
{ uint count = ref_stack_counttomark(pstack);
if ( count == 0 )
return_error(e_unmatchedmark);
count -= skip + 1;
if ( count & 1 )
return_error(e_rangecheck);
plist->procs = &ref_read_procs;
plist->u.r.read = stack_param_read;
plist->pstack = pstack;
plist->skip = skip;
return ref_param_read_init(iplist, count >> 1, ppolicies);
}
/* Implementation for putting parameters from a dictionary. */
private int
dict_param_read(iparam_list *plist, const ref *pkey, iparam_loc *ploc)
{
#define spdict (&((dict_param_list *)plist)->dict)
int code = dict_find(spdict, pkey, &ploc->pvalue);
if ( code != 1 )
return 1;
ploc->presult = &plist->results[dict_value_index(spdict, ploc->pvalue)];
#undef spdict
*ploc->presult = 1;
return 0;
}
int
dict_param_list_read(dict_param_list *plist, const ref *pdict,
const ref *ppolicies)
{ check_dict_read(*pdict);
plist->procs = &ref_read_procs;
plist->u.r.read = dict_param_read;
plist->dict = *pdict;
return ref_param_read_init(iplist, dict_maxlength(pdict), ppolicies);
}