Boost.Nowide
filebuf.hpp
1 //
2 // Copyright (c) 2012 Artyom Beilis (Tonkikh)
3 // Copyright (c) 2019-2020 Alexander Grund
4 //
5 // Distributed under the Boost Software License, Version 1.0. (See
6 // accompanying file LICENSE or copy at
7 // http://www.boost.org/LICENSE_1_0.txt)
8 //
9 #ifndef BOOST_NOWIDE_FILEBUF_HPP_INCLUDED
10 #define BOOST_NOWIDE_FILEBUF_HPP_INCLUDED
11 
12 #include <boost/nowide/config.hpp>
13 #if BOOST_NOWIDE_USE_FILEBUF_REPLACEMENT
14 #include <boost/nowide/cstdio.hpp>
15 #include <boost/nowide/stackstring.hpp>
16 #include <cassert>
17 #include <cstdio>
18 #include <ios>
19 #include <limits>
20 #include <locale>
21 #include <stdexcept>
22 #include <streambuf>
23 #else
24 #include <fstream>
25 #endif
26 
27 namespace boost {
28 namespace nowide {
29  namespace detail {
31  BOOST_NOWIDE_DECL std::streampos ftell(FILE* file);
33  BOOST_NOWIDE_DECL int fseek(FILE* file, std::streamoff offset, int origin);
34  } // namespace detail
35 
36 #if !BOOST_NOWIDE_USE_FILEBUF_REPLACEMENT && !defined(BOOST_NOWIDE_DOXYGEN)
37  using std::basic_filebuf;
38  using std::filebuf;
39 #else // Windows
40  template<typename CharType, typename Traits = std::char_traits<CharType>>
48 
55  template<>
56  class basic_filebuf<char> : public std::basic_streambuf<char>
57  {
58  using Traits = std::char_traits<char>;
59 
60  public:
61 #ifdef BOOST_MSVC
62 #pragma warning(push)
63 #pragma warning(disable : 4351) // new behavior : elements of array will be default initialized
64 #endif
65  basic_filebuf() :
69  buffer_size_(BUFSIZ), buffer_(0), file_(0), owns_buffer_(false), last_char_(),
70  mode_(std::ios_base::openmode(0))
71  {
72  setg(0, 0, 0);
73  setp(0, 0);
74  }
75 #ifdef BOOST_MSVC
76 #pragma warning(pop)
77 #endif
78  basic_filebuf(const basic_filebuf&) = delete;
79  basic_filebuf& operator=(const basic_filebuf&) = delete;
80  basic_filebuf(basic_filebuf&& other) noexcept : basic_filebuf()
81  {
82  swap(other);
83  }
84  basic_filebuf& operator=(basic_filebuf&& other) noexcept
85  {
86  swap(other);
87  return *this;
88  }
89  void swap(basic_filebuf& rhs)
90  {
91  std::basic_streambuf<char>::swap(rhs);
92  using std::swap;
93  swap(buffer_size_, rhs.buffer_size_);
94  swap(buffer_, rhs.buffer_);
95  swap(file_, rhs.file_);
96  swap(owns_buffer_, rhs.owns_buffer_);
97  swap(last_char_[0], rhs.last_char_[0]);
98  swap(mode_, rhs.mode_);
99 
100  // Fixup last_char references
101  if(pbase() == rhs.last_char_)
102  setp(last_char_, (pptr() == epptr()) ? last_char_ : last_char_ + 1);
103  if(eback() == rhs.last_char_)
104  setg(last_char_, (gptr() == rhs.last_char_) ? last_char_ : last_char_ + 1, last_char_ + 1);
105 
106  if(rhs.pbase() == last_char_)
107  rhs.setp(rhs.last_char_, (rhs.pptr() == rhs.epptr()) ? rhs.last_char_ : rhs.last_char_ + 1);
108  if(rhs.eback() == last_char_)
109  {
110  rhs.setg(rhs.last_char_,
111  (rhs.gptr() == last_char_) ? rhs.last_char_ : rhs.last_char_ + 1,
112  rhs.last_char_ + 1);
113  }
114  }
115 
116  virtual ~basic_filebuf()
117  {
118  close();
119  }
120 
124  basic_filebuf* open(const std::string& s, std::ios_base::openmode mode)
125  {
126  return open(s.c_str(), mode);
127  }
131  basic_filebuf* open(const char* s, std::ios_base::openmode mode)
132  {
133  const wstackstring name(s);
134  return open(name.get(), mode);
135  }
137  basic_filebuf* open(const wchar_t* s, std::ios_base::openmode mode)
138  {
139  if(is_open())
140  return NULL;
141  validate_cvt(this->getloc());
142  const bool ate = (mode & std::ios_base::ate) != 0;
143  if(ate)
144  mode &= ~std::ios_base::ate;
145  const wchar_t* smode = get_mode(mode);
146  if(!smode)
147  return 0;
148  file_ = detail::wfopen(s, smode);
149  if(!file_)
150  return 0;
151  if(ate && detail::fseek(file_, 0, SEEK_END) != 0)
152  {
153  close();
154  return 0;
155  }
156  mode_ = mode;
157  return this;
158  }
163  {
164  if(!is_open())
165  return NULL;
166  bool res = sync() == 0;
167  if(std::fclose(file_) != 0)
168  res = false;
169  file_ = NULL;
170  mode_ = std::ios_base::openmode(0);
171  if(owns_buffer_)
172  {
173  delete[] buffer_;
174  buffer_ = NULL;
175  owns_buffer_ = false;
176  }
177  setg(0, 0, 0);
178  setp(0, 0);
179  return res ? this : NULL;
180  }
184  bool is_open() const
185  {
186  return file_ != NULL;
187  }
188 
189  private:
190  void make_buffer()
191  {
192  if(buffer_)
193  return;
194  if(buffer_size_ > 0)
195  {
196  buffer_ = new char[buffer_size_];
197  owns_buffer_ = true;
198  }
199  }
200  void validate_cvt(const std::locale& loc)
201  {
202  if(!std::use_facet<std::codecvt<char, char, std::mbstate_t>>(loc).always_noconv())
203  throw std::runtime_error("Converting codecvts are not supported");
204  }
205 
206  protected:
207  std::streambuf* setbuf(char* s, std::streamsize n) override
208  {
209  assert(n >= 0);
210  // Maximum compatibility: Discard all local buffers and use user-provided values
211  // Users should call sync() before or better use it before any IO is done or any file is opened
212  setg(NULL, NULL, NULL);
213  setp(NULL, NULL);
214  if(owns_buffer_)
215  delete[] buffer_;
216  buffer_ = s;
217  buffer_size_ = (n >= 0) ? static_cast<size_t>(n) : 0;
218  return this;
219  }
220 
221  int overflow(int c = EOF) override
222  {
223  if(!(mode_ & std::ios_base::out))
224  return EOF;
225 
226  if(!stop_reading())
227  return EOF;
228 
229  size_t n = pptr() - pbase();
230  if(n > 0)
231  {
232  if(std::fwrite(pbase(), 1, n, file_) != n)
233  return EOF;
234  setp(buffer_, buffer_ + buffer_size_);
235  if(c != EOF)
236  {
237  *buffer_ = Traits::to_char_type(c);
238  pbump(1);
239  }
240  } else if(c != EOF)
241  {
242  if(buffer_size_ > 0)
243  {
244  make_buffer();
245  setp(buffer_, buffer_ + buffer_size_);
246  *buffer_ = Traits::to_char_type(c);
247  pbump(1);
248  } else if(std::fputc(c, file_) == EOF)
249  {
250  return EOF;
251  } else if(!pptr())
252  {
253  // Set to dummy value so we know we have written something
254  setp(last_char_, last_char_);
255  }
256  }
257  return Traits::not_eof(c);
258  }
259 
260  int sync() override
261  {
262  if(!file_)
263  return 0;
264  bool result;
265  if(pptr())
266  {
267  result = overflow() != EOF;
268  // Only flush if anything was written, otherwise behavior of fflush is undefined
269  if(std::fflush(file_) != 0)
270  return result = false;
271  } else
272  result = stop_reading();
273  return result ? 0 : -1;
274  }
275 
276  int underflow() override
277  {
278  if(!(mode_ & std::ios_base::in))
279  return EOF;
280  if(!stop_writing())
281  return EOF;
282  // In text mode we cannot use a buffer size of more than 1 (i.e. single char only)
283  // This is due to the need to seek back in case of a sync to "put back" unread chars.
284  // However determining the number of chars to seek back is impossible in case there are newlines
285  // as we cannot know if those were converted.
286  if(buffer_size_ == 0 || !(mode_ & std::ios_base::binary))
287  {
288  const int c = std::fgetc(file_);
289  if(c == EOF)
290  return EOF;
291  last_char_[0] = Traits::to_char_type(c);
292  setg(last_char_, last_char_, last_char_ + 1);
293  } else
294  {
295  make_buffer();
296  const size_t n = std::fread(buffer_, 1, buffer_size_, file_);
297  setg(buffer_, buffer_, buffer_ + n);
298  if(n == 0)
299  return EOF;
300  }
301  return Traits::to_int_type(*gptr());
302  }
303 
304  int pbackfail(int c = EOF) override
305  {
306  if(!(mode_ & std::ios_base::in))
307  return EOF;
308  if(!stop_writing())
309  return EOF;
310  if(gptr() > eback())
311  gbump(-1);
312  else if(seekoff(-1, std::ios_base::cur) != std::streampos(std::streamoff(-1)))
313  {
314  if(underflow() == EOF)
315  return EOF;
316  } else
317  return EOF;
318 
319  // Case 1: Caller just wanted space for 1 char
320  if(c == EOF)
321  return Traits::not_eof(c);
322  // Case 2: Caller wants to put back different char
323  // gptr now points to the (potentially newly read) previous char
324  if(*gptr() != c)
325  *gptr() = Traits::to_char_type(c);
326  return Traits::not_eof(c);
327  }
328 
329  std::streampos seekoff(std::streamoff off,
330  std::ios_base::seekdir seekdir,
331  std::ios_base::openmode = std::ios_base::in | std::ios_base::out) override
332  {
333  if(!file_)
334  return EOF;
335  // Switching between input<->output requires a seek
336  // So do NOT optimize for seekoff(0, cur) as No-OP
337 
338  // On some implementations a seek also flushes, so do a full sync
339  if(sync() != 0)
340  return EOF;
341  int whence;
342  switch(seekdir)
343  {
344  case std::ios_base::beg: whence = SEEK_SET; break;
345  case std::ios_base::cur: whence = SEEK_CUR; break;
346  case std::ios_base::end: whence = SEEK_END; break;
347  default: assert(false); return EOF;
348  }
349  if(detail::fseek(file_, off, whence) != 0)
350  return EOF;
351  return detail::ftell(file_);
352  }
353  std::streampos seekpos(std::streampos pos,
354  std::ios_base::openmode m = std::ios_base::in | std::ios_base::out) override
355  {
356  // Standard mandates "as-if fsetpos", but assume the effect is the same as fseek
357  return seekoff(pos, std::ios_base::beg, m);
358  }
359  void imbue(const std::locale& loc) override
360  {
361  validate_cvt(loc);
362  }
363 
364  private:
367  bool stop_reading()
368  {
369  if(!gptr())
370  return true;
371  const auto off = gptr() - egptr();
372  setg(0, 0, 0);
373  if(!off)
374  return true;
375 #if defined(__clang__)
376 #pragma clang diagnostic push
377 #pragma clang diagnostic ignored "-Wtautological-constant-out-of-range-compare"
378 #endif
379  // coverity[result_independent_of_operands]
380  if(off > std::numeric_limits<std::streamoff>::max())
381  return false;
382 #if defined(__clang__)
383 #pragma clang diagnostic pop
384 #endif
385  return detail::fseek(file_, static_cast<std::streamoff>(off), SEEK_CUR) == 0;
386  }
387 
390  bool stop_writing()
391  {
392  if(pptr())
393  {
394  const char* const base = pbase();
395  const size_t n = pptr() - base;
396  setp(0, 0);
397  if(n && std::fwrite(base, 1, n, file_) != n)
398  return false;
399  }
400  return true;
401  }
402 
403  void reset(FILE* f = 0)
404  {
405  sync();
406  if(file_)
407  {
408  fclose(file_);
409  file_ = 0;
410  }
411  file_ = f;
412  }
413 
414  static const wchar_t* get_mode(std::ios_base::openmode mode)
415  {
416  //
417  // done according to n2914 table 106 27.9.1.4
418  //
419 
420  // note can't use switch case as overload operator can't be used
421  // in constant expression
422  if(mode == (std::ios_base::out))
423  return L"w";
424  if(mode == (std::ios_base::out | std::ios_base::app))
425  return L"a";
426  if(mode == (std::ios_base::app))
427  return L"a";
428  if(mode == (std::ios_base::out | std::ios_base::trunc))
429  return L"w";
430  if(mode == (std::ios_base::in))
431  return L"r";
432  if(mode == (std::ios_base::in | std::ios_base::out))
433  return L"r+";
434  if(mode == (std::ios_base::in | std::ios_base::out | std::ios_base::trunc))
435  return L"w+";
436  if(mode == (std::ios_base::in | std::ios_base::out | std::ios_base::app))
437  return L"a+";
438  if(mode == (std::ios_base::in | std::ios_base::app))
439  return L"a+";
440  if(mode == (std::ios_base::binary | std::ios_base::out))
441  return L"wb";
442  if(mode == (std::ios_base::binary | std::ios_base::out | std::ios_base::app))
443  return L"ab";
444  if(mode == (std::ios_base::binary | std::ios_base::app))
445  return L"ab";
446  if(mode == (std::ios_base::binary | std::ios_base::out | std::ios_base::trunc))
447  return L"wb";
448  if(mode == (std::ios_base::binary | std::ios_base::in))
449  return L"rb";
450  if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::out))
451  return L"r+b";
452  if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::out | std::ios_base::trunc))
453  return L"w+b";
454  if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::out | std::ios_base::app))
455  return L"a+b";
456  if(mode == (std::ios_base::binary | std::ios_base::in | std::ios_base::app))
457  return L"a+b";
458  return 0;
459  }
460 
461  size_t buffer_size_;
462  char* buffer_;
463  FILE* file_;
464  bool owns_buffer_;
465  char last_char_[1];
466  std::ios::openmode mode_;
467  };
468 
473 
474 #endif // windows
475 
476 } // namespace nowide
477 } // namespace boost
478 
479 #endif
basic_filebuf * close()
Definition: filebuf.hpp:162
bool is_open() const
Definition: filebuf.hpp:184
This forward declaration defines the basic_filebuf type.
Definition: filebuf.hpp:47
basic_filebuf * open(const wchar_t *s, std::ios_base::openmode mode)
Opens the file with the given name, see std::filebuf::open.
Definition: filebuf.hpp:137
basic_filebuf * open(const char *s, std::ios_base::openmode mode)
Definition: filebuf.hpp:131
This is the implementation of std::filebuf.
Definition: filebuf.hpp:56
A class that allows to create a temporary wide or narrow UTF strings from wide or narrow UTF source.
Definition: stackstring.hpp:32
basic_filebuf * open(const std::string &s, std::ios_base::openmode mode)
Definition: filebuf.hpp:124
output_char * get()
Return the converted, NULL-terminated string or NULL if no string was converted.
Definition: stackstring.hpp:127