test unpacking of PackBits()

This commit is contained in:
Thomas Bernard 2019-11-15 23:46:56 +01:00
parent 3ba4472626
commit d56915edc6
No known key found for this signature in database
GPG Key ID: 0FF11B67A5C0863C
4 changed files with 131 additions and 2 deletions

View File

@ -819,7 +819,7 @@ endif
TESTSOBJS = $(patsubst %.c,%.o,$(wildcard tests/*.c)) \
miscfileformats.o fileformats.o oldies.o libraw2crtc.o \
loadsavefuncs.o \
loadsavefuncs.o packbits.o \
unicode.o \
io.o realpath.o version.o pversion.o \
gfx2surface.o \

69
src/packbits.c Normal file
View File

@ -0,0 +1,69 @@
/* vim:expandtab:ts=2 sw=2:
*/
/* Grafx2 - The Ultimate 256-color bitmap paint program
Copyright 2018-2019 Thomas Bernard
Copyright 2008 Yves Rizoud
Copyright 2007 Adrien Destugues
Copyright 1996-2001 Sunset Design (Guillaume Dorme & Karl Maritaud)
Grafx2 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; version 2
of the License.
Grafx2 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 Grafx2; if not, see <http://www.gnu.org/licenses/>
*/
///@file packbits.c
/// Packbits compression as used in IFF etc.
/// see http://fileformats.archiveteam.org/wiki/PackBits
#include <stdio.h>
#include <string.h>
#include "struct.h"
#include "gfx2log.h"
#include "packbits.h"
int PackBits_unpack_from_file(FILE * f, byte * dest, unsigned int count)
{
unsigned int i = 0;
while (i < count)
{
byte cmd;
if (fread(&cmd, 1, 1, f) != 1)
return PACKBITS_UNPACK_READ_ERROR;
if (cmd > 128)
{
// cmd > 128 => repeat (257 - cmd) the next byte
byte v;
if (fread(&v, 1, 1, f) != 1)
return PACKBITS_UNPACK_READ_ERROR;
if (count < (i + 257 - cmd))
return PACKBITS_UNPACK_OVERFLOW_ERROR;
memset(dest + i, v, (257 - cmd));
i += (257 - cmd);
}
else if (cmd < 128)
{
// cmd < 128 => copy (cmd + 1) bytes
if (count < (i + cmd + 1))
return PACKBITS_UNPACK_OVERFLOW_ERROR;
if (fread(dest + i, 1, (cmd + 1), f) != (unsigned)(cmd + 1))
return PACKBITS_UNPACK_READ_ERROR;
i += (cmd + 1);
}
else
{
// 128 = NOP
GFX2_Log(GFX2_WARNING, "NOP in packbits stream\n");
}
}
return PACKBITS_UNPACK_OK;
}

41
src/packbits.h Normal file
View File

@ -0,0 +1,41 @@
/* vim:expandtab:ts=2 sw=2:
*/
/* Grafx2 - The Ultimate 256-color bitmap paint program
Copyright 2018-2019 Thomas Bernard
Copyright 2008 Yves Rizoud
Copyright 2007 Adrien Destugues
Copyright 1996-2001 Sunset Design (Guillaume Dorme & Karl Maritaud)
Grafx2 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; version 2
of the License.
Grafx2 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 Grafx2; if not, see <http://www.gnu.org/licenses/>
*/
///@file packbits.h
/// Packbits compression as used in IFF etc.
#ifndef PACKBITS_H_INCLUDED
#define PACKBITS_H_INCLUDED
// error codes :
#define PACKBITS_UNPACK_OK 0
#define PACKBITS_UNPACK_READ_ERROR -1
#define PACKBITS_UNPACK_OVERFLOW_ERROR -2
/**
* @return PACKBITS_UNPACK_OK or PACKBITS_UNPACK_READ_ERROR or PACKBITS_UNPACK_OVERFLOW_ERROR
*/
int PackBits_unpack_from_file(FILE * f, byte * dest, unsigned int count);
#endif

View File

@ -33,6 +33,7 @@
#include <unistd.h>
#include "../struct.h"
#include "../oldies.h"
#include "../packbits.h"
#include "../gfx2log.h"
unsigned int MOTO_MAP_pack(byte * packed, const byte * unpacked, unsigned int unpacked_len);
@ -160,6 +161,7 @@ int Test_Packbits(void)
NULL
};
const long best_packed = 7 + 22 + 23;
byte buffer[1024];
snprintf(tempfilename, sizeof(tempfilename), "/tmp/gfx2test-packbits-%lx", random());
GFX2_Log(GFX2_DEBUG, "tempfile %s\n", tempfilename);
@ -190,13 +192,30 @@ int Test_Packbits(void)
return 0;
}
// TODO : test unpacking
// test unpacking
f = fopen(tempfilename, "rb");
if (f == NULL)
{
GFX2_Log(GFX2_ERROR, "Failed to open %s for reading\n", tempfilename);
return 0;
}
for (i = 0; tests[i]; i++)
{
size_t len = strlen(tests[i]);
memset(buffer, 0x80, len);
if (PackBits_unpack_from_file(f, buffer, len) != PACKBITS_UNPACK_OK)
{
GFX2_Log(GFX2_ERROR, "PackBits_unpack_from_file() failed\n");
return 0;
}
if (memcmp(buffer, tests[i], len) != 0)
{
GFX2_Log(GFX2_ERROR, "uncompressed stream mismatch !\n");
GFX2_LogHexDump(GFX2_ERROR, "original ", (const byte *)tests[i], 0, len);
GFX2_LogHexDump(GFX2_ERROR, "unpacked ", buffer, 0, len);
return 0;
}
}
fclose(f);
unlink(tempfilename);
return 1; // test OK