ClickHouse/dbms/src/IO/WriteBufferFromFileDescriptor.cpp

130 lines
3.0 KiB
C++
Raw Normal View History

#include <port/unistd.h>
2016-10-25 06:49:24 +00:00
#include <errno.h>
#include <Common/Exception.h>
#include <Common/ProfileEvents.h>
#include <Common/CurrentMetrics.h>
2016-10-25 06:49:24 +00:00
#include <IO/WriteBufferFromFileDescriptor.h>
#include <IO/WriteHelpers.h>
2016-10-25 06:49:24 +00:00
namespace ProfileEvents
{
extern const Event WriteBufferFromFileDescriptorWrite;
extern const Event WriteBufferFromFileDescriptorWriteFailed;
extern const Event WriteBufferFromFileDescriptorWriteBytes;
2016-10-25 06:49:24 +00:00
}
namespace CurrentMetrics
{
extern const Metric Write;
2016-10-25 06:49:24 +00:00
}
namespace DB
{
namespace ErrorCodes
{
extern const int CANNOT_WRITE_TO_FILE_DESCRIPTOR;
extern const int CANNOT_FSYNC;
extern const int CANNOT_SEEK_THROUGH_FILE;
extern const int CANNOT_TRUNCATE_FILE;
2016-10-25 06:49:24 +00:00
}
void WriteBufferFromFileDescriptor::nextImpl()
{
if (!offset())
return;
size_t bytes_written = 0;
while (bytes_written != offset())
{
ProfileEvents::increment(ProfileEvents::WriteBufferFromFileDescriptorWrite);
ssize_t res = 0;
{
CurrentMetrics::Increment metric_increment{CurrentMetrics::Write};
res = ::write(fd, working_buffer.begin() + bytes_written, offset() - bytes_written);
}
if ((-1 == res || 0 == res) && errno != EINTR)
{
ProfileEvents::increment(ProfileEvents::WriteBufferFromFileDescriptorWriteFailed);
throwFromErrno("Cannot write to file " + getFileName(), ErrorCodes::CANNOT_WRITE_TO_FILE_DESCRIPTOR);
}
if (res > 0)
bytes_written += res;
}
ProfileEvents::increment(ProfileEvents::WriteBufferFromFileDescriptorWriteBytes, bytes_written);
2016-10-25 06:49:24 +00:00
}
/// Name or some description of file.
std::string WriteBufferFromFileDescriptor::getFileName() const
{
return "(fd = " + toString(fd) + ")";
2016-10-25 06:49:24 +00:00
}
WriteBufferFromFileDescriptor::WriteBufferFromFileDescriptor(
int fd_,
size_t buf_size,
char * existing_memory,
size_t alignment)
: WriteBufferFromFileBase(buf_size, existing_memory, alignment), fd(fd_) {}
2016-10-25 06:49:24 +00:00
WriteBufferFromFileDescriptor::~WriteBufferFromFileDescriptor()
{
try
{
if (fd >= 0)
next();
}
catch (...)
{
tryLogCurrentException(__PRETTY_FUNCTION__);
}
2016-10-25 06:49:24 +00:00
}
off_t WriteBufferFromFileDescriptor::getPositionInFile()
{
return seek(0, SEEK_CUR);
2016-10-25 06:49:24 +00:00
}
void WriteBufferFromFileDescriptor::sync()
{
/// If buffer has pending data - write it.
next();
2016-10-25 06:49:24 +00:00
/// Request OS to sync data with storage medium.
int res = fsync(fd);
if (-1 == res)
throwFromErrno("Cannot fsync " + getFileName(), ErrorCodes::CANNOT_FSYNC);
2016-10-25 06:49:24 +00:00
}
off_t WriteBufferFromFileDescriptor::doSeek(off_t offset, int whence)
{
off_t res = lseek(fd, offset, whence);
if (-1 == res)
throwFromErrno("Cannot seek through file " + getFileName(), ErrorCodes::CANNOT_SEEK_THROUGH_FILE);
return res;
2016-10-25 06:49:24 +00:00
}
void WriteBufferFromFileDescriptor::doTruncate(off_t length)
{
int res = ftruncate(fd, length);
if (-1 == res)
throwFromErrno("Cannot truncate file " + getFileName(), ErrorCodes::CANNOT_TRUNCATE_FILE);
2016-10-25 06:49:24 +00:00
}
}