You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
175 lines
4.6 KiB
175 lines
4.6 KiB
3 months ago
|
/*
|
||
|
* Image duplicates testing
|
||
|
*
|
||
|
* (c) 2006 Pavel Charvat <pchar@ucw.cz>
|
||
|
*
|
||
|
* This software may be freely distributed and used according to the terms
|
||
|
* of the GNU General Public License.
|
||
|
*/
|
||
|
|
||
|
#include <ucw/lib.h>
|
||
|
#include <ucw/getopt.h>
|
||
|
#include <ucw/fastbuf.h>
|
||
|
#include <ucw/mempool.h>
|
||
|
#include <images/images.h>
|
||
|
#include <images/color.h>
|
||
|
#include <images/duplicates.h>
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <errno.h>
|
||
|
#include <stdio.h>
|
||
|
|
||
|
static void NONRET
|
||
|
usage(void)
|
||
|
{
|
||
|
fputs("\
|
||
|
Usage: ucw-image-dup-test [options] image1 image2 \n\
|
||
|
\n\
|
||
|
-q --quiet no progress messages\n\
|
||
|
-f --format-1 image1 format (jpeg, gif, png)\n\
|
||
|
-F --format-2 image2 format\n\
|
||
|
-g --background background color (hexadecimal RRGGBB)\n\
|
||
|
-t --transformations hexadecimal value of allowed transformtion (1=identity, FF=all)\n\
|
||
|
", stderr);
|
||
|
exit(1);
|
||
|
}
|
||
|
|
||
|
static char *shortopts = "qf:F:g:t:" CF_SHORT_OPTS;
|
||
|
static struct option longopts[] =
|
||
|
{
|
||
|
CF_LONG_OPTS
|
||
|
{ "quiet", 0, 0, 'q' },
|
||
|
{ "format-1", 0, 0, 'f' },
|
||
|
{ "format-2", 0, 0, 'F' },
|
||
|
{ "background", 0, 0, 'g' },
|
||
|
{ "transormations", 0, 0, 't' },
|
||
|
{ NULL, 0, 0, 0 }
|
||
|
};
|
||
|
|
||
|
static uint verbose = 1;
|
||
|
static byte *file_name_1;
|
||
|
static byte *file_name_2;
|
||
|
static enum image_format format_1;
|
||
|
static enum image_format format_2;
|
||
|
static struct color background_color;
|
||
|
static uint transformations = IMAGE_DUP_TRANS_ALL;
|
||
|
|
||
|
#define MSG(x...) do{ if (verbose) msg(L_INFO, ##x); }while(0)
|
||
|
|
||
|
int
|
||
|
main(int argc, char **argv)
|
||
|
{
|
||
|
log_init(argv[0]);
|
||
|
int opt;
|
||
|
while ((opt = cf_getopt(argc, argv, shortopts, longopts, NULL)) >= 0)
|
||
|
switch (opt)
|
||
|
{
|
||
|
case 'q':
|
||
|
verbose = 0;
|
||
|
break;
|
||
|
case 'f':
|
||
|
if (!(format_1 = image_extension_to_format(optarg)))
|
||
|
usage();
|
||
|
break;
|
||
|
case 'F':
|
||
|
if (!(format_2 = image_extension_to_format(optarg)))
|
||
|
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);
|
||
|
}
|
||
|
break;
|
||
|
case 't':
|
||
|
{
|
||
|
errno = 0;
|
||
|
char *end;
|
||
|
long int v = strtol(optarg, &end, 16);
|
||
|
if (errno || *end || v < 0 || v > 0xff)
|
||
|
usage();
|
||
|
transformations = v;
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
usage();
|
||
|
}
|
||
|
|
||
|
if (argc != optind + 2)
|
||
|
usage();
|
||
|
file_name_1 = argv[optind++];
|
||
|
file_name_2 = argv[optind];
|
||
|
|
||
|
#define TRY(x) do{ if (!(x)) exit(1); }while(0)
|
||
|
MSG("Initializing image library");
|
||
|
struct image_context ctx;
|
||
|
struct image_dup_context idc;
|
||
|
struct image_io io;
|
||
|
image_context_init(&ctx);
|
||
|
image_dup_context_init(&ctx, &idc);
|
||
|
|
||
|
struct image *img1, *img2;
|
||
|
|
||
|
TRY(image_io_init(&ctx, &io));
|
||
|
MSG("Reading %s", file_name_1);
|
||
|
io.fastbuf = bopen(file_name_1, O_RDONLY, 1 << 18);
|
||
|
io.format = format_1 ? : image_file_name_to_format(file_name_1);
|
||
|
TRY(image_io_read_header(&io));
|
||
|
io.flags = COLOR_SPACE_RGB | IMAGE_IO_USE_BACKGROUND;
|
||
|
if (background_color.color_space)
|
||
|
io.background_color = background_color;
|
||
|
else if (!io.background_color.color_space)
|
||
|
io.background_color = color_black;
|
||
|
TRY(image_io_read_data(&io, 1));
|
||
|
bclose(io.fastbuf);
|
||
|
img1 = io.image;
|
||
|
MSG("Image size=%ux%u", img1->cols, img1->rows);
|
||
|
|
||
|
image_io_reset(&io);
|
||
|
MSG("Reading %s", file_name_2);
|
||
|
io.fastbuf = bopen(file_name_2, O_RDONLY, 1 << 18);
|
||
|
io.format = format_2 ? : image_file_name_to_format(file_name_2);
|
||
|
TRY(image_io_read_header(&io));
|
||
|
io.flags = COLOR_SPACE_RGB | IMAGE_IO_USE_BACKGROUND;
|
||
|
if (background_color.color_space)
|
||
|
io.background_color = background_color;
|
||
|
else if (!io.background_color.color_space)
|
||
|
io.background_color = color_black;
|
||
|
TRY(image_io_read_data(&io, 1));
|
||
|
bclose(io.fastbuf);
|
||
|
img2 = io.image;
|
||
|
image_io_cleanup(&io);
|
||
|
MSG("Image size=%ux%u", img2->cols, img2->rows);
|
||
|
|
||
|
struct image_dup *dup1, *dup2;
|
||
|
struct mempool *pool = mp_new(1 << 18);
|
||
|
MSG("Creating internal structures");
|
||
|
dup1 = mp_start(pool, image_dup_estimate_size(img1->cols, img1->rows, 1, idc.qtree_limit));
|
||
|
uint size = image_dup_new(&idc, img1, dup1, 1);
|
||
|
TRY(size);
|
||
|
mp_end(pool, (void *)dup1 + size);
|
||
|
dup2 = mp_start(pool, image_dup_estimate_size(img2->cols, img2->rows, 1, idc.qtree_limit));
|
||
|
size = image_dup_new(&idc, img2, dup2, 1);
|
||
|
TRY(size);
|
||
|
mp_end(pool, (void *)dup2 + size);
|
||
|
|
||
|
idc.flags = transformations | IMAGE_DUP_SCALE | IMAGE_DUP_WANT_ALL;
|
||
|
MSG("Similarity bitmap %02x", image_dup_compare(&idc, dup1, dup2));
|
||
|
|
||
|
mp_delete(pool);
|
||
|
|
||
|
image_destroy(img1);
|
||
|
image_destroy(img2);
|
||
|
image_dup_context_cleanup(&idc);
|
||
|
image_context_cleanup(&ctx);
|
||
|
MSG("Done.");
|
||
|
return 0;
|
||
|
}
|