/* Copyright (C) 2001-2023 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., 39 Mesa Street, Suite 108A, San Francisco, CA 94129, USA, for further information. */ /* Painting procedures for Ghostscript library */ #include "math_.h" /* for fabs */ #include "gx.h" #include "gpcheck.h" #include "gserrors.h" #include "gsropt.h" /* for gxpaint.h */ #include "gxfixed.h" #include "gxmatrix.h" /* for gs_gstate */ #include "gspaint.h" #include "gspath.h" #include "gzpath.h" #include "gxpaint.h" #include "gxpcolor.h" /* for do_fill_stroke */ #include "gzstate.h" #include "gxdevice.h" #include "gxdevmem.h" #include "gzcpath.h" #include "gxhldevc.h" #include "gsutil.h" #include "gxscanc.h" #include "gxdevsop.h" #include "gsicc_cms.h" #include "gdevepo.h" #include "assert_.h" /* Define the nominal size for alpha buffers. */ #define abuf_nominal_SMALL 500 #define abuf_nominal_LARGE 2000 #if ARCH_SMALL_MEMORY # define abuf_nominal abuf_nominal_SMALL #else #ifdef DEBUG # define abuf_nominal\ (gs_debug_c('.') ? abuf_nominal_SMALL : abuf_nominal_LARGE) #else # define abuf_nominal abuf_nominal_LARGE #endif #endif /* Erase the page */ int gs_erasepage(gs_gstate * pgs) { /* * We can't just fill with device white; we must take the * transfer function into account. */ int code; if ((code = gs_gsave(pgs)) < 0) return code; if ((code = gs_setgray(pgs, 1.0)) >= 0) { /* Fill the page directly, ignoring clipping. */ code = gs_fillpage(pgs); } gs_grestore(pgs); return code; } /* Fill the page with the current color. */ int gs_fillpage(gs_gstate * pgs) { gx_device *dev = gs_currentdevice(pgs); int code; /* * No need to check for returning error code, * existing device will continue to operate as before. */ epo_check_and_install(dev); /* Deliberately use the terminal device here */ if (dev_proc(dev, get_color_mapping_procs) == gx_error_get_color_mapping_procs) { emprintf1(dev->memory, "\n *** Error: No get_color_mapping_procs for device: %s\n", dev->dname); return_error(gs_error_Fatal); } /* Processing a fill object operation, but this counts as "UNTOUCHED" */ gx_unset_dev_color(pgs); /* force update so we pick up the new tag */ gx_unset_alt_dev_color(pgs); dev_proc(pgs->device, set_graphics_type_tag)(pgs->device, GS_UNTOUCHED_TAG); code = gx_set_dev_color(pgs); if (code != 0) return code; code = (*dev_proc(dev, fillpage))(dev, pgs, gs_currentdevicecolor_inline(pgs)); if (code < 0) return code; /* If GrayDetection is set, make sure monitoring is enabled. */ if (dev->icc_struct != NULL && dev->icc_struct->graydetection && !dev->icc_struct->pageneutralcolor) { dev->icc_struct->pageneutralcolor = true; /* start detecting again */ code = gsicc_mcm_begin_monitor(pgs->icc_link_cache, dev); } if (code < 0) return code; return (*dev_proc(dev, sync_output)) (dev); } /* * Set up an alpha buffer for a stroke or fill operation. Return 0 * if no buffer could be allocated, 1 if a buffer was installed, * or the usual negative error code. * * The fill/stroke code sets up a clipping device if needed; however, * since we scale up all the path coordinates, we either need to scale up * the clipping region, or do clipping after, rather than before, * alpha buffering. Either of these is a little inconvenient, but * the former is less inconvenient. */ static int scale_paths(gs_gstate * pgs, int log2_scale_x, int log2_scale_y, bool do_path) { /* * Because of clip and clippath, any of path, clip_path, and view_clip * may be aliases for each other. The only reliable way to detect * this is by comparing the segments pointers. Note that we must * scale the non-segment parts of the paths even if the segments are * aliased. */ const gx_path_segments *seg_clip = (pgs->clip_path->path_valid ? pgs->clip_path->path.segments : 0); const gx_clip_rect_list *list_clip = pgs->clip_path->rect_list; const gx_path_segments *seg_view_clip; const gx_clip_rect_list *list_view_clip; const gx_path_segments *seg_effective_clip = (pgs->effective_clip_path->path_valid ? pgs->effective_clip_path->path.segments : 0); const gx_clip_rect_list *list_effective_clip = pgs->effective_clip_path->rect_list; gx_cpath_scale_exp2_shared(pgs->clip_path, log2_scale_x, log2_scale_y, false, false); if (pgs->view_clip != 0 && pgs->view_clip != pgs->clip_path) { seg_view_clip = (pgs->view_clip->path_valid ? pgs->view_clip->path.segments : 0); list_view_clip = pgs->view_clip->rect_list; gx_cpath_scale_exp2_shared(pgs->view_clip, log2_scale_x, log2_scale_y, list_view_clip == list_clip, seg_view_clip && seg_view_clip == seg_clip); } else seg_view_clip = 0, list_view_clip = 0; if (pgs->effective_clip_path != pgs->clip_path && pgs->effective_clip_path != pgs->view_clip ) gx_cpath_scale_exp2_shared(pgs->effective_clip_path, log2_scale_x, log2_scale_y, list_effective_clip == list_clip || list_effective_clip == list_view_clip, seg_effective_clip && (seg_effective_clip == seg_clip || seg_effective_clip == seg_view_clip)); if (do_path) { const gx_path_segments *seg_path = pgs->path->segments; gx_path_scale_exp2_shared(pgs->path, log2_scale_x, log2_scale_y, seg_path == seg_clip || seg_path == seg_view_clip || seg_path == seg_effective_clip); } return 0; } static void scale_dash_pattern(gs_gstate * pgs, double scale) { int i; for (i = 0; i < pgs->line_params.dash.pattern_size; ++i) pgs->line_params.dash.pattern[i] *= scale; pgs->line_params.dash.offset *= scale; pgs->line_params.dash.pattern_length *= scale; pgs->line_params.dash.init_dist_left *= scale; if (pgs->line_params.dot_length_absolute) pgs->line_params.dot_length *= scale; } /* Returns 0 for OK. Returns 1 for "OK, buffer needs releasing" Returns 2 for "Empty region" Returns -ve for error */ static int alpha_buffer_init(gs_gstate * pgs, fixed extra_x, fixed extra_y, int alpha_bits, bool devn) { gx_device *dev = gs_currentdevice_inline(pgs); int log2_alpha_bits = ilog2(alpha_bits); gs_fixed_rect bbox; gs_int_rect ibox; uint width, raster, band_space; uint height, height2; gs_log2_scale_point log2_scale; gs_memory_t *mem; gx_device_memory *mdev; log2_scale.x = log2_scale.y = log2_alpha_bits; gx_path_bbox(pgs->path, &bbox); ibox.p.x = fixed2int(bbox.p.x - extra_x) - 1; ibox.p.y = fixed2int(bbox.p.y - extra_y) - 1; ibox.q.x = fixed2int_ceiling(bbox.q.x + extra_x) + 1; ibox.q.y = fixed2int_ceiling(bbox.q.y + extra_y) + 1; (void)dev_proc(dev, dev_spec_op)(dev, gxdso_restrict_bbox, &ibox, sizeof(ibox)); if ((ibox.q.y <= ibox.p.y) || (ibox.q.x <= ibox.p.x)) return 2; width = (ibox.q.x - ibox.p.x) << log2_scale.x; raster = bitmap_raster(width); band_space = raster << log2_scale.y; height2 = (ibox.q.y - ibox.p.y); height = (abuf_nominal / band_space); if (height == 0) height = 1; if (height > height2) height = height2; height <<= log2_scale.y; /* We may have to update the marking parameters if we have a pdf14 device as our target. Need to do while dev is still active in pgs. Do this before allocating the device to simplify cleanup. */ if (dev_proc(dev, dev_spec_op)(dev, gxdso_is_pdf14_device, NULL, 0) > 0) { int code = gs_update_trans_marking_params(pgs); if (code < 0) return code; } mem = pgs->memory; mdev = gs_alloc_struct(mem, gx_device_memory, &st_device_memory, "alpha_buffer_init"); if (mdev == 0) return 0; /* if no room, don't buffer */ gs_make_mem_abuf_device(mdev, mem, dev, &log2_scale, alpha_bits, ibox.p.x << log2_scale.x, devn); mdev->width = width; mdev->height = height; mdev->bitmap_memory = mem; /* Set the horrible hacky flag that tells people that the width/height here * have been set for *our* convenience, rather than accurately depicting the * size of the device for callers. */ mdev->non_strict_bounds = 1; if ((*dev_proc(mdev, open_device)) ((gx_device *) mdev) < 0) { /* No room for bits, punt. */ gs_free_object(mem, mdev, "alpha_buffer_init"); return 0; } gx_set_device_only(pgs, (gx_device *) mdev); scale_paths(pgs, log2_scale.x, log2_scale.y, true); return 1; } /* Release an alpha buffer. */ static int alpha_buffer_release(gs_gstate * pgs, bool newpath) { gx_device_memory *mdev = (gx_device_memory *) gs_currentdevice_inline(pgs); int code = (*dev_proc(mdev, close_device)) ((gx_device *) mdev); if (code >= 0) scale_paths(pgs, -mdev->log2_scale.x, -mdev->log2_scale.y, !(newpath && !gx_path_is_shared(pgs->path))); /* Reference counting will free mdev. */ gx_set_device_only(pgs, mdev->target); return code; } /* Setup for black vector handling */ static inline bool black_vectors(gs_gstate *pgs, gx_device *dev) { if (dev->icc_struct != NULL && dev->icc_struct->blackvector && pgs->black_textvec_state == NULL) { return gsicc_setup_blacktextvec(pgs, dev, false); } return false; } static int do_fill(gs_gstate *pgs, int rule) { int code, abits, acode, rcode = 0; bool devn; bool black_vector = false; bool in_smask = (dev_proc(pgs->device, dev_spec_op)(pgs->device, gxdso_in_smask_construction, NULL, 0)) > 0; /* We need to distinguish text from vectors to set the object tag. To make that determination, we check for the show graphics state being stored in the current graphics state. This works even in the case of a glyph from a Type 3 Postscript/PDF font which has multiple, nested gsave/grestore pairs in the BuildGlyph/BuildChar procedure. Also, it works in the case of operating without a glyph cache or bypassing the cache because the glyph is too large or the cache being already full. Note that it doesn't work for a construction like: "(xyz) true charpath fill/stroke" where the show machinations have completed before we get to the fill operation. This has implications for how we handle PDF text rendering modes 1 and 2. To handle that, we'll have to add a flag to the path structure, or to the path segment structure (depending on how fine grained we require it to be). */ if (pgs->show_gstate == NULL && !in_smask) { ensure_tag_is_set(pgs, pgs->device, GS_VECTOR_TAG); /* NB: may unset_dev_color */ black_vector = black_vectors(pgs, pgs->device); /* Set vector fill to black */ } else ensure_tag_is_set(pgs, pgs->device, GS_TEXT_TAG); /* NB: may unset_dev_color */ code = gx_set_dev_color(pgs); if (code != 0) goto out; code = gs_gstate_color_load(pgs); if (code < 0) goto out; if (pgs->overprint || (!pgs->overprint && dev_proc(pgs->device, dev_spec_op)(pgs->device, gxdso_overprint_active, NULL, 0))) { gs_overprint_params_t op_params = { 0 }; if_debug0m(gs_debug_flag_overprint, pgs->memory, "[overprint] Fill Overprint\n"); code = gs_do_set_overprint(pgs); if (code < 0) goto out; op_params.op_state = OP_STATE_FILL; gs_gstate_update_overprint(pgs, &op_params); } abits = 0; { gx_device_color *col = gs_currentdevicecolor_inline(pgs); devn = color_is_devn(col); if (color_is_pure(col) || devn) abits = alpha_buffer_bits(pgs); } if (abits > 1) { acode = alpha_buffer_init(pgs, pgs->fill_adjust.x, pgs->fill_adjust.y, abits, devn); if (acode == 2) { /* Special case for no fill required */ code = 0; goto out; } if (acode < 0) { code = acode; goto out; } } else acode = 0; code = gx_fill_path(pgs->path, gs_currentdevicecolor_inline(pgs), pgs, rule, pgs->fill_adjust.x, pgs->fill_adjust.y); if (acode > 0) rcode = alpha_buffer_release(pgs, code >= 0); if (code >= 0 && rcode < 0) code = rcode; out: if (black_vector) { /* Restore color */ gsicc_restore_blacktextvec(pgs, false); } return code; } /* Fill the current path using a specified rule. */ static int fill_with_rule(gs_gstate * pgs, int rule) { int code; /* If we're inside a charpath, just merge the current path */ /* into the parent's path. */ if (pgs->in_charpath) code = gx_path_add_char_path(pgs->show_gstate->path, pgs->path, pgs->in_charpath); /* If we're rendering a glyph cached, the show machinery decides * whether to actually image it on the output or not, but uncached * will render directly to the output, so for text rendering * mode 3, we have to short circuit it here, but keep the * current point */ else if (gs_is_null_device(pgs->device) || (pgs->show_gstate && pgs->text_rendering_mode == 3 && pgs->in_cachedevice == CACHE_DEVICE_NOT_CACHING)) { /* Handle separately to prevent gs_gstate_color_load - bug 688308. */ gs_newpath(pgs); code = 0; } else { code = do_fill(pgs, rule); if (code >= 0) code = gs_newpath(pgs); } return code; } /* Fill using the winding number rule */ int gs_fill(gs_gstate * pgs) { pgs->device->sgr.stroke_stored = false; return fill_with_rule(pgs, gx_rule_winding_number); } /* Fill using the even/odd rule */ int gs_eofill(gs_gstate * pgs) { pgs->device->sgr.stroke_stored = false; return fill_with_rule(pgs, gx_rule_even_odd); } static int do_stroke(gs_gstate * pgs) { int code, abits, acode, rcode = 0; bool devn; bool is_fill_correct = true; bool black_vector = false; bool in_smask = (dev_proc(pgs->device, dev_spec_op)(pgs->device, gxdso_in_smask_construction, NULL, 0)) > 0; /* We need to distinguish text from vectors to set the object tag. To make that determination, we check for the show graphics state being stored in the current graphics state. This works even in the case of a glyph from a Type 3 Postscript/PDF font which has multiple, nested gsave/grestore pairs in the BuildGlyph/BuildChar procedure. Also, it works in the case of operating without a glyph cache or bypassing the cache because the glyph is too large or the cache being already full. Note that it doesn't work for a construction like: "(xyz) true charpath fill/stroke" where the show machinations have completed before we get to the fill operation. This has implications for how we handle PDF text rendering modes 1 and 2. To handle that, we'll have to add a flag to the path structure, or to the path segment structure (depending on how fine grained we require it to be). */ if (pgs->show_gstate == NULL && !in_smask) { ensure_tag_is_set(pgs, pgs->device, GS_VECTOR_TAG); /* NB: may unset_dev_color */ black_vector = black_vectors(pgs, pgs->device); } else ensure_tag_is_set(pgs, pgs->device, GS_TEXT_TAG); /* NB: may unset_dev_color */ code = gx_set_dev_color(pgs); if (code != 0) goto out; code = gs_gstate_color_load(pgs); if (code < 0) goto out; if (pgs->stroke_overprint || (!pgs->stroke_overprint && dev_proc(pgs->device, dev_spec_op)(pgs->device, gxdso_overprint_active, NULL, 0))) { gs_overprint_params_t op_params = { 0 }; /* PS2 does not have the concept of fill and stroke colors. Here we need to possibly correct for that in the graphic state during this operation */ if (pgs->is_fill_color) { is_fill_correct = false; pgs->is_fill_color = false; } if_debug0m(gs_debug_flag_overprint, pgs->memory, "[overprint] Stroke Overprint\n"); code = gs_do_set_overprint(pgs); if (code < 0) { if (!is_fill_correct) { pgs->is_fill_color = true; } goto out; } op_params.op_state = OP_STATE_STROKE; gs_gstate_update_overprint(pgs, &op_params); } abits = 0; { gx_device_color *col = gs_currentdevicecolor_inline(pgs); devn = color_is_devn(col); if (color_is_pure(col) || devn) abits = alpha_buffer_bits(pgs); } if (abits > 1) { /* * Expand the bounding box by the line width. * This is expensive to compute, so we only do it * if we know we're going to buffer. */ float xxyy = fabs(pgs->ctm.xx) + fabs(pgs->ctm.yy); float xyyx = fabs(pgs->ctm.xy) + fabs(pgs->ctm.yx); float scale = (float)(1 << (abits / 2)); float orig_width = gs_currentlinewidth(pgs); float new_width = orig_width * scale; fixed extra_adjust = float2fixed(max(xxyy, xyyx) * new_width / 2); float orig_flatness = gs_currentflat(pgs); gx_path spath; /* Scale up the line width, dash pattern, and flatness. */ if (extra_adjust < fixed_1) extra_adjust = fixed_1; acode = alpha_buffer_init(pgs, pgs->fill_adjust.x + extra_adjust, pgs->fill_adjust.y + extra_adjust, abits, devn); if (acode == 2) { /* Special code meaning no fill required */ if (!is_fill_correct) { pgs->is_fill_color = true; } code = 0; goto out; } if (acode < 0) { if (!is_fill_correct) { pgs->is_fill_color = true; } code = acode; goto out; } gs_setlinewidth(pgs, new_width); scale_dash_pattern(pgs, scale); gs_setflat(pgs, (double)(orig_flatness * scale)); /* * The alpha-buffer device requires that we fill the * entire path as a single unit. */ gx_path_init_local(&spath, pgs->memory); code = gx_stroke_add(pgs->path, &spath, pgs, false); gs_setlinewidth(pgs, orig_width); scale_dash_pattern(pgs, 1.0 / scale); if (code >= 0) code = gx_fill_path(&spath, gs_currentdevicecolor_inline(pgs), pgs, gx_rule_winding_number, pgs->fill_adjust.x, pgs->fill_adjust.y); gs_setflat(pgs, orig_flatness); gx_path_free(&spath, "gs_stroke"); if (acode > 0) rcode = alpha_buffer_release(pgs, code >= 0); } else code = gx_stroke_fill(pgs->path, pgs); if (code >= 0 && rcode < 0) code = rcode; if (!is_fill_correct) { pgs->is_fill_color = true; } out: if (black_vector) { /* Restore color */ gsicc_restore_blacktextvec(pgs, false); } return code; } /* Stroke the current path */ int gs_stroke(gs_gstate * pgs) { int code; /* * If we're inside a charpath, just merge the current path * into the parent's path. */ if (pgs->in_charpath) { if (pgs->in_charpath == cpm_true_charpath) { /* * A stroke inside a true charpath should do the * equivalent of strokepath. */ code = gs_strokepath(pgs); if (code < 0) return code; } code = gx_path_add_char_path(pgs->show_gstate->path, pgs->path, pgs->in_charpath); if (code < 0) return code; } if (!gs_is_null_device(pgs->device)) { code = do_stroke(pgs); if (code < 0) return code; } return gs_newpath(pgs); } /* Compute the stroked outline of the current path */ static int gs_strokepath_aux(gs_gstate * pgs, bool traditional) { gx_path spath; int code; gx_path_init_local(&spath, pgs->path->memory); code = gx_stroke_add(pgs->path, &spath, pgs, traditional); if (code < 0) { gx_path_free(&spath, "gs_strokepath"); return code; } pgs->device->sgr.stroke_stored = false; code = gx_path_assign_free(pgs->path, &spath); if (code < 0) return code; /* NB: needs testing with PCL */ if (gx_path_is_void(pgs->path)) pgs->current_point_valid = false; else { gx_setcurrentpoint(pgs, fixed2float(spath.position.x), fixed2float(spath.position.y)); } return 0; } int gs_strokepath(gs_gstate * pgs) { return gs_strokepath_aux(pgs, true); } int gs_strokepath2(gs_gstate * pgs) { return gs_strokepath_aux(pgs, false); } static int do_fill_stroke(gs_gstate *pgs, int rule, int *restart) { int code, abits, acode = 0, rcode = 0; bool devn; float orig_width, scale, orig_flatness; bool black_vector = false; bool in_smask = (dev_proc(pgs->device, dev_spec_op)(pgs->device, gxdso_in_smask_construction, NULL, 0)) > 0; gs_logical_operation_t orig_lop = pgs->log_op; /* It is either our first time, or the stroke was a pattern and we are coming back from the error if restart < 1 (0 is first time, 1 stroke is set, and we only need to finish out fill */ if (pgs->is_fill_color) { /* if the fill_color is a pattern, make sure the tile is locked so that */ /* it does not get evicted as the stroke color is loaded. */ if (gx_dc_is_pattern1_color(gs_currentdevicecolor_inline(pgs))) { gs_id id; if(gs_currentdevicecolor_inline(pgs)->colors.pattern.p_tile != NULL) { id = gs_currentdevicecolor_inline(pgs)->colors.pattern.p_tile->id; code = dev_proc(pgs->device, lock_pattern)(pgs->device, pgs, id, true); } else { code = 0; } if (code < 0) return code; /* lock failed -- tile not in cache? */ } gs_swapcolors_quick(pgs); /* switch to the stroke color */ } if (*restart < 1) { /* We need to distinguish text from vectors to set the object tag. To make that determination, we check for the show graphics state being stored in the current graphics state. This works even in the case of a glyph from a Type 3 Postscript/PDF font which has multiple, nested gsave/grestore pairs in the BuildGlyph/BuildChar procedure. Also, it works in the case of operating without a glyph cache or bypassing the cache because the glyph is too large or the cache being already full. Note that it doesn't work for a construction like: "(xyz) true charpath fill/stroke" where the show machinations have completed before we get to the fill operation. This has implications for how we handle PDF text rendering modes 1 and 2. To handle that, we'll have to add a flag to the path structure, or to the path segment structure (depending on how fine grained we require it to be). */ if (pgs->show_gstate == NULL && !in_smask) { ensure_tag_is_set(pgs, pgs->device, GS_VECTOR_TAG); /* NB: may unset_dev_color */ black_vector = black_vectors(pgs, pgs->device); } else ensure_tag_is_set(pgs, pgs->device, GS_TEXT_TAG); /* NB: may unset_dev_color */ /* if we are at restart == 0, we set the stroke color. */ code = gx_set_dev_color(pgs); if (code != 0) goto out2; /* may be gs_error_Remap_color or real error */ code = gs_gstate_color_load(pgs); if (code < 0) goto out2; /* If this was a pattern color, make sure and lock it in the pattern_cache */ if (gx_dc_is_pattern1_color(gs_currentdevicecolor_inline(pgs))) { gs_id id; if(gs_currentdevicecolor_inline(pgs)->colors.pattern.p_tile != NULL) { id = gs_currentdevicecolor_inline(pgs)->colors.pattern.p_tile->id; code = dev_proc(pgs->device, lock_pattern)(pgs->device, pgs, id, true); } else { code = 0; } if (code < 0) goto out2; /* lock failed -- tile not in cache? */ } } if (pgs->stroke_overprint || (!pgs->stroke_overprint && dev_proc(pgs->device, dev_spec_op)(pgs->device, gxdso_overprint_active, NULL, 0))) { if_debug0m(gs_debug_flag_overprint, pgs->memory, "[overprint] StrokeFill Stroke Set Overprint\n"); code = gs_do_set_overprint(pgs); if (code < 0) goto out2; } *restart = 1; /* finished, successfully with stroke_color */ gs_swapcolors_quick(pgs); /* switch to fill color */ /* Have to set the fill color too */ if (pgs->show_gstate == NULL) ensure_tag_is_set(pgs, pgs->device, GS_VECTOR_TAG); /* NB: may unset_dev_color */ else ensure_tag_is_set(pgs, pgs->device, GS_TEXT_TAG); /* NB: may unset_dev_color */ code = gx_set_dev_color(pgs); if (code != 0) { goto out2; } code = gs_gstate_color_load(pgs); if (code < 0) { /* color is set for fill, but a failure here is a problem */ /* i.e., something other than error_Remap_Color */ *restart = 2; /* we shouldn't re-enter with '2' */ goto out; } if (pgs->overprint || (!pgs->overprint && dev_proc(pgs->device, dev_spec_op)(pgs->device, gxdso_overprint_active, NULL, 0))) { if_debug0m(gs_debug_flag_overprint, pgs->memory, "[overprint] StrokeFill Fill Set Overprint\n"); code = gs_do_set_overprint(pgs); if (code < 0) goto out; /* fatal */ } abits = 0; { gx_device_color *col_fill = gs_currentdevicecolor_inline(pgs); gx_device_color *col_stroke = gs_swappeddevicecolor_inline(pgs); devn = color_is_devn(col_fill) && color_is_devn(col_stroke); /* could be devn and masked_devn */ if ((color_is_pure(col_fill) && color_is_pure(col_stroke)) || devn) abits = alpha_buffer_bits(pgs); } if (abits > 1) { /* * Expand the bounding box by the line width. * This is expensive to compute, so we only do it * if we know we're going to buffer. */ float new_width; fixed extra_adjust; float xxyy = fabs(pgs->ctm.xx) + fabs(pgs->ctm.yy); float xyyx = fabs(pgs->ctm.xy) + fabs(pgs->ctm.yx); pgs->log_op |= lop_pdf14; /* Force stroking to happen all in 1 go */ scale = (float)(1 << (abits / 2)); orig_width = gs_currentlinewidth(pgs); new_width = orig_width * scale; extra_adjust = float2fixed(max(xxyy, xyyx) * new_width / 2); orig_flatness = gs_currentflat(pgs); /* Scale up the line width, dash pattern, and flatness. */ if (extra_adjust < fixed_1) extra_adjust = fixed_1; acode = alpha_buffer_init(pgs, pgs->fill_adjust.x + extra_adjust, pgs->fill_adjust.y + extra_adjust, abits, devn); if (acode == 2) /* Special case for no fill required */ goto out; if (acode < 0) goto out; gs_setlinewidth(pgs, new_width); scale_dash_pattern(pgs, scale); gs_setflat(pgs, (double)(orig_flatness * scale)); } else acode = 0; code = gx_fill_stroke_path(pgs, rule); if (abits > 1) { gs_setlinewidth(pgs, orig_width); scale_dash_pattern(pgs, 1.0 / scale); gs_setflat(pgs, orig_flatness); acode = alpha_buffer_release(pgs, code >= 0); pgs->log_op = orig_lop; } if (pgs->is_fill_color) { /* The color _should_ be the fill color, so make sure it is unlocked */ if (gx_dc_is_pattern1_color(gs_currentdevicecolor_inline(pgs))) { gs_id id; if(gs_currentdevicecolor_inline(pgs)->colors.pattern.p_tile != NULL) { id = gs_currentdevicecolor_inline(pgs)->colors.pattern.p_tile->id; code = dev_proc(pgs->device, lock_pattern)(pgs->device, pgs, id, false); } else { code = 0; } if (code < 0) goto out2; /* lock failed -- tile not in cache? */ } } out: if (black_vector) { /* Restore color */ gsicc_restore_blacktextvec(pgs, false); } if (gx_dc_is_pattern1_color(gs_swappeddevicecolor_inline(pgs))) { gs_id id; if (gs_swappeddevicecolor_inline(pgs)->colors.pattern.p_tile != NULL) { id = gs_swappeddevicecolor_inline(pgs)->colors.pattern.p_tile->id; rcode = dev_proc(pgs->device, lock_pattern)(pgs->device, pgs, id, false); if (rcode < 0) return rcode; /* unlock failed -- shouldn't be possible */ } else { code = 0; } } if (code >= 0 && acode < 0) code = acode; return code; out2: if (black_vector) { /* Restore color */ gsicc_restore_blacktextvec(pgs, false); } return code; } /* Fill the current path using a specified rule. */ static int fill_stroke_with_rule(gs_gstate * pgs, int rule, int *restart) { int code; /* If we're inside a charpath, just merge the current path */ /* into the parent's path. */ if (pgs->in_charpath) { /* If we're rendering a glyph cached, the show machinery decides * whether to actually image it on the output or not, but uncached * will render directly to the output, so for text rendering * mode 3, we have to short circuit it here, but keep the * current point */ *restart = 0; code = gx_path_add_char_path(pgs->show_gstate->path, pgs->path, pgs->in_charpath); if (code < 0) return code; if (pgs->in_charpath == cpm_true_charpath) { /* * A stroke inside a true charpath should do the * equivalent of strokepath. */ code = gs_strokepath(pgs); if (code < 0) return code; code = gx_path_add_char_path(pgs->show_gstate->path, pgs->path, pgs->in_charpath); if (code < 0) return code; } } else if (gs_is_null_device(pgs->device) || (pgs->show_gstate && pgs->text_rendering_mode == 3 && pgs->in_cachedevice == CACHE_DEVICE_NOT_CACHING)) { /* Text Rendering Mode = 3 => Neither stroke, nor fill */ /* Handle separately to prevent gs_gstate_color_load - bug 688308. */ *restart = 0; gs_newpath(pgs); code = 0; } else { code = do_fill_stroke(pgs, rule, restart); if (code >= 0) gs_newpath(pgs); } return code; } /* Fill using the winding number rule */ int gs_fillstroke(gs_gstate * pgs, int *restart) { pgs->device->sgr.stroke_stored = false; return fill_stroke_with_rule(pgs, gx_rule_winding_number, restart); } /* Fill using the even/odd rule */ int gs_eofillstroke(gs_gstate * pgs, int *restart) { pgs->device->sgr.stroke_stored = false; return fill_stroke_with_rule(pgs, gx_rule_even_odd, restart); }