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 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368
|
/* Copyright (C) 2001-2021 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., 1305 Grant Avenue - Suite 200, Novato,
CA 94945, U.S.A., +1(415)492-9861, for further information.
*/
/* Default image rendering state structure */
/* Requires gxcpath.h, gxdevmem.h, gxdcolor.h, gzpath.h */
#ifndef gximage_INCLUDED
# define gximage_INCLUDED
#include "gsiparam.h"
#include "gxcspace.h"
#include "strimpl.h" /* for sisparam.h */
#include "sisparam.h"
#include "gxdda.h"
#include "gxiclass.h"
#include "gxiparam.h"
#include "gxsample.h"
#include "gscms.h"
#include "gxcpath.h"
/* Define the abstract type for the image enumerator state. */
/*typedef struct gx_image_enum_s gx_image_enum;*/ /* in gxiclass.h */
/*
* Incoming samples may go through two different transformations:
*
* - For N-bit input samples with N <= 8, N-to-8-bit expansion
* may involve a lookup map. Currently this map is either an
* identity function or a subtraction from 1 (inversion).
*
* - The 8-bit or frac expanded sample may undergo decoding (a linear
* transformation) before being handed off to the color mapping
* machinery.
*
* If the decoding function's range is [0..1], we fold it into the
* expansion lookup; otherwise we must compute it separately.
* For speed, we distinguish 3 different cases of the decoding step:
*/
typedef enum {
sd_none, /* decoded during expansion */
sd_lookup, /* use lookup_decode table */
sd_compute /* compute using base and factor */
} sample_decoding;
struct sample_map_s {
sample_lookup_t table;
/*
* If an 8-bit fraction doesn't represent the decoded value
* accurately enough, but the samples have 4 bits or fewer,
* we precompute the decoded values into a table.
* Different entries are used depending on bits/sample:
* 1,8,12 bits/sample: 0,15
* 2 bits/sample: 0,5,10,15
* 4 bits/sample: all
*/
float decode_lookup[16];
#define decode_base decode_lookup[0]
#define decode_max decode_lookup[15]
/*
* In the worst case, we have to do the decoding on the fly.
* The value is base + sample * factor, where the sample is
* an 8-bit (unsigned) integer or a frac.
*/
float decode_factor;
sample_decoding decoding;
/*
* If decoding is sd_none for a non-mask image, we still need to know
* whether the table includes an inversion, so that we can transform
* mask values correctly.
*/
bool inverted;
};
/* Decode an 8-bit sample into a floating point color component. */
/* penum points to the gx_image_enum structure. */
#define decode_sample(sample_value, cc, i)\
switch ( penum->map[i].decoding )\
{\
case sd_none:\
cc.paint.values[i] = (sample_value) * (1.0f / 255.0f); /* faster than / */\
break;\
case sd_lookup: /* <= 4 significant bits */\
cc.paint.values[i] =\
penum->map[i].decode_lookup[(sample_value) >> 4];\
break;\
case sd_compute:\
cc.paint.values[i] =\
penum->map[i].decode_base + (sample_value) * penum->map[i].decode_factor;\
}
/* Decode a frac value similarly. */
#define decode_frac(frac_value, cc, i)\
cc.paint.values[i] =\
penum->map[i].decode_base + (frac_value) * penum->map[i].decode_factor
/* Decode a frac value, to our 16 bit frac form. */
#define DECODE_FRAC_FRAC(frac_value, frac_value_out, i)\
frac_value_out =\
gx_unit_frac(penum->map[i].decode_base + (frac_value) * penum->map[i].decode_factor)
/* Define the distinct postures of an image. */
/* Each posture includes its reflected variant. */
typedef enum {
image_portrait = 0, /* 0 or 180 degrees */
image_landscape, /* 90 or 270 degrees */
image_skewed /* any other transformation */
} image_posture;
/* Interpolation request allows for off, on or forced */
typedef enum {
interp_off = 0, /* no interpolation */
interp_on, /* interpolation requested, but not forced */
interp_force /* force interpolation */
} image_interp;
/*
* Define an entry in the image color table. For single-source-plane
* images, the table index is the sample value, and the key is not used;
* for multiple-plane (color) images, the table index is a hash of the key,
* which is the concatenation of the source pixel components.
* "Clue" = Color LookUp Entry (by analogy with CLUT).
*/
typedef struct gx_image_clue_s {
gx_device_color dev_color;
bits32 key;
} gx_image_clue;
typedef struct gx_image_color_cache_s {
bool *is_transparent;
byte *device_contone;
} gx_image_color_cache_t;
/* Main state structure */
typedef struct gx_device_rop_texture_s gx_device_rop_texture;
typedef struct gx_image_icc_setup_s {
bool need_decode; /* used in icc processing */
bool is_lab; /* used in icc processing */
bool must_halftone; /* used in icc processing */
bool has_transfer; /* used in icc processing */
} gx_image_icc_setup_t;
struct gx_image_enum_s {
gx_image_enum_common;
/* We really want the map structure to be long-aligned, */
/* so we choose shorter types for some flags. */
/* Following are set at structure initialization */
int Width; /* Full image width */
int Height; /* Full image height */
byte bps; /* bits per sample: 1, 2, 4, 8, 12 */
byte unpack_bps; /* bps for computing unpack proc, */
/* set to 8 if no unpacking */
byte log2_xbytes; /* log2(bytes per expanded sample): */
/* 0 if bps <= 8, log2(sizeof(frac)) */
/* if bps > 8 */
byte spp; /* samples per pixel */
gs_image_alpha_t alpha; /* Alpha from image structure */
struct mc_ {
uint values[GS_IMAGE_MAX_COMPONENTS * 2]; /* MaskColor values, */
/* always as ranges, guaranteed in range */
/* and in order (v0 <= v1) */
bits32 mask, test; /* (if spp > 1, bps <= 8) */
/* mask & test value for quick filtering */
bool exact; /* (if spp > 1, bps <= 8) */
/* if true, mask/test filter is exact */
} mask_color; /* (if ImageType 4) */
byte use_mask_color; /* true if color masking is being used */
/*byte num_planes; */ /* (in common part) */
byte spread; /* (spp if multi-plane, 1 if not) */
/* << log2_xbytes */
byte masked; /* 0 = [color]image, 1 = imagemask */
image_interp interpolate; /* interpolation: off, on, forced */
gs_matrix matrix; /* image space -> device space */
/* We send 3 rectangles, rect >= drect >= rrect */
struct r_ {
int x, y, w, h; /* subrectangle for which data is supplied */
} rect;
struct {
int x, y, w, h; /* subrectangle that actually needs to be decoded */
} drect;
struct {
int x, y, w, h; /* subrectangle that actually needs to be rendered */
} rrect;
fixed dst_height; /* Full height covered by the transformed image
* in the device space (for siscale.c only);
* assumes posture == image_portrait or
* image_landscape. */
fixed dst_width; /* Full width covered by the transformed image
* in the device space (for siscale.c only);
* assumes posture == image_portrait or
* image_landscape. */
gs_fixed_point x_extent, y_extent; /* extent of one row of rect */
SAMPLE_UNPACK_PROC((*unpack));
irender_proc((*render));
int (*skip_next_line)(gx_image_enum *penum, gx_device *dev);
const gs_gstate *pgs;
gs_color_space *pcs; /* color space of image */
byte *buffer; /* for expanding samples to a */
/* byte or frac */
uint buffer_size;
byte *line; /* buffer for an output scan line */
uint line_size;
uint line_width; /* width of line in device pixels */
image_posture posture;
byte use_rop; /* true if CombineWithColor requested */
byte clip_image; /* mask, see below */
/* Either we are clipping to a rectangle, in which case */
/* the individual x/y flags may be set, or we are clipping */
/* to a general region, in which case only clip_region */
/* is set. */
#define image_clip_xmin 1
#define image_clip_xmax 2
#define image_clip_ymin 4
#define image_clip_ymax 8
#define image_clip_region 0x10
byte slow_loop; /* true if must use slower loop */
/* (if needed) */
byte device_color; /* true if device color space and */
/* standard decoding */
gs_fixed_rect clip_outer; /* outer box of clip path */
gs_fixed_rect clip_inner; /* inner box of clip path */
gs_logical_operation_t log_op; /* logical operation */
fixed adjust; /* adjustment when rendering */
/* characters */
fixed dxx, dxy; /* fixed versions of matrix */
/* components (as needed) */
gx_device_clip *clip_dev; /* clipping device (if needed) */
gx_device_rop_texture *rop_dev; /* RasterOp device (if needed) */
stream_image_scale_state *scaler; /* scale state for Interpolate */
/* (if needed) */
/* Following are updated dynamically */
int y; /* next source y */
gs_int_point used; /* amount of data already used, if */
/* interrupted by error */
gs_fixed_point cur, prev; /* device x, y of current & */
/* previous row */
struct dd_ {
gx_dda_fixed_point row; /* DDA for row origin, has been */
/* advanced when render proc called */
gx_dda_fixed_point strip; /* row + rect.x */
gx_dda_fixed_point pixel0; /* DDA for first pixel to render, */
/* strip + used.x */
} dda;
int line_xy; /* x or y value at start of buffered line */
int xi_next; /* expected xci of next row */
/* (landscape only) */
gs_int_point xyi; /* integer origin of row */
/* (Interpolate only) */
int yi0; /* integer y of entire image origin. */
int yci, hci; /* integer y & h of row (portrait) */
int xci, wci; /* integer x & w of row (landscape) */
/* The maps are set at initialization. We put them here */
/* so that the scalars will have smaller offsets. */
sample_map map[GS_IMAGE_MAX_COMPONENTS];
/* Entries 0 and 255 of the following are set at initialization */
/* for monochrome images; other entries are updated dynamically. */
gx_image_clue *clues;
gx_device_color icolor0_val; /* This is used if clues is not used */
gx_device_color icolor1_val;
gx_device_color *icolor0;
gx_device_color *icolor1;
gsicc_link_t *icc_link; /* ICC link to avoid recreation with every line */
gx_image_color_cache_t *color_cache; /* A cache that is con-tone values */
byte *ht_buffer; /* A buffer to contain halftoned data */
int ht_stride;
int ht_offset_bits; /* An offset adjustement to allow aligned copies */
int ht_plane_height; /* Needed during the copy_planes operation */
byte *thresh_buffer; /* A buffer to hold threshold values for HT */
int thresh_stride;
gs_image_parent_t image_parent_type; /* Need to avoid threshold of type3 images */
ht_landscape_info_t ht_landscape;
gx_image_icc_setup_t icc_setup;
bool use_cie_range; /* Needed potentially if CS was PS CIE based */
void *tpr_state;
/* The following pointer is only used if the CAL library is
* in place, but it's too much makefile upheaval to only have
* it here in CAL builds, so we live with it in all builds. */
void *cal_ht; /* CAL halftone state pointer */
};
/* Enumerate the pointers in an image enumerator. */
#define gx_image_enum_do_ptrs(m)\
m(0,pgs) m(1,pcs) m(2,dev) m(3,buffer) m(4,line)\
m(5,clip_dev) m(6,rop_dev) m(7,scaler) m(8,icc_link)\
m(9,color_cache) m(10,ht_buffer) m(11,thresh_buffer) \
m(12,clues)
#define gx_image_enum_num_ptrs 13
#define private_st_gx_image_enum() /* in gsimage.c */\
gs_private_st_composite(st_gx_image_enum, gx_image_enum, "gx_image_enum",\
image_enum_enum_ptrs, image_enum_reloc_ptrs)
/* Compare two device colors for equality. */
/* We can special-case this for speed later if we care. */
#define dev_color_eq(devc1, devc2)\
gx_device_color_equal(&(devc1), &(devc2))
/*
* Scale a pair of mask_color values to match the scaling of each sample to
* a full byte, and complement and swap them if the map incorporates
* a Decode = [1 0] inversion.
*/
void gx_image_scale_mask_colors(gx_image_enum *penum,
int component_index);
/* Used by icc processing to detect decode cases */
bool gx_has_transfer(const gs_gstate *pgs, int num_comps);
/* Compute the image matrix combining the ImageMatrix with either the pmat or the pgs ctm */
/* Exported for use outside gx_image_enum_begin */
int gx_image_compute_mat(const gs_gstate *pgs, const gs_matrix *pmat, const gs_matrix *ImageMatrix,
gs_matrix_double *rmat);
/*
* Do common initialization for processing an ImageType 1 or 4 image.
* Allocate the enumerator and fill in the following members:
* rect
*/
int
gx_image_enum_alloc(const gs_image_common_t * pic,
const gs_int_rect * prect,
gs_memory_t * mem, gx_image_enum **ppenum);
/*
* Finish initialization for processing an ImageType 1 or 4 image.
* Assumes the following members of *penum are set in addition to those
* set by gx_image_enum_alloc:
* alpha, use_mask_color, mask_color (if use_mask_color is true),
* masked, adjust
*/
int
gx_image_enum_begin(gx_device * dev, const gs_gstate * pgs,
const gs_matrix *pmat, const gs_image_common_t * pic,
const gx_drawing_color * pdcolor,
const gx_clip_path * pcpath,
gs_memory_t * mem, gx_image_enum *penum);
/*
* Clear the relevant clues. Exported for use by image_render_*
* when ht_tile cache is invalidated.
*/
void
image_init_clues(gx_image_enum * penum, int bps, int spp);
/* Initialize and do all the required mapping to get the con-tone device
values right away */
int
image_init_color_cache(gx_image_enum * penum, int bps, int spp);
#endif /* gximage_INCLUDED */
|