/* */ #include "AbstractDiskWriter.h" #include "File.h" #include "Util.h" #include "message.h" #include "LogFactory.h" #include "a2io.h" #include "DlAbortEx.h" #include #include #include #include #include AbstractDiskWriter::AbstractDiskWriter(): fd(-1), logger(LogFactory::getInstance()) {} AbstractDiskWriter::~AbstractDiskWriter() { closeFile(); } void AbstractDiskWriter::openFile(const string& filename, int64_t totalLength) { File f(filename); if(f.exists()) { openExistingFile(filename, totalLength); } else { initAndOpenFile(filename, totalLength); } } void AbstractDiskWriter::closeFile() { if(fd >= 0) { close(fd); fd = -1; } } void AbstractDiskWriter::openExistingFile(const string& filename, int64_t totalLength) { this->filename = filename; File f(filename); if(!f.isFile()) { throw new DlAbortEx(EX_FILE_OPEN, filename.c_str(), MSG_FILE_NOT_FOUND); } if((fd = open(filename.c_str(), O_RDWR|O_BINARY, OPEN_MODE)) < 0) { throw new DlAbortEx(EX_FILE_OPEN, filename.c_str(), strerror(errno)); } } void AbstractDiskWriter::createFile(const string& filename, int32_t addFlags) { this->filename = filename; assert(filename.size()); Util::mkdirs(File(filename).getDirname()); if((fd = open(filename.c_str(), O_CREAT|O_RDWR|O_TRUNC|O_BINARY|addFlags, OPEN_MODE)) < 0) { throw new DlAbortEx(EX_FILE_OPEN, filename.c_str(), strerror(errno)); } } int32_t AbstractDiskWriter::writeDataInternal(const unsigned char* data, int32_t len) { int32_t writtenLength = 0; while(writtenLength < len) { int32_t ret = 0; while((ret = write(fd, data+writtenLength, len-writtenLength)) == -1 && errno == EINTR); if(ret == -1) { return -1; } writtenLength += ret; } return writtenLength; } int32_t AbstractDiskWriter::readDataInternal(unsigned char* data, int32_t len) { int32_t ret = 0; while((ret = read(fd, data, len)) == -1 && errno == EINTR); return ret; } void AbstractDiskWriter::seek(int64_t offset) { if(offset != lseek(fd, offset, SEEK_SET)) { throw new DlAbortEx(EX_FILE_SEEK, filename.c_str(), strerror(errno)); } } void AbstractDiskWriter::writeData(const unsigned char* data, int32_t len, int64_t offset) { seek(offset); if(writeDataInternal(data, len) < 0) { throw new DlAbortEx(EX_FILE_WRITE, filename.c_str(), strerror(errno)); } } int32_t AbstractDiskWriter::readData(unsigned char* data, int32_t len, int64_t offset) { int32_t ret; seek(offset); if((ret = readDataInternal(data, len)) < 0) { throw new DlAbortEx(EX_FILE_READ, filename.c_str(), strerror(errno)); } return ret; } void AbstractDiskWriter::truncate(int64_t length) { if(fd == -1) { throw new DlAbortEx("File not opened."); } ftruncate(fd, length); } // TODO the file descriptor fd must be opened before calling this function. int64_t AbstractDiskWriter::size() const { if(fd == -1) { throw new DlAbortEx("File not opened."); } struct stat fileStat; if(fstat(fd, &fileStat) < 0) { return 0; } return fileStat.st_size; } void AbstractDiskWriter::enableDirectIO() { #ifdef ENABLE_DIRECT_IO int32_t flg; while((flg = fcntl(fd, F_GETFL)) == -1 && errno == EINTR); while(fcntl(fd, F_SETFL, flg|O_DIRECT) == -1 && errno == EINTR); #endif // ENABLE_DIRECT_IO } void AbstractDiskWriter::disableDirectIO() { #ifdef ENABLE_DIRECT_IO int32_t flg; while((flg = fcntl(fd, F_GETFL)) == -1 && errno == EINTR); while(fcntl(fd, F_SETFL, flg&(~O_DIRECT)) == -1 && errno == EINTR); #endif // ENABLE_DIRECT_IO }