2020-09-30 16:53:55 +02:00

375 lines
11 KiB
Plaintext

// fstream standard header
#ifndef _FSTREAM_
#define _FSTREAM_
#include <istream>
#include <new>
#ifdef _MSC_VER
#pragma pack(push,8)
#endif /* _MSC_VER */
_STD_BEGIN
extern _CRTIMP FILE *__cdecl __Fiopen(const char *,ios_base::openmode);
// TEMPLATE FUNCTION _Fgetc
template<class _E> inline
bool _Fgetc(_E& _C, _Filet *_Fi)
{return (fread(&_C, sizeof (_E), 1, _Fi) == 1); }
inline bool _Fgetc(char& _C, _Filet *_Fi)
{int _Ch;
if ((_Ch = fgetc(_Fi)) == EOF)
return (false);
else
{_C = (char)_Ch;
return (true); }}
inline bool _Fgetc(wchar_t& _C, _Filet *_Fi)
{_Wint_t _Ch;
if ((_Ch = fgetwc(_Fi)) == WEOF)
return (false);
else
{_C = _Ch;
return (true); }}
// TEMPLATE FUNCTION _Fputc
template<class _E> inline
bool _Fputc(_E _C, _Filet *_Fi)
{return (fwrite(&_C, sizeof (_E), 1, _Fi) == 1); }
inline bool _Fputc(char _C, _Filet *_Fi)
{return (fputc(_C, _Fi) != EOF); }
inline bool _Fputc(wchar_t _C, _Filet *_Fi)
{return (fputwc(_C, _Fi) != WEOF); }
// TEMPLATE FUNCTION _Ungetc
template<class _E> inline
bool _Ungetc(const _E& _C, _Filet *_Fi, size_t _N)
{const unsigned char *_P = (const unsigned char *)&_C;
for (_P += _N; 0 < _N && ungetc(*--_P, _Fi) != EOF; --_N)
;
if (_N == 0)
return (true);
else
{for (; _N < sizeof (_E); ++_N)
fgetc(_Fi);
return (false); }}
template<class _E> inline
bool _Ungetc(const _E& _C, _Filet *_Fi)
{return (_Ungetc(_C, _Fi, sizeof (_E))); }
inline bool _Ungetc(char _C, _Filet *_Fi)
{return (ungetc((unsigned char)_C, _Fi) != EOF); }
inline bool _Ungetc(wchar_t _C, _Filet *_Fi)
{return (ungetwc(_C, _Fi) != WEOF); }
// TEMPLATE CLASS basic_filebuf
template<class _E, class _Tr /* = char_traits<_E> */ >
class basic_filebuf : public basic_streambuf<_E, _Tr> {
public:
typedef basic_filebuf<_E, _Tr> _Myt;
typedef basic_streambuf<_E, _Tr> _Mysb;
typedef codecvt<_E, char, typename _Tr::state_type> _Cvt;
basic_filebuf(_Filet *_F = 0)
: _Loc(), _Mysb() {_Init(_F, _Newfl); }
basic_filebuf(_Uninitialized)
: _Loc(_Noinit), _Mysb(_Noinit) {}
virtual ~basic_filebuf()
{if (_Closef)
close();
delete _Str; }
enum _Initfl {_Newfl, _Openfl, _Closefl};
bool is_open() const
{return (_File != 0); }
_Myt *open(const char *_S, ios_base::openmode _M)
{_Filet *_Fp;
if (_File != 0 || (_Fp = __Fiopen(_S, _M)) == 0)
return (0);
_Init(_Fp, _Openfl);
_Initcvt();
return (this); }
_Myt *open(const char *_N, ios::open_mode _M)
{return (open(_N, (ios::openmode)_M)); }
_Myt *close()
{if (_File != 0 && fclose(_File) == 0)
{_Init(0, _Closefl);
return (this); }
else
return (0); }
protected:
virtual int_type overflow(int_type _C = _Tr::eof())
{if (_Tr::eq_int_type(_Tr::eof(), _C))
return (_Tr::not_eof(_C));
else if (pptr() != 0 && pptr() < epptr())
{*_Pninc() = _Tr::to_char_type(_C);
return (_C); }
else if (_File == 0)
return (_Tr::eof());
else if (_Pcvt == 0)
return (_Fputc(_Tr::to_char_type(_C), _File)
? _C : _Tr::eof());
else
{const int _NC = 8;
const _E _X = _Tr::to_char_type(_C);
const _E *_S;
char *_D;
_Str->erase();
for (size_t _I = _NC; ; _I += _NC)
{_Str->append(_NC, '\0');
switch (_Pcvt->out(_State,
&_X, &_X + 1, _S,
_Str->begin(), _Str->end(), _D))
{case codecvt_base::partial:
if (_S == &_X)
return (_Tr::eof());
case codecvt_base::ok: // can fall through
{size_t _N = _D - _Str->begin();
return (fwrite(_Str->begin(), 1, _N, _File)
== _N ? _C : _Tr::eof()); }
case codecvt_base::noconv:
return (_Fputc(_X, _File) ? _C : _Tr::eof());
default:
return (_Tr::eof()); }}}}
virtual int_type pbackfail(int_type _C = _Tr::eof())
{if (gptr() != 0 && eback() < gptr()
&& (_Tr::eq_int_type(_Tr::eof(), _C)
|| _Tr::eq_int_type(_Tr::to_int_type(gptr()[-1]),
_C)))
{_Gndec();
return (_Tr::not_eof(_C)); }
else if (_File == 0 || _Tr::eq_int_type(_Tr::eof(), _C))
return (_Tr::eof());
else if (_Pcvt == 0)
return (_Ungetc(_Tr::to_char_type(_C), _File)
? _C : _Tr::eof());
else if (0 < _Str->size()
&& _Ungetc(*_Str->begin(), _File, _Str->size()))
{_Str->erase();
_State = _State0;
return (_C); }
else
return (_Tr::eof()); }
virtual int_type underflow()
{if (gptr() != 0 && gptr() < egptr())
return (_Tr::to_int_type(*gptr()));
else
return (pbackfail(uflow())); }
virtual int_type uflow()
{if (gptr() != 0 && gptr() < egptr())
return (_Tr::to_int_type(*_Gninc()));
else if (_File == 0)
return (_Tr::eof());
else if (_Pcvt == 0)
{_E _C;
return (_Fgetc(_C, _File) ? _Tr::to_int_type(_C)
: _Tr::eof()); }
else
for (_State0 = _State, _Str->erase(); ; )
{_E _X, *_D;
const char *_S;
int _C = fgetc(_File);
if (_C == EOF)
return (_Tr::eof()); // partial char?
_Str->append(1, (char)_C);
_State = _State0;
switch (_Pcvt->in(_State,
_Str->begin(), _Str->end(), _S,
&_X, &_X + 1, _D))
{case codecvt_base::partial:
break;
case codecvt_base::noconv:
if (_Str->size() < sizeof (_E))
break;
memcpy(&_X, _Str->begin(), sizeof (_E));
case codecvt_base::ok: // can fall through
return (_Tr::to_int_type(_X));
default:
return (_Tr::eof()); }}}
virtual pos_type seekoff(off_type _O, ios_base::seekdir _Way,
ios_base::openmode =
(ios_base::openmode)(ios_base::in | ios_base::out))
{fpos_t _Fp;
if (_File == 0 || fseek(_File, (long)_O, _Way) != 0
|| fgetpos(_File, &_Fp) != 0)
return (pos_type(_BADOFF));
return (pos_type(_State, _Fp)); }
virtual pos_type seekpos(pos_type _P,
ios_base::openmode =
(ios_base::openmode)(ios_base::in | ios_base::out))
{fpos_t _Fp = _P.get_fpos_t();
off_type _Off = (off_type)_P - _FPOSOFF(_Fp);
if (_File == 0
|| fsetpos(_File, &_Fp) != 0
|| _Off != 0 && fseek(_File, (long)_Off, SEEK_CUR) != 0
|| fgetpos(_File, &_Fp) != 0)
return (pos_type(_BADOFF));
if (_Str != 0)
_State = _P.state(), _Str->erase();
return (pos_type(_State, _Fp)); }
virtual _Mysb *setbuf(_E *_S, streamsize _N)
{return (_File == 0 || setvbuf(_File, (char *)_S,
_IOFBF, _N * sizeof (_E)) != 0 ? 0 : this); }
virtual int sync()
{return (_File == 0 || 0 <= fflush(_File) ? 0 : -1); }
void _Init(_Filet *_Fp, _Initfl _Which)
{static _Tr::state_type _Stinit;
_Closef = _Which == _Openfl;
if (_Which == _Newfl)
{_Loc.locale::~locale();
new (&_Loc) locale;
_Str = 0; }
_Mysb::_Init();
if (_Fp != 0 && !_Closef && sizeof (_E) == 1)
{_Mysb::_Init((_E **)&_Fp->_base,
(_E **)&_Fp->_ptr, &_Fp->_cnt,
(_E **)&_Fp->_base, (_E **)&_Fp->_ptr,
&_Fp->_cnt); }
_File = _Fp;
_State = _Stinit;
_State0 = _Stinit;
_Pcvt = 0; }
void _Initcvt()
{_Pcvt = (_Cvt *)&_USE(getloc(), _Cvt);
_Loc = _ADDFAC(_Loc, _Pcvt);
if (_Pcvt->always_noconv())
_Pcvt = 0;
if (_Str == 0)
_Str = new string; }
private:
_Cvt *_Pcvt;
typename _Tr::state_type _State0;
typename _Tr::state_type _State;
string *_Str;
bool _Closef;
locale _Loc;
_Filet *_File;
};
#ifdef _DLL
#pragma warning(disable:4231) /* the extern before template is a non-standard extension */
extern template class _CRTIMP basic_filebuf<char, char_traits<char> >;
extern template class _CRTIMP basic_filebuf<wchar_t, char_traits<wchar_t> >;
#pragma warning(default:4231) /* restore previous warning */
#endif // _DLL
// TEMPLATE CLASS basic_ifstream
template<class _E, class _Tr /* = char_traits<_E> */ >
class basic_ifstream : public basic_istream<_E, _Tr> {
public:
typedef basic_ifstream<_E, _Tr> _Myt;
typedef basic_filebuf<_E, _Tr> _Myfb;
basic_ifstream()
: basic_istream<_E, _Tr>(&_Fb) {}
explicit basic_ifstream(const char *_S,
ios_base::openmode _M = in)
: basic_istream<_E, _Tr>(&_Fb)
{if (_Fb.open(_S, _M | in) == 0)
setstate(failbit); }
virtual ~basic_ifstream()
{}
_Myfb *rdbuf() const
{return ((_Myfb *)&_Fb); }
bool is_open() const
{return (_Fb.is_open()); }
void open(const char *_S, ios_base::openmode _M = in)
{if (_Fb.open(_S, _M | in) == 0)
setstate(failbit); }
void open(const char *_S, ios_base::open_mode _M)
{open(_S, (openmode)_M); }
void close()
{if (_Fb.close() == 0)
setstate(failbit); }
private:
_Myfb _Fb;
};
#ifdef _DLL
#pragma warning(disable:4231) /* the extern before template is a non-standard extension */
extern template class _CRTIMP basic_ifstream<char, char_traits<char> >;
extern template class _CRTIMP basic_ifstream<wchar_t, char_traits<wchar_t> >;
#pragma warning(default:4231) /* restore previous warning */
#endif // _DLL
// TEMPLATE CLASS basic_ofstream
template<class _E, class _Tr /* = char_traits<_E> */ >
class basic_ofstream : public basic_ostream<_E, _Tr> {
public:
typedef basic_ofstream<_E, _Tr> _Myt;
typedef basic_filebuf<_E, _Tr> _Myfb;
basic_ofstream()
: basic_ostream<_E, _Tr>(&_Fb) {}
explicit basic_ofstream(const char *_S,
ios_base::openmode _M = out | trunc)
: basic_ostream<_E, _Tr>(&_Fb)
{if (_Fb.open(_S, _M | out) == 0)
setstate(failbit); }
virtual ~basic_ofstream()
{}
_Myfb *rdbuf() const
{return ((_Myfb *)&_Fb); }
bool is_open() const
{return (_Fb.is_open()); }
void open(const char *_S, ios_base::openmode _M = out | trunc)
{if (_Fb.open(_S, _M | out) == 0)
setstate(failbit); }
void open(const char *_S, ios_base::open_mode _M)
{open(_S, (openmode)_M); }
void close()
{if (_Fb.close() == 0)
setstate(failbit); }
private:
_Myfb _Fb;
};
#ifdef _DLL
#pragma warning(disable:4231) /* the extern before template is a non-standard extension */
extern template class _CRTIMP basic_ofstream<char, char_traits<char> >;
extern template class _CRTIMP basic_ofstream<wchar_t, char_traits<wchar_t> >;
#pragma warning(default:4231) /* restore previous warning */
#endif // _DLL
// TEMPLATE CLASS basic_fstream
template<class _E, class _Tr /* = char_traits<_E> */ >
class basic_fstream : public basic_iostream<_E, _Tr> {
public:
basic_fstream()
: basic_iostream<_E, _Tr>(&_Fb) {}
explicit basic_fstream(const char *_S,
ios_base::openmode _M = in | out)
: basic_iostream<_E, _Tr>(&_Fb)
{if (_Fb.open(_S, _M) == 0)
setstate(failbit); }
virtual ~basic_fstream()
{}
basic_filebuf<_E, _Tr> *rdbuf() const
{return ((basic_filebuf<_E, _Tr> *)&_Fb); }
bool is_open() const
{return (_Fb.is_open()); }
void open(const char *_S, ios_base::openmode _M = in | out)
{if (_Fb.open(_S, _M) == 0)
setstate(failbit); }
void open(const char *_S, ios_base::open_mode _M)
{open(_S, (openmode)_M); }
void close()
{if (_Fb.close() == 0)
setstate(failbit); }
private:
basic_filebuf<_E, _Tr> _Fb;
};
#ifdef _DLL
#pragma warning(disable:4231) /* the extern before template is a non-standard extension */
extern template class _CRTIMP basic_fstream<char, char_traits<char> >;
extern template class _CRTIMP basic_fstream<wchar_t, char_traits<wchar_t> >;
#pragma warning(default:4231) /* restore previous warning */
#endif // _DLL
_STD_END
#ifdef _MSC_VER
#pragma pack(pop)
#endif /* _MSC_VER */
#endif /* _FSTREAM_ */
/*
* Copyright (c) 1994 by P.J. Plauger. ALL RIGHTS RESERVED.
* Consult your license regarding permissions and restrictions.
*/