2008-09-03 19:02:00 +00:00
|
|
|
/*
|
|
|
|
* SpanDSP - a series of DSP components for telephony
|
|
|
|
*
|
|
|
|
* t4_tests.c - ITU T.4 FAX image to and from TIFF file tests
|
|
|
|
*
|
|
|
|
* Written by Steve Underwood <steveu@coppice.org>
|
|
|
|
*
|
|
|
|
* Copyright (C) 2003 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*! \file */
|
|
|
|
|
|
|
|
/*! \page t4_tests_page T.4 tests
|
|
|
|
\section t4_tests_page_sec_1 What does it do
|
|
|
|
These tests exercise the image compression and decompression methods defined
|
|
|
|
in ITU specifications T.4 and T.6.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if defined(HAVE_CONFIG_H)
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <memory.h>
|
|
|
|
|
2009-01-28 04:48:03 +00:00
|
|
|
#define SPANDSP_EXPOSE_INTERNAL_STRUCTURES
|
|
|
|
|
2008-09-03 19:02:00 +00:00
|
|
|
#include "spandsp.h"
|
|
|
|
|
|
|
|
#define IN_FILE_NAME "../test-data/itu/fax/itutests.tif"
|
|
|
|
#define OUT_FILE_NAME "t4_tests_receive.tif"
|
|
|
|
|
|
|
|
#define XSIZE 1728
|
|
|
|
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_tx_state_t *send_state;
|
|
|
|
t4_rx_state_t *receive_state;
|
2008-09-03 19:02:00 +00:00
|
|
|
|
2010-07-24 19:29:44 +00:00
|
|
|
int rows_written = 0;
|
|
|
|
int rows_read = 0;
|
|
|
|
|
2012-07-15 05:37:39 +00:00
|
|
|
static void dump_image_as_xxx(t4_rx_state_t *state)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2012-07-15 05:37:39 +00:00
|
|
|
#if 0
|
|
|
|
uint8_t *s;
|
2008-09-03 19:02:00 +00:00
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
int k;
|
|
|
|
|
|
|
|
/* Dump the entire image as text 'X's and spaces */
|
2012-07-15 05:37:39 +00:00
|
|
|
printf("Image (%d x %d):\n", state->t4_t6.image_width, state->t4_t6.image_length);
|
|
|
|
s = state->image_buffer;
|
|
|
|
for (i = 0; i < state->t4_t6.image_length; i++)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2012-07-15 05:37:39 +00:00
|
|
|
for (j = 0; j < state->t4_t6.bytes_per_row; j++)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
for (k = 0; k < 8; k++)
|
2012-07-15 05:37:39 +00:00
|
|
|
printf((state->image_buffer[i*state->t4_t6.bytes_per_row + j] & (0x80 >> k)) ? "X" : " ");
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
2012-07-15 05:37:39 +00:00
|
|
|
#endif
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
|
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
|
2012-07-15 05:37:39 +00:00
|
|
|
static void display_page_stats(t4_rx_state_t *s)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
t4_stats_t stats;
|
|
|
|
|
2010-07-24 19:29:44 +00:00
|
|
|
t4_rx_get_transfer_statistics(s, &stats);
|
2008-09-03 19:02:00 +00:00
|
|
|
printf("Pages = %d\n", stats.pages_transferred);
|
2013-05-29 02:07:08 +00:00
|
|
|
printf("Compression = %s\n", t4_compression_to_str(stats.compression));
|
2010-07-24 19:29:44 +00:00
|
|
|
printf("Compressed size = %d\n", stats.line_image_size);
|
2013-03-10 12:55:21 +00:00
|
|
|
printf("Raw image size = %d pels x %d pels\n", stats.image_width, stats.image_length);
|
2009-12-23 01:38:36 +00:00
|
|
|
printf("Image size = %d pels x %d pels\n", stats.width, stats.length);
|
2013-03-10 12:55:21 +00:00
|
|
|
printf("Raw image resolution = %d pels/m x %d pels/m\n", stats.image_x_resolution, stats.image_y_resolution);
|
2009-12-23 01:38:36 +00:00
|
|
|
printf("Image resolution = %d pels/m x %d pels/m\n", stats.x_resolution, stats.y_resolution);
|
2008-09-03 19:02:00 +00:00
|
|
|
printf("Bad rows = %d\n", stats.bad_rows);
|
|
|
|
printf("Longest bad row run = %d\n", stats.longest_bad_row_run);
|
2012-07-15 05:37:39 +00:00
|
|
|
printf("Bits per row - min %d, max %d\n", s->decoder.t4_t6.min_row_bits, s->decoder.t4_t6.max_row_bits);
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
|
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
|
2013-07-19 07:40:22 +00:00
|
|
|
static int detect_non_ecm_page_end(int bit, int page_ended)
|
2009-05-14 21:52:52 +00:00
|
|
|
{
|
|
|
|
static int consecutive_eols;
|
|
|
|
static int max_consecutive_eols;
|
|
|
|
static int consecutive_zeros;
|
|
|
|
static int consecutive_ones;
|
|
|
|
static int eol_zeros;
|
|
|
|
static int eol_ones;
|
|
|
|
static int expected_eols;
|
|
|
|
static int end_marks;
|
|
|
|
|
2013-07-19 07:40:22 +00:00
|
|
|
/* Check the EOLs are added properly to the end of a non-ECM image. We can't rely
|
|
|
|
on the decoder giving the right answer, as a full set of EOLs is not needed for
|
|
|
|
the decoder to work. */
|
|
|
|
if (bit == -1)
|
2009-05-14 21:52:52 +00:00
|
|
|
{
|
|
|
|
/* Reset */
|
|
|
|
consecutive_eols = 0;
|
|
|
|
max_consecutive_eols = 0;
|
|
|
|
consecutive_zeros = 0;
|
|
|
|
consecutive_ones = 0;
|
|
|
|
end_marks = 0;
|
|
|
|
|
|
|
|
eol_zeros = 11;
|
2013-04-01 16:18:21 +00:00
|
|
|
eol_ones = (page_ended == T4_COMPRESSION_T4_2D) ? 2 : 1;
|
|
|
|
expected_eols = (page_ended == T4_COMPRESSION_T6) ? 2 : 6;
|
2010-07-24 19:29:44 +00:00
|
|
|
return 0;
|
2009-05-14 21:52:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Monitor whether the EOLs are there in the correct amount */
|
|
|
|
if (bit == 0)
|
|
|
|
{
|
|
|
|
consecutive_zeros++;
|
|
|
|
consecutive_ones = 0;
|
|
|
|
}
|
|
|
|
else if (bit == 1)
|
|
|
|
{
|
|
|
|
if (++consecutive_ones == eol_ones)
|
|
|
|
{
|
|
|
|
if (consecutive_eols == 0 && consecutive_zeros >= eol_zeros)
|
|
|
|
consecutive_eols++;
|
|
|
|
else if (consecutive_zeros == eol_zeros)
|
|
|
|
consecutive_eols++;
|
|
|
|
else
|
|
|
|
consecutive_eols = 0;
|
|
|
|
consecutive_zeros = 0;
|
|
|
|
consecutive_ones = 0;
|
|
|
|
}
|
|
|
|
if (max_consecutive_eols < consecutive_eols)
|
|
|
|
max_consecutive_eols = consecutive_eols;
|
|
|
|
}
|
|
|
|
else if (bit == SIG_STATUS_END_OF_DATA)
|
|
|
|
{
|
|
|
|
if (end_marks == 0)
|
|
|
|
{
|
|
|
|
if (max_consecutive_eols != expected_eols)
|
|
|
|
{
|
|
|
|
printf("Only %d EOLs (should be %d)\n", max_consecutive_eols, expected_eols);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
consecutive_zeros = 0;
|
|
|
|
consecutive_eols = 0;
|
|
|
|
max_consecutive_eols = 0;
|
|
|
|
}
|
|
|
|
if (!page_ended)
|
|
|
|
{
|
|
|
|
/* We might need to push a few bits to get the receiver to report the
|
2012-08-12 14:11:06 +00:00
|
|
|
end of page condition (at least with T.6). */
|
2009-05-14 21:52:52 +00:00
|
|
|
if (++end_marks > 50)
|
|
|
|
{
|
|
|
|
printf("Receiver missed the end of page mark\n");
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/*- End of function --------------------------------------------------------*/
|
2008-09-03 19:02:00 +00:00
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2009-05-14 21:52:52 +00:00
|
|
|
static const int compression_sequence[] =
|
|
|
|
{
|
2009-12-23 01:38:36 +00:00
|
|
|
//T4_COMPRESSION_NONE,
|
2013-04-01 16:18:21 +00:00
|
|
|
T4_COMPRESSION_T4_1D,
|
|
|
|
T4_COMPRESSION_T4_2D,
|
|
|
|
T4_COMPRESSION_T6,
|
2013-04-04 15:40:00 +00:00
|
|
|
T4_COMPRESSION_T85,
|
|
|
|
T4_COMPRESSION_T85_L0,
|
2013-07-19 07:40:22 +00:00
|
|
|
#if defined(SPANDSP_SUPPORT_T88x)
|
|
|
|
T4_COMPRESSION_T88,
|
|
|
|
#endif
|
2012-03-28 13:43:13 +00:00
|
|
|
#if defined(SPANDSP_SUPPORT_T42x)
|
2013-04-04 15:40:00 +00:00
|
|
|
T4_COMPRESSION_T42_T81,
|
|
|
|
T4_COMPRESSION_SYCC_T81,
|
2013-03-13 21:04:43 +00:00
|
|
|
#endif
|
2012-03-28 13:43:13 +00:00
|
|
|
#if defined(SPANDSP_SUPPORT_T43x)
|
2013-04-01 16:18:21 +00:00
|
|
|
T4_COMPRESSION_T43,
|
2013-03-13 21:04:43 +00:00
|
|
|
#endif
|
2013-07-19 07:40:22 +00:00
|
|
|
#if defined(SPANDSP_SUPPORT_T45x)
|
|
|
|
T4_COMPRESSION_T45,
|
|
|
|
#endif
|
2010-07-24 19:29:44 +00:00
|
|
|
-1
|
2009-05-14 21:52:52 +00:00
|
|
|
};
|
2008-09-03 19:02:00 +00:00
|
|
|
int sends;
|
|
|
|
int bit;
|
|
|
|
int end_of_page;
|
|
|
|
int end_marks;
|
2009-05-14 21:52:52 +00:00
|
|
|
int res;
|
2008-09-03 19:02:00 +00:00
|
|
|
int compression;
|
2018-01-22 15:19:38 +00:00
|
|
|
int x_resolution;
|
|
|
|
int y_resolution;
|
2008-09-03 19:02:00 +00:00
|
|
|
int compression_step;
|
|
|
|
int min_row_bits;
|
|
|
|
int block_size;
|
|
|
|
char buf[1024];
|
|
|
|
uint8_t block[1024];
|
|
|
|
const char *in_file_name;
|
2009-05-14 21:52:52 +00:00
|
|
|
const char *decode_file_name;
|
2010-07-24 19:29:44 +00:00
|
|
|
const char *page_header_tz;
|
2011-12-05 17:00:22 +00:00
|
|
|
tz_t tz;
|
2008-09-03 19:02:00 +00:00
|
|
|
int opt;
|
2012-08-12 14:11:06 +00:00
|
|
|
int len;
|
2008-09-03 19:02:00 +00:00
|
|
|
int i;
|
|
|
|
int bit_error_rate;
|
|
|
|
int tests_failed;
|
2018-01-22 15:19:38 +00:00
|
|
|
int match_pos;
|
2013-08-08 13:40:28 +00:00
|
|
|
bool restart_pages;
|
|
|
|
bool add_page_headers;
|
|
|
|
bool overlay_page_headers;
|
|
|
|
bool dump_as_xxx;
|
2009-01-28 04:48:03 +00:00
|
|
|
unsigned int last_pkt_no;
|
|
|
|
unsigned int pkt_no;
|
2009-05-14 21:52:52 +00:00
|
|
|
int page_ended;
|
|
|
|
FILE *file;
|
2008-09-03 19:02:00 +00:00
|
|
|
|
|
|
|
tests_failed = 0;
|
|
|
|
compression = -1;
|
2009-05-14 21:52:52 +00:00
|
|
|
compression_step = 0;
|
2018-01-22 15:19:38 +00:00
|
|
|
x_resolution = -1;
|
|
|
|
y_resolution = -1;
|
2013-08-08 13:40:28 +00:00
|
|
|
add_page_headers = false;
|
|
|
|
overlay_page_headers = false;
|
|
|
|
restart_pages = false;
|
2008-09-03 19:02:00 +00:00
|
|
|
in_file_name = IN_FILE_NAME;
|
2009-05-14 21:52:52 +00:00
|
|
|
decode_file_name = NULL;
|
2010-07-24 19:29:44 +00:00
|
|
|
page_header_tz = NULL;
|
2009-05-14 21:52:52 +00:00
|
|
|
/* Use a non-zero default minimum row length to ensure we test the consecutive EOLs part
|
|
|
|
properly. */
|
|
|
|
min_row_bits = 50;
|
2010-07-24 19:29:44 +00:00
|
|
|
block_size = 1;
|
2008-09-03 19:02:00 +00:00
|
|
|
bit_error_rate = 0;
|
2013-08-08 13:40:28 +00:00
|
|
|
dump_as_xxx = false;
|
2018-01-22 15:19:38 +00:00
|
|
|
while ((opt = getopt(argc, argv, "b:c:d:ehHrR:i:m:t:x")) != -1)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
switch (opt)
|
|
|
|
{
|
|
|
|
case 'b':
|
|
|
|
block_size = atoi(optarg);
|
|
|
|
if (block_size > 1024)
|
2013-07-19 07:40:22 +00:00
|
|
|
{
|
|
|
|
printf("Block size too large. Must be 1024 or less\n");
|
|
|
|
exit(2);
|
|
|
|
}
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
2012-03-28 13:43:13 +00:00
|
|
|
case 'c':
|
|
|
|
if (strcmp(optarg, "T41D") == 0)
|
|
|
|
{
|
2013-04-01 16:18:21 +00:00
|
|
|
compression = T4_COMPRESSION_T4_1D;
|
2012-03-28 13:43:13 +00:00
|
|
|
compression_step = -1;
|
|
|
|
}
|
|
|
|
else if (strcmp(optarg, "T42D") == 0)
|
|
|
|
{
|
2013-04-01 16:18:21 +00:00
|
|
|
compression = T4_COMPRESSION_T4_2D;
|
2012-03-28 13:43:13 +00:00
|
|
|
compression_step = -1;
|
|
|
|
}
|
|
|
|
else if (strcmp(optarg, "T6") == 0)
|
|
|
|
{
|
2013-04-01 16:18:21 +00:00
|
|
|
compression = T4_COMPRESSION_T6;
|
2012-03-28 13:43:13 +00:00
|
|
|
compression_step = -1;
|
|
|
|
}
|
2013-04-04 15:40:00 +00:00
|
|
|
else if (strcmp(optarg, "T85") == 0)
|
|
|
|
{
|
|
|
|
compression = T4_COMPRESSION_T85;
|
|
|
|
compression_step = -1;
|
|
|
|
}
|
2013-07-19 07:40:22 +00:00
|
|
|
#if defined(SPANDSP_SUPPORT_T88)
|
|
|
|
else if (strcmp(optarg, "T88") == 0)
|
|
|
|
{
|
|
|
|
compression = T4_COMPRESSION_T88;
|
|
|
|
compression_step = -1;
|
|
|
|
}
|
|
|
|
#endif
|
2013-04-04 15:40:00 +00:00
|
|
|
else if (strcmp(optarg, "T81") == 0)
|
2012-03-28 13:43:13 +00:00
|
|
|
{
|
2013-04-04 15:40:00 +00:00
|
|
|
compression = T4_COMPRESSION_T42_T81;
|
2012-03-28 13:43:13 +00:00
|
|
|
compression_step = -1;
|
|
|
|
}
|
|
|
|
else if (strcmp(optarg, "T43") == 0)
|
|
|
|
{
|
2013-04-01 16:18:21 +00:00
|
|
|
compression = T4_COMPRESSION_T43;
|
2012-03-28 13:43:13 +00:00
|
|
|
compression_step = -1;
|
|
|
|
}
|
2013-07-19 07:40:22 +00:00
|
|
|
#if defined(SPANDSP_SUPPORT_T45)
|
|
|
|
else if (strcmp(optarg, "T45") == 0)
|
|
|
|
{
|
|
|
|
compression = T4_COMPRESSION_T45;
|
|
|
|
compression_step = -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Unrecognised compression.\n");
|
|
|
|
exit(2);
|
|
|
|
}
|
2012-03-28 13:43:13 +00:00
|
|
|
break;
|
2008-09-03 19:02:00 +00:00
|
|
|
case 'd':
|
2009-05-14 21:52:52 +00:00
|
|
|
decode_file_name = optarg;
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
|
|
|
case 'e':
|
|
|
|
bit_error_rate = 0x3FF;
|
|
|
|
break;
|
|
|
|
case 'h':
|
2013-08-08 13:40:28 +00:00
|
|
|
add_page_headers = true;
|
|
|
|
overlay_page_headers = false;
|
2010-07-24 19:29:44 +00:00
|
|
|
break;
|
|
|
|
case 'H':
|
2013-08-08 13:40:28 +00:00
|
|
|
add_page_headers = true;
|
|
|
|
overlay_page_headers = true;
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
|
|
|
case 'r':
|
2013-08-08 13:40:28 +00:00
|
|
|
restart_pages = true;
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
2018-01-22 15:19:38 +00:00
|
|
|
case 'R':
|
|
|
|
if (strcmp(optarg, "standard") == 0)
|
|
|
|
{
|
|
|
|
y_resolution = T4_Y_RESOLUTION_STANDARD;
|
|
|
|
}
|
|
|
|
else if (strcmp(optarg, "fine") == 0)
|
|
|
|
{
|
|
|
|
y_resolution = T4_Y_RESOLUTION_FINE;
|
|
|
|
}
|
|
|
|
else if (strcmp(optarg, "superfine") == 0)
|
|
|
|
{
|
|
|
|
y_resolution = T4_Y_RESOLUTION_SUPERFINE;
|
|
|
|
}
|
|
|
|
break;
|
2008-09-03 19:02:00 +00:00
|
|
|
case 'i':
|
|
|
|
in_file_name = optarg;
|
|
|
|
break;
|
|
|
|
case 'm':
|
|
|
|
min_row_bits = atoi(optarg);
|
|
|
|
break;
|
2010-07-24 19:29:44 +00:00
|
|
|
case 't':
|
|
|
|
page_header_tz = optarg;
|
|
|
|
break;
|
2008-09-03 19:02:00 +00:00
|
|
|
case 'x':
|
2013-08-08 13:40:28 +00:00
|
|
|
dump_as_xxx = true;
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
//usage();
|
|
|
|
exit(2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-12 14:11:06 +00:00
|
|
|
end_of_page = T4_DECODE_MORE_DATA;
|
2009-05-14 21:52:52 +00:00
|
|
|
if (decode_file_name)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
if (compression < 0)
|
2013-04-01 16:18:21 +00:00
|
|
|
compression = T4_COMPRESSION_T4_1D;
|
2018-01-22 15:19:38 +00:00
|
|
|
if (x_resolution < 0)
|
|
|
|
x_resolution = T4_X_RESOLUTION_R8;
|
|
|
|
if (y_resolution < 0)
|
|
|
|
y_resolution = T4_Y_RESOLUTION_STANDARD;
|
2008-09-03 19:02:00 +00:00
|
|
|
/* Receive end puts TIFF to a new file. We assume the receive width here. */
|
2013-04-01 16:18:21 +00:00
|
|
|
if ((receive_state = t4_rx_init(NULL, OUT_FILE_NAME, T4_COMPRESSION_T4_2D)) == NULL)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
printf("Failed to init T.4 rx\n");
|
|
|
|
exit(2);
|
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
span_log_set_level(t4_rx_get_logging_state(receive_state), SPAN_LOG_SHOW_SEVERITY | SPAN_LOG_SHOW_PROTOCOL | SPAN_LOG_SHOW_TAG | SPAN_LOG_FLOW);
|
|
|
|
t4_rx_set_rx_encoding(receive_state, compression);
|
2018-01-22 15:19:38 +00:00
|
|
|
t4_rx_set_x_resolution(receive_state, x_resolution);
|
|
|
|
t4_rx_set_y_resolution(receive_state, y_resolution);
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_rx_set_image_width(receive_state, XSIZE);
|
|
|
|
|
|
|
|
t4_rx_start_page(receive_state);
|
2009-01-28 04:48:03 +00:00
|
|
|
last_pkt_no = 0;
|
2009-05-14 21:52:52 +00:00
|
|
|
file = fopen(decode_file_name, "r");
|
|
|
|
while (fgets(buf, 1024, file))
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2009-01-28 04:48:03 +00:00
|
|
|
if (sscanf(buf, "HDLC: FCD: 06 %x", &pkt_no) == 1)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
/* Useful for breaking up T.38 ECM logs */
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
2008-09-09 17:04:42 +00:00
|
|
|
if (sscanf(&buf[18 + 3*i], "%x", (unsigned int *) &bit) != 1)
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
2012-08-12 14:11:06 +00:00
|
|
|
block[i] = bit;
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, i);
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
2009-01-28 04:48:03 +00:00
|
|
|
else if (sscanf(buf, "HDLC: %x", &pkt_no) == 1)
|
|
|
|
{
|
|
|
|
/* Useful for breaking up HDLC decodes of ECM logs */
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
|
|
|
if (sscanf(&buf[19 + 3*i], "%x", (unsigned int *) &bit) != 1)
|
|
|
|
break;
|
2012-08-12 14:11:06 +00:00
|
|
|
block[i] = bit;
|
2009-01-28 04:48:03 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, i);
|
2009-01-28 04:48:03 +00:00
|
|
|
}
|
2018-01-22 15:19:38 +00:00
|
|
|
else if (sscanf(buf, "%*d:%*d:%*d.%*d T.38 Rx %d: IFP %x %x %x %x %x %n", &pkt_no, (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit, &match_pos) == 6)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
/* Useful for breaking up T.38 non-ECM logs */
|
2009-01-28 04:48:03 +00:00
|
|
|
if (pkt_no != last_pkt_no + 1)
|
|
|
|
printf("Packet %u\n", pkt_no);
|
|
|
|
last_pkt_no = pkt_no;
|
2008-09-03 19:02:00 +00:00
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
2018-01-22 15:19:38 +00:00
|
|
|
if (sscanf(&buf[match_pos + 3*i], "%x", (unsigned int *) &bit) != 1)
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
2012-08-12 14:11:06 +00:00
|
|
|
block[i] = bit_reverse8(bit);
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, i);
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
2012-09-08 17:08:15 +00:00
|
|
|
else if (strlen(buf) > 2 && sscanf(buf, "T.30 Rx: %x %x %x %x", (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &pkt_no) == 4)
|
2010-07-24 19:29:44 +00:00
|
|
|
{
|
|
|
|
/* Useful for breaking up ECM logs */
|
|
|
|
if (pkt_no != last_pkt_no + 1)
|
|
|
|
printf("Packet %u\n", pkt_no);
|
|
|
|
last_pkt_no = pkt_no;
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
|
|
|
if (sscanf(&buf[22 + 3*i], "%x", (unsigned int *) &bit) != 1)
|
|
|
|
break;
|
2012-08-12 14:11:06 +00:00
|
|
|
block[i] = bit_reverse8(bit);
|
2010-07-24 19:29:44 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, i);
|
2010-07-24 19:29:44 +00:00
|
|
|
}
|
|
|
|
else if (sscanf(buf, "%04x %02x %02x %02x", (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit) == 4)
|
|
|
|
{
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
{
|
|
|
|
if (sscanf(&buf[6 + 3*i], "%x", (unsigned int *) &bit) != 1)
|
|
|
|
break;
|
2012-08-12 14:11:06 +00:00
|
|
|
block[i] = bit_reverse8(bit);
|
2010-07-24 19:29:44 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, i);
|
2010-07-24 19:29:44 +00:00
|
|
|
}
|
2009-03-02 16:24:30 +00:00
|
|
|
else if (sscanf(buf, "%08x %02x %02x %02x", (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit, (unsigned int *) &bit) == 4)
|
|
|
|
{
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
{
|
|
|
|
if (sscanf(&buf[10 + 3*i], "%x", (unsigned int *) &bit) != 1)
|
|
|
|
break;
|
2012-08-12 14:11:06 +00:00
|
|
|
block[i] = bit_reverse8(bit);
|
2009-03-02 16:24:30 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, i);
|
2009-03-02 16:24:30 +00:00
|
|
|
}
|
2008-09-03 19:02:00 +00:00
|
|
|
else if (sscanf(buf, "Rx bit %*d - %d", &bit) == 1)
|
|
|
|
{
|
2013-01-01 15:07:55 +00:00
|
|
|
if ((end_of_page = t4_rx_put_bit(receive_state, bit)))
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
printf("End of page detected\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-05-14 21:52:52 +00:00
|
|
|
fclose(file);
|
2008-09-03 19:02:00 +00:00
|
|
|
if (dump_as_xxx)
|
2013-01-01 15:07:55 +00:00
|
|
|
dump_image_as_xxx(receive_state);
|
|
|
|
t4_rx_end_page(receive_state);
|
|
|
|
display_page_stats(receive_state);
|
2014-06-14 11:49:05 +00:00
|
|
|
t4_rx_free(receive_state);
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-05-14 21:52:52 +00:00
|
|
|
#if 1
|
|
|
|
printf("Testing TIFF->compress->decompress->TIFF cycle\n");
|
2008-09-03 19:02:00 +00:00
|
|
|
/* Send end gets TIFF from a file */
|
2013-01-01 15:07:55 +00:00
|
|
|
if ((send_state = t4_tx_init(NULL, in_file_name, -1, -1)) == NULL)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
printf("Failed to init T.4 send\n");
|
|
|
|
exit(2);
|
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
span_log_set_level(t4_tx_get_logging_state(send_state), SPAN_LOG_SHOW_SEVERITY | SPAN_LOG_SHOW_PROTOCOL | SPAN_LOG_SHOW_TAG | SPAN_LOG_FLOW);
|
|
|
|
t4_tx_set_min_bits_per_row(send_state, min_row_bits);
|
|
|
|
t4_tx_set_local_ident(send_state, "111 2222 3333");
|
2008-09-03 19:02:00 +00:00
|
|
|
|
|
|
|
/* Receive end puts TIFF to a new file. */
|
2013-04-01 16:18:21 +00:00
|
|
|
if ((receive_state = t4_rx_init(NULL, OUT_FILE_NAME, T4_COMPRESSION_T4_2D)) == NULL)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2009-05-14 21:52:52 +00:00
|
|
|
printf("Failed to init T.4 rx for '%s'\n", OUT_FILE_NAME);
|
2008-09-03 19:02:00 +00:00
|
|
|
exit(2);
|
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
span_log_set_level(t4_rx_get_logging_state(receive_state), SPAN_LOG_SHOW_SEVERITY | SPAN_LOG_SHOW_PROTOCOL | SPAN_LOG_SHOW_TAG | SPAN_LOG_FLOW);
|
2008-09-03 19:02:00 +00:00
|
|
|
|
|
|
|
/* Now send and receive all the pages in the source TIFF file */
|
|
|
|
sends = 0;
|
2009-05-14 21:52:52 +00:00
|
|
|
/* If we are stepping around the compression schemes, reset to the start of the sequence. */
|
|
|
|
if (compression_step > 0)
|
|
|
|
compression_step = 0;
|
2008-09-03 19:02:00 +00:00
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
end_marks = 0;
|
|
|
|
/* Add a header line to alternate pages, if required */
|
|
|
|
if (add_page_headers && (sends & 2))
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_tx_set_header_info(send_state, "Header");
|
2008-09-03 19:02:00 +00:00
|
|
|
else
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_tx_set_header_info(send_state, NULL);
|
2010-07-24 19:29:44 +00:00
|
|
|
if (page_header_tz && page_header_tz[0])
|
2011-12-05 17:00:22 +00:00
|
|
|
{
|
|
|
|
if (tz_init(&tz, page_header_tz))
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_tx_set_header_tz(send_state, &tz);
|
2011-12-05 17:00:22 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_tx_set_header_overlays_image(send_state, overlay_page_headers);
|
2008-09-03 19:02:00 +00:00
|
|
|
if (restart_pages && (sends & 1))
|
|
|
|
{
|
|
|
|
/* Use restart, to send the page a second time */
|
2013-01-01 15:07:55 +00:00
|
|
|
if (t4_tx_restart_page(send_state))
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-05-14 21:52:52 +00:00
|
|
|
if (compression_step >= 0)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2009-05-14 21:52:52 +00:00
|
|
|
compression = compression_sequence[compression_step++];
|
2012-08-12 14:11:06 +00:00
|
|
|
if (compression < 0 || (block_size == 0 && compression_step >= 3))
|
2010-07-24 19:29:44 +00:00
|
|
|
{
|
2009-05-14 21:52:52 +00:00
|
|
|
compression_step = 0;
|
2010-07-24 19:29:44 +00:00
|
|
|
compression = compression_sequence[compression_step++];
|
|
|
|
}
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
2013-07-19 07:40:22 +00:00
|
|
|
if (t4_tx_set_tx_image_format(send_state,
|
|
|
|
compression,
|
|
|
|
T4_SUPPORT_WIDTH_215MM
|
|
|
|
| T4_SUPPORT_LENGTH_US_LETTER
|
|
|
|
| T4_SUPPORT_LENGTH_US_LEGAL
|
|
|
|
| T4_SUPPORT_LENGTH_UNLIMITED,
|
|
|
|
T4_RESOLUTION_R8_STANDARD
|
|
|
|
| T4_RESOLUTION_R8_FINE
|
|
|
|
| T4_RESOLUTION_R8_SUPERFINE
|
|
|
|
| T4_RESOLUTION_R16_SUPERFINE
|
|
|
|
| T4_RESOLUTION_200_100
|
|
|
|
| T4_RESOLUTION_200_200
|
|
|
|
| T4_RESOLUTION_200_400
|
|
|
|
| T4_RESOLUTION_300_300
|
|
|
|
| T4_RESOLUTION_300_600
|
|
|
|
| T4_RESOLUTION_400_400
|
|
|
|
| T4_RESOLUTION_400_800
|
|
|
|
| T4_RESOLUTION_600_600
|
|
|
|
| T4_RESOLUTION_600_1200
|
|
|
|
| T4_RESOLUTION_1200_1200,
|
|
|
|
T4_RESOLUTION_100_100
|
|
|
|
| T4_RESOLUTION_200_200
|
|
|
|
| T4_RESOLUTION_300_300
|
|
|
|
| T4_RESOLUTION_400_400
|
|
|
|
| T4_RESOLUTION_600_600
|
|
|
|
| T4_RESOLUTION_1200_1200) < 0)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_rx_set_rx_encoding(receive_state, compression);
|
2008-09-03 19:02:00 +00:00
|
|
|
|
2013-01-01 15:07:55 +00:00
|
|
|
if (t4_tx_start_page(send_state))
|
2008-09-03 19:02:00 +00:00
|
|
|
break;
|
2013-07-23 16:46:47 +00:00
|
|
|
t4_rx_set_x_resolution(receive_state, t4_tx_get_tx_x_resolution(send_state));
|
|
|
|
t4_rx_set_y_resolution(receive_state, t4_tx_get_tx_y_resolution(send_state));
|
|
|
|
t4_rx_set_image_width(receive_state, t4_tx_get_tx_image_width(send_state));
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_rx_start_page(receive_state);
|
2013-07-19 07:40:22 +00:00
|
|
|
detect_non_ecm_page_end(-1, compression);
|
2013-08-08 13:40:28 +00:00
|
|
|
page_ended = false;
|
2010-07-24 19:29:44 +00:00
|
|
|
switch (block_size)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2010-07-24 19:29:44 +00:00
|
|
|
case 0:
|
2013-07-19 07:40:22 +00:00
|
|
|
/* Bit by bit operation. This is only appropriate for T.4 1D and 2D,
|
|
|
|
which are used without ECM. */
|
2013-01-01 15:07:55 +00:00
|
|
|
while ((bit = t4_tx_get_bit(send_state)) >= 0)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2009-05-14 21:52:52 +00:00
|
|
|
/* Monitor whether the EOLs are there in the correct amount */
|
2013-07-19 07:40:22 +00:00
|
|
|
if ((res = detect_non_ecm_page_end(bit, page_ended)))
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2012-08-12 14:11:06 +00:00
|
|
|
printf("Incorrect EOLs - %d\n", res);
|
2009-05-14 21:52:52 +00:00
|
|
|
tests_failed += (res - 1);
|
|
|
|
break;
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
2012-08-12 14:11:06 +00:00
|
|
|
if (bit_error_rate)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2012-08-12 14:11:06 +00:00
|
|
|
if ((rand() % bit_error_rate) == 0)
|
|
|
|
bit ^= 1;
|
|
|
|
}
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put_bit(receive_state, bit);
|
2012-08-12 14:11:06 +00:00
|
|
|
}
|
|
|
|
while (end_of_page != T4_DECODE_OK)
|
|
|
|
{
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put_bit(receive_state, 0);
|
2012-08-12 14:11:06 +00:00
|
|
|
if (++end_marks > 50)
|
|
|
|
{
|
|
|
|
printf("Receiver missed the end of page mark\n");
|
|
|
|
tests_failed++;
|
|
|
|
break;
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
|
|
|
}
|
2009-05-14 21:52:52 +00:00
|
|
|
/* Now throw junk at the receive context, to ensure stuff occuring
|
|
|
|
after the end of page condition has no bad effect. */
|
|
|
|
for (i = 0; i < 1000; i++)
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_rx_put_bit(receive_state, (rand() >> 10) & 1);
|
2010-07-24 19:29:44 +00:00
|
|
|
break;
|
2012-08-12 14:11:06 +00:00
|
|
|
default:
|
2008-09-03 19:02:00 +00:00
|
|
|
do
|
|
|
|
{
|
2013-01-01 15:07:55 +00:00
|
|
|
len = t4_tx_get(send_state, block, block_size);
|
2012-08-12 14:11:06 +00:00
|
|
|
if (len > 0)
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, len);
|
2010-07-24 19:29:44 +00:00
|
|
|
}
|
2012-08-12 14:11:06 +00:00
|
|
|
while (len > 0);
|
2013-07-19 07:40:22 +00:00
|
|
|
/* Some decoders require a few extra bits before they recognise the end
|
2012-08-12 14:11:06 +00:00
|
|
|
of an image, so be prepared to offer it a few. */
|
|
|
|
while (end_of_page != T4_DECODE_OK)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2012-08-12 14:11:06 +00:00
|
|
|
block[0] = 0;
|
2013-01-01 15:07:55 +00:00
|
|
|
end_of_page = t4_rx_put(receive_state, block, 1);
|
2012-08-12 14:11:06 +00:00
|
|
|
if (++end_marks > 5)
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
2012-08-12 14:11:06 +00:00
|
|
|
printf("Receiver missed the end of page mark\n");
|
|
|
|
tests_failed++;
|
|
|
|
break;
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
|
|
|
}
|
2010-07-24 19:29:44 +00:00
|
|
|
break;
|
2008-09-03 19:02:00 +00:00
|
|
|
}
|
|
|
|
if (dump_as_xxx)
|
2013-01-01 15:07:55 +00:00
|
|
|
dump_image_as_xxx(receive_state);
|
|
|
|
display_page_stats(receive_state);
|
2008-09-03 19:02:00 +00:00
|
|
|
if (!restart_pages || (sends & 1))
|
2013-01-01 15:07:55 +00:00
|
|
|
t4_tx_end_page(send_state);
|
|
|
|
t4_rx_end_page(receive_state);
|
2008-09-03 19:02:00 +00:00
|
|
|
sends++;
|
|
|
|
}
|
2014-06-14 11:49:05 +00:00
|
|
|
t4_tx_free(send_state);
|
|
|
|
t4_rx_free(receive_state);
|
2008-09-03 19:02:00 +00:00
|
|
|
/* And we should now have a matching received TIFF file. Note this will only match
|
|
|
|
at the image level. TIFF files allow a lot of ways to express the same thing,
|
|
|
|
so bit matching of the files is not the normal case. */
|
|
|
|
fflush(stdout);
|
2009-05-14 21:52:52 +00:00
|
|
|
sprintf(buf, "tiffcmp -t %s %s", in_file_name, OUT_FILE_NAME);
|
|
|
|
if (tests_failed || system(buf))
|
2008-09-03 19:02:00 +00:00
|
|
|
{
|
|
|
|
printf("Tests failed\n");
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
printf("Tests passed\n");
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
/*- End of file ------------------------------------------------------------*/
|