diff options
author | Robin Watts <robin.watts@artifex.com> | 2017-04-26 12:03:07 +0100 |
---|---|---|
committer | Robin Watts <Robin.Watts@artifex.com> | 2017-06-13 16:03:59 +0100 |
commit | f1386c6778baded82a3b98215264c2613efc7fe7 (patch) | |
tree | 63c093d906d2a69f3d978577930484968eff7f3e /source/fitz/draw-edge.c | |
parent | 1bcfc08545f4bed3d1197af1756f7465af11cf96 (diff) | |
download | mupdf-f1386c6778baded82a3b98215264c2613efc7fe7.tar.xz |
Introduce fz_rasterizer encapsulation
This is intended to be a way to allow us to implement
compiletime/runtime selection of different scan converter
implementations.
Diffstat (limited to 'source/fitz/draw-edge.c')
-rw-r--r-- | source/fitz/draw-edge.c | 480 |
1 files changed, 100 insertions, 380 deletions
diff --git a/source/fitz/draw-edge.c b/source/fitz/draw-edge.c index 68c98690..9b417345 100644 --- a/source/fitz/draw-edge.c +++ b/source/fitz/draw-edge.c @@ -1,241 +1,6 @@ #include "mupdf/fitz.h" #include "draw-imp.h" -#include <string.h> -#include <math.h> -#include <stdlib.h> -#include <limits.h> -#include <assert.h> - -#define BBOX_MIN -(1<<20) -#define BBOX_MAX (1<<20) - -/* divide and floor towards -inf */ -static inline int fz_idiv(int a, int b) -{ - return a < 0 ? (a - b + 1) / b : a / b; -} - -/* divide and ceil towards inf */ -static inline int fz_idiv_up(int a, int b) -{ - return a < 0 ? a / b : (a + b - 1) / b; -} - -/* If AA_BITS is defined, then we assume constant N bits of antialiasing. We - * will attempt to provide at least that number of bits of accuracy in the - * antialiasing (to a maximum of 8). If it is defined to be 0 then no - * antialiasing is done. If it is undefined to we will leave the antialiasing - * accuracy as a run time choice. - */ -struct fz_aa_context_s -{ - int hscale; - int vscale; - int scale; - int bits; - int text_bits; - float min_line_width; -}; - -void fz_new_aa_context(fz_context *ctx) -{ -#ifndef AA_BITS - ctx->aa = fz_malloc_struct(ctx, fz_aa_context); - ctx->aa->hscale = 17; - ctx->aa->vscale = 15; - ctx->aa->scale = 256; - ctx->aa->bits = 8; - ctx->aa->text_bits = 8; - -#define fz_aa_hscale (ctx->aa->hscale) -#define fz_aa_vscale (ctx->aa->vscale) -#define fz_aa_scale (ctx->aa->scale) -#define fz_aa_bits (ctx->aa->bits) -#define fz_aa_text_bits (ctx->aa->text_bits) -#define AA_SCALE(scale, x) ((x * scale) >> 8) - -#endif -} - -void fz_copy_aa_context(fz_context *dst, fz_context *src) -{ - if (dst && dst->aa && src && src->aa) - memcpy(dst->aa, src->aa, sizeof(*src->aa)); -} - -void fz_drop_aa_context(fz_context *ctx) -{ - if (!ctx) - return; -#ifndef AA_BITS - fz_free(ctx, ctx->aa); - ctx->aa = NULL; -#endif -} - -#ifdef AA_BITS - -#define fz_aa_scale 0 - -#if AA_BITS > 6 -#define AA_SCALE(s, x) (x) -#define fz_aa_hscale 17 -#define fz_aa_vscale 15 -#define fz_aa_bits 8 -#define fz_aa_text_bits 8 - -#elif AA_BITS > 4 -#define AA_SCALE(s, x) ((x * 255) >> 6) -#define fz_aa_hscale 8 -#define fz_aa_vscale 8 -#define fz_aa_bits 6 -#define fz_aa_text_bits 6 - -#elif AA_BITS > 2 -#define AA_SCALE(s, x) (x * 17) -#define fz_aa_hscale 5 -#define fz_aa_vscale 3 -#define fz_aa_bits 4 -#define fz_aa_text_bits 4 - -#elif AA_BITS > 0 -#define AA_SCALE(s, x) ((x * 255) >> 2) -#define fz_aa_hscale 2 -#define fz_aa_vscale 2 -#define fz_aa_bits 2 -#define fz_aa_text_bits 2 - -#else -#define AA_SCALE(s, x) (x * 255) -#define fz_aa_hscale 1 -#define fz_aa_vscale 1 -#define fz_aa_bits 0 -#define fz_aa_text_bits 0 - -#endif -#endif - -int -fz_aa_level(fz_context *ctx) -{ - return fz_aa_bits; -} - -int -fz_graphics_aa_level(fz_context *ctx) -{ - return fz_aa_bits; -} - -int -fz_text_aa_level(fz_context *ctx) -{ - return fz_aa_text_bits; -} - -#ifndef AA_BITS -static void -set_gfx_level(fz_context *ctx, int level) -{ - if (level > 6) - { - fz_aa_hscale = 17; - fz_aa_vscale = 15; - fz_aa_bits = 8; - } - else if (level > 4) - { - fz_aa_hscale = 8; - fz_aa_vscale = 8; - fz_aa_bits = 6; - } - else if (level > 2) - { - fz_aa_hscale = 5; - fz_aa_vscale = 3; - fz_aa_bits = 4; - } - else if (level > 0) - { - fz_aa_hscale = 2; - fz_aa_vscale = 2; - fz_aa_bits = 2; - } - else - { - fz_aa_hscale = 1; - fz_aa_vscale = 1; - fz_aa_bits = 0; - } - fz_aa_scale = 0xFF00 / (fz_aa_hscale * fz_aa_vscale); -} - -static void -set_txt_level(fz_context *ctx, int level) -{ - if (level > 6) - fz_aa_text_bits = 8; - else if (level > 4) - fz_aa_text_bits = 6; - else if (level > 2) - fz_aa_text_bits = 4; - else if (level > 0) - fz_aa_text_bits = 2; - else - fz_aa_text_bits = 0; -} -#endif /* AA_BITS */ - -void -fz_set_aa_level(fz_context *ctx, int level) -{ -#ifdef AA_BITS - fz_warn(ctx, "anti-aliasing was compiled with a fixed precision of %d bits", fz_aa_bits); -#else - set_gfx_level(ctx, level); - set_txt_level(ctx, level); -#endif -} - -void -fz_set_text_aa_level(fz_context *ctx, int level) -{ -#ifdef AA_BITS - fz_warn(ctx, "anti-aliasing was compiled with a fixed precision of %d bits", fz_aa_bits); -#else - set_txt_level(ctx, level); -#endif -} - -void -fz_set_graphics_aa_level(fz_context *ctx, int level) -{ -#ifdef AA_BITS - fz_warn(ctx, "anti-aliasing was compiled with a fixed precision of %d bits", fz_aa_bits); -#else - set_gfx_level(ctx, level); -#endif -} - -void -fz_set_graphics_min_line_width(fz_context *ctx, float min_line_width) -{ - if (!ctx || !ctx->aa) - return; - - ctx->aa->min_line_width = min_line_width; -} - -float -fz_graphics_min_line_width(fz_context *ctx) -{ - if (!ctx || !ctx->aa) - return 0; - - return ctx->aa->min_line_width; -} - /* * Global Edge List -- list of straight path segments for scan conversion * @@ -244,88 +9,38 @@ fz_graphics_min_line_width(fz_context *ctx) * See Mike Abrash -- Graphics Programming Black Book (notably chapter 40) */ -typedef struct fz_edge_s fz_edge; - -struct fz_edge_s +typedef struct fz_edge_s { int x, e, h, y; int adj_up, adj_down; int xmove; int xdir, ydir; /* -1 or +1 */ -}; +} fz_edge; -struct fz_gel_s +typedef struct fz_gel_s { - fz_rect clip; - fz_irect bbox; + fz_rasterizer super; int cap, len; fz_edge *edges; int acap, alen; fz_edge **active; -}; - -fz_gel * -fz_new_gel(fz_context *ctx) -{ - fz_gel *gel; - - gel = fz_malloc_struct(ctx, fz_gel); - fz_try(ctx) - { - gel->edges = NULL; - gel->cap = 512; - gel->len = 0; - gel->edges = fz_malloc_array(ctx, gel->cap, sizeof(fz_edge)); - - gel->clip.x0 = gel->clip.y0 = BBOX_MIN; - gel->clip.x1 = gel->clip.y1 = BBOX_MAX; - - gel->bbox.x0 = gel->bbox.y0 = BBOX_MAX; - gel->bbox.x1 = gel->bbox.y1 = BBOX_MIN; - - gel->acap = 64; - gel->alen = 0; - gel->active = fz_malloc_array(ctx, gel->acap, sizeof(fz_edge*)); - } - fz_catch(ctx) - { - if (gel) - fz_free(ctx, gel->edges); - fz_free(ctx, gel); - fz_rethrow(ctx); - } - - return gel; -} +} fz_gel; -void -fz_reset_gel(fz_context *ctx, fz_gel *gel, const fz_irect *clip) +static int +fz_reset_gel(fz_context *ctx, fz_rasterizer *rast) { - const int hscale = fz_aa_hscale; - const int vscale = fz_aa_vscale; - - if (fz_is_infinite_irect(clip)) - { - gel->clip.x0 = gel->clip.y0 = BBOX_MIN; - gel->clip.x1 = gel->clip.y1 = BBOX_MAX; - } - else { - gel->clip.x0 = clip->x0 * hscale; - gel->clip.x1 = clip->x1 * hscale; - gel->clip.y0 = clip->y0 * vscale; - gel->clip.y1 = clip->y1 * vscale; - } - - gel->bbox.x0 = gel->bbox.y0 = BBOX_MAX; - gel->bbox.x1 = gel->bbox.y1 = BBOX_MIN; + fz_gel *gel = (fz_gel *)rast; gel->len = 0; gel->alen = 0; + + return 0; } -void -fz_drop_gel(fz_context *ctx, fz_gel *gel) +static void +fz_drop_gel(fz_context *ctx, fz_rasterizer *rast) { + fz_gel *gel = (fz_gel *)rast; if (gel == NULL) return; fz_free(ctx, gel->active); @@ -333,40 +48,6 @@ fz_drop_gel(fz_context *ctx, fz_gel *gel) fz_free(ctx, gel); } -fz_irect * -fz_bound_gel(fz_context *ctx, const fz_gel *gel, fz_irect *bbox) -{ - const int hscale = fz_aa_hscale; - const int vscale = fz_aa_vscale; - - if (gel->len == 0) - { - *bbox = fz_empty_irect; - } - else - { - bbox->x0 = fz_idiv(gel->bbox.x0, hscale); - bbox->y0 = fz_idiv(gel->bbox.y0, vscale); - bbox->x1 = fz_idiv_up(gel->bbox.x1, hscale); - bbox->y1 = fz_idiv_up(gel->bbox.y1, vscale); - } - return bbox; -} - -fz_rect * -fz_gel_scissor(fz_context *ctx, const fz_gel *gel, fz_rect *r) -{ - const int hscale = fz_aa_hscale; - const int vscale = fz_aa_vscale; - - r->x0 = gel->clip.x0 / hscale; - r->x1 = gel->clip.x1 / vscale; - r->y0 = gel->clip.y0 / hscale; - r->y1 = gel->clip.y1 / vscale; - - return r; -} - enum { INSIDE, OUTSIDE, LEAVE, ENTER }; #define clip_lerp_y(v,m,x0,y0,x1,y1,t) clip_lerp_x(v,m,y0,x0,y1,x1,t) @@ -397,8 +78,9 @@ clip_lerp_x(int val, int m, int x0, int y0, int x1, int y1, int *out) } static void -fz_insert_gel_raw(fz_context *ctx, fz_gel *gel, int x0, int y0, int x1, int y1) +fz_insert_gel_raw(fz_context *ctx, fz_rasterizer *ras, int x0, int y0, int x1, int y1) { + fz_gel *gel = (fz_gel *)ras; fz_edge *edge; int dx, dy; int winding; @@ -416,13 +98,13 @@ fz_insert_gel_raw(fz_context *ctx, fz_gel *gel, int x0, int y0, int x1, int y1) else winding = 1; - if (x0 < gel->bbox.x0) gel->bbox.x0 = x0; - if (x0 > gel->bbox.x1) gel->bbox.x1 = x0; - if (x1 < gel->bbox.x0) gel->bbox.x0 = x1; - if (x1 > gel->bbox.x1) gel->bbox.x1 = x1; + if (x0 < gel->super.bbox.x0) gel->super.bbox.x0 = x0; + if (x0 > gel->super.bbox.x1) gel->super.bbox.x1 = x0; + if (x1 < gel->super.bbox.x0) gel->super.bbox.x0 = x1; + if (x1 > gel->super.bbox.x1) gel->super.bbox.x1 = x1; - if (y0 < gel->bbox.y0) gel->bbox.y0 = y0; - if (y1 > gel->bbox.y1) gel->bbox.y1 = y1; + if (y0 < gel->super.bbox.y0) gel->super.bbox.y0 = y0; + if (y1 > gel->super.bbox.y1) gel->super.bbox.y1 = y1; if (gel->len + 1 == gel->cap) { int new_cap = gel->cap * 2; @@ -462,8 +144,8 @@ fz_insert_gel_raw(fz_context *ctx, fz_gel *gel, int x0, int y0, int x1, int y1) } } -void -fz_insert_gel(fz_context *ctx, fz_gel *gel, float fx0, float fy0, float fx1, float fy1) +static void +fz_insert_gel(fz_context *ctx, fz_rasterizer *ras, float fx0, float fy0, float fx1, float fy1, int rev) { int x0, y0, x1, y1; int d, v; @@ -484,51 +166,51 @@ fz_insert_gel(fz_context *ctx, fz_gel *gel, float fx0, float fy0, float fx1, flo x1 = (int)fz_clamp(fx1, BBOX_MIN * hscale, BBOX_MAX * hscale); y1 = (int)fz_clamp(fy1, BBOX_MIN * vscale, BBOX_MAX * vscale); - d = clip_lerp_y(gel->clip.y0, 0, x0, y0, x1, y1, &v); + d = clip_lerp_y(ras->clip.y0, 0, x0, y0, x1, y1, &v); if (d == OUTSIDE) return; - if (d == LEAVE) { y1 = gel->clip.y0; x1 = v; } - if (d == ENTER) { y0 = gel->clip.y0; x0 = v; } + if (d == LEAVE) { y1 = ras->clip.y0; x1 = v; } + if (d == ENTER) { y0 = ras->clip.y0; x0 = v; } - d = clip_lerp_y(gel->clip.y1, 1, x0, y0, x1, y1, &v); + d = clip_lerp_y(ras->clip.y1, 1, x0, y0, x1, y1, &v); if (d == OUTSIDE) return; - if (d == LEAVE) { y1 = gel->clip.y1; x1 = v; } - if (d == ENTER) { y0 = gel->clip.y1; x0 = v; } + if (d == LEAVE) { y1 = ras->clip.y1; x1 = v; } + if (d == ENTER) { y0 = ras->clip.y1; x0 = v; } - d = clip_lerp_x(gel->clip.x0, 0, x0, y0, x1, y1, &v); + d = clip_lerp_x(ras->clip.x0, 0, x0, y0, x1, y1, &v); if (d == OUTSIDE) { - x0 = x1 = gel->clip.x0; + x0 = x1 = ras->clip.x0; } if (d == LEAVE) { - fz_insert_gel_raw(ctx, gel, gel->clip.x0, v, gel->clip.x0, y1); - x1 = gel->clip.x0; + fz_insert_gel_raw(ctx, ras, ras->clip.x0, v, ras->clip.x0, y1); + x1 = ras->clip.x0; y1 = v; } if (d == ENTER) { - fz_insert_gel_raw(ctx, gel, gel->clip.x0, y0, gel->clip.x0, v); - x0 = gel->clip.x0; + fz_insert_gel_raw(ctx, ras, ras->clip.x0, y0, ras->clip.x0, v); + x0 = ras->clip.x0; y0 = v; } - d = clip_lerp_x(gel->clip.x1, 1, x0, y0, x1, y1, &v); + d = clip_lerp_x(ras->clip.x1, 1, x0, y0, x1, y1, &v); if (d == OUTSIDE) { - x0 = x1 = gel->clip.x1; + x0 = x1 = ras->clip.x1; } if (d == LEAVE) { - fz_insert_gel_raw(ctx, gel, gel->clip.x1, v, gel->clip.x1, y1); - x1 = gel->clip.x1; + fz_insert_gel_raw(ctx, ras, ras->clip.x1, v, ras->clip.x1, y1); + x1 = ras->clip.x1; y1 = v; } if (d == ENTER) { - fz_insert_gel_raw(ctx, gel, gel->clip.x1, y0, gel->clip.x1, v); - x0 = gel->clip.x1; + fz_insert_gel_raw(ctx, ras, ras->clip.x1, y0, ras->clip.x1, v); + x0 = ras->clip.x1; y0 = v; } - fz_insert_gel_raw(ctx, gel, x0, y0, x1, y1); + fz_insert_gel_raw(ctx, ras, x0, y0, x1, y1); } -void -fz_insert_gel_rect(fz_context *ctx, fz_gel *gel, float fx0, float fy0, float fx1, float fy1) +static void +fz_insert_gel_rect(fz_context *ctx, fz_rasterizer *ras, float fx0, float fy0, float fx1, float fy1) { int x0, y0, x1, y1; const int hscale = fz_aa_hscale; @@ -555,10 +237,10 @@ fz_insert_gel_rect(fz_context *ctx, fz_gel *gel, float fx0, float fy0, float fx1 fy1 = floorf(fy1 * vscale); } - fx0 = fz_clamp(fx0, gel->clip.x0, gel->clip.x1); - fx1 = fz_clamp(fx1, gel->clip.x0, gel->clip.x1); - fy0 = fz_clamp(fy0, gel->clip.y0, gel->clip.y1); - fy1 = fz_clamp(fy1, gel->clip.y0, gel->clip.y1); + fx0 = fz_clamp(fx0, ras->clip.x0, ras->clip.x1); + fx1 = fz_clamp(fx1, ras->clip.x0, ras->clip.x1); + fy0 = fz_clamp(fy0, ras->clip.y0, ras->clip.y1); + fy1 = fz_clamp(fy1, ras->clip.y0, ras->clip.y1); /* Call fz_clamp so that clamping is done in the float domain, THEN * cast down to an int. Calling fz_clampi causes problems due to the @@ -569,8 +251,8 @@ fz_insert_gel_rect(fz_context *ctx, fz_gel *gel, float fx0, float fy0, float fx1 x1 = (int)fz_clamp(fx1, BBOX_MIN * hscale, BBOX_MAX * hscale); y1 = (int)fz_clamp(fy1, BBOX_MIN * vscale, BBOX_MAX * vscale); - fz_insert_gel_raw(ctx, gel, x1, y0, x1, y1); - fz_insert_gel_raw(ctx, gel, x0, y1, x0, y0); + fz_insert_gel_raw(ctx, ras, x1, y0, x1, y1); + fz_insert_gel_raw(ctx, ras, x0, y1, x0, y0); } static int @@ -631,9 +313,10 @@ sort_gel(fz_context *ctx, fz_gel *gel) #endif } -int -fz_is_rect_gel(fz_context *ctx, fz_gel *gel) +static int +fz_is_rect_gel(fz_context *ctx, fz_rasterizer *ras) { + fz_gel *gel = (fz_gel *)ras; /* a rectangular path is converted into two vertical edges of identical height */ if (gel->len == 2) { @@ -868,8 +551,8 @@ fz_scan_convert_aa(fz_context *ctx, fz_gel *gel, int eofill, const fz_irect *cli const int hscale = fz_aa_hscale; const int vscale = fz_aa_vscale; - int xmin = fz_idiv(gel->bbox.x0, hscale); - int xmax = fz_idiv_up(gel->bbox.x1, hscale); + int xmin = fz_idiv(gel->super.bbox.x0, hscale); + int xmax = fz_idiv_up(gel->super.bbox.x1, hscale); int xofs = xmin * hscale; @@ -1147,13 +830,10 @@ fz_scan_convert_sharp(fz_context *ctx, } } -void -fz_scan_convert(fz_context *ctx, fz_gel *gel, int eofill, const fz_irect *clip, fz_pixmap *dst, unsigned char *color) +static void +fz_convert_gel(fz_context *ctx, fz_rasterizer *rast, int eofill, const fz_irect *clip, fz_pixmap *dst, unsigned char *color) { - fz_irect local_clip; - - if (fz_is_empty_irect(fz_intersect_irect(fz_pixmap_bbox_no_ctx(dst, &local_clip), clip))) - return; + fz_gel *gel = (fz_gel *)rast; sort_gel(ctx, gel); @@ -1167,7 +847,7 @@ fz_scan_convert(fz_context *ctx, fz_gel *gel, int eofill, const fz_irect *clip, assert(fn); if (fn == NULL) return; - fz_scan_convert_aa(ctx, gel, eofill, &local_clip, dst, color, fn); + fz_scan_convert_aa(ctx, gel, eofill, clip, dst, color, fn); } else { @@ -1175,6 +855,46 @@ fz_scan_convert(fz_context *ctx, fz_gel *gel, int eofill, const fz_irect *clip, assert(fn); if (fn == NULL) return; - fz_scan_convert_sharp(ctx, gel, eofill, &local_clip, dst, color, (fz_solid_color_painter_t *)fn); + fz_scan_convert_sharp(ctx, gel, eofill, clip, dst, color, (fz_solid_color_painter_t *)fn); + } +} + +static const fz_rasterizer_fns gel_rasterizer = +{ + fz_drop_gel, + fz_reset_gel, + NULL, /* postindex */ + fz_insert_gel, + fz_insert_gel_rect, + NULL, /* gap */ + fz_convert_gel, + fz_is_rect_gel, + 0 /* Not reusable */ +}; + +fz_rasterizer * +fz_new_gel(fz_context *ctx) +{ + fz_gel *gel; + + gel = fz_new_derived_rasterizer(ctx, fz_gel, &gel_rasterizer); + fz_try(ctx) + { + gel->edges = NULL; + gel->cap = 512; + gel->len = 0; + gel->edges = fz_malloc_array(ctx, gel->cap, sizeof(fz_edge)); + + gel->acap = 64; + gel->alen = 0; + gel->active = fz_malloc_array(ctx, gel->acap, sizeof(fz_edge*)); + } + fz_catch(ctx) + { + fz_free(ctx, gel->edges); + fz_free(ctx, gel); + fz_rethrow(ctx); } + + return &gel->super; } |