123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546 |
- //
- // Copyright (c) 2012 Artyom Beilis (Tonkikh)
- // Copyright (c) 2019-2020 Alexander Grund
- //
- // Distributed under the Boost Software License, Version 1.0.
- // https://www.boost.org/LICENSE_1_0.txt
- #ifndef BOOST_NOWIDE_FILEBUF_HPP_INCLUDED
- #define BOOST_NOWIDE_FILEBUF_HPP_INCLUDED
- #include <boost/nowide/config.hpp>
- #if BOOST_NOWIDE_USE_FILEBUF_REPLACEMENT
- #include <boost/nowide/cstdio.hpp>
- #include <boost/nowide/detail/is_path.hpp>
- #include <boost/nowide/stackstring.hpp>
- #include <cassert>
- #include <cstdio>
- #include <ios>
- #include <limits>
- #include <locale>
- #include <stdexcept>
- #include <streambuf>
- #else
- #include <fstream>
- #endif
- namespace boost {
- namespace nowide {
- namespace detail {
- /// Same as std::ftell but potentially with Large File Support
- BOOST_NOWIDE_DECL std::streampos ftell(FILE* file);
- /// Same as std::fseek but potentially with Large File Support
- BOOST_NOWIDE_DECL int fseek(FILE* file, std::streamoff offset, int origin);
- } // namespace detail
- #if !BOOST_NOWIDE_USE_FILEBUF_REPLACEMENT && !defined(BOOST_NOWIDE_DOXYGEN)
- using std::basic_filebuf;
- using std::filebuf;
- #else // Windows
- ///
- /// \brief This forward declaration defines the basic_filebuf type
- /// which is used when #BOOST_NOWIDE_USE_FILEBUF_REPLACEMENT is set, e.g. on Windows.
- ///
- /// It is implemented and specialized for CharType = char, it
- /// implements std::filebuf over standard C I/O
- ///
- template<typename CharType, typename Traits = std::char_traits<CharType>>
- class basic_filebuf;
- ///
- /// \brief This is the implementation of std::filebuf
- /// which is used when #BOOST_NOWIDE_USE_FILEBUF_REPLACEMENT is set, e.g. on Windows.
- ///
- /// It is implemented and specialized for CharType = char, it
- /// implements std::filebuf over standard C I/O
- ///
- template<>
- class basic_filebuf<char> : public std::basic_streambuf<char>
- {
- using Traits = std::char_traits<char>;
- public:
- #ifdef BOOST_MSVC
- #pragma warning(push)
- #pragma warning(disable : 4351) // new behavior : elements of array will be default initialized
- #endif
- ///
- /// Creates new filebuf
- ///
- basic_filebuf() :
- file_(nullptr), buffer_(nullptr), buffer_size_(BUFSIZ), owns_buffer_(false), unbuffered_read_(false),
- last_char_(), mode_(std::ios_base::openmode(0))
- {
- setg(nullptr, nullptr, nullptr);
- setp(nullptr, nullptr);
- }
- #ifdef BOOST_MSVC
- #pragma warning(pop)
- #endif
- basic_filebuf(const basic_filebuf&) = delete;
- basic_filebuf& operator=(const basic_filebuf&) = delete;
- basic_filebuf(basic_filebuf&& other) noexcept : basic_filebuf()
- {
- swap(other);
- }
- basic_filebuf& operator=(basic_filebuf&& other) noexcept
- {
- close();
- swap(other);
- return *this;
- }
- void swap(basic_filebuf& rhs)
- {
- std::basic_streambuf<char>::swap(rhs);
- using std::swap;
- swap(file_, rhs.file_);
- swap(buffer_, rhs.buffer_);
- swap(buffer_size_, rhs.buffer_size_);
- swap(owns_buffer_, rhs.owns_buffer_);
- swap(unbuffered_read_, rhs.unbuffered_read_);
- swap(last_char_[0], rhs.last_char_[0]);
- swap(mode_, rhs.mode_);
- // Fixup last_char references
- if(pbase() == rhs.last_char_)
- setp(last_char_, (pptr() == epptr()) ? last_char_ : last_char_ + 1);
- if(eback() == rhs.last_char_)
- setg(last_char_, (gptr() == rhs.last_char_) ? last_char_ : last_char_ + 1, last_char_ + 1);
- if(rhs.pbase() == last_char_)
- rhs.setp(rhs.last_char_, (rhs.pptr() == rhs.epptr()) ? rhs.last_char_ : rhs.last_char_ + 1);
- if(rhs.eback() == last_char_)
- {
- rhs.setg(rhs.last_char_,
- (rhs.gptr() == last_char_) ? rhs.last_char_ : rhs.last_char_ + 1,
- rhs.last_char_ + 1);
- }
- }
- virtual ~basic_filebuf()
- {
- close();
- }
- ///
- /// Same as std::filebuf::open but s is UTF-8 string
- ///
- basic_filebuf* open(const std::string& s, std::ios_base::openmode mode)
- {
- return open(s.c_str(), mode);
- }
- ///
- /// Same as std::filebuf::open but s is UTF-8 string
- ///
- basic_filebuf* open(const char* s, std::ios_base::openmode mode)
- {
- const wstackstring name(s);
- return open(name.get(), mode);
- }
- /// Opens the file with the given name, see std::filebuf::open
- basic_filebuf* open(const wchar_t* s, std::ios_base::openmode mode)
- {
- if(is_open())
- return nullptr;
- validate_cvt(this->getloc());
- const bool ate = (mode & std::ios_base::ate) != 0;
- if(ate)
- mode &= ~std::ios_base::ate;
- const wchar_t* smode = get_mode(mode);
- if(!smode)
- return nullptr;
- file_ = detail::wfopen(s, smode);
- if(!file_)
- return nullptr;
- if(ate && detail::fseek(file_, 0, SEEK_END) != 0)
- {
- close();
- return nullptr;
- }
- mode_ = mode;
- set_unbuffered_read();
- return this;
- }
- template<typename Path>
- detail::enable_if_path_t<Path, basic_filebuf*> open(const Path& file_name, std::ios_base::openmode mode)
- {
- return open(file_name.c_str(), mode);
- }
- ///
- /// Same as std::filebuf::close()
- ///
- basic_filebuf* close()
- {
- if(!is_open())
- return nullptr;
- bool res = sync() == 0;
- if(std::fclose(file_) != 0)
- res = false;
- file_ = nullptr;
- mode_ = std::ios_base::openmode(0);
- if(owns_buffer_)
- {
- delete[] buffer_;
- buffer_ = nullptr;
- owns_buffer_ = false;
- }
- setg(nullptr, nullptr, nullptr);
- setp(nullptr, nullptr);
- return res ? this : nullptr;
- }
- ///
- /// Same as std::filebuf::is_open()
- ///
- bool is_open() const
- {
- return file_ != nullptr;
- }
- protected:
- std::streambuf* setbuf(char* s, std::streamsize n) override
- {
- assert(n >= 0);
- // Maximum compatibility: Discard all local buffers and use user-provided values
- // Users should call sync() before or better use it before any IO is done or any file is opened
- setg(nullptr, nullptr, nullptr);
- setp(nullptr, nullptr);
- if(owns_buffer_)
- {
- delete[] buffer_;
- owns_buffer_ = false;
- }
- buffer_ = s;
- buffer_size_ = (n >= 0) ? static_cast<size_t>(n) : 0;
- set_unbuffered_read();
- return this;
- }
- int sync() override
- {
- if(!file_)
- return 0;
- bool result;
- if(pptr())
- {
- // Only flush if anything was written, otherwise behavior of fflush is undefined. I.e.:
- // - Buffered mode: pptr was set to buffer_ and advanced
- // - Unbuffered mode: pptr set to last_char_
- const bool has_prev_write = pptr() != buffer_;
- result = overflow() != EOF;
- if(has_prev_write && std::fflush(file_) != 0)
- result = false;
- } else
- result = stop_reading();
- return result ? 0 : -1;
- }
- int overflow(int c = EOF) override
- {
- if(!(mode_ & (std::ios_base::out | std::ios_base::app)))
- return EOF;
- if(!stop_reading())
- return EOF;
- size_t n = pptr() - pbase();
- if(n > 0)
- {
- if(std::fwrite(pbase(), 1, n, file_) != n)
- return EOF;
- assert(buffer_);
- setp(buffer_, buffer_ + buffer_size_);
- if(c != EOF)
- {
- *buffer_ = Traits::to_char_type(c);
- pbump(1);
- }
- } else if(c != EOF)
- {
- if(buffer_size_ > 0)
- {
- make_buffer();
- setp(buffer_, buffer_ + buffer_size_);
- *buffer_ = Traits::to_char_type(c);
- pbump(1);
- } else if(std::fputc(c, file_) == EOF)
- {
- return EOF;
- } else if(!pptr())
- {
- // Set to dummy value so we know we have written something
- setp(last_char_, last_char_);
- }
- }
- return Traits::not_eof(c);
- }
- std::streamsize xsputn(const char* s, std::streamsize n) override
- {
- // Only optimize when writing more than a buffer worth of data
- if(n <= static_cast<std::streamsize>(buffer_size_))
- return std::basic_streambuf<char>::xsputn(s, n);
- if(!(mode_ & (std::ios_base::out | std::ios_base::app)) || !stop_reading())
- return 0;
- assert(n >= 0);
- // First empty the remaining put area, if any
- const char* const base = pbase();
- const size_t num_buffered = pptr() - base;
- if(num_buffered != 0)
- {
- const auto num_written = std::fwrite(base, 1, num_buffered, file_);
- setp(const_cast<char*>(base + num_written), epptr()); // i.e. pbump(num_written)
- if(num_written != num_buffered)
- return 0; // Error writing buffered chars
- }
- // Then write directly to file
- const auto num_written = std::fwrite(s, 1, static_cast<size_t>(n), file_);
- if(num_written > 0u && base != last_char_)
- setp(last_char_, last_char_); // Mark as "written" if not done yet
- return num_written;
- }
- int underflow() override
- {
- if(!(mode_ & std::ios_base::in) || !stop_writing())
- return EOF;
- if(unbuffered_read_)
- {
- const int c = std::fgetc(file_);
- if(c == EOF)
- return EOF;
- last_char_[0] = Traits::to_char_type(c);
- setg(last_char_, last_char_, last_char_ + 1);
- } else
- {
- make_buffer();
- const size_t n = std::fread(buffer_, 1, buffer_size_, file_);
- setg(buffer_, buffer_, buffer_ + n);
- if(n == 0)
- return EOF;
- }
- return Traits::to_int_type(*gptr());
- }
- std::streamsize xsgetn(char* s, std::streamsize n) override
- {
- // Only optimize when reading more than a buffer worth of data
- if(n <= static_cast<std::streamsize>(unbuffered_read_ ? 1u : buffer_size_))
- return std::basic_streambuf<char>::xsgetn(s, n);
- if(!(mode_ & std::ios_base::in) || !stop_writing())
- return 0;
- assert(n >= 0);
- std::streamsize num_copied = 0;
- // First empty the remaining get area, if any
- const auto num_buffered = egptr() - gptr();
- if(num_buffered != 0)
- {
- const auto num_read = num_buffered > n ? n : num_buffered;
- traits_type::copy(s, gptr(), static_cast<size_t>(num_read));
- s += num_read;
- n -= num_read;
- num_copied = num_read;
- setg(eback(), gptr() + num_read, egptr()); // i.e. gbump(num_read)
- }
- // Then read directly from file (loop as number of bytes read may be less than requested)
- while(n > 0)
- {
- const auto num_read = std::fread(s, 1, static_cast<size_t>(n), file_);
- if(num_read == 0) // EOF or error
- break;
- s += num_read;
- n -= num_read;
- num_copied += num_read;
- }
- return num_copied;
- }
- int pbackfail(int c = EOF) override
- {
- // For simplicity we only allow putting back into our read buffer
- // So putting back more chars than we have read from the buffer will fail
- if(gptr() > eback())
- gbump(-1);
- else
- return EOF;
- // Assign the new value if requested
- if(c != EOF && *gptr() != Traits::to_char_type(c))
- *gptr() = Traits::to_char_type(c);
- return Traits::not_eof(c);
- }
- std::streampos seekoff(std::streamoff off,
- std::ios_base::seekdir seekdir,
- std::ios_base::openmode = std::ios_base::in | std::ios_base::out) override
- {
- if(!file_)
- return EOF;
- // Switching between input<->output requires a seek
- // So do NOT optimize for seekoff(0, cur) as No-OP
- // On some implementations a seek also flushes, so do a full sync
- if(sync() != 0)
- return EOF;
- int whence;
- switch(seekdir)
- {
- case std::ios_base::beg: whence = SEEK_SET; break;
- case std::ios_base::cur: whence = SEEK_CUR; break;
- case std::ios_base::end: whence = SEEK_END; break;
- default: assert(false); return EOF;
- }
- if(detail::fseek(file_, off, whence) != 0)
- return EOF;
- return detail::ftell(file_);
- }
- std::streampos seekpos(std::streampos pos,
- std::ios_base::openmode m = std::ios_base::in | std::ios_base::out) override
- {
- // Standard mandates "as-if fsetpos", but assume the effect is the same as fseek
- return seekoff(pos, std::ios_base::beg, m);
- }
- void imbue(const std::locale& loc) override
- {
- validate_cvt(loc);
- }
- private:
- void make_buffer()
- {
- if(buffer_)
- return;
- if(buffer_size_ > 0)
- {
- buffer_ = new char[buffer_size_];
- owns_buffer_ = true;
- }
- }
- void set_unbuffered_read()
- {
- // In text mode we cannot use buffering as we are required to know the (file) position of each
- // char in the get area and to seek back in case of a sync to "put back" unread chars.
- // However std::fseek with non-zero offsets is unsupported for text files and the (file) offset
- // to seek back is unknown anyway due to newlines which may got converted.
- unbuffered_read_ = !(mode_ & std::ios_base::binary) || buffer_size_ == 0u;
- }
- void validate_cvt(const std::locale& loc)
- {
- if(!std::use_facet<std::codecvt<char, char, std::mbstate_t>>(loc).always_noconv())
- throw std::runtime_error("Converting codecvts are not supported");
- }
- /// Stop reading adjusting the file pointer if necessary
- /// Postcondition: gptr() == nullptr
- bool stop_reading()
- {
- if(!gptr())
- return true;
- const auto off = gptr() - egptr();
- setg(nullptr, nullptr, nullptr);
- if(!off)
- return true;
- #if defined(__clang__)
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wtautological-constant-out-of-range-compare"
- #endif
- // coverity[result_independent_of_operands]
- if(off < std::numeric_limits<std::streamoff>::min())
- return false;
- #if defined(__clang__)
- #pragma clang diagnostic pop
- #endif
- return detail::fseek(file_, static_cast<std::streamoff>(off), SEEK_CUR) == 0;
- }
- /// Stop writing. If any bytes are to be written, writes them to file
- /// Postcondition: pptr() == nullptr
- bool stop_writing()
- {
- if(pptr())
- {
- const char* const base = pbase();
- const size_t n = pptr() - base;
- setp(nullptr, nullptr);
- if(n && std::fwrite(base, 1, n, file_) != n)
- return false;
- }
- return true;
- }
- static const wchar_t* get_mode(std::ios_base::openmode mode)
- {
- //
- // done according to n2914 table 106 27.9.1.4
- //
- // note can't use switch case as overload operator can't be used
- // in constant expression
- if(mode == (std::ios_base::out))
- return L"w";
- if(mode == (std::ios_base::out | std::ios_base::app))
- return L"a";
- if(mode == (std::ios_base::app))
- return L"a";
- if(mode == (std::ios_base::out | std::ios_base::trunc))
- return L"w";
- if(mode == (std::ios_base::in))
- return L"r";
- if(mode == (std::ios_base::in | std::ios_base::out))
- return L"r+";
- if(mode == (std::ios_base::in | std::ios_base::out | std::ios_base::trunc))
- return L"w+";
- if(mode == (std::ios_base::in | std::ios_base::out | std::ios_base::app))
- return L"a+";
- if(mode == (std::ios_base::in | std::ios_base::app))
- return L"a+";
- if(mode == (std::ios_base::binary | std::ios_base::out))
- return L"wb";
- if(mode == (std::ios_base::binary | std::ios_base::out | std::ios_base::app))
- return L"ab";
- if(mode == (std::ios_base::binary | std::ios_base::app))
- return L"ab";
- if(mode == (std::ios_base::binary | std::ios_base::out | std::ios_base::trunc))
- return L"wb";
- if(mode == (std::ios_base::binary | std::ios_base::in))
- return L"rb";
- if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::out))
- return L"r+b";
- if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::out | std::ios_base::trunc))
- return L"w+b";
- if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::out | std::ios_base::app))
- return L"a+b";
- if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::app))
- return L"a+b";
- return nullptr;
- }
- FILE* file_;
- char* buffer_;
- size_t buffer_size_;
- bool owns_buffer_;
- bool unbuffered_read_; // True to read char by char
- char last_char_[1];
- std::ios::openmode mode_;
- };
- ///
- /// \brief Convenience typedef
- ///
- using filebuf = basic_filebuf<char>;
- /// Swap the basic_filebuf instances
- template<typename CharType, typename Traits>
- void swap(basic_filebuf<CharType, Traits>& lhs, basic_filebuf<CharType, Traits>& rhs)
- {
- lhs.swap(rhs);
- }
- #endif // windows
- } // namespace nowide
- } // namespace boost
- #endif
|