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
|
/* 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.
*/
/* Support for printer devices with planar buffering. */
#include "gdevprn.h"
#include "gdevmpla.h"
#include "gdevppla.h"
#include "gxdevsop.h"
static int
prn_planar_dev_spec_op(gx_device *pdev, int dev_spec_op,
void *data, int size)
{
if (dev_spec_op == gxdso_is_native_planar)
return 1;
return gx_default_dev_spec_op(pdev, dev_spec_op, data, size);
}
/* Set the buf_procs in a printer device to planar mode. */
int
gdev_prn_set_procs_planar(gx_device *dev)
{
gx_device_printer * const pdev = (gx_device_printer *)dev;
pdev->printer_procs.buf_procs.create_buf_device =
gdev_prn_create_buf_planar;
pdev->printer_procs.buf_procs.size_buf_device =
gdev_prn_size_buf_planar;
if (dev_proc(pdev, dev_spec_op) == gx_default_dev_spec_op)
set_dev_proc(pdev, dev_spec_op, prn_planar_dev_spec_op);
return 0;
}
/* Open a printer device, conditionally setting it to be planar. */
int
gdev_prn_open_planar(gx_device *dev, bool upb)
{
if (upb)
gdev_prn_set_procs_planar(dev);
return gdev_prn_open(dev);
}
/* Augment get/put_params to add UsePlanarBuffer. */
int
gdev_prn_get_params_planar(gx_device * pdev, gs_param_list * plist,
bool *pupb)
{
int ecode = gdev_prn_get_params(pdev, plist);
if (ecode < 0)
return ecode;
return param_write_bool(plist, "UsePlanarBuffer", pupb);
}
int
gdev_prn_put_params_planar(gx_device * pdev, gs_param_list * plist,
bool *pupb)
{
bool upb = *pupb;
int ecode = 0, code;
if (pdev->color_info.num_components > 1)
ecode = param_read_bool(plist, "UsePlanarBuffer", &upb);
code = gdev_prn_put_params(pdev, plist);
if (ecode >= 0)
ecode = code;
if (ecode >= 0)
*pupb = upb;
return ecode;
}
/* Set the buffer device to planar mode. */
static int
gdev_prn_set_planar(gx_device_memory *mdev, const gx_device *tdev)
{
int num_comp = tdev->color_info.num_components;
gx_render_plane_t planes[GX_DEVICE_COLOR_MAX_COMPONENTS];
int depth = tdev->color_info.depth / num_comp;
int k;
if (num_comp < 1 || num_comp > GX_DEVICE_COLOR_MAX_COMPONENTS)
return_error(gs_error_rangecheck);
/* Round up the depth per plane to a power of 2. */
while (depth & (depth - 1))
--depth, depth = (depth | (depth >> 1)) + 1;
/* We want the most significant plane to come out first. */
planes[num_comp-1].shift = 0;
planes[num_comp-1].depth = depth;
for (k = (num_comp - 2); k >= 0; k--) {
planes[k].depth = depth;
planes[k].shift = planes[k + 1].shift + depth;
}
return gdev_mem_set_planar(mdev, num_comp, planes);
}
/* Create a planar buffer device. */
int
gdev_prn_create_buf_planar(gx_device **pbdev, gx_device *target, int y,
const gx_render_plane_t *render_plane,
gs_memory_t *mem, gx_band_complexity_t *for_band)
{
int code = gx_default_create_buf_device(pbdev, target, y, render_plane, mem,
for_band);
if (code < 0)
return code;
if (gs_device_is_memory(*pbdev) /* == render_plane->index < 0 */) {
code = gdev_prn_set_planar((gx_device_memory *)*pbdev, *pbdev);
}
return code;
}
/* Determine the space needed by a planar buffer device. */
int
gdev_prn_size_buf_planar(gx_device_buf_space_t *space, gx_device *target,
const gx_render_plane_t *render_plane,
int height, bool for_band)
{
gx_device_memory mdev;
int code;
if (render_plane && render_plane->index >= 0)
return gx_default_size_buf_device(space, target, render_plane,
height, for_band);
mdev.color_info = target->color_info;
code = gdev_prn_set_planar(&mdev, target);
if (code < 0)
return code;
if (gdev_mem_bits_size(&mdev, target->width, height, &(space->bits)) < 0)
return_error(gs_error_VMerror);
space->line_ptrs = gdev_mem_line_ptrs_size(&mdev, target->width, height);
space->raster = bitmap_raster(target->width * mdev.planes[0].depth);
return 0;
}
|