// // 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 #include #include #include #include #include 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(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(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