/* Copyright (C) 1989, 1992, 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. */ /* zdevice.c */ /* Device-related operators */ #include "ghost.h" #include "errors.h" #include "oper.h" #include "ialloc.h" #include "idict.h" #include "igstate.h" #include "iname.h" #include "interp.h" #include "iparam.h" #include "ivmspace.h" #include "gsmatrix.h" #include "gsstate.h" #include "gxdevice.h" #include "store.h" /* copydevice */ int zcopydevice(register os_ptr op) { gx_device *new_dev; int code; check_type(*op, t_device); code = gs_copydevice(&new_dev, op->value.pdevice, imemory); if ( code < 0 ) return code; new_dev->memory = imemory; make_tav(op, t_device, icurrent_space, pdevice, new_dev); return 0; } /* copyscanlines */ int zcopyscanlines(register os_ptr op) { os_ptr op1 = op - 1; os_ptr op2 = op - 2; gx_device *dev; int code; uint bytes_copied; check_type(*op2, t_device); dev = op2->value.pdevice; check_type(*op1, t_integer); if ( op1->value.intval < 0 || op1->value.intval > dev->height ) return_error(e_rangecheck); check_write_type(*op, t_string); code = gs_copyscanlines(dev, (int)op1->value.intval, op->value.bytes, r_size(op), NULL, &bytes_copied); if ( code < 0 ) return_error(e_rangecheck); /* not a memory device */ *op2 = *op; r_set_size(op2, bytes_copied); pop(2); return 0; } /* - currentdevice */ int zcurrentdevice(register os_ptr op) { gx_device *dev = gs_currentdevice(igs); gs_ref_memory_t *mem = (gs_ref_memory_t *)dev->memory; push(1); make_tav(op, t_device, (mem == 0 ? avm_foreign : imemory_space(mem)), pdevice, dev); return 0; } /* - flushpage - */ int zflushpage(register os_ptr op) { return gs_flushpage(igs); } /* .getdevice */ int zgetdevice(register os_ptr op) { gx_device *dev; check_type(*op, t_integer); if ( op->value.intval != (int)(op->value.intval) ) return_error(e_rangecheck); /* won't fit in an int */ dev = gs_getdevice((int)(op->value.intval)); if ( dev == 0 ) /* index out of range */ return_error(e_rangecheck); make_tav(op, t_device, avm_foreign, pdevice, dev); return 0; } /* .getdeviceparams ... */ int zgetdeviceparams(os_ptr op) { ref rkeys; gx_device *dev; stack_param_list list; int code; ref *pmark; check_type(op[-1], t_device); rkeys = *op; dev = op[-1].value.pdevice; pop(1); stack_param_list_write(&list, &o_stack, &rkeys); code = gs_getdeviceparams(dev, (gs_param_list *)&list); if ( code < 0 ) { /* We have to put back the top argument. */ if ( list.count > 0 ) ref_stack_pop(&o_stack, list.count * 2 - 1); else ref_stack_push(&o_stack, 1); *osp = rkeys; return code; } pmark = ref_stack_index(&o_stack, list.count * 2); make_mark(pmark); return 0; } /* makeimagedevice */ int zmakeimagedevice(register os_ptr op) { gs_matrix imat; gx_device *new_dev; const byte *colors; int colors_size; int code; check_int_leu(op[-2], max_uint >> 1); /* width */ check_int_leu(op[-1], max_uint >> 1); /* height */ if ( r_has_type(op, t_null) ) /* true color */ { colors = 0; colors_size = -24; /* 24-bit true color */ } else { check_type(*op, t_string); /* palette */ if ( r_size(op) > 3*256 ) return_error(e_rangecheck); colors = op->value.bytes; colors_size = r_size(op); } if ( (code = read_matrix(op - 3, &imat)) < 0 ) return code; /* Everything OK, create device */ code = gs_makeimagedevice(&new_dev, &imat, (int)op[-2].value.intval, (int)op[-1].value.intval, colors, colors_size, imemory); if ( code == 0 ) { new_dev->memory = imemory; make_tav(op - 3, t_device, imemory_space(iimemory), pdevice, new_dev); pop(3); } return code; } /* - nulldevice - */ int znulldevice(register os_ptr op) { gs_nulldevice(igs); make_null(&istate->pagedevice); return 0; } /* .outputpage - */ int zoutputpage(register os_ptr op) { int code; check_type(op[-1], t_integer); check_type(*op, t_boolean); code = gs_output_page(igs, (int)op[-1].value.intval, op->value.boolval); if ( code < 0 ) return code; pop(2); return 0; } /* ... */ /* .putdeviceparams */ /* (on success) */ /* (on failure) */ /* ... */ /* For a key that simply was not recognized, if require_all is true, */ /* the result will be an /undefined error; if require_all is false, */ /* the key will be ignored. */ int zputdeviceparams(os_ptr op) { uint count = ref_stack_counttomark(&o_stack); ref *prequire_all; ref *ppolicy; ref *pdev; gx_device *dev; stack_param_list list; int code; int old_width, old_height; int i, dest; if ( count == 0 ) return_error(e_unmatchedmark); prequire_all = ref_stack_index(&o_stack, count); ppolicy = ref_stack_index(&o_stack, count + 1); pdev = ref_stack_index(&o_stack, count + 2); if ( pdev == 0 ) return_error(e_stackunderflow); check_type_only(*prequire_all, t_boolean); check_type_only(*pdev, t_device); dev = pdev->value.pdevice; code = stack_param_list_read(&list, &o_stack, 0, ppolicy, prequire_all->value.boolval); if ( code < 0 ) return code; old_width = dev->width; old_height = dev->height; code = gs_putdeviceparams(dev, (gs_param_list *)&list); /* Check for names that were undefined or caused errors. */ for ( dest = count - 2, i = 0; i < count >> 1; i++ ) if ( list.results[i] < 0 ) { *ref_stack_index(&o_stack, dest) = *ref_stack_index(&o_stack, count - (i << 1) - 2); gs_errorname(list.results[i], ref_stack_index(&o_stack, dest - 1)); dest -= 2; } if ( code < 0 ) { /* There were errors reported. */ ref_stack_pop(&o_stack, dest + 1); return 0; } if ( code > 0 || (code == 0 && (dev->width != old_width || dev->height != old_height)) ) { /* The device was open and is now closed, */ /* or its dimensions have changed. */ /* If it was the current device, */ /* call setdevice to reinstall it and erase the page. */ /****** DOESN'T FIND ALL THE GSTATES THAT REFERENCE THE DEVICE. ******/ if ( gs_currentdevice(igs) == dev ) { bool was_open = dev->is_open; code = gs_setdevice_no_erase(igs, dev); /* If the device wasn't closed, */ /* setdevice won't erase the page. */ if ( was_open && code >= 0 ) code = 1; } } if ( code < 0 ) return code; ref_stack_pop(&o_stack, count + 1); make_bool(osp, code); make_null(&istate->pagedevice); return 0; } /* .setdevice */ int zsetdevice(register os_ptr op) { int code; check_type(*op, t_device); code = gs_setdevice_no_erase(igs, op->value.pdevice); if ( code < 0 ) return code; make_bool(op, 1); make_null(&istate->pagedevice); return code; } /* ------ Initialization procedure ------ */ BEGIN_OP_DEFS(zdevice_op_defs) { {"1copydevice", zcopydevice}, {"3copyscanlines", zcopyscanlines}, {"0currentdevice", zcurrentdevice}, {"0flushpage", zflushpage}, {"1.getdevice", zgetdevice}, {"2.getdeviceparams", zgetdeviceparams}, {"4makeimagedevice", zmakeimagedevice}, {"0nulldevice", znulldevice}, {"2.outputpage", zoutputpage}, {"3.putdeviceparams", zputdeviceparams}, {"1.setdevice", zsetdevice}, END_OP_DEFS(0) }