mirror of
git://projects.qi-hardware.com/eda-tools.git
synced 2025-04-21 12:27:27 +03:00
Initial commit of files moved over from ben-wpan.
This commit is contained in:
3
schhist/ppmdiff/Makefile
Normal file
3
schhist/ppmdiff/Makefile
Normal file
@@ -0,0 +1,3 @@
|
||||
CFLAGS=-Wall -g
|
||||
|
||||
ppmdiff:
|
||||
412
schhist/ppmdiff/ppmdiff.c
Normal file
412
schhist/ppmdiff/ppmdiff.c
Normal file
@@ -0,0 +1,412 @@
|
||||
/*
|
||||
* ppmdiff.c - Mark differences in two PPM files
|
||||
*
|
||||
* Written 2010 by Werner Almesberger
|
||||
* Copyright 2010 Werner Almesberger
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*/
|
||||
|
||||
|
||||
#include <stdint.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include <string.h>
|
||||
|
||||
|
||||
static uint8_t a_only[3] = { 255, 0, 0 };
|
||||
static uint8_t b_only[3] = { 0, 255, 0 };
|
||||
static uint8_t both[3] = { 220, 220, 220 };
|
||||
static uint8_t frame[3] = { 0, 0, 255 };
|
||||
static uint8_t frame_fill[3] = { 255, 255, 200 };
|
||||
static int frame_dist = 40;
|
||||
static int frame_width = 2;
|
||||
|
||||
|
||||
static uint8_t *load_ppm(const char *name, int *x, int *y)
|
||||
{
|
||||
FILE *file;
|
||||
char line[100];
|
||||
int this_x, this_y, depth;
|
||||
int n;
|
||||
uint8_t *img;
|
||||
|
||||
file = fopen(name, "r");
|
||||
if (!file) {
|
||||
perror(name);
|
||||
exit(1);
|
||||
}
|
||||
if (!fgets(line, sizeof(line), file)) {
|
||||
fprintf(stderr, "can't read file type\n");
|
||||
exit(1);
|
||||
}
|
||||
if (strcmp(line, "P6\n")) {
|
||||
fprintf(stderr, "file type must be P6, not %s", line);
|
||||
exit(1);
|
||||
}
|
||||
if (!fgets(line, sizeof(line), file)) {
|
||||
fprintf(stderr, "can't read resolution\n");
|
||||
exit(1);
|
||||
}
|
||||
if (sscanf(line, "%d %d", &this_x, &this_y) != 2) {
|
||||
fprintf(stderr, "can't parse resolution: %s", line);
|
||||
exit(1);
|
||||
}
|
||||
if (*x || *y) {
|
||||
if (*x != this_x || *y != this_y) {
|
||||
fprintf(stderr,
|
||||
"resolution changed from %dx%d to %dx%d\n",
|
||||
*x, *y, this_x, this_y);
|
||||
exit(1);
|
||||
}
|
||||
} else {
|
||||
*x = this_x;
|
||||
*y = this_y;
|
||||
}
|
||||
if (!fgets(line, sizeof(line), file)) {
|
||||
fprintf(stderr, "can't read depth\n");
|
||||
exit(1);
|
||||
}
|
||||
if (sscanf(line, "%d", &depth) != 1) {
|
||||
fprintf(stderr, "can't parse depth: %s", line);
|
||||
exit(1);
|
||||
}
|
||||
if (depth != 255) {
|
||||
fprintf(stderr, "depth must be 255, not %d\n", depth);
|
||||
exit(1);
|
||||
}
|
||||
n = *x**y*3;
|
||||
img = malloc(n);
|
||||
if (!img) {
|
||||
perror("malloc");
|
||||
exit(1);
|
||||
}
|
||||
if (fread(img, 1, n, file) != n) {
|
||||
fprintf(stderr, "can't read %d bytes\n", n);
|
||||
exit(1);
|
||||
}
|
||||
fclose(file);
|
||||
return img;
|
||||
}
|
||||
|
||||
|
||||
static struct area {
|
||||
int x0, y0, x1, y1;
|
||||
struct area *next;
|
||||
} *areas = NULL;
|
||||
|
||||
|
||||
static void add_area(struct area **root, int x0, int y0, int x1, int y1)
|
||||
{
|
||||
while (*root) {
|
||||
struct area *area = *root;
|
||||
|
||||
if (area->x0 > x1 || area->y0 > y1 ||
|
||||
area->x1 < x0 || area->y1 < y0) {
|
||||
root = &(*root)->next;
|
||||
continue;
|
||||
}
|
||||
x0 = x0 < area->x0 ? x0 : area->x0;
|
||||
y0 = y0 < area->y0 ? y0 : area->y0;
|
||||
x1 = x1 > area->x1 ? x1 : area->x1;
|
||||
y1 = y1 > area->y1 ? y1 : area->y1;
|
||||
*root = area->next;
|
||||
free(area);
|
||||
add_area(&areas, x0, y0, x1, y1);
|
||||
return;
|
||||
}
|
||||
*root = malloc(sizeof(**root));
|
||||
if (!*root) {
|
||||
perror("malloc");
|
||||
exit(1);
|
||||
}
|
||||
(*root)->x0 = x0;
|
||||
(*root)->y0 = y0;
|
||||
(*root)->x1 = x1;
|
||||
(*root)->y1 = y1;
|
||||
(*root)->next = NULL;
|
||||
}
|
||||
|
||||
|
||||
static void change(int x, int y)
|
||||
{
|
||||
add_area(&areas,
|
||||
x-frame_dist, y-frame_dist, x+frame_dist, y+frame_dist);
|
||||
}
|
||||
|
||||
|
||||
static void set_pixel(uint8_t *p, const uint8_t *color, const uint8_t *value)
|
||||
{
|
||||
double f;
|
||||
int i;
|
||||
|
||||
f = (255-(value[0] | value[1] | value[2]))/255.0;
|
||||
for (i = 0; i != 3; i++)
|
||||
p[i] = 255-(255-color[i])*f;
|
||||
}
|
||||
|
||||
|
||||
static uint8_t *diff(const uint8_t *a, const uint8_t *b, int xres, int yres,
|
||||
int mark_areas)
|
||||
{
|
||||
uint8_t *res, *p;
|
||||
int x, y;
|
||||
unsigned val_a, val_b;
|
||||
|
||||
res = p = malloc(xres*yres*3);
|
||||
if (!res) {
|
||||
perror("malloc");
|
||||
exit(1);
|
||||
}
|
||||
for (y = 0; y != yres; y++)
|
||||
for (x = 0; x != xres; x++) {
|
||||
val_a = a[0]+a[1]+a[2];
|
||||
val_b = b[0]+b[1]+b[2];
|
||||
if (val_a == val_b) {
|
||||
set_pixel(p, both, b);
|
||||
} else if (val_a < val_b) {
|
||||
set_pixel(p, a_only, a);
|
||||
if (mark_areas)
|
||||
change(x, y);
|
||||
} else if (val_a > val_b) {
|
||||
set_pixel(p, b_only, b);
|
||||
if (mark_areas)
|
||||
change(x, y);
|
||||
} else {
|
||||
abort(); /* no longer used */
|
||||
memset(p, 255, 3);
|
||||
// memcpy(p, "\0\0\xff", 3);
|
||||
}
|
||||
a += 3;
|
||||
b += 3;
|
||||
p += 3;
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
|
||||
static void shadow_diff(const uint8_t *a, const uint8_t *b, int xres, int yres)
|
||||
{
|
||||
int x, y;
|
||||
|
||||
for (y = 0; y != yres; y++)
|
||||
for (x = 0; x != xres; x++) {
|
||||
if (memcmp(a, b, 3))
|
||||
change(x, y);
|
||||
a += 3;
|
||||
b += 3;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void point(uint8_t *img, int x, int y, int xres, int yres)
|
||||
{
|
||||
uint8_t *p;
|
||||
|
||||
if (x < 0 || y < 0 || x >= xres || y >= yres)
|
||||
return;
|
||||
p = img+(y*xres+x)*3;
|
||||
if ((p[0] & p[1] & p[2]) != 255)
|
||||
return;
|
||||
memcpy(p, frame, 3);
|
||||
}
|
||||
|
||||
|
||||
static void hline(uint8_t *img, int x0, int x1, int y, int xres, int yres)
|
||||
{
|
||||
int x;
|
||||
|
||||
for (x = x0; x <= x1; x++)
|
||||
point(img, x, y, xres, yres);
|
||||
}
|
||||
|
||||
|
||||
static void vline(uint8_t *img, int y0, int y1, int x, int xres, int yres)
|
||||
{
|
||||
int y;
|
||||
|
||||
for (y = y0; y <= y1; y++)
|
||||
point(img, x, y, xres, yres);
|
||||
}
|
||||
|
||||
|
||||
static void fill(uint8_t *img, int x0, int y0, int x1, int y1,
|
||||
int xres, int yres)
|
||||
{
|
||||
int x, y;
|
||||
uint8_t *p;
|
||||
|
||||
for (y = y0; y <= y1; y++) {
|
||||
if (y < 0 || y >= yres)
|
||||
continue;
|
||||
p = img+(xres*y+x0)*3;
|
||||
for (x = x0; x <= x1; x++) {
|
||||
if (x >= 0 && x < xres && (p[0] & p[1] & p[2]) == 255)
|
||||
memcpy(p, frame_fill, 3);
|
||||
p += 3;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void mark_areas(uint8_t *img, int x, int y)
|
||||
{
|
||||
const struct area *area;
|
||||
int r1 = 0, r2 = 0, i;
|
||||
|
||||
if (frame_width) {
|
||||
r1 = (frame_width-1)/2;
|
||||
r2 = (frame_width-1)-r1;
|
||||
}
|
||||
for (area = areas; area; area = area->next) {
|
||||
if (frame_width)
|
||||
for (i = -r1; i <= r2; i++) {
|
||||
hline(img, area->x0-r1, area->x1+r2, area->y0+i,
|
||||
x, y);
|
||||
hline(img, area->x0-r1, area->x1+r2, area->y1+i,
|
||||
x, y);
|
||||
vline(img, area->y0+r1, area->y1-r2, area->x0+i,
|
||||
x, y);
|
||||
vline(img, area->y0+r1, area->y1-r2, area->x1+i,
|
||||
x, y);
|
||||
}
|
||||
fill(img,
|
||||
area->x0+r1, area->y0+r1, area->x1-r2, area->y1-r2, x, y);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void usage(const char *name)
|
||||
{
|
||||
fprintf(stderr,
|
||||
"usage: %s [-f] [-a color] [-b color] [-c color] [-d pixels]\n"
|
||||
"%6s %*s [-m color] [-n color] [-w pixels] file_a.ppm file_b.ppm\n"
|
||||
"%6s %*s [file_a'.ppm file_b'.ppm] [out.ppm]\n\n"
|
||||
" file_a.ppm and file_b.ppm are two input images\n"
|
||||
" file_a'.ppm and file_b'.ppm if present, are searched for changes as well\n"
|
||||
" out.ppm output file (default: standard output)\n\n"
|
||||
" -f generate output (and return success) even if there is no change\n"
|
||||
" -a color color of items only in image A\n"
|
||||
" -b color color of items only in image B\n"
|
||||
" -c color color of items in both images\n"
|
||||
" -d pixels distance between change and marker box. 0 disables markers.\n"
|
||||
" -m color color of the frame of the marker box.\n"
|
||||
" -n color color of the background of the marker box\n"
|
||||
" -w pixels width of the frame of the marker box. 0 disables frames.\n\n"
|
||||
" color is specified as R,B,G with each component as a floating-point\n"
|
||||
" value from 0 to 1. E.g., 1,1,1 is white.\n\n"
|
||||
" The images are expected to have dark colors on a perfectly white\n"
|
||||
" background.\n"
|
||||
, name, "", (int) strlen(name), "", "", (int) strlen(name), "");
|
||||
exit(1);
|
||||
}
|
||||
|
||||
|
||||
static void parse_color(uint8_t *c, const char *s, const char *name)
|
||||
{
|
||||
float r, g, b;
|
||||
|
||||
if (sscanf(s, "%f,%f,%f", &r, &g, &b) != 3)
|
||||
usage(name);
|
||||
c[0] = 255*r;
|
||||
c[1] = 255*g;
|
||||
c[2] = 255*b;
|
||||
}
|
||||
|
||||
|
||||
int main(int argc, char *const *argv)
|
||||
{
|
||||
int force = 0;
|
||||
int x = 0, y = 0;
|
||||
uint8_t *old, *new, *d, *a, *b;
|
||||
char *shadow_old = NULL, *shadow_new = NULL, *out_name = NULL;
|
||||
FILE *out;
|
||||
char *end;
|
||||
int c;
|
||||
|
||||
while ((c = getopt(argc, argv, "a:b:c:d:fm:n:w:")) != EOF)
|
||||
switch (c) {
|
||||
case 'a':
|
||||
parse_color(a_only, optarg, *argv);
|
||||
break;
|
||||
case 'b':
|
||||
parse_color(b_only, optarg, *argv);
|
||||
break;
|
||||
case 'c':
|
||||
parse_color(both, optarg, *argv);
|
||||
break;
|
||||
case 'd':
|
||||
frame_dist = strtoul(optarg, &end, 0);
|
||||
if (*end)
|
||||
usage(*argv);
|
||||
break;
|
||||
case 'f':
|
||||
force = 1;
|
||||
break;
|
||||
case 'm':
|
||||
parse_color(frame, optarg, *argv);
|
||||
break;
|
||||
case 'n':
|
||||
parse_color(frame_fill, optarg, *argv);
|
||||
break;
|
||||
case 'w':
|
||||
frame_width = strtoul(optarg, &end, 0);
|
||||
if (*end)
|
||||
usage(*argv);
|
||||
break;
|
||||
default:
|
||||
usage(*argv);
|
||||
}
|
||||
switch (argc-optind) {
|
||||
case 2:
|
||||
break;
|
||||
case 3:
|
||||
out_name = argv[optind+2];
|
||||
break;
|
||||
case 5:
|
||||
out_name = argv[optind+4];
|
||||
/* fall through */
|
||||
case 4:
|
||||
shadow_old = argv[optind+2];
|
||||
shadow_new = argv[optind+3];
|
||||
break;
|
||||
default:
|
||||
usage(*argv);
|
||||
}
|
||||
|
||||
old = load_ppm(argv[optind], &x, &y);
|
||||
new = load_ppm(argv[optind+1], &x, &y);
|
||||
if (shadow_old) {
|
||||
a = load_ppm(shadow_old, &x, &y);
|
||||
b = load_ppm(shadow_new, &x, &y);
|
||||
if (!force && !memcmp(a, b, x*y*3))
|
||||
return 1;
|
||||
shadow_diff(a, b, x, y);
|
||||
}
|
||||
if (!force && !areas && !memcmp(old, new, x*y*3))
|
||||
return 1;
|
||||
d = diff(old, new, x, y, !shadow_old);
|
||||
if (frame_dist)
|
||||
mark_areas(d, x, y);
|
||||
|
||||
if (out_name) {
|
||||
out = fopen(out_name, "w");
|
||||
if (!out) {
|
||||
perror(out_name);
|
||||
exit(1);
|
||||
}
|
||||
} else {
|
||||
out = stdout;
|
||||
}
|
||||
fprintf(out, "P6\n%d %d\n255\n", x, y);
|
||||
fwrite(d, 1, x*y*3, out);
|
||||
if (fclose(out) == EOF) {
|
||||
perror("fclose");
|
||||
exit(1);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
Reference in New Issue
Block a user