Files
wxWidgets/tests/streams/datastreamtest.cpp
Vadim Zeitlin 789ab84044 Add IEEE 754 single/double precision support to wxDataStream classes.
Allow to optionally raed/write float/double values in IEEE 754 single/double
precision formats, respectively, instead of always using the extended
precision format for both of them.

This makes the code more flexible, allowing for better interoperability with
the other programs, and also allows to implement floating point functions in
these classes even when wxUSE_APPLE_IEEE is turned off (as can be the case
because of the licencing concerns for the code in extended.c).

Closes #10625.

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@73938 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
2013-05-06 00:31:03 +00:00

294 lines
8.3 KiB
C++

///////////////////////////////////////////////////////////////////////////////
// Name: tests/streams/datastreamtest.cpp
// Purpose: wxDataXXXStream Unit Test
// Author: Ryan Norton
// Created: 2004-08-14
// RCS-ID: $Id$
// Copyright: (c) 2004 Ryan Norton
///////////////////////////////////////////////////////////////////////////////
// ----------------------------------------------------------------------------
// headers
// ----------------------------------------------------------------------------
#include "testprec.h"
#ifdef __BORLANDC__
#pragma hdrstop
#endif
#ifndef WX_PRECOMP
#include "wx/wx.h"
#endif // WX_PRECOMP
#include <vector>
#include "wx/datstrm.h"
#include "wx/wfstream.h"
#include "wx/math.h"
// ----------------------------------------------------------------------------
// test class
// ----------------------------------------------------------------------------
class DataStreamTestCase : public CppUnit::TestCase
{
public:
DataStreamTestCase();
private:
CPPUNIT_TEST_SUITE( DataStreamTestCase );
CPPUNIT_TEST( FloatRW );
CPPUNIT_TEST( DoubleRW );
#if wxUSE_LONGLONG
CPPUNIT_TEST( LongLongRW );
#endif
#if wxHAS_INT64
CPPUNIT_TEST( Int64RW );
#endif
CPPUNIT_TEST( NaNRW );
CPPUNIT_TEST( PseudoTest_UseBigEndian );
CPPUNIT_TEST( FloatRW );
CPPUNIT_TEST( DoubleRW );
// Only test standard IEEE 754 formats if we're using IEEE extended
// format by default, otherwise the tests above already covered them.
#if wxUSE_APPLE_IEEE
CPPUNIT_TEST( PseudoTest_UseIEEE754 );
CPPUNIT_TEST( FloatRW );
CPPUNIT_TEST( DoubleRW );
// Also retest little endian version with standard formats.
CPPUNIT_TEST( PseudoTest_UseLittleEndian );
CPPUNIT_TEST( FloatRW );
CPPUNIT_TEST( DoubleRW );
#endif // wxUSE_APPLE_IEEE
CPPUNIT_TEST_SUITE_END();
wxFloat64 TestFloatRW(wxFloat64 fValue);
void FloatRW();
void DoubleRW();
#if wxUSE_LONGLONG
void LongLongRW();
#endif
#if wxHAS_INT64
void Int64RW();
#endif
void NaNRW();
void PseudoTest_UseBigEndian() { ms_useBigEndianFormat = true; }
void PseudoTest_UseLittleEndian() { ms_useBigEndianFormat = false; }
#if wxUSE_APPLE_IEEE
void PseudoTest_UseIEEE754() { ms_useIEEE754 = true; }
#endif // wxUSE_APPLE_IEEE
static bool ms_useBigEndianFormat;
#if wxUSE_APPLE_IEEE
static bool ms_useIEEE754;
#endif // wxUSE_APPLE_IEEE
DECLARE_NO_COPY_CLASS(DataStreamTestCase)
};
// register in the unnamed registry so that these tests are run by default
CPPUNIT_TEST_SUITE_REGISTRATION( DataStreamTestCase );
// also include in its own registry so that these tests can be run alone
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( DataStreamTestCase, "DataStreamTestCase" );
bool DataStreamTestCase::ms_useBigEndianFormat = false;
#if wxUSE_APPLE_IEEE
bool DataStreamTestCase::ms_useIEEE754 = false;
#endif // wxUSE_APPLE_IEEE
DataStreamTestCase::DataStreamTestCase()
{
}
wxFloat64 DataStreamTestCase::TestFloatRW(wxFloat64 fValue)
{
{
wxFileOutputStream pFileOutput( wxT("mytext.dat") );
wxDataOutputStream pDataOutput( pFileOutput );
if ( ms_useBigEndianFormat )
pDataOutput.BigEndianOrdered(true);
#if wxUSE_APPLE_IEEE
if ( ms_useIEEE754 )
pDataOutput.UseBasicPrecisions();
#endif // wxUSE_APPLE_IEEE
pDataOutput << fValue;
}
wxFileInputStream pFileInput( wxT("mytext.dat") );
wxDataInputStream pDataInput( pFileInput );
if ( ms_useBigEndianFormat )
pDataInput.BigEndianOrdered(true);
#if wxUSE_APPLE_IEEE
if ( ms_useIEEE754 )
pDataInput.UseBasicPrecisions();
#endif // wxUSE_APPLE_IEEE
wxFloat64 fInFloat;
pDataInput >> fInFloat;
return fInFloat;
}
template <class T>
class TestMultiRW {
public:
typedef std::vector<T> ValueArray;
typedef void (wxDataOutputStream::*FnWriter)(const T *buffer, size_t size);
typedef void (wxDataInputStream::*FnReader)(T *buffer, size_t size);
private:
bool m_ok;
private:
void ProcessData(const T *Values,
typename ValueArray::size_type Size,
FnWriter pfnWriter,
FnReader pfnReader)
{
ValueArray InValues(Size);
{
wxFileOutputStream FileOutput( wxT("mytext.dat") );
wxDataOutputStream DataOutput( FileOutput );
(DataOutput.*pfnWriter)(Values, Size);
}
{
wxFileInputStream FileInput( wxT("mytext.dat") );
wxDataInputStream DataInput( FileInput );
(DataInput.*pfnReader)(&*InValues.begin(), InValues.size());
}
m_ok = true;
for (typename ValueArray::size_type idx=0; idx!=Size; ++idx) {
if (InValues[idx]!=Values[idx]) {
m_ok = false;
break;
}
}
}
public:
TestMultiRW(const T *Values,
size_t Size,
FnWriter pfnWriter,
FnReader pfnReader)
{
ProcessData(Values, (typename ValueArray::size_type) Size, pfnWriter, pfnReader);
}
TestMultiRW(const ValueArray &Values,
FnWriter pfnWriter,
FnReader pfnReader)
{
ProcessData(&*Values.begin(), Values.size(), pfnWriter, pfnReader);
}
bool IsOk() const
{
return m_ok;
}
};
template <class T>
static
T TestRW(const T &Value)
{
T InValue;
{
wxFileOutputStream FileOutput( wxT("mytext.dat") );
wxDataOutputStream DataOutput( FileOutput );
DataOutput << Value;
}
{
wxFileInputStream FileInput( wxT("mytext.dat") );
wxDataInputStream DataInput( FileInput );
DataInput >> InValue;
}
return InValue;
}
void DataStreamTestCase::FloatRW()
{
CPPUNIT_ASSERT( TestFloatRW(5.5) == 5.5 );
CPPUNIT_ASSERT( TestFloatRW(5) == 5 );
CPPUNIT_ASSERT( TestFloatRW(5.55) == 5.55 );
CPPUNIT_ASSERT( TestFloatRW(55555.555555) == 55555.555555 );
}
void DataStreamTestCase::DoubleRW()
{
CPPUNIT_ASSERT( TestFloatRW(2132131.1232132) == 2132131.1232132 );
CPPUNIT_ASSERT( TestFloatRW(21321343431.1232143432) == 21321343431.1232143432 );
}
#if wxUSE_LONGLONG
void DataStreamTestCase::LongLongRW()
{
TestMultiRW<wxLongLong>::ValueArray ValuesLL;
TestMultiRW<wxULongLong>::ValueArray ValuesULL;
ValuesLL.push_back(wxLongLong(0l));
ValuesLL.push_back(wxLongLong(1l));
ValuesLL.push_back(wxLongLong(-1l));
ValuesLL.push_back(wxLongLong(0x12345678l));
ValuesLL.push_back(wxLongLong(0x12345678l, 0xabcdef01l));
ValuesULL.push_back(wxULongLong(0l));
ValuesULL.push_back(wxULongLong(1l));
ValuesULL.push_back(wxULongLong(0x12345678l));
ValuesULL.push_back(wxULongLong(0x12345678l, 0xabcdef01l));
CPPUNIT_ASSERT( TestRW(wxLongLong(0x12345678l)) == wxLongLong(0x12345678l) );
CPPUNIT_ASSERT( TestRW(wxLongLong(0x12345678l, 0xabcdef01l)) == wxLongLong(0x12345678l, 0xabcdef01l) );
CPPUNIT_ASSERT( TestMultiRW<wxLongLong>(ValuesLL, &wxDataOutputStream::WriteLL, &wxDataInputStream::ReadLL).IsOk() );
CPPUNIT_ASSERT( TestMultiRW<wxULongLong>(ValuesULL, &wxDataOutputStream::WriteLL, &wxDataInputStream::ReadLL).IsOk() );
}
#endif
#if wxHAS_INT64
void DataStreamTestCase::Int64RW()
{
TestMultiRW<wxInt64>::ValueArray ValuesI64;
TestMultiRW<wxUint64>::ValueArray ValuesUI64;
ValuesI64.push_back(wxInt64(0l));
ValuesI64.push_back(wxInt64(1l));
ValuesI64.push_back(wxInt64(-1l));
ValuesI64.push_back(wxInt64(0x12345678l));
ValuesI64.push_back((wxInt64(0x12345678l) << 32) + wxInt64(0xabcdef01l));
ValuesUI64.push_back(wxUint64(0l));
ValuesUI64.push_back(wxUint64(1l));
ValuesUI64.push_back(wxUint64(0x12345678l));
ValuesUI64.push_back((wxUint64(0x12345678l) << 32) + wxUint64(0xabcdef01l));
CPPUNIT_ASSERT( TestRW(wxUint64(0x12345678l)) == wxUint64(0x12345678l) );
CPPUNIT_ASSERT( TestRW((wxUint64(0x12345678l) << 32) + wxUint64(0xabcdef01l)) == (wxUint64(0x12345678l) << 32) + wxUint64(0xabcdef01l) );
CPPUNIT_ASSERT( TestMultiRW<wxInt64>(ValuesI64, &wxDataOutputStream::Write64, &wxDataInputStream::Read64).IsOk() );
CPPUNIT_ASSERT( TestMultiRW<wxUint64>(ValuesUI64, &wxDataOutputStream::Write64, &wxDataInputStream::Read64).IsOk() );
}
#endif
void DataStreamTestCase::NaNRW()
{
//TODO?
}