fputs("\
Usage: image-tool [options] infile [outfile]\n\
\n\
--q --quiet no progress messages\n\
--f --input-format input image format (jpeg, gif, png)\n\
--F --output-format output image format\n\
--s --size force output dimensions (100x200)\n\
--b --fit-to-box scale to fit the box (100x200)\n\
--c --colorspace force output colorspace (Gray, GrayAlpha, RGB, RGBAlpha)\n\
--Q --jpeg-quality JPEG quality (1..100)\n\
--g --background background color (hexadecimal RRGGBB)\n\
+-q --quiet no progress messages\n\
+-f --input-format input image format (jpeg, gif, png)\n\
+-F --output-format output image format\n\
+-s --size force output dimensions (100x200)\n\
+-b --fit-to-box scale to fit the box (100x200)\n\
+-c --colorspace force output colorspace (Gray, GrayAlpha, RGB, RGBAlpha)\n\
+-Q --jpeg-quality JPEG quality (1..100)\n\
+-g --background background color (hexadecimal RRGGBB)\n\
+-G --default-background background applied only if the image contains no background info (RRGGBB, default=FFFFFF)\n\
+-a --remove-alpha remove alpha channel\n\
", stderr);
exit(1);
}
-static char *shortopts = "qf:F:s:b:c:Q:g:";
+static char *shortopts = "qf:F:s:b:c:Q:g:G:a";
static struct option longopts[] =
{
- { "quiet", 0, 0, 'q' },
- { "input-format", 0, 0, 'f' },
- { "output-format", 0, 0, 'F' },
- { "size", 0, 0, 's' },
- { "fit-to-box", 0, 0, 'b' },
- { "colorspace", 0, 0, 'c' },
- { "jpeg-quality", 0, 0, 'Q' },
- { "background", 0, 0, 'g' },
- { NULL, 0, 0, 0 }
+ { "quiet", 0, 0, 'q' },
+ { "input-format", 0, 0, 'f' },
+ { "output-format", 0, 0, 'F' },
+ { "size", 0, 0, 's' },
+ { "fit-to-box", 0, 0, 'b' },
+ { "colorspace", 0, 0, 'c' },
+ { "jpeg-quality", 0, 0, 'Q' },
+ { "background", 0, 0, 'g' },
+ { "default-background", 0, 0, 'G' },
+ { "remove-alpha", 0, 0, 'a' },
+ { NULL, 0, 0, 0 }
};
static uns verbose = 1;
static uns channels_format;
static uns jpeg_quality;
static struct color background_color;
+static struct color default_background_color;
+static uns remove_alpha;
+
+static void
+parse_color(struct color *color, byte *s)
+{
+ if (strlen(s) != 6)
+ usage();
+ s = 0;
+ char *end;
+ long int v = strtol(s, &end, 16);
+ if (errno || *end || v < 0)
+ usage();
+ color_make_rgb(color, (v >> 16) & 255, (v >> 8) & 255, v & 255);
+}
#define MSG(x...) do{ if (verbose) log(L_INFO, ##x); }while(0)
{
log_init(argv[0]);
int opt;
+ default_background_color = color_white;
while ((opt = getopt_long(argc, argv, shortopts, longopts, NULL)) >= 0)
switch (opt)
{
usage();
break;
case 'g':
- {
- if (strlen(optarg) != 6)
- usage();
- errno = 0;
- char *end;
- long int v = strtol(optarg, &end, 16);
- if (errno || *end || v < 0)
- usage();
- color_make_rgb(&background_color, (v >> 16) & 255, (v >> 8) & 255, v & 255);
- }
+ parse_color(&background_color, optarg);
+ break;
+ case 'G':
+ parse_color(&default_background_color, optarg);
+ break;
+ case 'a':
+ remove_alpha++;
break;
default:
usage();
struct image_thread it;
struct image_io io;
image_thread_init(&it);
- image_io_init(&it, &io);
+ if (!image_io_init(&it, &io))
+ die("Cannot initialize image I/O (%s)", it.err_msg);
MSG("Reading %s", input_file_name);
io.fastbuf = bopen(input_file_name, O_RDONLY, 1 << 18);
if (background_color.color_space)
io.background_color = background_color;
else if (!io.background_color.color_space)
- io.background_color = color_white;
+ io.background_color = default_background_color;
+ if (remove_alpha)
+ io.flags &= ~IMAGE_ALPHA;
if (channels_format)
io.flags = io.flags & ~IMAGE_PIXEL_FORMAT | channels_format;
if (!(io.flags & IMAGE_ALPHA))
#define BYTE_TO_QUANTUM(x) ((uns)(x) << QUANTUM_SCALE)
#define OPACITY_MAX ((1 << QuantumDepth) - 1)
+static uns libmagick_counter;
+
struct magick_read_data {
ExceptionInfo exception;
ImageInfo *info;
Image *image;
};
-static inline void
+int
+libmagick_init(struct image_io *io UNUSED)
+{
+ // FIXME: lock
+ if (!libmagick_counter++)
+ InitializeMagick(NULL);
+ return 1;
+}
+
+void
+libmagick_cleanup(struct image_io *io UNUSED)
+{
+ // FIXME: lock
+ if (!--libmagick_counter)
+ DestroyMagick();
+}
+
+static void
libmagick_destroy_read_data(struct magick_read_data *rd)
{
if (rd->image)
DestroyImage(rd->image);
DestroyImageInfo(rd->info);
DestroyExceptionInfo(&rd->exception);
- DestroyMagick();
}
static void
DBG("libmagick_read_cancel()");
struct magick_read_data *rd = io->read_data;
-
- DestroyImage(rd->image);
libmagick_destroy_read_data(rd);
}
breadb(io->fastbuf, buf, buf_size);
/* Allocate read structure */
- struct magick_read_data *rd = io->read_data = mp_alloc(io->internal_pool, sizeof(*rd));
+ struct magick_read_data *rd = io->read_data = mp_alloc_zero(io->internal_pool, sizeof(*rd));
/* Initialize GraphicsMagick */
- InitializeMagick(NULL);
GetExceptionInfo(&rd->exception);
rd->info = CloneImageInfo(NULL);
rd->info->subrange = 1;
switch (rd->image->colorspace)
{
case GRAYColorspace:
- io->flags = COLOR_SPACE_GRAYSCALE | IMAGE_ALPHA;
+ io->flags = COLOR_SPACE_GRAYSCALE;
break;
default:
- io->flags = COLOR_SPACE_RGB | IMAGE_ALPHA;
+ io->flags = COLOR_SPACE_RGB;
break;
}
+ if (rd->image->matte)
+ io->flags |= IMAGE_ALPHA;
io->number_of_colors = rd->image->colors;
if (rd->image->storage_class == PseudoClass && rd->image->compression != JPEGCompression)
io->flags |= IMAGE_IO_HAS_PALETTE;
int result = 0;
ExceptionInfo exception;
ImageInfo *info;
- InitializeMagick(NULL);
GetExceptionInfo(&exception);
info = CloneImageInfo(NULL);
err:
DestroyImageInfo(info);
DestroyExceptionInfo(&exception);
- DestroyMagick();
return result;
}
#ifndef _IMAGES_IO_MAIN_H
#define _IMAGES_IO_MAIN_H
+static inline int libjpeg_init(struct image_io *io UNUSED) { return 1; }
+static inline void libjpeg_cleanup(struct image_io *io UNUSED) {}
int libjpeg_read_header(struct image_io *io);
int libjpeg_read_data(struct image_io *io);
int libjpeg_write(struct image_io *io);
+static inline int libpng_init(struct image_io *io UNUSED) { return 1; }
+static inline void libpng_cleanup(struct image_io *io UNUSED) {}
int libpng_read_header(struct image_io *io);
int libpng_read_data(struct image_io *io);
int libpng_write(struct image_io *io);
+static inline int libungif_init(struct image_io *io UNUSED) { return 1; }
+static inline void libungif_cleanup(struct image_io *io UNUSED) {}
int libungif_read_header(struct image_io *io);
int libungif_read_data(struct image_io *io);
+int libmagick_init(struct image_io *io);
+void libmagick_cleanup(struct image_io *io);
int libmagick_read_header(struct image_io *io);
int libmagick_read_data(struct image_io *io);
int libmagick_write(struct image_io *io);