2009-05-19 11:23:13 +00:00
|
|
|
#include <cxxtest/TestSuite.h>
|
|
|
|
|
|
|
|
#include "common/serializer.h"
|
|
|
|
#include "common/stream.h"
|
|
|
|
|
|
|
|
class SerializerTestSuite : public CxxTest::TestSuite {
|
|
|
|
Common::SeekableReadStream *_inStreamV1;
|
|
|
|
Common::SeekableReadStream *_inStreamV2;
|
|
|
|
public:
|
|
|
|
void setUp() {
|
|
|
|
// Our pseudo data format is as follows:
|
|
|
|
// * magic id - string "MAGI"
|
|
|
|
// * Version - uint32, LE
|
|
|
|
// * uint32, LE (available in v2 onward)
|
|
|
|
// * uint16, BE (available in v1 onward)
|
|
|
|
// * sint16, LE (available only in v1)
|
|
|
|
// * byte (always available)
|
|
|
|
static const byte contents_v1[] = {
|
|
|
|
'M', 'A', 'G', 'I', // magic id
|
|
|
|
0x01, 0x00, 0x00, 0x00, // Version
|
|
|
|
0x06, 0x07, // uint16, BE (available in v1 onward)
|
|
|
|
0xfe, 0xff, // sint16, LE (available only in v1)
|
|
|
|
0x0a // byte (always available)
|
|
|
|
};
|
|
|
|
static const byte contents_v2[] = {
|
|
|
|
'M', 'A', 'G', 'I', // magic id
|
|
|
|
0x02, 0x00, 0x00, 0x00, // Version
|
|
|
|
0x02, 0x03, 0x04, 0x05, // uint32, LE (available in v2 onward)
|
|
|
|
0x06, 0x07, // uint16, BE (available in v1 onward)
|
|
|
|
0x0a // byte (always available)
|
|
|
|
};
|
|
|
|
|
|
|
|
_inStreamV1 = new Common::MemoryReadStream(contents_v1, sizeof(contents_v1));
|
|
|
|
_inStreamV2 = new Common::MemoryReadStream(contents_v2, sizeof(contents_v2));
|
|
|
|
}
|
|
|
|
|
2009-05-19 12:39:53 +00:00
|
|
|
void tearDown() {
|
|
|
|
delete _inStreamV1;
|
|
|
|
delete _inStreamV2;
|
|
|
|
}
|
2009-05-19 11:23:13 +00:00
|
|
|
|
|
|
|
// A method which reads a v1 file
|
2009-05-19 12:39:53 +00:00
|
|
|
void readVersioned_v1(Common::SeekableReadStream *stream, Common::Serializer::Version version) {
|
2009-05-19 11:23:13 +00:00
|
|
|
Common::Serializer ser(stream, 0);
|
|
|
|
|
2009-05-26 11:31:45 +00:00
|
|
|
TS_ASSERT(ser.matchBytes("MAGI", 4));
|
2009-05-19 11:23:13 +00:00
|
|
|
|
|
|
|
TS_ASSERT(ser.syncVersion(1));
|
|
|
|
TS_ASSERT_EQUALS(ser.getVersion(), version);
|
|
|
|
|
2009-05-20 17:51:37 +00:00
|
|
|
uint32 tmp = 0;
|
2009-05-19 11:23:13 +00:00
|
|
|
|
|
|
|
ser.syncAsUint16BE(tmp, Common::Serializer::Version(1));
|
2009-05-19 12:39:53 +00:00
|
|
|
TS_ASSERT_EQUALS(tmp, (uint16)0x0607);
|
2009-05-19 11:23:13 +00:00
|
|
|
|
|
|
|
ser.syncAsSint16LE(tmp, Common::Serializer::Version(1));
|
2009-05-19 12:39:53 +00:00
|
|
|
TS_ASSERT_EQUALS((int16)tmp, -2);
|
2009-05-19 11:23:13 +00:00
|
|
|
|
|
|
|
ser.syncAsByte(tmp);
|
2009-05-19 12:39:53 +00:00
|
|
|
TS_ASSERT_EQUALS(tmp, (uint8)0x0a);
|
2009-05-19 11:23:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// A method which reads a v2 file
|
2009-05-19 12:39:53 +00:00
|
|
|
void readVersioned_v2(Common::SeekableReadStream *stream, Common::Serializer::Version version) {
|
2009-05-19 11:23:13 +00:00
|
|
|
Common::Serializer ser(stream, 0);
|
|
|
|
|
2009-05-26 11:31:45 +00:00
|
|
|
TS_ASSERT(ser.matchBytes("MAGI", 4));
|
2009-05-19 11:23:13 +00:00
|
|
|
|
|
|
|
TS_ASSERT(ser.syncVersion(2));
|
|
|
|
TS_ASSERT_EQUALS(ser.getVersion(), version);
|
|
|
|
|
|
|
|
uint32 tmp;
|
|
|
|
|
|
|
|
// Read a value only available starting with v2.
|
|
|
|
// Thus if we load an old save, it must be
|
|
|
|
// manually set. To simplify that, no sync method should
|
|
|
|
// modify the value passed to it if nothing was read!
|
|
|
|
tmp = 0x12345678;
|
|
|
|
ser.syncAsUint32LE(tmp, Common::Serializer::Version(2));
|
|
|
|
if (ser.getVersion() < 2) {
|
2009-05-19 12:39:53 +00:00
|
|
|
TS_ASSERT_EQUALS(tmp, (uint32)0x12345678);
|
2009-05-19 11:23:13 +00:00
|
|
|
} else {
|
2009-05-19 12:39:53 +00:00
|
|
|
TS_ASSERT_EQUALS(tmp, (uint32)0x05040302);
|
2009-05-19 11:23:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ser.syncAsUint16BE(tmp, Common::Serializer::Version(1));
|
2009-05-19 12:39:53 +00:00
|
|
|
TS_ASSERT_EQUALS(tmp, (uint32)0x0607);
|
2009-05-19 11:23:13 +00:00
|
|
|
|
|
|
|
// Skip over obsolete data
|
|
|
|
ser.skip(2, Common::Serializer::Version(1), Common::Serializer::Version(1));
|
|
|
|
|
|
|
|
ser.syncAsByte(tmp);
|
2009-05-19 12:39:53 +00:00
|
|
|
TS_ASSERT_EQUALS(tmp, (uint8)0x0a);
|
2009-05-19 11:23:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void test_read_v1_as_v1() {
|
|
|
|
readVersioned_v1(_inStreamV1, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// There is no test_read_v2_as_v1() because a v1 parser cannot possibly
|
|
|
|
// read v2 data correctly. It should instead error out if it
|
|
|
|
// detects a version newer than its current version.
|
|
|
|
|
2009-05-19 12:39:53 +00:00
|
|
|
void test_read_v1_as_v2() {
|
2009-05-19 11:23:13 +00:00
|
|
|
readVersioned_v2(_inStreamV1, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_read_v2_as_v2() {
|
|
|
|
readVersioned_v2(_inStreamV2, 2);
|
|
|
|
}
|
|
|
|
};
|