home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
PC World Komputer 1996 February
/
PCWK0296.iso
/
sharewar
/
dos
/
program
/
gs300sr1
/
gs300sr1.exe
/
ZDEVICE.C
< prev
next >
Wrap
C/C++ Source or Header
|
1994-07-27
|
9KB
|
298 lines
/* 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 "gsmatrix.h"
#include "gsstate.h"
#include "gxdevice.h"
#include "store.h"
/* <device> copydevice <newdevice> */
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, ialloc_local_attr(idmemory), pdevice, new_dev);
return 0;
}
/* <device> <y> <string> copyscanlines <substring> */
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_typecheck); /* not a memory device */
*op2 = *op;
r_set_size(op2, bytes_copied);
pop(2);
return 0;
}
/* - currentdevice <device> */
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 ? 0 : imemory_local_attr(mem)),
pdevice, dev);
return 0;
}
/* <device> <matrix> deviceinitialmatrix <matrix> */
int
zdeviceinitialmatrix(register os_ptr op)
{ gs_matrix mat;
int code;
check_type(op[-1], t_device);
gs_deviceinitialmatrix(op[-1].value.pdevice, &mat);
code = write_matrix(op, &mat);
if ( code < 0 )
return code;
op[-1] = *op;
pop(1);
return 0;
}
/* - flushpage - */
int
zflushpage(register os_ptr op)
{ return gs_flushpage(igs);
}
/* <int> .getdevice <device> */
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, a_foreign, pdevice, dev);
return 0;
}
/* <device> <key_dict|null> .getdeviceparams <mark> <name> <value> ... */
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;
}
/* <matrix> <width> <height> <palette> makeimagedevice <device> */
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_local_attr(iimemory),
pdevice, new_dev);
pop(3);
}
return code;
}
/* - nulldevice - */
int
znulldevice(register os_ptr op)
{ gs_nulldevice(igs);
return 0;
}
/* <num_copies> <flush_bool> .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;
}
/* <device> <policy_dict|null> <mark> <name> <value> ... .putdeviceparams */
/* (on success) <device> <eraseflag> */
/* (on failure) <device> <policy_dict|null> <mark> <name1> <error1> ... */
/* For a name that simply was not recognized, the error will be /undefined. */
int
zputdeviceparams(os_ptr op)
{ uint count = ref_stack_counttomark(&o_stack);
ref *pdev;
ref *ppolicy;
gx_device *dev;
stack_param_list list;
int code;
int old_width, old_height;
int i, dest;
int ecode = 0;
if ( count == 0 )
return_error(e_unmatchedmark);
ppolicy = ref_stack_index(&o_stack, count);
pdev = ref_stack_index(&o_stack, count + 1);
if ( pdev == 0 )
return_error(e_stackunderflow);
check_type(*pdev, t_device);
dev = pdev->value.pdevice;
code = stack_param_list_read(&list, &o_stack, 0, ppolicy);
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] != 1 )
{ ecode = list.results[i];
if ( ecode == 0 )
ecode = e_undefined;
*ref_stack_index(&o_stack, dest) =
*ref_stack_index(&o_stack, count - (i << 1) - 2);
gs_errorname(ecode, ref_stack_index(&o_stack, dest - 1));
dest -= 2;
}
if ( ecode != 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);
make_bool(osp, code);
return 0;
}
/* <device> .setdevice <eraseflag> */
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 )
make_bool(op, 1);
return code;
}
/* ------ Initialization procedure ------ */
op_def zdevice_op_defs[] = {
{"1copydevice", zcopydevice},
{"3copyscanlines", zcopyscanlines},
{"0currentdevice", zcurrentdevice},
{"2deviceinitialmatrix", zdeviceinitialmatrix},
{"0flushpage", zflushpage},
{"1.getdevice", zgetdevice},
{"2.getdeviceparams", zgetdeviceparams},
{"4makeimagedevice", zmakeimagedevice},
{"0nulldevice", znulldevice},
{"2.outputpage", zoutputpage},
{"3.putdeviceparams", zputdeviceparams},
{"1.setdevice", zsetdevice},
op_def_end(0)
};