#include "images/images.h"
#include <string.h>
-#define MAX_IMAGE_SIZE (1 << 30)
+#define MAX_IMAGE_BYTES (1 << 30)
void
image_thread_init(struct image_thread *it)
image_new(struct image_thread *it, uns cols, uns rows, uns flags, struct mempool *pool)
{
DBG("image_new(cols=%u rows=%u flags=0x%x pool=%p)", cols, rows, flags, pool);
- if (cols >= 0x10000 || rows >= 0x10000)
+ if (cols > IMAGE_MAX_SIZE || rows > IMAGE_MAX_SIZE)
{
image_thread_err(it, IMAGE_ERR_INVALID_DIMENSIONS, "Image dimension(s) too large");
return NULL;
ASSERT(0);
}
if (flags & IMAGE_SSE_ALIGNED)
- align = MAX(16, sizeof(uns));
+ align = IMAGE_SSE_ALIGN_SIZE;
else if (flags & IMAGE_PIXELS_ALIGNED)
align = pixel_size;
else
align = 1;
row_size = cols * pixel_size;
row_size = ALIGN(row_size, align);
- u64 image_size_64 = row_size * rows;
- if (image_size_64 > MAX_IMAGE_SIZE)
+ u64 image_size_64 = (u64)row_size * rows;
+ u64 bytes_64 = image_size_64 + (sizeof(struct image) + IMAGE_SSE_ALIGN_SIZE - 1 + sizeof(uns));
+ if (bytes_64 > MAX_IMAGE_BYTES)
{
image_thread_err(it, IMAGE_ERR_INVALID_DIMENSIONS, "Image does not fit in memory");
return NULL;
image_thread_err(it, IMAGE_ERR_INVALID_DIMENSIONS, "Zero dimension(s)");
return NULL;
}
- uns size = sizeof(struct image) + image_size + MAX(16, sizeof(uns)) - 1 + sizeof(uns);
- img = pool ? mp_alloc(pool, size) : xmalloc(size);
+ img = pool ? mp_alloc(pool, (uns)bytes_64) : xmalloc((uns)bytes_64);
bzero(img, sizeof(struct image));
byte *p = (byte *)img + sizeof(struct image);
- img->pixels = ALIGN_PTR(p, MAX(16, sizeof(uns)));
+ img->pixels = ALIGN_PTR(p, IMAGE_SSE_ALIGN_SIZE);
img->flags = flags;
img->pixel_size = pixel_size;
img->cols = cols;
/*
- * Image Library -- GrapgicsMagick
+ * Image Library -- GraphicsMagick
*
* (c) 2006 Pavel Charvat <pchar@ucw.cz>
*
#define LOCAL_DEBUG
#include "lib/lib.h"
+#include "lib/mempool.h"
+#include "lib/fastbuf.h"
#include "images/images.h"
#include <sys/types.h>
#include <stdlib.h>
#include <stdio.h>
#include <magick/api.h>
-int
-libmagick_read_header(struct image_io *io)
-{
- image_thread_err(io->thread, IMAGE_ERR_NOT_IMPLEMENTED, "GraphicsMagick read not implemented.");
- return 0;
-}
-
-int
-libmagick_read_data(struct image_io *io UNUSED)
-{
- ASSERT(0);
-}
-
-int
-libmagick_write(struct image_io *io)
-{
- image_thread_err(io->thread, IMAGE_ERR_NOT_IMPLEMENTED, "GraphicsMagick write not implemented.");
- return 0;
-}
+#define MAX_FILE_SIZE (1 << 30)
+#define QUANTUM_SCALE (QuantumDepth - 8)
+#define QUANTUM_TO_BYTE(x) ((uns)(x) >> QUANTUM_SCALE)
-#if 0
-struct magick_internals {
+struct magick_read_data {
ExceptionInfo exception;
- QuantizeInfo quantize;
ImageInfo *info;
+ Image *image;
};
static inline void
-magick_cleanup(struct image_io *io)
+libmagick_destroy_read_data(struct magick_read_data *rd)
{
- DestroyImageInfo(io->internals->info);
- DestroyExceptionInfo(&io->internals->exception);
+ if (rd->image)
+ DestroyImage(rd->image);
+ DestroyImageInfo(rd->info);
+ DestroyExceptionInfo(&rd->exception);
DestroyMagick();
}
-static int
-magick_read_header(struct image_io *io)
+static void
+libmagick_read_cancel(struct image_io *io)
{
- DBG("magick_read_header()");
- struct magick_internals *i = io->internals = mp_alloc(io->pool, sizeof(*i));
+ DBG("libmagick_read_cancel()");
- InitializeMagick(NULL);
- GetExceptionInfo(&i->exception);
- i->info = CloneImageInfo(NULL);
- i->info->subrange = 1;
- GetQuantizeInfo(&i->quantize);
- i->quantize.colorspace = RGBColorspace;
+ struct magick_read_data *rd = io->read_data;
+
+ DestroyImage(rd->image);
+ libmagick_destroy_read_data(rd);
+}
- uns len = bfilesize(io->fastbuf);
- byte *buf = mp_alloc(io->pool, len);
- len = bread(io->fastbuf, buf, len);
+int
+libmagick_read_header(struct image_io *io)
+{
+ DBG("libmagick_read_header()");
- Image *image = BlobToImage(magick_info, imo->thumb_data, imo->thumb_size, &magick_exception);
- if (unlikely(!image))
- goto error;
+ /* Read entire stream */
+ sh_off_t file_size = bfilesize(io->fastbuf) - btell(io->fastbuf);
+ if (file_size > MAX_FILE_SIZE)
+ {
+ image_thread_err(io->thread, IMAGE_ERR_READ_FAILED, "Too long stream.");
+ return 0;
+ }
+ uns buf_size = file_size;
+ byte *buf = xmalloc(buf_size);
+ bread(io->fastbuf, buf, buf_size);
+
+ /* Allocate read structure */
+ struct magick_read_data *rd = io->read_data = mp_alloc(io->internal_pool, sizeof(*rd));
+
+ /* Initialize GraphicsMagick */
+ InitializeMagick(NULL);
+ GetExceptionInfo(&rd->exception);
+ rd->info = CloneImageInfo(NULL);
+ rd->info->subrange = 1;
+
+ /* Read the image */
+ rd->image = BlobToImage(rd->info, buf, buf_size, &rd->exception);
+ xfree(buf);
+ if (!rd->image)
+ {
+ image_thread_err(io->thread, IMAGE_ERR_READ_FAILED, "GraphicsMagick failed to read the image.");
+ goto err;
+ }
+ if (rd->image->columns > IMAGE_MAX_SIZE || rd->image->rows > IMAGE_MAX_SIZE)
+ {
+ image_thread_err(io->thread, IMAGE_ERR_INVALID_DIMENSIONS, "Image too large.");
+ goto err;
+ }
- // FIXME
+ /* Fill image parameters */
+ if (!io->cols)
+ io->cols = rd->image->columns;
+ if (!io->rows)
+ io->rows = rd->image->rows;
+ if (!(io->flags & IMAGE_CHANNELS_FORMAT))
+ {
+ switch (rd->image->colorspace)
+ {
+ case GRAYColorspace:
+ io->flags |= COLOR_SPACE_GRAYSCALE | IMAGE_ALPHA;
+ break;
+ default:
+ io->flags |= COLOR_SPACE_RGB | IMAGE_ALPHA;
+ break;
+ }
+ }
+
+ io->read_cancel = libmagick_read_cancel;
return 1;
-error:
- magick_cleanup(io);
+
+err:
+ libmagick_destroy_read_data(rd);
return 0;
}
-static int
-magick_read_data(struct image_io *io)
+static inline byte
+libmagick_pixel_to_gray(PixelPacket *pixel)
{
- DBG("magick_read_data()");
-
- // FIXME
-
- magick_cleanup(io);
- return 1;
+ return ((uns)pixel->red * 19660 + (uns)pixel->green * 38666 + (uns)pixel->blue * 7210) >> (16 + QUANTUM_SCALE);
}
-static int
-magick_decompress_thumbnail(struct image_obj *imo)
+int
+libmagick_read_data(struct image_io *io)
{
- DBG("Quantizing image");
- QuantizeImage(&magick_quantize, image);
- DBG("Converting pixels");
- PixelPacket *pixels = (PixelPacket *)AcquireImagePixels(image, 0, 0, image->columns, image->rows, &magick_exception);
- ASSERT(pixels);
- uns size = image->columns * image->rows;
- byte *p = imo->thumb.pixels = mp_alloc(imo->pool, imo->thumb.size = size * 3);
- for (uns i = 0; i < size; i++)
+ DBG("libmagick_read_data()");
+
+ struct magick_read_data *rd = io->read_data;
+
+ /* Quantize image */
+ switch (rd->image->colorspace)
+ {
+ case RGBColorspace:
+ case GRAYColorspace:
+ break;
+ default: ;
+ QuantizeInfo quantize;
+ GetQuantizeInfo(&quantize);
+ quantize.colorspace = RGBColorspace;
+ QuantizeImage(&quantize, rd->image);
+ break;
+ }
+
+ /* Allocate image for conversion */
+ int need_scale = io->cols != rd->image->columns || io->rows != rd->image->rows;
+ int need_destroy = need_scale || !io->pool;
+ struct image *img = need_scale ?
+ image_new(io->thread, rd->image->columns, rd->image->rows, io->flags & IMAGE_CHANNELS_FORMAT, NULL) :
+ image_new(io->thread, io->cols, io->rows, io->flags, io->pool);
+ if (!img)
+ goto err;
+
+ /* Acquire pixels */
+ PixelPacket *src = (PixelPacket *)AcquireImagePixels(rd->image, 0, 0, rd->image->columns, rd->image->rows, &rd->exception);
+ if (!src)
+ {
+ image_thread_err(io->thread, IMAGE_ERR_READ_FAILED, "Cannot acquire image pixels.");
+ goto err;
+ }
+
+ /* Convert pixels */
+ switch (img->pixel_size)
+ {
+ case 1:
+# define IMAGE_WALK_INLINE
+# define IMAGE_WALK_UNROLL 4
+# define IMAGE_WALK_COL_STEP 1
+# define IMAGE_WALK_DO_STEP do{ \
+ pos[0] = libmagick_pixel_to_gray(src); \
+ src++; }while(0)
+# include "images/image-walk.h"
+ break;
+
+ case 2:
+# define IMAGE_WALK_INLINE
+# define IMAGE_WALK_UNROLL 4
+# define IMAGE_WALK_COL_STEP 2
+# define IMAGE_WALK_DO_STEP do{ \
+ pos[0] = libmagick_pixel_to_gray(src); \
+ pos[1] = QUANTUM_TO_BYTE(src->opacity); \
+ src++; }while(0)
+# include "images/image-walk.h"
+ break;
+
+ case 3:
+# define IMAGE_WALK_INLINE
+# define IMAGE_WALK_UNROLL 4
+# define IMAGE_WALK_COL_STEP 3
+# define IMAGE_WALK_DO_STEP do{ \
+ pos[0] = QUANTUM_TO_BYTE(src->red); \
+ pos[1] = QUANTUM_TO_BYTE(src->green); \
+ pos[2] = QUANTUM_TO_BYTE(src->blue); \
+ src++; }while(0)
+# include "images/image-walk.h"
+ break;
+
+ case 4:
+# define IMAGE_WALK_INLINE
+# define IMAGE_WALK_UNROLL 4
+# define IMAGE_WALK_COL_STEP 4
+# define IMAGE_WALK_DO_STEP do{ \
+ pos[0] = QUANTUM_TO_BYTE(src->red); \
+ pos[1] = QUANTUM_TO_BYTE(src->green); \
+ pos[2] = QUANTUM_TO_BYTE(src->blue); \
+ pos[3] = QUANTUM_TO_BYTE(src->opacity); \
+ src++; }while(0)
+# include "images/image-walk.h"
+ break;
+
+ default:
+ ASSERT(0);
+ }
+
+ /* Free GraphicsMagick structures */
+ libmagick_destroy_read_data(rd);
+
+ /* Scale image */
+ if (need_scale)
{
- p[0] = pixels->red >> (QuantumDepth - 8);
- p[1] = pixels->green >> (QuantumDepth - 8);
- p[2] = pixels->blue >> (QuantumDepth - 8);
- p += 3;
- pixels++;
+ struct image *img2 = image_new(io->thread, io->cols, io->rows, io->flags, io->pool);
+ if (!img2)
+ goto err2;
+ int result = image_scale(io->thread, img2, img);
+ image_destroy(io->thread, img);
+ img = img2;
+ need_destroy = !io->pool;
+ if (!result)
+ goto err2;
}
- DestroyImage(image);
+
+ /* Success */
+ io->image = img;
+ io->image_destroy = need_destroy;
return 1;
+
+ /* Free structures */
+err:
+ libmagick_destroy_read_data(rd);
+err2:
+ if (need_destroy)
+ image_destroy(io->thread, img);
+ return 0;
+}
+
+int
+libmagick_write(struct image_io *io)
+{
+ image_thread_err(io->thread, IMAGE_ERR_NOT_IMPLEMENTED, "GraphicsMagick write not implemented.");
+ return 0;
}
-#endif