370 lines
12 KiB
C
370 lines
12 KiB
C
/*
|
|
* SpanDSP - a series of DSP components for telephony
|
|
*
|
|
* t85_tests.c - ITU T.85 FAX image compression and decompression tests
|
|
*
|
|
* Written by Steve Underwood <steveu@coppice.org>
|
|
*
|
|
* Copyright (C) 2009 Steve Underwood
|
|
*
|
|
* All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
/*
|
|
* These tests are based on code from Markus Kuhn's jbigkit. See
|
|
* http://www.cl.cam.ac.uk/~mgk25/
|
|
*
|
|
* jbigkit is GPL2 licenced. This file is also GPL2 licenced, and our
|
|
* T.85 code is LGPL2.1 licenced. There are no licence incompatibilities
|
|
* in this reuse of Markus's work.
|
|
*/
|
|
|
|
/*! \file */
|
|
|
|
/*! \page t85_tests_page T.85 image compress and decompression tests
|
|
\section t85_tests_page_sec_1 What does it do
|
|
These tests exercise the image compression and decompression methods defined
|
|
in ITU specifications T.85.
|
|
*/
|
|
|
|
#if defined(HAVE_CONFIG_H)
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <inttypes.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stddef.h>
|
|
#include <string.h>
|
|
|
|
//#if defined(WITH_SPANDSP_INTERNALS)
|
|
#define SPANDSP_EXPOSE_INTERNAL_STRUCTURES
|
|
//#endif
|
|
|
|
#include "spandsp.h"
|
|
|
|
#define TESTBUF_SIZE 400000
|
|
#define TEST_IMAGE_SIZE (1951*1960/8)
|
|
|
|
#define FALSE 0
|
|
#define TRUE (!FALSE)
|
|
|
|
uint8_t testbuf[TESTBUF_SIZE];
|
|
uint8_t test_image[TEST_IMAGE_SIZE];
|
|
|
|
size_t testbuf_len;
|
|
|
|
int read_row = 0;
|
|
int write_row = 0;
|
|
|
|
int clip_to_row = 0;
|
|
|
|
static int row_read_handler(void *user_data, uint8_t buf[], size_t len)
|
|
{
|
|
//t85_encode_state_t *s;
|
|
|
|
//s = (t85_encode_state_t *) user_data;
|
|
|
|
memcpy(buf, &test_image[len*read_row], len);
|
|
//printf("Read row %d\n", read_row);
|
|
if (clip_to_row && read_row == clip_to_row)
|
|
{
|
|
clip_to_row = 0;
|
|
return 0;
|
|
}
|
|
read_row++;
|
|
return len;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static int row_write_handler(void *user_data, const uint8_t buf[], size_t len)
|
|
{
|
|
uint8_t *bitmap;
|
|
|
|
bitmap = (uint8_t *) user_data;
|
|
memcpy(&bitmap[len*write_row], buf, len);
|
|
//printf("Write row %d\n", write_row);
|
|
write_row++;
|
|
return 0;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static int comment_handler(void *user_data, const uint8_t buf[], size_t len)
|
|
{
|
|
if (buf)
|
|
printf("Comment (%lu): %s\n", (unsigned long int) len, buf);
|
|
else
|
|
printf("Comment (%lu): ---\n", (unsigned long int) len);
|
|
return 0;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static void create_test_image(uint8_t *pic)
|
|
{
|
|
int i;
|
|
int j;
|
|
uint32_t sum;
|
|
uint32_t prsg;
|
|
uint32_t repeat[8];
|
|
uint8_t *p;
|
|
|
|
/* Cook up the test image defined in T.82/7.2.1. This image is 1960 x 1951
|
|
pixels, and occupies a single plane (which it has to for T.85). */
|
|
memset(pic, 0, TEST_IMAGE_SIZE);
|
|
p = pic;
|
|
prsg = 1;
|
|
for (i = 0; i < 1951; i++)
|
|
{
|
|
for (j = 0; j < 1960; j++)
|
|
{
|
|
if (i >= 192)
|
|
{
|
|
if (i < 1023 || (j & (3 << 3)) == 0)
|
|
{
|
|
sum = (prsg & 1)
|
|
+ ((prsg >> 2) & 1)
|
|
+ ((prsg >> 11) & 1)
|
|
+ ((prsg >> 15) & 1);
|
|
prsg = (prsg << 1) + (sum & 1);
|
|
if ((prsg & 3) == 0)
|
|
{
|
|
*p |= (1 << (7 - (j & 7)));
|
|
repeat[j & 7] = 1;
|
|
}
|
|
else
|
|
{
|
|
repeat[j & 7] = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (repeat[j & 7])
|
|
*p |= 1 << (7 - (j & 7));
|
|
}
|
|
}
|
|
if ((j & 7) == 7)
|
|
++p;
|
|
}
|
|
}
|
|
|
|
/* Verify the test image has been generated OK, by checking the number of set pixels */
|
|
sum = 0;
|
|
for (i = 0; i < TEST_IMAGE_SIZE; i++)
|
|
{
|
|
for (j = 0; j < 8; j++)
|
|
sum += ((pic[i] >> j) & 1);
|
|
}
|
|
if (sum != 861965)
|
|
{
|
|
printf("WARNING: Test image has %" PRIu32 " foreground pixels. There should be 861965.\n",
|
|
sum);
|
|
}
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
/* Perform a test cycle, as defined in T.82/7, with one set of parameters. */
|
|
static int test_cycle(const char *test_id,
|
|
const uint8_t *image,
|
|
uint32_t width,
|
|
uint32_t height,
|
|
uint32_t l0,
|
|
int mx,
|
|
int options,
|
|
int optionsx,
|
|
const uint8_t *comment,
|
|
size_t correct_length)
|
|
{
|
|
t85_encode_state_t t85_enc;
|
|
t85_decode_state_t t85_dec;
|
|
long int l;
|
|
size_t image_size;
|
|
int result;
|
|
int len;
|
|
int max_len;
|
|
size_t bytes_per_row;
|
|
size_t cnt_a;
|
|
size_t cnt_b;
|
|
uint8_t *decoded_image;
|
|
|
|
printf("%s: TPBON=%d, LRLTWO=%d, Mx=%d, L0=%" PRIu32 "\n",
|
|
test_id,
|
|
(options & T85_TPBON) ? 1 : 0,
|
|
(options & T85_LRLTWO) ? 1 : 0,
|
|
mx,
|
|
l0);
|
|
|
|
printf("%s.1: Encode\n", test_id);
|
|
bytes_per_row = (width + 7)/8;
|
|
image_size = bytes_per_row*height;
|
|
|
|
if ((optionsx & T85_VLENGTH))
|
|
{
|
|
t85_encode_init(&t85_enc, width, height + 10, row_read_handler, &t85_enc);
|
|
clip_to_row = height;
|
|
}
|
|
else
|
|
{
|
|
t85_encode_init(&t85_enc, width, height, row_read_handler, &t85_enc);
|
|
clip_to_row = 0;
|
|
}
|
|
read_row = 0;
|
|
t85_encode_set_options(&t85_enc, l0, mx, options);
|
|
/* A comment inserted here should always succeed. The later one, inserted some way
|
|
down the image, will only succeed if a new chunk is started afterwards. */
|
|
if (comment)
|
|
t85_encode_comment(&t85_enc, comment, strlen((const char *) comment) + 1);
|
|
|
|
testbuf_len = 0;
|
|
max_len = 100;
|
|
while ((len = t85_encode_get_chunk(&t85_enc, &testbuf[testbuf_len], max_len)) > 0)
|
|
{
|
|
testbuf_len += len;
|
|
max_len = 100;
|
|
if (testbuf_len + 100 > TESTBUF_SIZE)
|
|
max_len = TESTBUF_SIZE - testbuf_len;
|
|
if (comment && testbuf_len == 1000)
|
|
t85_encode_comment(&t85_enc, comment, strlen((const char *) comment) + 1);
|
|
}
|
|
t85_encode_release(&t85_enc);
|
|
printf("Encoded BIE has %lu bytes\n", (unsigned long int) testbuf_len);
|
|
if (correct_length > 0)
|
|
{
|
|
if (testbuf_len != correct_length)
|
|
{
|
|
printf("Incorrect encoded length. Should have been %lu\n", (unsigned long int) correct_length);
|
|
printf("Test failed\n");
|
|
exit(2);
|
|
}
|
|
printf("Test passed\n");
|
|
}
|
|
|
|
printf("%s.2: Decode in one big chunk\n", test_id);
|
|
if ((decoded_image = (uint8_t *) malloc(image_size)) == NULL)
|
|
{
|
|
fprintf(stderr, "Out of memory!\n");
|
|
exit(2);
|
|
}
|
|
t85_decode_init(&t85_dec, row_write_handler, decoded_image);
|
|
if (comment && comment[0] != 'X')
|
|
t85_decode_set_comment_handler(&t85_dec, 1000, comment_handler, NULL);
|
|
write_row = 0;
|
|
result = t85_decode_put_chunk(&t85_dec, testbuf, testbuf_len);
|
|
if (result == T85_MORE_DATA)
|
|
result = t85_decode_put_byte(&t85_dec, SIG_STATUS_END_OF_DATA);
|
|
cnt_a = t85_encode_get_compressed_image_size(&t85_enc);
|
|
cnt_b = t85_decode_get_compressed_image_size(&t85_dec);
|
|
if (cnt_a != cnt_b || cnt_a != testbuf_len*8 || result != T85_OK)
|
|
{
|
|
printf("Decode result %d\n", result);
|
|
printf("%ld/%ld bits of %ld bits of BIE read. %lu lines decoded.\n",
|
|
(long int) cnt_a,
|
|
(long int) cnt_b,
|
|
(unsigned long int) testbuf_len*8,
|
|
(unsigned long int) t85_dec.y);
|
|
printf("Test failed\n");
|
|
exit(2);
|
|
}
|
|
if (memcmp(decoded_image, image, image_size))
|
|
{
|
|
printf("Image mismatch\n");
|
|
printf("Test failed\n");
|
|
exit(2);
|
|
}
|
|
free(decoded_image);
|
|
t85_decode_release(&t85_dec);
|
|
printf("Test passed\n");
|
|
|
|
printf("%s.3: Decode byte by byte\n", test_id);
|
|
if ((decoded_image = (uint8_t *) malloc(image_size)) == NULL)
|
|
{
|
|
fprintf(stderr, "Out of memory!\n");
|
|
exit(2);
|
|
}
|
|
t85_decode_init(&t85_dec, row_write_handler, decoded_image);
|
|
if (comment && comment[0] != 'X')
|
|
t85_decode_set_comment_handler(&t85_dec, 1000, comment_handler, NULL);
|
|
write_row = 0;
|
|
result = T85_MORE_DATA;
|
|
for (l = 0; l < testbuf_len; l++)
|
|
{
|
|
result = t85_decode_put_chunk(&t85_dec, &testbuf[l], 1);
|
|
if (result != T85_MORE_DATA)
|
|
{
|
|
l++;
|
|
break;
|
|
}
|
|
}
|
|
if (result == T85_MORE_DATA)
|
|
result = t85_decode_put_byte(&t85_dec, SIG_STATUS_END_OF_DATA);
|
|
if (l != testbuf_len || result != T85_OK)
|
|
{
|
|
printf("Decode result %d\n", result);
|
|
printf("%ld bytes of %ld bytes of BIE read. %lu lines decoded.\n",
|
|
(long int) l,
|
|
(unsigned long int) testbuf_len,
|
|
(unsigned long int) t85_dec.y);
|
|
printf("Test failed\n");
|
|
exit(2);
|
|
}
|
|
if (memcmp(decoded_image, image, image_size))
|
|
{
|
|
printf("Image mismatch\n");
|
|
printf("Test failed\n");
|
|
exit(2);
|
|
}
|
|
free(decoded_image);
|
|
t85_decode_release(&t85_dec);
|
|
printf("Test passed\n");
|
|
|
|
return 0;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
printf("T.85 JBIG for FAX encoder and decoder tests, from ITU-T T.82\n\n");
|
|
|
|
printf("Generating the test image from T.82...\n");
|
|
create_test_image(test_image);
|
|
|
|
/* Run through the tests in T.82/7.2, which are applicable to T.85 */
|
|
test_cycle("1", test_image, 1960, 1951, 1951, 0, 0, 0, NULL, 317384);
|
|
test_cycle("2", test_image, 1960, 1951, 1951, 0, T85_LRLTWO, 0, NULL, 317132);
|
|
test_cycle("3", test_image, 1960, 1951, 128, 8, T85_TPBON, 0, NULL, 253653);
|
|
/* Again with a comment added and handled */
|
|
test_cycle("4", test_image, 1960, 1951, 1951, 0, 0, 0, (const uint8_t *) "Comment 4", 317384 + 16);
|
|
test_cycle("5", test_image, 1960, 1951, 1951, 0, T85_LRLTWO, 0, (const uint8_t *) "Comment 5", 317132 + 16);
|
|
test_cycle("6", test_image, 1960, 1951, 128, 8, T85_TPBON, 0, (const uint8_t *) "Comment 6", 253653 + 2*16);
|
|
/* Again with a comment added, but not handled */
|
|
test_cycle("7", test_image, 1960, 1951, 1951, 0, 0, 0, (const uint8_t *) "Xomment 7", 317384 + 16);
|
|
test_cycle("8", test_image, 1960, 1951, 1951, 0, T85_LRLTWO, 0, (const uint8_t *) "Xomment 8", 317132 + 16);
|
|
test_cycle("9", test_image, 1960, 1951, 128, 8, T85_TPBON, 0, (const uint8_t *) "Xomment 9", 253653 + 2*16);
|
|
/* Again with the image variable length and prematurely terminated */
|
|
test_cycle("10", test_image, 1960, 1951, 1951, 0, T85_VLENGTH, T85_VLENGTH, NULL, 317384 + 8);
|
|
test_cycle("11", test_image, 1960, 1951, 1951, 0, T85_VLENGTH | T85_LRLTWO, T85_VLENGTH, NULL, 317132 + 8);
|
|
test_cycle("12", test_image, 1960, 1951, 128, 8, T85_VLENGTH | T85_TPBON, T85_VLENGTH, NULL, 253653 + 8);
|
|
/* Again with the image variable length but not prematurely terminated */
|
|
test_cycle("13", test_image, 1960, 1951, 1951, 0, T85_VLENGTH, 0, NULL, 317384);
|
|
test_cycle("14", test_image, 1960, 1951, 1951, 0, T85_VLENGTH | T85_LRLTWO, 0, NULL, 317132);
|
|
test_cycle("15", test_image, 1960, 1951, 128, 8, T85_VLENGTH | T85_TPBON, 0, NULL, 253653);
|
|
|
|
printf("Tests passed\n");
|
|
|
|
return 0;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
/*- End of file ------------------------------------------------------------*/
|