#include <limits.h> #include <stdio.h> #ifdef _MSC_VER #include <io.h> #else // _MSC_VER #include <unistd.h> #endif // _MSC_VER #include "reader.h" namespace marisa_alpha { Reader::Reader() : file_(NULL), fd_(-1), stream_(NULL), needs_fclose_(false) {} Reader::Reader(std::FILE *file) : file_(file), fd_(-1), stream_(NULL), needs_fclose_(false) {} Reader::Reader(int fd) : file_(NULL), fd_(fd), stream_(NULL), needs_fclose_(false) {} Reader::Reader(std::istream *stream) : file_(NULL), fd_(-1), stream_(stream), needs_fclose_(false) {} Reader::~Reader() { if (needs_fclose_) { ::fclose(file_); } } void Reader::open(const char *filename, long offset, int whence) { MARISA_ALPHA_THROW_IF(is_open(), MARISA_ALPHA_STATE_ERROR); MARISA_ALPHA_THROW_IF(filename == NULL, MARISA_ALPHA_PARAM_ERROR); #ifdef _MSC_VER std::FILE *file = NULL; if (::fopen_s(&file, filename, "rb") != 0) { MARISA_ALPHA_THROW(MARISA_ALPHA_IO_ERROR); } #else // _MSC_VER std::FILE * const file = ::fopen(filename, "rb"); MARISA_ALPHA_THROW_IF(file == NULL, MARISA_ALPHA_IO_ERROR); #endif // _MSC_VER if (::fseek(file, offset, whence) != 0) { ::fclose(file); MARISA_ALPHA_THROW(MARISA_ALPHA_IO_ERROR); } file_ = file; needs_fclose_ = true; } void Reader::clear() { Reader().swap(this); } void Reader::swap(Reader *rhs) { MARISA_ALPHA_THROW_IF(rhs == NULL, MARISA_ALPHA_PARAM_ERROR); Swap(&file_, &rhs->file_); Swap(&fd_, &rhs->fd_); Swap(&stream_, &rhs->stream_); Swap(&needs_fclose_, &rhs->needs_fclose_); } void Reader::read_data(void *buf, std::size_t size) { if (fd_ != -1) { while (size != 0) { #ifdef _MSC_VER const unsigned int count = (size < INT_MAX) ? size : INT_MAX; const int size_read = _read(fd_, buf, count); #else // _MSC_VER const ::size_t count = (size < SSIZE_MAX) ? size : SSIZE_MAX; const ::ssize_t size_read = ::read(fd_, buf, count); #endif // _MSC_VER MARISA_ALPHA_THROW_IF(size_read <= 0, MARISA_ALPHA_IO_ERROR); buf = static_cast<char *>(buf) + size_read; size -= size_read; } } else if (file_ != NULL) { if (::fread(buf, 1, size, file_) != size) { MARISA_ALPHA_THROW(MARISA_ALPHA_IO_ERROR); } } else if (stream_ != NULL) { try { if (!stream_->read(static_cast<char *>(buf), size)) { MARISA_ALPHA_THROW(MARISA_ALPHA_IO_ERROR); } } catch (const std::ios_base::failure &) { MARISA_ALPHA_THROW(MARISA_ALPHA_IO_ERROR); } } else { MARISA_ALPHA_THROW(MARISA_ALPHA_STATE_ERROR); } } } // namespace marisa_alpha