summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--Makefile7
-rw-r--r--include/mupdf/fitz/document.h1
-rw-r--r--platform/android/jni/Core.mk3
-rw-r--r--platform/win32/libmupdf.vcproj8
-rw-r--r--source/fitz/document-all.c1
-rw-r--r--source/gprf/gprf-doc.c751
6 files changed, 769 insertions, 2 deletions
diff --git a/Makefile b/Makefile
index 1bd50ff5..938ec742 100644
--- a/Makefile
+++ b/Makefile
@@ -67,6 +67,7 @@ ALL_DIR += $(OUT)/cbz
ALL_DIR += $(OUT)/img
ALL_DIR += $(OUT)/tiff
ALL_DIR += $(OUT)/html
+ALL_DIR += $(OUT)/gprf
ALL_DIR += $(OUT)/tools
ALL_DIR += $(OUT)/platform/x11
ALL_DIR += $(OUT)/platform/x11/curl
@@ -81,17 +82,20 @@ PDF_SRC := $(wildcard source/pdf/*.c)
XPS_SRC := $(wildcard source/xps/*.c)
CBZ_SRC := $(wildcard source/cbz/*.c)
HTML_SRC := $(wildcard source/html/*.c)
+GPRF_SRC := $(wildcard source/gprf/*.c)
FITZ_SRC_HDR := $(wildcard source/fitz/*.h)
PDF_SRC_HDR := $(wildcard source/pdf/*.h) source/pdf/pdf-name-table.h
XPS_SRC_HDR := $(wildcard source/xps/*.h)
HTML_SRC_HDR := $(wildcard source/html/*.h)
+GPRF_SRC_HDR := $(wildcard source/gprf/*.h)
FITZ_OBJ := $(subst source/, $(OUT)/, $(addsuffix .o, $(basename $(FITZ_SRC))))
PDF_OBJ := $(subst source/, $(OUT)/, $(addsuffix .o, $(basename $(PDF_SRC))))
XPS_OBJ := $(subst source/, $(OUT)/, $(addsuffix .o, $(basename $(XPS_SRC))))
CBZ_OBJ := $(subst source/, $(OUT)/, $(addsuffix .o, $(basename $(CBZ_SRC))))
HTML_OBJ := $(subst source/, $(OUT)/, $(addsuffix .o, $(basename $(HTML_SRC))))
+GPRF_OBJ := $(subst source/, $(OUT)/, $(addsuffix .o, $(basename $(GPRF_SRC))))
# --- Choice of Javascript library ---
@@ -120,12 +124,13 @@ $(PDF_OBJ) : $(FITZ_HDR) $(PDF_HDR) $(PDF_SRC_HDR)
$(XPS_OBJ) : $(FITZ_HDR) $(XPS_HDR) $(XPS_SRC_HDR)
$(CBZ_OBJ) : $(FITZ_HDR)
$(HTML_OBJ) : $(FITZ_HDR) $(HTML_HDR) $(HTML_SRC_HDR)
+$(GPRF_OBJ) : $(FITZ_HDR) $(GPRF_HDR) $(GPRF_SRC_HDR)
# --- Library ---
MUPDF_LIB := $(OUT)/libmupdf.a
-$(MUPDF_LIB) : $(FITZ_OBJ) $(PDF_OBJ) $(XPS_OBJ) $(CBZ_OBJ) $(HTML_OBJ)
+$(MUPDF_LIB) : $(FITZ_OBJ) $(PDF_OBJ) $(XPS_OBJ) $(CBZ_OBJ) $(HTML_OBJ) $(GPRF_OBJ)
INSTALL_LIBS := $(MUPDF_LIB)
diff --git a/include/mupdf/fitz/document.h b/include/mupdf/fitz/document.h
index f7552ec0..58cdc63a 100644
--- a/include/mupdf/fitz/document.h
+++ b/include/mupdf/fitz/document.h
@@ -106,6 +106,7 @@ extern fz_document_handler img_document_handler;
extern fz_document_handler tiff_document_handler;
extern fz_document_handler html_document_handler;
extern fz_document_handler epub_document_handler;
+extern fz_document_handler gprf_document_handler;
void fz_register_document_handler(fz_context *ctx, const fz_document_handler *handler);
diff --git a/platform/android/jni/Core.mk b/platform/android/jni/Core.mk
index c1eb07dc..70ba5938 100644
--- a/platform/android/jni/Core.mk
+++ b/platform/android/jni/Core.mk
@@ -51,7 +51,8 @@ LOCAL_SRC_FILES := \
$(wildcard $(MY_ROOT)/source/pdf/*.c) \
$(wildcard $(MY_ROOT)/source/xps/*.c) \
$(wildcard $(MY_ROOT)/source/cbz/*.c) \
- $(wildcard $(MY_ROOT)/source/html/*.c)
+ $(wildcard $(MY_ROOT)/source/html/*.c) \
+ $(wildcard $(MY_ROOT)/source/gprf/*.c)
LOCAL_SRC_FILES += \
$(MY_ROOT)/source/pdf/js/pdf-js.c \
$(MY_ROOT)/source/pdf/js/pdf-jsimp-mu.c
diff --git a/platform/win32/libmupdf.vcproj b/platform/win32/libmupdf.vcproj
index 50b64556..0b83bb98 100644
--- a/platform/win32/libmupdf.vcproj
+++ b/platform/win32/libmupdf.vcproj
@@ -1582,6 +1582,14 @@
>
</File>
</Filter>
+ <Filter
+ Name="gprf"
+ >
+ <File
+ RelativePath="..\..\source\gprf\gprf-doc.c"
+ >
+ </File>
+ </Filter>
</Files>
<Globals>
</Globals>
diff --git a/source/fitz/document-all.c b/source/fitz/document-all.c
index bfe57592..8517e0ee 100644
--- a/source/fitz/document-all.c
+++ b/source/fitz/document-all.c
@@ -9,4 +9,5 @@ void fz_register_document_handlers(fz_context *ctx)
fz_register_document_handler(ctx, &tiff_document_handler);
fz_register_document_handler(ctx, &html_document_handler);
fz_register_document_handler(ctx, &epub_document_handler);
+ fz_register_document_handler(ctx, &gprf_document_handler);
}
diff --git a/source/gprf/gprf-doc.c b/source/gprf/gprf-doc.c
new file mode 100644
index 00000000..d9cd2938
--- /dev/null
+++ b/source/gprf/gprf-doc.c
@@ -0,0 +1,751 @@
+#include "mupdf/fitz.h"
+
+typedef struct gprf_document_s gprf_document;
+typedef struct gprf_chapter_s gprf_chapter;
+typedef struct gprf_page_s gprf_page;
+
+enum
+{
+ GPRF_TILESIZE = 256
+};
+
+struct gprf_document_s
+{
+ fz_document super;
+ char *pdf_filename;
+ int res;
+ int num_pages;
+ struct {
+ int w;
+ int h;
+ } *page_dims;
+};
+
+typedef struct gprf_file_s
+{
+ int refs;
+ char *filename;
+} gprf_file;
+
+static gprf_file *
+fz_new_gprf_file(fz_context *ctx, char *filename)
+{
+ gprf_file *file = fz_malloc_struct(ctx, gprf_file);
+ file->refs = 1;
+ file->filename = filename;
+
+ return file;
+}
+
+static gprf_file *
+fz_keep_gprf_file(fz_context *ctx, gprf_file *file)
+{
+ if (!ctx || !file)
+ return NULL;
+
+ fz_lock(ctx, FZ_LOCK_ALLOC);
+ file->refs++;
+ fz_unlock(ctx, FZ_LOCK_ALLOC);
+
+ return file;
+}
+
+static void
+fz_drop_gprf_file(fz_context *ctx, gprf_file *file)
+{
+ int i;
+
+ if (!ctx || !file)
+ return;
+
+ fz_lock(ctx, FZ_LOCK_ALLOC);
+ i = --file->refs;
+ fz_unlock(ctx, FZ_LOCK_ALLOC);
+ if (i == 0)
+ {
+ unlink(file->filename);
+ fz_free(ctx, file->filename);
+ fz_free(ctx, file);
+ }
+}
+
+struct gprf_page_s
+{
+ fz_page super;
+ gprf_document *doc;
+ gprf_file *file;
+ int number;
+ fz_separations *separations;
+ int width;
+ int height;
+ int tile_width;
+ int tile_height;
+ int num_tiles;
+ fz_image **tiles;
+};
+
+typedef struct fz_image_gprf_s
+{
+ fz_image base;
+ fz_off_t offset[FZ_MAX_SEPARATIONS+3+1]; /* + RGB + END */
+ gprf_file *file;
+ fz_separations *separations;
+} fz_image_gprf;
+
+static int
+gprf_count_pages(fz_context *ctx, fz_document *doc_)
+{
+ gprf_document *doc = (gprf_document*)doc_;
+ return doc->num_pages;
+}
+
+static void
+gprf_drop_page_imp(fz_context *ctx, fz_page *page_)
+{
+ gprf_page *page = (gprf_page*)page_;
+ gprf_document *doc = page->doc;
+ int i;
+
+ fz_drop_document(ctx, (fz_document *)doc);
+ if (page->tiles)
+ {
+ for (i = 0; i < page->num_tiles; i++)
+ fz_drop_image(ctx, page->tiles[i]);
+ fz_free(ctx, page->tiles);
+ }
+ fz_drop_separations(ctx, page->separations);
+ fz_drop_gprf_file(ctx, page->file);
+}
+
+static fz_rect *
+gprf_bound_page(fz_context *ctx, fz_page *page_, fz_rect *bbox)
+{
+ gprf_page *page = (gprf_page*)page_;
+ gprf_document *doc = page->doc;
+
+ /* BBox is in points, not pixels */
+ bbox->x0 = 0;
+ bbox->y0 = 0;
+ bbox->x1 = 72.0 * page->width / doc->res;
+ bbox->y1 = 72.0 * page->height / doc->res;
+
+ return bbox;
+}
+
+static void
+fz_drop_image_gprf_imp(fz_context *ctx, fz_storable *image_)
+{
+ fz_image_gprf *image = (fz_image_gprf *)image_;
+
+ fz_drop_gprf_file(ctx, image->file);
+ fz_drop_separations(ctx, image->separations);
+ fz_drop_image_imp(ctx, &image->base.storable);
+}
+
+static inline unsigned char *cmyk_to_rgba(unsigned char *out, uint32_t c, uint32_t m, uint32_t y, uint32_t k)
+{
+ /* c m y k in 0 to 65535 range on entry */
+ uint32_t r, g, b;
+#ifdef SLOWCMYK /* FP version originally from poppler */
+ uint32_t x;
+ uint32_t cm = (c * m)>>16;
+ uint32_t c1m = m - cm;
+ uint32_t cm1 = c - cm;
+ uint32_t c1m1 = 65536 - m - cm1;
+ uint32_t c1m1y = (c1m1 * y)>>16;
+ uint32_t c1m1y1 = c1m1 - c1m1y;
+ uint32_t c1my = (c1m * y)>>16;
+ uint32_t c1my1 = c1m - c1my;
+ uint32_t cm1y = (cm1 * y)>>16;
+ uint32_t cm1y1 = cm1 - cm1y;
+ uint32_t cmy = (cm * y)>>16;
+ uint32_t cmy1 = cm - cmy;
+
+#define CONST16(x) ((int)(x * 65536.0 + 0.5))
+
+ /* this is a matrix multiplication, unrolled for performance */
+ x = (c1m1y1 * k)>>16; /* 0 0 0 1 */
+ r = g = b = c1m1y1 - x; /* 0 0 0 0 */
+ r += (CONST16(0.1373) * x)>>16;
+ g += (CONST16(0.1216) * x)>>16;
+ b += (CONST16(0.1255) * x)>>16;
+
+ x = c1m1y * k; /* 0 0 1 1 */
+ r += (CONST16(0.1098) * x)>>16;
+ g += (CONST16(0.1020) * x)>>16;
+ x = c1m1y - x; /* 0 0 1 0 */
+ r += x;
+ g += (CONST16(0.9490) * x)>>16;
+
+ x = c1my1 * k; /* 0 1 0 1 */
+ r += (CONST16(0.1412) * x)>>16;
+ x = c1my1 - x; /* 0 1 0 0 */
+ r += (CONST16(0.9255) * x)>>16;
+ b += (CONST16(0.5490) * x)>>16;
+
+ x = c1my * k; /* 0 1 1 1 */
+ r += (CONST16(0.1333) * x)>>16;
+ x = c1my - x; /* 0 1 1 0 */
+ r += (CONST16(0.9294) * x)>>16;
+ g += (CONST16(0.1098) * x)>>16;
+ b += (CONST16(0.1412) * x)>>16;
+
+ x = cm1y1 * k; /* 1 0 0 1 */
+ g += (CONST16(0.0588) * x)>>16;
+ b += (CONST16(0.1412) * x)>>16;
+ x = cm1y1 - x; /* 1 0 0 0 */
+ g += (CONST16(0.6784) * x)>>16;
+ b += (CONST16(0.9373) * x)>>16;
+
+ x = cm1y * k; /* 1 0 1 1 */
+ g += (CONST16(0.0745) * x)>>16;
+ x = cm1y - x; /* 1 0 1 0 */
+ g += (CONST16(0.6510) * x)>>16;
+ b += (CONST16(0.3137) * x)>>16;
+
+ x = cmy1 * k; /* 1 1 0 1 */
+ b += (CONST16(0.0078) * x)>>16;
+ x = cmy1 - x; /* 1 1 0 0 */
+ r += (CONST16(0.1804) * x)>>16;
+ g += (CONST16(0.1922) * x)>>16;
+ b += (CONST16(0.5725) * x)>>16;
+
+ x = cmy * (1-k); /* 1 1 1 0 */
+ r += (CONST16(0.2118) * x)>>16;
+ g += (CONST16(0.2119) * x)>>16;
+ b += (CONST16(0.2235) * x)>>16;
+#else
+ k = 63356 - k;
+ r = k - c;
+ g = k - m;
+ b = k - y;
+#endif
+ r >>= 8;
+ if (r < 0)
+ r = 0;
+ else if (r > 255)
+ r = 255;
+ g >>= 8;
+ if (g < 0)
+ g = 0;
+ else if (g > 255)
+ g = 255;
+ b >>= 8;
+ if (b < 0)
+ b = 0;
+ else if (b > 255)
+ b = 255;
+
+ *out++ = r;
+ *out++ = g;
+ *out++ = b;
+ *out++ = 0xFF;
+ return out;
+}
+
+unsigned char undelta(unsigned char delta, unsigned char *ptr, int len)
+{
+ do
+ {
+ delta = (*ptr++ += delta);
+ }
+ while (--len);
+
+ return delta;
+}
+
+static fz_pixmap *
+gprf_get_pixmap(fz_context *ctx, fz_image *image_, int w, int h, int *l2factor)
+{
+ /* The file contains RGB + up to FZ_MAX_SEPARATIONS. Hence the
+ * "3 + FZ_MAX_SEPARATIONS" usage in all the arrays below. */
+ fz_image_gprf *image = (fz_image_gprf *)image_;
+ fz_pixmap *pix = fz_new_pixmap(ctx, image->base.colorspace, image->base.w, image->base.h);
+ fz_stream *file[3 + FZ_MAX_SEPARATIONS] = { NULL };
+ int read_sep[3 + FZ_MAX_SEPARATIONS] = { 0 };
+ int num_seps, i, j, n;
+ enum { decode_chunk_size = 64 };
+ unsigned char data[(3 + FZ_MAX_SEPARATIONS) * decode_chunk_size];
+ unsigned char delta[3 + FZ_MAX_SEPARATIONS] = { 0 };
+ unsigned char equiv[3 + FZ_MAX_SEPARATIONS][4];
+ int bytes_per_channel = image->base.w * image->base.h;
+ unsigned char *out = fz_pixmap_samples(ctx, pix);
+
+ fz_var(file);
+
+ fz_try(ctx)
+ {
+ /* First off, figure out of we are doing RGB or separations
+ * decoding. */
+ num_seps = 3 + fz_count_separations(ctx, image->separations);
+ if (fz_separations_all_enabled(ctx, image->separations))
+ {
+ num_seps = 3;
+ for (i = 0; i < 3; i++)
+ read_sep[i] = 1;
+ }
+ else
+ {
+ for (i = 3; i < num_seps; i++)
+ {
+ read_sep[i] = !fz_separation_disabled(ctx, image->separations, i-3);
+ if (read_sep[i])
+ {
+ uint32_t rgb, cmyk;
+
+ (void)fz_get_separation(ctx, image->separations, i - 3, &rgb, &cmyk);
+ equiv[i][0] = (cmyk>> 0) & 0xFF;
+ equiv[i][1] = (cmyk>> 8) & 0xFF;
+ equiv[i][2] = (cmyk>>16) & 0xFF;
+ equiv[i][3] = (cmyk>>24) & 0xFF;
+ }
+ }
+ }
+
+ /* Open 1 file handle per channel */
+ for (i = 0; i < num_seps; i++)
+ {
+ if (!read_sep[i])
+ continue;
+
+ if (image->offset[i] == image->offset[i+1])
+ {
+ read_sep[i] = 2;
+ memset(&data[i * decode_chunk_size], 0, decode_chunk_size);
+ }
+ file[i] = fz_open_file(ctx, image->file->filename);
+ fz_seek(ctx, file[i], image->offset[i], SEEK_SET);
+ file[i] = fz_open_flated(ctx, file[i], 15);
+ }
+
+ /* Now actually do the decode */
+ for (j = 0; j < bytes_per_channel; j += decode_chunk_size)
+ {
+ int len = bytes_per_channel - j;
+ if (len > decode_chunk_size)
+ len = decode_chunk_size;
+
+ /* Load data with the unpacked channel bytes */
+ for (i = 0; i < num_seps; i++)
+ {
+ if (read_sep[i] == 1)
+ {
+ fz_read(ctx, file[i], &data[i * decode_chunk_size], len);
+ delta[i] = undelta(delta[i], &data[i * decode_chunk_size], len);
+ }
+ }
+
+ /* And unpack */
+ if (num_seps == 3)
+ {
+ for (n = 0; n < len; n++)
+ {
+ *out++ = data[0 * decode_chunk_size + n];
+ *out++ = data[1 * decode_chunk_size + n];
+ *out++ = data[2 * decode_chunk_size + n];
+ *out++ = 0xFF; /* Alpha */
+ }
+ }
+ else
+ {
+ int c, m, y, k;
+
+ c = m = y = k = 0;
+ for (n = 0; n < len; n++)
+ {
+ for (i = 3; i < num_seps; i++)
+ {
+ int v;
+ if (read_sep[n] != 1)
+ continue;
+ v = data[i * decode_chunk_size + n];
+ c += v * equiv[i][0];
+ m += v * equiv[i][1];
+ y += v * equiv[i][2];
+ k += v * equiv[i][3];
+ }
+ }
+ out = cmyk_to_rgba(out, c, m, y, k);
+ }
+ }
+ }
+ fz_always(ctx)
+ {
+ for (i = 0; i < 3+num_seps; i++)
+ fz_drop_stream(ctx, file[i]);
+ }
+ fz_catch(ctx)
+ {
+ fz_drop_pixmap(ctx, pix);
+ fz_rethrow(ctx);
+ }
+
+ return pix;
+}
+
+static fz_image *
+fz_new_gprf_image(fz_context *ctx, gprf_page *page, int imagenum, fz_off_t offsets[], fz_off_t end)
+{
+ fz_image_gprf *image = fz_malloc_struct(ctx, fz_image_gprf);
+ int tile_x = imagenum % page->tile_width;
+ int tile_y = imagenum / page->tile_width;
+ int w = GPRF_TILESIZE;
+ int h = GPRF_TILESIZE;
+ int seps = fz_count_separations(ctx, page->separations);
+
+ if (tile_x == page->tile_width-1)
+ {
+ w = page->width % GPRF_TILESIZE;
+ if (w == 0)
+ w = GPRF_TILESIZE;
+ }
+ if (tile_y == page->tile_height-1)
+ {
+ h = page->height % GPRF_TILESIZE;
+ if (h == 0)
+ h = GPRF_TILESIZE;
+ }
+
+ FZ_INIT_STORABLE(&image->base, 1, fz_drop_image_gprf_imp);
+ image->base.w = w;
+ image->base.h = h;
+ image->base.n = 4; /* Always RGB + Alpha */
+ image->base.colorspace = fz_keep_colorspace(ctx, fz_device_rgb(ctx));
+ image->base.bpc = 8;
+ image->base.buffer = NULL;
+ image->base.get_pixmap = gprf_get_pixmap;
+ image->base.xres = page->doc->res;
+ image->base.yres = page->doc->res;
+ image->base.tile = NULL;
+ image->base.mask = NULL;
+ image->file = fz_keep_gprf_file(ctx, page->file);
+ memcpy(image->offset, offsets, sizeof(fz_off_t) * (3+seps));
+ image->offset[seps+3] = end;
+ image->separations = fz_keep_separations(ctx, page->separations);
+
+ return &image->base;
+}
+
+static void
+fz_system(fz_context *ctx, const char *cmd)
+{
+ int ret = system(cmd);
+
+ if (ret != 0)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "child process reported error %d", ret);
+}
+
+static void
+generate_page(fz_context *ctx, gprf_page *page)
+{
+ gprf_document *doc = page->doc;
+ char nameroot[32];
+ char *filename;
+ char gs_command[1024];
+
+ sprintf(nameroot, "gprf_%d_", page->number);
+ filename = fz_tempfilename(ctx, nameroot, doc->pdf_filename);
+
+ /* Invoke gs to convert to a temp file. */
+ sprintf(gs_command, "gswin32c.exe -sDEVICE=gproof -r%d -o \"%s\" -dFirstPage=%d -dLastPage=%d %s",
+ doc->res, filename, page->number+1, page->number+1, doc->pdf_filename);
+
+ fz_try(ctx)
+ {
+ fz_system(ctx, gs_command);
+
+ page->file = fz_new_gprf_file(ctx, filename);
+ }
+ fz_catch(ctx)
+ {
+ unlink(filename);
+ fz_free(ctx, filename);
+ fz_rethrow(ctx);
+ }
+}
+
+static void
+read_tiles(fz_context *ctx, gprf_page *page)
+{
+ fz_stream *file;
+ int32_t val;
+ uint64_t offset;
+ int num_tiles;
+ int num_seps;
+ int i, x, y;
+ int64_t off;
+
+ /* Clear up any aborted attempts before (unlikely) */
+ fz_drop_separations(ctx, page->separations);
+ page->separations = NULL;
+
+ file = fz_open_file(ctx, page->file->filename);
+
+ fz_try(ctx)
+ {
+ val = fz_read_int32le(ctx, file);
+ if (val != 0x46505347)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Unexpected signature in GSPF file");
+
+ val = fz_read_int16le(ctx, file);
+ if (val != 1)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Unexpected version in GSPF file");
+
+ val = fz_read_int16le(ctx, file);
+ if (val != 0)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Unexpected compression in GSPF file");
+
+ val = fz_read_int32le(ctx, file);
+ if (val != page->width)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Unexpected width in GSPF file");
+
+ val = fz_read_int32le(ctx, file);
+ if (val != page->height)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Unexpected height in GSPF file");
+
+ val = fz_read_int16le(ctx, file);
+ if (val != 8)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Unexpected bpc in GSPF file");
+
+ num_seps = fz_read_int16le(ctx, file);
+ if (num_seps < 0 || num_seps > FZ_MAX_SEPARATIONS)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Unexpected number of separations in GSPF file");
+
+ offset = fz_read_int64le(ctx, file); /* Ignore the ICC for now */
+
+ /* Read the table offset */
+ offset = fz_read_int64le(ctx, file);
+
+ /* Skip to the separations */
+ fz_seek(ctx, file, 64, SEEK_SET);
+ page->separations = fz_new_separations(ctx);
+ for (i = 0; i < num_seps; i++)
+ {
+ char blatter[4096];
+ int32_t rgba = fz_read_int32le(ctx, file);
+ int32_t cmyk = fz_read_int32le(ctx, file);
+ fz_read_line(ctx, file, blatter, sizeof(blatter));
+ fz_add_separation(ctx, page->separations, rgba, cmyk, blatter);
+ }
+
+ /* Seek to the image data */
+ fz_seek(ctx, file, (fz_off_t)offset, SEEK_SET);
+
+ num_tiles = page->tile_width * page->tile_height;
+ page->tiles = fz_calloc(ctx, num_tiles, sizeof(fz_image *));
+
+ i = 0;
+ off = fz_read_int64le(ctx, file);
+ for (y = 0; y < page->tile_height; y++)
+ {
+ for (x = 0; x < page->tile_width; x++)
+ {
+ fz_off_t offsets[FZ_MAX_SEPARATIONS + 3]; /* SEPARATIONS + RGB */
+ int j;
+
+ for (j = 0; j < num_seps+3; j++)
+ {
+ offsets[j] = (fz_off_t)off;
+ off = fz_read_int64le(ctx, file);
+ }
+
+ page->tiles[i] = fz_new_gprf_image(ctx, page, i, offsets, (fz_off_t)off);
+ i++;
+ }
+ }
+ }
+ fz_always(ctx)
+ {
+ fz_drop_stream(ctx, file);
+ }
+ fz_catch(ctx)
+ {
+ /* Free any tiles made so far */
+ for (i = num_tiles - 1; i >= 0; i--)
+ {
+ fz_drop_image(ctx, page->tiles[i]);
+ }
+ fz_free(ctx, page->tiles);
+
+ fz_rethrow(ctx);
+ }
+ page->num_tiles = num_tiles;
+}
+
+static void
+gprf_run_page(fz_context *ctx, fz_page *page_, fz_device *dev, const fz_matrix *ctm, fz_cookie *cookie)
+{
+ gprf_page *page = (gprf_page*)page_;
+ gprf_document *doc = page->doc;
+ fz_rect page_rect;
+ int i, y, x;
+
+ /* If we have no page, generate it. */
+ if (page->file == NULL)
+ {
+ generate_page(ctx, page);
+ }
+ /* If we have no tiles, generate them. */
+ if (page->num_tiles == 0)
+ {
+ read_tiles(ctx, page);
+ }
+
+ /* Send the images to the page */
+ page_rect.x0 = 0;
+ page_rect.y0 = 0;
+ page_rect.x1 = 72.0 * page->width / doc->res;
+ page_rect.y1 = 72.0 * page->height / doc->res;
+ fz_begin_page(ctx, dev, &page_rect, ctm);
+
+ i = 0;
+ for (y = 0; y < page->tile_height; y++)
+ {
+ double scale = GPRF_TILESIZE * 72.0 / doc->res;
+ for (x = 0; x < page->tile_width; x++)
+ {
+ fz_matrix local;
+ double scale_x = page->tiles[i]->w * 72.0 / doc->res;
+ double scale_y = page->tiles[i]->h * 72.0 / doc->res;
+ local.a = scale_x;
+ local.b = 0;
+ local.c = 0;
+ local.d = scale_y;
+ local.e = x * scale;
+ local.f = y * scale;
+ fz_concat(&local, &local, ctm);
+ fz_fill_image(ctx, dev, page->tiles[i++], &local, 1.0);
+ }
+ }
+ fz_end_page(ctx, dev);
+}
+
+
+static fz_page *
+gprf_load_page(fz_context *ctx, fz_document *doc_, int number)
+{
+ gprf_document *doc = (gprf_document*)doc_;
+ gprf_page *page = fz_new_page(ctx, sizeof *page);
+
+ fz_try(ctx)
+ {
+ page->super.bound_page = gprf_bound_page;
+ page->super.run_page_contents = gprf_run_page;
+ page->super.drop_page_imp = gprf_drop_page_imp;
+ page->doc = (gprf_document *)fz_keep_document(ctx, &doc->super);
+ page->number = number;
+ page->separations = fz_new_separations(ctx);
+ page->width = doc->page_dims[number].w;
+ page->height = doc->page_dims[number].h;
+ page->tile_width = (page->width + GPRF_TILESIZE-1)/GPRF_TILESIZE;
+ page->tile_height = (page->height + GPRF_TILESIZE-1)/GPRF_TILESIZE;
+ }
+ fz_catch(ctx)
+ {
+ fz_drop_page(ctx, (fz_page *)page);
+ fz_rethrow(ctx);
+ }
+
+ return (fz_page*)page;
+}
+
+static void
+gprf_close_document(fz_context *ctx, fz_document *doc_)
+{
+ gprf_document *doc = (gprf_document*)doc_;
+
+ if (doc == NULL)
+ return;
+ fz_free(ctx, doc->page_dims);
+ fz_free(ctx, doc->pdf_filename);
+
+ fz_free(ctx, doc);
+}
+
+static fz_document *
+gprf_open_document_with_stream(fz_context *ctx, fz_stream *file)
+{
+ gprf_document *doc;
+
+ doc = Memento_label(fz_new_document(ctx, sizeof(gprf_document)), "gprf_document");
+ doc->super.close = gprf_close_document;
+ doc->super.count_pages = gprf_count_pages;
+ doc->super.load_page = gprf_load_page;
+
+ fz_try(ctx)
+ {
+ int32_t val;
+ int i;
+ char buf[4096];
+
+ val = fz_read_int32le(ctx, file);
+ if (val != 0x4f525047)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Invalid file signature in gproof file");
+ val = fz_read_byte(ctx, file);
+ val |= fz_read_byte(ctx, file)<<8;
+ if (val != 1)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Invalid version in gproof file");
+ doc->res = fz_read_int32le(ctx, file);
+ if (doc->res < 0)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Invalid resolution in gproof file");
+ doc->num_pages = fz_read_int32le(ctx, file);
+ if (doc->num_pages < 0)
+ fz_throw(ctx, FZ_ERROR_GENERIC, "Invalid resolution in gproof file");
+ doc->page_dims = fz_calloc(ctx, doc->num_pages, sizeof(*doc->page_dims));
+
+ for (i = 0; i < doc->num_pages; i++)
+ {
+ doc->page_dims[i].w = fz_read_int32le(ctx, file);
+ doc->page_dims[i].h = fz_read_int32le(ctx, file);
+ }
+ fz_read_line(ctx, file, buf, sizeof(buf));
+ doc->pdf_filename = fz_strdup(ctx, buf);
+ }
+ fz_catch(ctx)
+ {
+ gprf_close_document(ctx, (fz_document*)doc);
+ fz_rethrow(ctx);
+ }
+
+ return (fz_document*)doc;
+}
+
+static fz_document *
+gprf_open_document(fz_context *ctx, const char *filename)
+{
+ fz_stream *file = fz_open_file(ctx, filename);
+ fz_document *doc;
+
+ fz_try(ctx)
+ {
+ doc = gprf_open_document_with_stream(ctx, file);
+ }
+ fz_always(ctx)
+ {
+ fz_drop_stream(ctx, file);
+ }
+ fz_catch(ctx)
+ {
+ fz_rethrow(ctx);
+ }
+ return doc;
+}
+
+static int
+gprf_recognize(fz_context *doc, const char *magic)
+{
+ char *ext = strrchr(magic, '.');
+ if (ext)
+ if (!fz_strcasecmp(ext, ".gproof"))
+ return 100;
+ if (!strcmp(magic, "application/ghostproof"))
+ return 100;
+ return 0;
+}
+
+fz_document_handler gprf_document_handler =
+{
+ &gprf_recognize,
+ &gprf_open_document,
+ &gprf_open_document_with_stream
+};