home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
PC World Komputer 1996 February
/
PCWK0296.iso
/
sharewar
/
dos
/
program
/
gs300sr1
/
gs300sr1.exe
/
GXPCMAP.C
< prev
next >
Wrap
C/C++ Source or Header
|
1994-07-27
|
15KB
|
458 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.
*/
/* gxpcmap.c */
/* Pattern color mapping for Ghostscript library */
#include "memory_.h"
#include "gx.h"
#include "gserrors.h"
#include "gsstruct.h"
#include "gsutil.h" /* for gs_next_ids */
#include "gxfixed.h"
#include "gsmatrix.h"
#include "gscspace.h" /* for gscolor2.h */
#include "gxcolor2.h"
#include "gxdevice.h"
#include "gxdevmem.h"
#include "gxpcolor.h"
#include "gzstate.h"
/* Define the size of the Pattern cache. These are adhoc.... */
/*** Big memory machines ***/
#define max_cached_patterns_LARGE 50
#define max_pattern_bits_LARGE 100000
/*** Small memory machines ***/
#define max_cached_patterns_SMALL 5
#define max_pattern_bits_SMALL 1000
#if arch_ints_are_short
const uint pattern_cache_default_max_tiles = max_cached_patterns_SMALL;
const ulong pattern_cache_default_max_bits = max_pattern_bits_SMALL;
#else
const uint pattern_cache_default_max_tiles = max_cached_patterns_LARGE;
const ulong pattern_cache_default_max_bits = max_pattern_bits_LARGE;
#endif
/* Define the structures for Pattern rendering and caching. */
private_st_color_tile();
private_st_color_tile_element();
private_st_pattern_cache();
private_st_device_pattern_accum();
/* ------ Pattern rendering ------ */
/* Device procedures */
private dev_proc_open_device(pattern_accum_open);
private dev_proc_close_device(pattern_accum_close);
private dev_proc_fill_rectangle(pattern_accum_fill_rectangle);
private dev_proc_copy_mono(pattern_accum_copy_mono);
private dev_proc_copy_color(pattern_accum_copy_color);
/* The device descriptor */
private const gx_device_pattern_accum gs_pattern_accum_device =
{ sizeof(gx_device_pattern_accum),
0, /* &gs_pattern_accum_device.std_procs */
"pattern accumulator",
0, 0, 72, 72, no_margins, dci_black_and_white,
dev_init_misc_open(true),
{ pattern_accum_open,
NULL,
NULL,
NULL,
pattern_accum_close,
NULL,
NULL,
pattern_accum_fill_rectangle,
NULL,
pattern_accum_copy_mono,
pattern_accum_copy_color,
NULL,
NULL,
NULL,
NULL,
NULL,
NULL,
NULL,
NULL,
NULL
},
0, /* target */
0, 0, 0, 0 /* bitmap_memory, bits, mask, instance */
};
#define padev ((gx_device_pattern_accum *)dev)
/* Allocate a pattern accumulator. */
gx_device_pattern_accum *
gx_pattern_accum_alloc(gs_memory_t *mem, client_name_t cname)
{ gx_device_pattern_accum *adev =
gs_alloc_struct(mem, gx_device_pattern_accum,
&st_device_pattern_accum, cname);
if ( adev == 0 )
return 0;
*adev = gs_pattern_accum_device;
adev->memory = mem;
gx_device_forward_fill_in_procs((gx_device_forward *)adev); /* (should only do once) */
return adev;
}
/* Initialize a pattern accumulator. */
/* Client must already have set instance and bitmap_memory. */
private int
pattern_accum_open(gx_device *dev)
{ const gs_pattern_instance *pinst = padev->instance;
gs_memory_t *mem = padev->bitmap_memory;
gx_device_memory *mask =
gs_alloc_struct(mem, gx_device_memory, &st_device_memory,
"pattern_accum_open(mask)");
gx_device_memory *bits = 0;
gx_device *target = gs_currentdevice(pinst->saved);
int width = pinst->size.x;
int height = pinst->size.y;
int code;
if ( mask == 0 )
return_error(gs_error_VMerror);
#define pdset(dev)\
(dev)->width = width, (dev)->height = height,\
(dev)->x_pixels_per_inch = target->x_pixels_per_inch,\
(dev)->y_pixels_per_inch = target->y_pixels_per_inch
pdset(padev);
padev->color_info = target->color_info;
gs_make_mem_mono_device(mask, mem);
pdset(mask);
mask->bitmap_memory = mem;
mask->base = 0;
code = (*dev_proc(mask, open_device))((gx_device *)mask);
if ( code >= 0 )
{ memset(mask->base, 0, mask->raster * mask->height);
switch ( pinst->template.PaintType )
{
case 2: /* uncolored */
padev->target = target;
break;
case 1: /* colored */
bits = gs_alloc_struct(mem, gx_device_memory,
&st_device_memory,
"pattern_accum_open(bits)");
if ( bits == 0 )
return_error(gs_error_VMerror);
padev->target = (gx_device *)bits;
gs_make_mem_device(bits, gdev_mem_device_for_bits(target->color_info.depth), mem, -1);
pdset(bits);
#undef pdset
bits->color_info = target->color_info;
bits->target = target;
bits->bitmap_memory = mem;
/* Forward the color mappers to the target. */
/* (Maybe the memory device should do this by */
/* default if there is a target?) */
set_dev_proc(bits, map_rgb_color,
gx_forward_map_rgb_color);
set_dev_proc(bits, map_color_rgb,
gx_forward_map_color_rgb);
set_dev_proc(bits, map_cmyk_color,
gx_forward_map_cmyk_color);
set_dev_proc(bits, map_rgb_alpha_color,
gx_forward_map_rgb_alpha_color);
code = (*dev_proc(bits, open_device))((gx_device *)bits);
}
}
if ( code < 0 )
{ if ( bits != 0 )
gs_free_object(mem, bits, "pattern_accum_open(bits)");
(*dev_proc(mask, close_device))((gx_device *)mask);
gs_free_object(mem, mask, "pattern_accum_open(mask)");
return code;
}
padev->mask = mask;
padev->bits = bits;
return code;
}
/* Close an accumulator. */
private int
pattern_accum_close(gx_device *dev)
{ gs_memory_t *mem = padev->bitmap_memory;
if ( padev->bits != 0 )
{ (*dev_proc(padev->bits, close_device))((gx_device *)padev->bits);
gs_free_object(mem, padev->bits, "pattern_accum_close(bits)");
padev->bits = 0;
}
(*dev_proc(padev->mask, close_device))((gx_device *)padev->mask);
gs_free_object(mem, padev->mask, "pattern_accum_close(mask)");
padev->mask = 0;
return 0;
}
/* Fill a rectangle */
private int
pattern_accum_fill_rectangle(gx_device *dev, int x, int y, int w, int h,
gx_color_index color)
{ if ( padev->bits )
(*dev_proc(padev->target, fill_rectangle))(padev->target,
x, y, w, h, color);
return (*dev_proc(padev->mask, fill_rectangle))((gx_device *)padev->mask,
x, y, w, h, (gx_color_index)1);
}
/* Copy a monochrome bitmap. */
private int
pattern_accum_copy_mono(gx_device *dev, const byte *data, int data_x,
int raster, gx_bitmap_id id, int x, int y, int w, int h,
gx_color_index color0, gx_color_index color1)
{ if ( padev->bits )
(*dev_proc(padev->target, copy_mono))(padev->target,
data, data_x, raster, id, x, y, w, h, color0, color1);
if ( color0 != gx_no_color_index )
color0 = 1;
if ( color1 != gx_no_color_index )
color1 = 1;
if ( color0 == 1 && color1 == 1 )
return (*dev_proc(padev->mask, fill_rectangle))((gx_device *)padev->mask,
x, y, w, h, (gx_color_index)1);
else
return (*dev_proc(padev->mask, copy_mono))((gx_device *)padev->mask,
data, data_x, raster, id, x, y, w, h, color0, color1);
}
/* Copy a color bitmap. */
private int
pattern_accum_copy_color(gx_device *dev, const byte *data, int data_x,
int raster, gx_bitmap_id id, int x, int y, int w, int h)
{ if ( padev->bits )
(*dev_proc(padev->target, copy_color))(padev->target,
data, data_x, raster, id, x, y, w, h);
return (*dev_proc(padev->mask, fill_rectangle))((gx_device *)padev->mask,
x, y, w, h, (gx_color_index)1);
}
#undef padev
/* ------ Color space implementation ------ */
/* Allocate a Pattern cache. */
gx_pattern_cache *
gx_pattern_alloc_cache(gs_memory_t *mem, uint num_tiles, ulong max_bits)
{ gx_pattern_cache *pcache =
gs_alloc_struct(mem, gx_pattern_cache, &st_pattern_cache,
"pattern_cache_alloc(struct)");
gx_color_tile *tiles =
gs_alloc_struct_array(mem, num_tiles, gx_color_tile,
&st_color_tile_element,
"pattern_cache_alloc(tiles)");
uint i;
if ( pcache == 0 || tiles == 0 )
{ gs_free_object(mem, tiles, "pattern_cache_alloc(tiles)");
gs_free_object(mem, pcache, "pattern_cache_alloc(struct)");
return 0;
}
pcache->memory = mem;
pcache->tiles = tiles;
pcache->num_tiles = num_tiles;
pcache->tiles_used = 0;
pcache->next = 0;
pcache->bits_used = 0;
pcache->max_bits = max_bits;
for ( i = 0; i < num_tiles; tiles++, i++ )
{ tiles->id = gx_no_bitmap_id;
/* Clear the pointers to pacify the GC. */
tiles->bits.data = 0;
tiles->mask.data = 0;
}
return pcache;
}
/* Ensure that a gstate has a Pattern cache. */
private int
ensure_pattern_cache(gs_state *pgs)
{ if ( pgs->pattern_cache == 0 )
{ gx_pattern_cache *pcache =
gx_pattern_alloc_cache(pgs->memory,
pattern_cache_default_max_tiles,
pattern_cache_default_max_bits);
if ( pcache == 0 )
return_error(gs_error_VMerror);
pgs->pattern_cache = pcache;
}
return 0;
}
/* Get and set the Pattern cache in a gstate. */
gx_pattern_cache *
gstate_pattern_cache(gs_state *pgs)
{ return pgs->pattern_cache;
}
void
gstate_set_pattern_cache(gs_state *pgs, gx_pattern_cache *pcache)
{ pgs->pattern_cache = pcache;
}
/* Free a Pattern cache entry. */
private void
gx_pattern_cache_free_entry(gx_pattern_cache *pcache, gx_color_tile *ctile)
{ if ( ctile->id != gx_no_bitmap_id )
{ if ( ctile->mask.data != 0 )
{ pcache->bits_used -=
(ulong)ctile->mask.raster * ctile->mask.size.y;
gs_free_object(pcache->memory, ctile->mask.data,
"free_pattern_cache_entry(mask data)");
}
if ( ctile->bits.data != 0 )
{ pcache->bits_used -=
(ulong)ctile->bits.raster * ctile->bits.size.y;
gs_free_object(pcache->memory, ctile->bits.data,
"free_pattern_cache_entry(bits data)");
}
ctile->id = gx_no_bitmap_id;
pcache->tiles_used--;
}
}
/* Add a Pattern cache entry. This is exported for the interpreter. */
private void make_bitmap(P3(gx_tile_bitmap *, const gx_device_memory *, gx_bitmap_id));
int
gx_pattern_cache_add_entry(gs_state *pgs, gx_device_pattern_accum *padev,
gx_color_tile **pctile)
{ const gx_device_memory *mbits = padev->bits;
const gx_device_memory *mmask = padev->mask;
gx_pattern_cache *pcache;
ulong used = 0;
gx_bitmap_id id = padev->instance->id;
gx_color_tile *ctile;
int code = ensure_pattern_cache(pgs);
if ( code < 0 )
return code;
pcache = pgs->pattern_cache;
/* Check whether the pattern completely fills its box. */
/* If so, we can avoid the expensive masking operations */
/* when using the pattern. */
{ int y;
for ( y = 0; y < mmask->height; y++ )
{ const byte *row = scan_line_base(mmask, y);
int w;
for ( w = mmask->width; w > 8; w -= 8 )
if ( *row++ != 0xff )
goto keep;
if ( (*row | (0xff >> w)) != 0xff )
goto keep;
}
/* We don't need a mask. */
mmask = 0;
keep: ;
}
if ( mbits != 0 )
used += gdev_mem_bitmap_size(mbits);
if ( mmask != 0 )
used += gdev_mem_bitmap_size(mmask);
ctile = &pcache->tiles[id % pcache->num_tiles];
gx_pattern_cache_free_entry(pcache, ctile);
while ( pcache->bits_used + used > pcache->max_bits &&
pcache->bits_used != 0 /* allow 1 oversized entry (?) */
)
{ pcache->next = (pcache->next + 1) % pcache->num_tiles;
gx_pattern_cache_free_entry(pcache, &pcache->tiles[pcache->next]);
}
ctile->id = id;
ctile->depth = padev->color_info.depth;
if ( mbits != 0 )
make_bitmap(&ctile->bits, mbits, gs_next_ids(1));
else
ctile->bits.data = 0;
if ( mmask != 0 )
make_bitmap(&ctile->mask, mmask, id);
else
ctile->mask.data = 0;
pcache->bits_used += used;
*pctile = ctile;
return 0;
}
private void
make_bitmap(register gx_tile_bitmap *pbm, const gx_device_memory *mdev,
gx_bitmap_id id)
{ pbm->data = mdev->base;
pbm->raster = mdev->raster;
pbm->rep_width = pbm->size.x = mdev->width;
pbm->rep_height = pbm->size.y = mdev->height;
pbm->id = id;
}
/* Remap a Pattern color. */
int
gx_remap_Pattern(const gs_client_color *pc, const gs_color_space *pcs,
gx_device_color *pdc, gs_state *pgs)
{ gs_pattern_instance *pinst = pc->pattern;
gs_memory_t *mem = pgs->memory;
gs_state *saved;
gx_device_pattern_accum adev;
gx_color_tile *ctile;
int code;
if ( pinst == 0 )
{ /* Null pattern */
color_set_pattern(pdc, 0);
pdc->mask = 0;
return 0;
}
if ( pinst->template.PaintType == 2 ) /* uncolored */
{ code = (*pcs->params.pattern.base_space.type->remap_color)
(pc, (const gs_color_space *)&pcs->params.pattern.base_space, pdc, pgs);
if ( code < 0 )
return code;
#define replace_dc_type(t, pt)\
if ( pdc->type == &t ) pdc->type = &pt
replace_dc_type(gx_dc_pure, gx_dc_pure_masked);
else
replace_dc_type(gx_dc_ht_binary, gx_dc_binary_masked);
else
replace_dc_type(gx_dc_ht_colored, gx_dc_colored_masked);
else
return_error(gs_error_unregistered);
#undef replace_dc_type
}
else
color_set_pattern(pdc, 0);
pdc->id = pinst->id;
pdc->mask = 0;
if ( gx_pattern_cache_lookup(pdc, pgs) )
return 0;
code = ensure_pattern_cache(pgs);
if ( code < 0 )
return code;
adev = gs_pattern_accum_device;
gx_device_forward_fill_in_procs((gx_device_forward *)&adev); /* (should only do once) */
adev.instance = pinst;
adev.bitmap_memory = mem;
code = (*dev_proc(&adev, open_device))((gx_device *)&adev);
if ( code < 0 )
return code;
saved = gs_gstate(pinst->saved);
if ( saved == 0 )
return_error(gs_error_VMerror);
if ( saved->pattern_cache == 0 )
saved->pattern_cache = pgs->pattern_cache;
gx_set_device_only(saved, (gx_device *)&adev);
code = (*pinst->template.PaintProc)(pc, saved);
gs_free_object(mem, saved, "gx_remap_Pattern(saved)");
if ( code < 0 )
{ (*dev_proc(&adev, close_device))((gx_device *)&adev);
return code;
}
code = gx_pattern_cache_add_entry(pgs, &adev, &ctile);
if ( code < 0 )
(*dev_proc(&adev, close_device))((gx_device *)&adev);
return code;
}