mirror of
https://github.com/pocoproject/poco.git
synced 2025-04-02 17:50:53 +02:00

Larger buffers improve performance significantly when streaming large quantity of data on very fast devices.
487 lines
11 KiB
C++
487 lines
11 KiB
C++
//
|
|
// FileStreamTest.cpp
|
|
//
|
|
// Copyright (c) 2007, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#include "FileStreamTest.h"
|
|
#include "CppUnit/TestCaller.h"
|
|
#include "CppUnit/TestSuite.h"
|
|
#include "Poco/FileStream.h"
|
|
#include "Poco/File.h"
|
|
#include "Poco/TemporaryFile.h"
|
|
#include "Poco/Exception.h"
|
|
#include "Poco/Stopwatch.h"
|
|
#include <iostream>
|
|
|
|
|
|
FileStreamTest::FileStreamTest(const std::string& name): CppUnit::TestCase(name)
|
|
{
|
|
}
|
|
|
|
|
|
FileStreamTest::~FileStreamTest()
|
|
{
|
|
}
|
|
|
|
|
|
void FileStreamTest::testRead()
|
|
{
|
|
#if defined(POCO_OS_FAMILY_WINDOWS)
|
|
char tmp[]={'\xc3', '\x84', '\xc3', '\x96', '\xc3', '\x9c', '\xc3', '\xa4', '\xc3', '\xb6', '\xc3', '\xbc', '\0'};
|
|
std::string file(tmp);
|
|
file.append(".txt");
|
|
#else
|
|
std::string file("testfile.txt");
|
|
#endif
|
|
|
|
Poco::TemporaryFile::registerForDeletion(file);
|
|
|
|
Poco::FileOutputStream fos(file, std::ios::binary);
|
|
fos << "sometestdata";
|
|
fos.close();
|
|
|
|
Poco::FileInputStream fis(file);
|
|
assertTrue (fis.good());
|
|
std::string read;
|
|
fis >> read;
|
|
assertTrue (!read.empty());
|
|
}
|
|
|
|
|
|
#if defined(POCO_OS_FAMILY_WINDOWS)
|
|
#include "Poco/UnicodeConverter.h"
|
|
#else
|
|
#include <fcntl.h>
|
|
#endif
|
|
|
|
void FileStreamTest::testWriteReadNativeHandle()
|
|
{
|
|
Poco::FileOutputStream fos;
|
|
Poco::FileInputStream fis;
|
|
Poco::FileIOS::NativeHandle outHandle;
|
|
|
|
#if defined(POCO_OS_FAMILY_WINDOWS)
|
|
char tmp[]={'\xc3', '\x84', '\xc3', '\x96', '\xc3', '\x9c', '\xc3', '\xa4', '\xc3', '\xb6', '\xc3', '\xbc', '\0'};
|
|
std::string file(tmp);
|
|
file.append(".txt");
|
|
std::wstring utf16Path;
|
|
Poco::UnicodeConverter::toUTF16(file, utf16Path);
|
|
outHandle = CreateFileW(utf16Path.c_str(), GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_WRITE, nullptr, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, nullptr);
|
|
assertTrue(outHandle != INVALID_HANDLE_VALUE);
|
|
#else
|
|
std::string file("testfile.txt");
|
|
outHandle = ::open(file.c_str(), O_WRONLY, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH);
|
|
#endif
|
|
|
|
Poco::TemporaryFile::registerForDeletion(file);
|
|
|
|
fos.openHandle(outHandle, std::ios::binary | std::ios::out | std::ios::trunc);
|
|
fos << "sometestdata";
|
|
fos.close();
|
|
|
|
Poco::FileIOS::NativeHandle inHandle;
|
|
|
|
#if defined(POCO_OS_FAMILY_WINDOWS)
|
|
inHandle = CreateFileW(utf16Path.c_str(), GENERIC_READ, FILE_SHARE_READ, nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, nullptr);
|
|
assertTrue(inHandle != INVALID_HANDLE_VALUE);
|
|
#else
|
|
inHandle = ::open(file.c_str(), O_RDONLY, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH);
|
|
#endif
|
|
|
|
fis.openHandle(inHandle, std::ios::in);
|
|
assertTrue (fis.good());
|
|
std::string read;
|
|
fis >> read;
|
|
assertTrue (!read.empty());
|
|
}
|
|
|
|
|
|
void FileStreamTest::testWrite()
|
|
{
|
|
#if defined(POCO_OS_FAMILY_WINDOWS)
|
|
char tmp[]={'\xc3', '\x9f', '\xc3', '\x84', '\xc3', '\x96', '\xc3', '\x9c', '\xc3', '\xa4', '\xc3', '\xb6', '\xc3', '\xbc', '\0'};
|
|
std::string file(tmp);
|
|
file = "dummy_" + file + ".txt";
|
|
#else
|
|
std::string file("dummy_file.txt");
|
|
#endif
|
|
|
|
Poco::TemporaryFile::registerForDeletion(file);
|
|
|
|
Poco::FileOutputStream fos(file);
|
|
assertTrue (fos.good());
|
|
fos << "hiho";
|
|
fos.close();
|
|
|
|
Poco::FileInputStream fis(file);
|
|
assertTrue (fis.good());
|
|
std::string read;
|
|
fis >> read;
|
|
assertTrue (read == "hiho");
|
|
}
|
|
|
|
|
|
void FileStreamTest::testReadWrite()
|
|
{
|
|
char tmp[]={'\xc3', '\x9f', '\xc3', '\x84', '\xc3', '\x96', '\xc3', '\x9c', '\xc3', '\xa4', '\xc3', '\xb6', '\xc3', '\xbc', '\0'};
|
|
std::string file(tmp);
|
|
file = "dummy_" + file + ".txt";
|
|
|
|
Poco::TemporaryFile::registerForDeletion(file);
|
|
|
|
Poco::FileStream fos(file);
|
|
assertTrue (fos.good());
|
|
fos << "hiho";
|
|
fos.seekg(0, std::ios::beg);
|
|
std::string read;
|
|
fos >> read;
|
|
assertTrue (read == "hiho");
|
|
}
|
|
|
|
|
|
void FileStreamTest::testOpen()
|
|
{
|
|
Poco::FileOutputStream ostr;
|
|
ostr.open("test.txt", std::ios::out);
|
|
assertTrue (ostr.good());
|
|
ostr.close();
|
|
}
|
|
|
|
|
|
void FileStreamTest::testOpenModeIn()
|
|
{
|
|
Poco::File f("nonexistent.txt");
|
|
if (f.exists())
|
|
f.remove();
|
|
|
|
try
|
|
{
|
|
Poco::FileInputStream istr("nonexistent.txt");
|
|
fail("nonexistent file - must throw");
|
|
}
|
|
catch (Poco::Exception&)
|
|
{
|
|
}
|
|
|
|
f.createFile();
|
|
Poco::FileInputStream istr("nonexistent.txt");
|
|
assertTrue (istr.good());
|
|
}
|
|
|
|
|
|
void FileStreamTest::testOpenModeOut()
|
|
{
|
|
Poco::File f("test.txt");
|
|
if (f.exists())
|
|
f.remove();
|
|
|
|
Poco::FileOutputStream ostr1("test.txt");
|
|
ostr1 << "Hello, world!";
|
|
ostr1.close();
|
|
|
|
assertTrue (f.exists());
|
|
assertTrue (f.getSize() != 0);
|
|
|
|
Poco::FileStream str1("test.txt");
|
|
str1.close();
|
|
|
|
assertTrue (f.exists());
|
|
assertTrue (f.getSize() != 0);
|
|
|
|
Poco::FileOutputStream ostr2("test.txt");
|
|
ostr2.close();
|
|
|
|
assertTrue (f.exists());
|
|
assertTrue (f.getSize() == 0);
|
|
|
|
f.remove();
|
|
}
|
|
|
|
|
|
void FileStreamTest::testOpenModeTrunc()
|
|
{
|
|
Poco::File f("test.txt");
|
|
if (f.exists())
|
|
f.remove();
|
|
|
|
Poco::FileOutputStream ostr1("test.txt");
|
|
ostr1 << "Hello, world!";
|
|
ostr1.close();
|
|
|
|
assertTrue (f.exists());
|
|
assertTrue (f.getSize() != 0);
|
|
|
|
Poco::FileStream str1("test.txt", std::ios::trunc);
|
|
str1.close();
|
|
|
|
assertTrue (f.exists());
|
|
assertTrue (f.getSize() == 0);
|
|
|
|
f.remove();
|
|
}
|
|
|
|
|
|
void FileStreamTest::testOpenModeAte()
|
|
{
|
|
Poco::FileOutputStream ostr("test.txt");
|
|
ostr << "0123456789";
|
|
ostr.close();
|
|
|
|
Poco::FileStream str1("test.txt", std::ios::in | std::ios::ate);
|
|
int c = str1.get();
|
|
assertTrue (str1.eof());
|
|
|
|
str1.clear();
|
|
str1.seekg(0);
|
|
c = str1.get();
|
|
assertTrue (c == '0');
|
|
|
|
str1.close();
|
|
|
|
Poco::FileStream str2("test.txt", std::ios::in | std::ios::out | std::ios::ate);
|
|
str2 << "abcdef";
|
|
str2.seekg(0);
|
|
std::string s;
|
|
str2 >> s;
|
|
assertTrue (s == "0123456789abcdef");
|
|
str2.close();
|
|
}
|
|
|
|
|
|
void FileStreamTest::testOpenModeApp()
|
|
{
|
|
Poco::FileOutputStream ostr("test.txt");
|
|
ostr << "0123456789";
|
|
ostr.close();
|
|
|
|
Poco::FileStream str1("test.txt", std::ios::in | std::ios::out | std::ios::app);
|
|
|
|
str1 << "abc";
|
|
|
|
str1.seekp(0);
|
|
|
|
str1 << "def";
|
|
|
|
str1.close();
|
|
|
|
Poco::FileInputStream istr("test.txt");
|
|
std::string s;
|
|
istr >> s;
|
|
assertTrue (s == "0123456789abcdef");
|
|
istr.close();
|
|
}
|
|
|
|
|
|
void FileStreamTest::testSeek()
|
|
{
|
|
Poco::FileStream str("test.txt", std::ios::in | std::ios::out | std::ios::trunc);
|
|
str << "0123456789abcdef";
|
|
|
|
str.seekg(0);
|
|
int c = str.get();
|
|
assertTrue (c == '0');
|
|
|
|
str.seekg(10);
|
|
assertTrue (str.tellg() == std::streampos(10));
|
|
c = str.get();
|
|
assertTrue (c == 'a');
|
|
assertTrue (str.tellg() == std::streampos(11));
|
|
|
|
str.seekg(-1, std::ios::end);
|
|
assertTrue (str.tellg() == std::streampos(15));
|
|
c = str.get();
|
|
assertTrue (c == 'f');
|
|
assertTrue (str.tellg() == std::streampos(16));
|
|
|
|
str.seekg(-1, std::ios::cur);
|
|
assertTrue (str.tellg() == std::streampos(15));
|
|
c = str.get();
|
|
assertTrue (c == 'f');
|
|
assertTrue (str.tellg() == std::streampos(16));
|
|
|
|
str.seekg(-4, std::ios::cur);
|
|
assertTrue (str.tellg() == std::streampos(12));
|
|
c = str.get();
|
|
assertTrue (c == 'c');
|
|
assertTrue (str.tellg() == std::streampos(13));
|
|
|
|
str.seekg(1, std::ios::cur);
|
|
assertTrue (str.tellg() == std::streampos(14));
|
|
c = str.get();
|
|
assertTrue (c == 'e');
|
|
assertTrue (str.tellg() == std::streampos(15));
|
|
}
|
|
|
|
|
|
void FileStreamTest::testMultiOpen()
|
|
{
|
|
Poco::FileStream str("test.txt", std::ios::out | std::ios::trunc);
|
|
str << "0123456789\n";
|
|
str << "abcdefghij\n";
|
|
str << "klmnopqrst\n";
|
|
str.close();
|
|
|
|
std::string s;
|
|
str.open("test.txt", std::ios::in);
|
|
std::getline(str, s);
|
|
assertTrue (s == "0123456789");
|
|
str.close();
|
|
|
|
str.open("test.txt", std::ios::in);
|
|
std::getline(str, s);
|
|
assertTrue (s == "0123456789");
|
|
str.close();
|
|
}
|
|
|
|
void FileStreamTest::testBufferSize()
|
|
{
|
|
// Tests correctness of stream operation when buffer is resized.
|
|
Poco::FileStream str("test.txt", std::ios::out | std::ios::trunc);
|
|
str << "0123456789\n";
|
|
str << "abcdefghij\n";
|
|
str << "klmnopqrst\n";
|
|
|
|
// Can't resize buffer when file is open.
|
|
assertFalse(str.rdbuf()->resizeBuffer(8000));
|
|
str.close();
|
|
|
|
std::string s;
|
|
assertTrue(str.rdbuf()->resizeBuffer(8000));
|
|
str.open("test.txt", std::ios::in);
|
|
std::getline(str, s);
|
|
assertTrue (s == "0123456789");
|
|
std::getline(str, s);
|
|
assertTrue (s == "abcdefghij");
|
|
str.close();
|
|
|
|
assertTrue(str.rdbuf()->resizeBuffer(8000));
|
|
str.open("test.txt", std::ios::out | std::ios::trunc);
|
|
str << "0123456789\n";
|
|
str << "abcdefghij\n";
|
|
str << "klmnopqrst\n";
|
|
str.close();
|
|
|
|
assertTrue(str.rdbuf()->resizeBuffer(4000));
|
|
str.open("test.txt", std::ios::in);
|
|
std::getline(str, s);
|
|
assertTrue (s == "0123456789");
|
|
std::getline(str, s);
|
|
assertTrue (s == "abcdefghij");
|
|
str.close();
|
|
}
|
|
|
|
void FileStreamTest::testBufferSizePerformance()
|
|
{
|
|
Poco::Stopwatch watch;
|
|
|
|
// Write with default buffer size
|
|
Poco::FileStream str("test.txt", std::ios::out | std::ios::trunc);
|
|
const std::string outStrBuf(150LL*1024LL, '#');
|
|
watch.start();
|
|
for (int r{0}; r < 10; r++)
|
|
{
|
|
for (int i{0}; i < 4000; i++)
|
|
{
|
|
str.write(outStrBuf.data(), outStrBuf.size());
|
|
}
|
|
str.seekp(0);
|
|
}
|
|
const auto writeDefUs { watch.elapsed() };
|
|
str.close();
|
|
|
|
// Read with default buffer size
|
|
std::string strBuf(150LL*1024LL, '0');
|
|
str.open("test.txt", std::ios::in);
|
|
|
|
watch.restart();
|
|
for (int r{0}; r < 10; r++)
|
|
{
|
|
for (int i{0}; i < 4000; i++)
|
|
{
|
|
str.read(strBuf.data(), strBuf.size());
|
|
}
|
|
str.seekg(0);
|
|
}
|
|
const auto readDefUs { watch.elapsed() };
|
|
str.close();
|
|
|
|
// Write with increased buffer
|
|
str.rdbuf()->resizeBuffer(128LL*1024LL);
|
|
|
|
str.open("test.txt", std::ios::out | std::ios::trunc);
|
|
watch.restart();
|
|
for (int r{0}; r < 10; r++)
|
|
{
|
|
for (int i{0}; i < 4000; i++)
|
|
{
|
|
str.write(outStrBuf.data(), outStrBuf.size());
|
|
}
|
|
str.seekp(0);
|
|
}
|
|
const auto writeLargeUs { watch.elapsed() };
|
|
str.close();
|
|
|
|
// Read with increased buffer size
|
|
str.open("test.txt", std::ios::in);
|
|
watch.restart();
|
|
for (int r{0}; r < 10; r++)
|
|
{
|
|
for (int i{0}; i < 4000; i++)
|
|
{
|
|
str.read(strBuf.data(), strBuf.size());
|
|
}
|
|
str.seekg(0);
|
|
}
|
|
const auto readLargeUs { watch.elapsed() };
|
|
str.close();
|
|
|
|
watch.stop();
|
|
|
|
Poco::File("test.txt").remove();
|
|
|
|
std::cout << std::endl;
|
|
std::cout << "Write (us) default/128k: " << writeDefUs << "/" << writeLargeUs << std::endl;
|
|
std::cout << "Read (us) default/128k: " << readDefUs << "/" << readLargeUs << std::endl;
|
|
|
|
// Measurements in test environment are not reliable enough.
|
|
//assertTrue(writeDefUs > writeLargeUs);
|
|
//assertTrue(readDefUs > readLargeUs);
|
|
}
|
|
|
|
void FileStreamTest::setUp()
|
|
{
|
|
}
|
|
|
|
|
|
void FileStreamTest::tearDown()
|
|
{
|
|
}
|
|
|
|
|
|
CppUnit::Test* FileStreamTest::suite()
|
|
{
|
|
CppUnit::TestSuite* pSuite = new CppUnit::TestSuite("FileStreamTest");
|
|
|
|
CppUnit_addTest(pSuite, FileStreamTest, testRead);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testWriteReadNativeHandle);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testWrite);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testReadWrite);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testOpen);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testOpenModeIn);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testOpenModeOut);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testOpenModeTrunc);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testOpenModeAte);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testOpenModeApp);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testSeek);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testMultiOpen);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testBufferSize);
|
|
CppUnit_addTest(pSuite, FileStreamTest, testBufferSizePerformance);
|
|
|
|
return pSuite;
|
|
}
|