/* Copyright (C) 1996, 1997, 1998 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. */ /* User and system parameter operators */ #include "memory_.h" #include "string_.h" #include "ghost.h" #include "oper.h" #include "gscdefs.h" #include "gsstruct.h" /* for gxht.h */ #include "gsfont.h" /* for user params */ #include "gxht.h" /* for user params */ #include "gsutil.h" #include "estack.h" #include "ialloc.h" /* for imemory for status */ #include "icontext.h" /* for set_user_params prototype */ #include "idict.h" #include "idparam.h" #include "iparam.h" #include "dstack.h" #include "iname.h" #include "iutil2.h" #include "store.h" /* The (global) font directory */ extern gs_font_dir *ifont_dir; /* in zfont.c */ /* Import the GC parameters from zvmem2.c. */ extern int set_vm_reclaim(P1(long)); extern int set_vm_threshold(P1(long)); /* Define an individual user or system parameter. */ /* Eventually this will be made public. */ #define param_def_common\ const char *pname typedef struct param_def_s { param_def_common; } param_def_t; typedef struct long_param_def_s { param_def_common; long min_value, max_value; long (*current)(P0()); int (*set)(P1(long)); } long_param_def_t; #if arch_sizeof_long > arch_sizeof_int # define MAX_UINT_PARAM max_uint #else # define MAX_UINT_PARAM max_long #endif typedef struct bool_param_def_s { param_def_common; bool (*current)(P0()); int (*set)(P1(bool)); } bool_param_def_t; typedef struct string_param_def_s { param_def_common; void (*current)(P1(gs_param_string *)); int (*set)(P1(gs_param_string *)); } string_param_def_t; /* Define a parameter set (user or system). */ typedef struct param_set_s { const long_param_def_t *long_defs; uint long_count; const bool_param_def_t *bool_defs; uint bool_count; const string_param_def_t *string_defs; uint string_count; } param_set; /* Forward references */ private int setparams(P2(gs_param_list *, const param_set *)); private int currentparams(P2(os_ptr, const param_set *)); private int currentparam1(P2(os_ptr, const param_set *)); /* ------ Passwords ------ */ /* .checkpassword <0|1|2> */ private int zcheckpassword(register os_ptr op) { ref params[2]; array_param_list list; gs_param_list *const plist = (gs_param_list *)&list; int result = 0; int code = name_ref((const byte *)"Password", 8, ¶ms[0], 0); password pass; if (code < 0) return code; params[1] = *op; array_param_list_read(&list, params, 2, NULL, false); if (dict_read_password(&pass, systemdict, "StartJobPassword") >= 0 && param_check_password(plist, &pass) == 0 ) result = 1; if (dict_read_password(&pass, systemdict, "SystemParamsPassword") >= 0 && param_check_password(plist, &pass) == 0 ) result = 2; iparam_list_release(&list); make_int(op, result); return 0; } /* ------ System parameters ------ */ /* Integer values */ private long current_BuildTime(void) { return gs_buildtime; } private long current_MaxFontCache(void) { return gs_currentcachesize(ifont_dir); } private int set_MaxFontCache(long val) { return gs_setcachesize(ifont_dir, (uint)(val < 0 ? 0 : val > max_uint ? max_uint : val)); } private long current_CurFontCache(void) { uint cstat[7]; gs_cachestatus(ifont_dir, cstat); return cstat[0]; } private long current_MaxGlobalVM(void) { gs_memory_gc_status_t stat; gs_memory_gc_status(iimemory_global, &stat); return stat.max_vm; } private int set_MaxGlobalVM(long val) { gs_memory_gc_status_t stat; gs_memory_gc_status(iimemory_global, &stat); stat.max_vm = max(val, 0); gs_memory_set_gc_status(iimemory_global, &stat); return 0; } private long current_Revision(void) { return gs_revision; } private const long_param_def_t system_long_params[] = { {"BuildTime", min_long, max_long, current_BuildTime, NULL}, {"MaxFontCache", 0, MAX_UINT_PARAM, current_MaxFontCache, set_MaxFontCache}, {"CurFontCache", 0, MAX_UINT_PARAM, current_CurFontCache, NULL}, {"Revision", min_long, max_long, current_Revision, NULL}, /* Extensions */ {"MaxGlobalVM", 0, max_long, current_MaxGlobalVM, set_MaxGlobalVM} }; /* Boolean values */ private bool current_ByteOrder(void) { return !arch_is_big_endian; } private const bool_param_def_t system_bool_params[] = { {"ByteOrder", current_ByteOrder, NULL} }; /* String values */ private void current_RealFormat(gs_param_string * pval) { #if arch_floats_are_IEEE static const char *const rfs = "IEEE"; #else static const char *const rfs = "not IEEE"; #endif pval->data = (const byte *)rfs; pval->size = strlen(rfs); pval->persistent = true; } private const string_param_def_t system_string_params[] = { {"RealFormat", current_RealFormat, NULL} }; /* The system parameter set */ private const param_set system_param_set = { system_long_params, countof(system_long_params), system_bool_params, countof(system_bool_params), system_string_params, countof(system_string_params) }; /* .setsystemparams - */ private int zsetsystemparams(register os_ptr op) { int code; dict_param_list list; gs_param_list *const plist = (gs_param_list *)&list; password pass; check_type(*op, t_dictionary); code = dict_param_list_read(&list, op, NULL, false); if (code < 0) return code; code = dict_read_password(&pass, systemdict, "SystemParamsPassword"); if (code < 0) return code; code = param_check_password(plist, &pass); if (code != 0) { if (code > 0) code = gs_note_error(e_invalidaccess); goto out; } code = param_read_password(plist, "StartJobPassword", &pass); switch (code) { default: /* invalid */ goto out; case 1: /* missing */ break; case 0: code = dict_write_password(&pass, systemdict, "StartJobPassword"); if (code < 0) goto out; } code = param_read_password(plist, "SystemParamsPassword", &pass); switch (code) { default: /* invalid */ goto out; case 1: /* missing */ break; case 0: code = dict_write_password(&pass, systemdict, "SystemParamsPassword"); if (code < 0) goto out; } code = setparams(plist, &system_param_set); out: iparam_list_release(&list); if (code < 0) return code; pop(1); return 0; } /* - .currentsystemparams ... */ private int zcurrentsystemparams(os_ptr op) { return currentparams(op, &system_param_set); } /* .getsystemparam */ private int zgetsystemparam(os_ptr op) { return currentparam1(op, &system_param_set); } /* ------ User parameters ------ */ /* Integer values */ private long current_JobTimeout(void) { return 0; } private int set_JobTimeout(long val) { return 0; } private long current_MaxFontItem(void) { return gs_currentcacheupper(ifont_dir); } private int set_MaxFontItem(long val) { return gs_setcacheupper(ifont_dir, val); } private long current_MinFontCompress(void) { return gs_currentcachelower(ifont_dir); } private int set_MinFontCompress(long val) { return gs_setcachelower(ifont_dir, val); } private long current_MaxOpStack(void) { return ref_stack_max_count(&o_stack); } private int set_MaxOpStack(long val) { return ref_stack_set_max_count(&o_stack, val); } private long current_MaxDictStack(void) { return ref_stack_max_count(&d_stack); } private int set_MaxDictStack(long val) { return ref_stack_set_max_count(&d_stack, val); } private long current_MaxExecStack(void) { return ref_stack_max_count(&e_stack); } private int set_MaxExecStack(long val) { return ref_stack_set_max_count(&e_stack, val); } private long current_MaxLocalVM(void) { gs_memory_gc_status_t stat; gs_memory_gc_status(iimemory_local, &stat); return stat.max_vm; } private int set_MaxLocalVM(long val) { gs_memory_gc_status_t stat; gs_memory_gc_status(iimemory_local, &stat); stat.max_vm = max(val, 0); gs_memory_set_gc_status(iimemory_local, &stat); return 0; } private long current_VMReclaim(void) { gs_memory_gc_status_t gstat, lstat; gs_memory_gc_status(iimemory_global, &gstat); gs_memory_gc_status(iimemory_local, &lstat); return (!gstat.enabled ? -2 : !lstat.enabled ? -1 : 0); } private long current_VMThreshold(void) { gs_memory_gc_status_t stat; gs_memory_gc_status(iimemory_local, &stat); return stat.vm_threshold; } private long current_WaitTimeout(void) { return 0; } private int set_WaitTimeout(long val) { return 0; } private long current_MinScreenLevels(void) { return gs_currentminscreenlevels(); } private int set_MinScreenLevels(long val) { gs_setminscreenlevels((uint) val); return 0; } private const long_param_def_t user_long_params[] = { {"JobTimeout", 0, MAX_UINT_PARAM, current_JobTimeout, set_JobTimeout}, {"MaxFontItem", 0, MAX_UINT_PARAM, current_MaxFontItem, set_MaxFontItem}, {"MinFontCompress", 0, MAX_UINT_PARAM, current_MinFontCompress, set_MinFontCompress}, {"MaxOpStack", 0, MAX_UINT_PARAM, current_MaxOpStack, set_MaxOpStack}, {"MaxDictStack", 0, MAX_UINT_PARAM, current_MaxDictStack, set_MaxDictStack}, {"MaxExecStack", 0, MAX_UINT_PARAM, current_MaxExecStack, set_MaxExecStack}, {"MaxLocalVM", 0, max_long, current_MaxLocalVM, set_MaxLocalVM}, {"VMReclaim", -2, 0, current_VMReclaim, set_vm_reclaim}, {"VMThreshold", -1, max_long, current_VMThreshold, set_vm_threshold}, {"WaitTimeout", 0, MAX_UINT_PARAM, current_WaitTimeout, set_WaitTimeout}, /* Extensions */ {"MinScreenLevels", 0, MAX_UINT_PARAM, current_MinScreenLevels, set_MinScreenLevels} }; /* Boolean values */ private bool current_AccurateScreens(void) { return gs_currentaccuratescreens(); } private int set_AccurateScreens(bool val) { gs_setaccuratescreens(val); return 0; } private const bool_param_def_t user_bool_params[] = { {"AccurateScreens", current_AccurateScreens, set_AccurateScreens} }; /* The user parameter set */ private const param_set user_param_set = { user_long_params, countof(user_long_params), user_bool_params, countof(user_bool_params), 0, 0 }; /* .setuserparams - */ /* We break this out for use when switching contexts. */ int set_user_params(const ref * op) { dict_param_list list; int code; check_type(*op, t_dictionary); code = dict_param_list_read(&list, op, NULL, false); if (code < 0) return code; code = setparams((gs_param_list *)&list, &user_param_set); iparam_list_release(&list); return code; } private int zsetuserparams(register os_ptr op) { int code = set_user_params(op); if (code >= 0) pop(1); return code; } /* - .currentuserparams ... */ private int zcurrentuserparams(os_ptr op) { return currentparams(op, &user_param_set); } /* .getuserparam */ private int zgetuserparam(os_ptr op) { return currentparam1(op, &user_param_set); } /* ------ Initialization procedure ------ */ const op_def zusparam_op_defs[] = { /* User and system parameters are accessible even in Level 1 */ /* (if this is a Level 2 system). */ {"0.currentsystemparams", zcurrentsystemparams}, {"0.currentuserparams", zcurrentuserparams}, {"1.getsystemparam", zgetsystemparam}, {"1.getuserparam", zgetuserparam}, {"1.setsystemparams", zsetsystemparams}, {"1.setuserparams", zsetuserparams}, /* The rest of the operators are defined only in Level 2. */ op_def_begin_level2(), {"1.checkpassword", zcheckpassword}, op_def_end(0) }; /* ------ Internal procedures ------ */ /* Set the values of a parameter set from a parameter list. */ /* We don't attempt to back out if anything fails. */ private int setparams(gs_param_list * plist, const param_set * pset) { int i, code; for (i = 0; i < pset->long_count; i++) { const long_param_def_t *pdef = &pset->long_defs[i]; long val; if (pdef->set == NULL) continue; code = param_read_long(plist, pdef->pname, &val); switch (code) { default: /* invalid */ return code; case 1: /* missing */ break; case 0: if (val < pdef->min_value || val > pdef->max_value) return_error(e_rangecheck); code = (*pdef->set)(val); if (code < 0) return code; } } for (i = 0; i < pset->bool_count; i++) { const bool_param_def_t *pdef = &pset->bool_defs[i]; bool val; if (pdef->set == NULL) continue; code = param_read_bool(plist, pdef->pname, &val); if (code == 0) code = (*pdef->set)(val); if (code < 0) return code; } /****** WE SHOULD DO STRINGS AND STRING ARRAYS, BUT WE DON'T YET ******/ return 0; } /* Get the current values of a parameter set to the stack. */ private bool pname_matches(const char *pname, const ref * psref) { return (psref == 0 || !bytes_compare((const byte *)pname, strlen(pname), psref->value.const_bytes, r_size(psref))); } private int current_param_list(os_ptr op, const param_set * pset, const ref * psref /*t_string */ ) { stack_param_list list; gs_param_list *const plist = (gs_param_list *)&list; int i; stack_param_list_write(&list, &o_stack, NULL); for (i = 0; i < pset->long_count; i++) { const char *pname = pset->long_defs[i].pname; if (pname_matches(pname, psref)) { long val = (*pset->long_defs[i].current)(); int code = param_write_long(plist, pname, &val); if (code < 0) return code; } } for (i = 0; i < pset->bool_count; i++) { const char *pname = pset->bool_defs[i].pname; if (pname_matches(pname, psref)) { bool val = (*pset->bool_defs[i].current)(); int code = param_write_bool(plist, pname, &val); if (code < 0) return code; } } for (i = 0; i < pset->string_count; i++) { const char *pname = pset->string_defs[i].pname; if (pname_matches(pname, psref)) { gs_param_string val; int code; (*pset->string_defs[i].current)(&val); code = param_write_string(plist, pname, &val); if (code < 0) return code; } } return 0; } /* Get the current values of a parameter set to the stack. */ private int currentparams(os_ptr op, const param_set * pset) { return current_param_list(op, pset, NULL); } /* Get the value of a single parameter to the stack, or signal an error. */ private int currentparam1(os_ptr op, const param_set * pset) { ref sref; int code; check_type(*op, t_name); check_ostack(2); name_string_ref((const ref *)op, &sref); code = current_param_list(op, pset, &sref); if (code < 0) return code; if (osp == op) return_error(e_undefined); /* We know osp == op + 2. */ ref_assign(op, op + 2); pop(2); return code; }