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.
229 lines
3.8 KiB
C++
229 lines
3.8 KiB
C++
//
|
|
// FileStream_POSIX.cpp
|
|
//
|
|
// Library: Foundation
|
|
// Package: Streams
|
|
// Module: FileStream
|
|
//
|
|
// Copyright (c) 2007, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#include "Poco/FileStream.h"
|
|
#include "Poco/File.h"
|
|
#include "Poco/Exception.h"
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
|
|
|
|
namespace Poco {
|
|
|
|
|
|
FileStreamBuf::FileStreamBuf():
|
|
BufferedBidirectionalStreamBuf(BUFFER_SIZE, std::ios::in | std::ios::out),
|
|
_fd(-1),
|
|
_pos(0)
|
|
{
|
|
}
|
|
|
|
|
|
FileStreamBuf::~FileStreamBuf()
|
|
{
|
|
close();
|
|
}
|
|
|
|
|
|
void FileStreamBuf::open(const std::string& path, std::ios::openmode mode)
|
|
{
|
|
poco_assert (_fd == -1);
|
|
|
|
_pos = 0;
|
|
_path = path;
|
|
setMode(mode);
|
|
resetBuffers();
|
|
|
|
int flags(0);
|
|
if (mode & std::ios::trunc)
|
|
flags |= O_TRUNC;
|
|
if (mode & std::ios::app)
|
|
flags |= O_APPEND;
|
|
if (mode & std::ios::out)
|
|
flags |= O_CREAT;
|
|
if ((mode & std::ios::in) && (mode & std::ios::out))
|
|
flags |= O_RDWR;
|
|
else if (mode & std::ios::in)
|
|
flags |= O_RDONLY;
|
|
else
|
|
flags |= O_WRONLY;
|
|
|
|
_fd = ::open(path.c_str(), flags, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH);
|
|
if (_fd == -1)
|
|
File::handleLastError(_path);
|
|
|
|
if ((mode & std::ios::app) || (mode & std::ios::ate))
|
|
seekoff(0, std::ios::end, mode);
|
|
}
|
|
|
|
|
|
void FileStreamBuf::openHandle(NativeHandle fd, std::ios::openmode mode)
|
|
{
|
|
poco_assert(_fd == -1);
|
|
poco_assert(fd != -1);
|
|
|
|
_pos = 0;
|
|
setMode(mode);
|
|
resetBuffers();
|
|
|
|
_fd = fd;
|
|
|
|
if ((mode & std::ios::app) || (mode & std::ios::ate))
|
|
seekoff(0, std::ios::end, mode);
|
|
}
|
|
|
|
|
|
int FileStreamBuf::readFromDevice(char* buffer, std::streamsize length)
|
|
{
|
|
if (_fd == -1) return -1;
|
|
|
|
if (getMode() & std::ios::out)
|
|
sync();
|
|
|
|
int n = ::read(_fd, buffer, length);
|
|
if (n == -1)
|
|
File::handleLastError(_path);
|
|
_pos += n;
|
|
return n;
|
|
}
|
|
|
|
|
|
int FileStreamBuf::writeToDevice(const char* buffer, std::streamsize length)
|
|
{
|
|
if (_fd == -1) return -1;
|
|
|
|
#if defined(POCO_VXWORKS)
|
|
int n = ::write(_fd, const_cast<char*>(buffer), length);
|
|
#else
|
|
int n = ::write(_fd, buffer, length);
|
|
#endif
|
|
if (n == -1)
|
|
File::handleLastError(_path);
|
|
_pos += n;
|
|
return n;
|
|
}
|
|
|
|
|
|
bool FileStreamBuf::close()
|
|
{
|
|
bool success = true;
|
|
if (_fd != -1)
|
|
{
|
|
try
|
|
{
|
|
sync();
|
|
}
|
|
catch (...)
|
|
{
|
|
success = false;
|
|
}
|
|
::close(_fd);
|
|
_fd = -1;
|
|
}
|
|
return success;
|
|
}
|
|
|
|
|
|
bool FileStreamBuf::resizeBuffer(std::streamsize bufferSize)
|
|
{
|
|
if (_fd != -1)
|
|
return false;
|
|
|
|
if (bufferSize < BUFFER_SIZE)
|
|
bufferSize = BUFFER_SIZE;
|
|
|
|
return BufferedBidirectionalStreamBuf::resizeBuffer(bufferSize);
|
|
}
|
|
|
|
|
|
std::streampos FileStreamBuf::seekoff(std::streamoff off, std::ios::seekdir dir, std::ios::openmode mode)
|
|
{
|
|
if (_fd == -1 || !(getMode() & mode))
|
|
return -1;
|
|
|
|
if (getMode() & std::ios::out)
|
|
sync();
|
|
|
|
std::streamoff adj;
|
|
if (mode & std::ios::in)
|
|
adj = static_cast<std::streamoff>(egptr() - gptr());
|
|
else
|
|
adj = 0;
|
|
|
|
resetBuffers();
|
|
|
|
int whence = SEEK_SET;
|
|
if (dir == std::ios::cur)
|
|
{
|
|
whence = SEEK_CUR;
|
|
off -= adj;
|
|
}
|
|
else if (dir == std::ios::end)
|
|
{
|
|
whence = SEEK_END;
|
|
}
|
|
_pos = ::lseek(_fd, off, whence);
|
|
return _pos;
|
|
}
|
|
|
|
|
|
std::streampos FileStreamBuf::seekpos(std::streampos pos, std::ios::openmode mode)
|
|
{
|
|
if (_fd == -1 || !(getMode() & mode))
|
|
return -1;
|
|
|
|
if (getMode() & std::ios::out)
|
|
sync();
|
|
|
|
resetBuffers();
|
|
|
|
_pos = ::lseek(_fd, pos, SEEK_SET);
|
|
return _pos;
|
|
}
|
|
|
|
|
|
void FileStreamBuf::flushToDisk()
|
|
{
|
|
if (getMode() & std::ios::out)
|
|
{
|
|
sync();
|
|
if (::fsync(_fd) != 0)
|
|
File::handleLastError(_path);
|
|
}
|
|
}
|
|
|
|
|
|
FileStreamBuf::NativeHandle FileStreamBuf::nativeHandle() const
|
|
{
|
|
return _fd;
|
|
}
|
|
|
|
Poco::UInt64 FileStreamBuf::size() const
|
|
{
|
|
struct stat stat_buf;
|
|
int rc = ::fstat(_fd, &stat_buf);
|
|
if (rc < 0)
|
|
{
|
|
Poco::SystemException(strerror(errno), errno);
|
|
}
|
|
return stat_buf.st_size;
|
|
}
|
|
|
|
|
|
} // namespace Poco
|