]>
git.saurik.com Git - wxWidgets.git/blob - src/common/stream.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/stream.cpp
3 // Purpose: wxStream base classes
4 // Author: Guilhem Lavaux
5 // Modified by: VZ (23.11.00) to fix realloc()ing new[]ed memory,
9 // Copyright: (c) Guilhem Lavaux
10 // Licence: wxWindows licence
11 /////////////////////////////////////////////////////////////////////////////
13 // ============================================================================
15 // ============================================================================
17 // ----------------------------------------------------------------------------
19 // ----------------------------------------------------------------------------
21 // For compilers that support precompilation, includes "wx.h".
22 #include "wx/wxprec.h"
35 #include "wx/stream.h"
36 #include "wx/datstrm.h"
37 #include "wx/textfile.h"
40 // ----------------------------------------------------------------------------
42 // ----------------------------------------------------------------------------
44 // the temporary buffer size used when copying from stream to stream
45 #define BUF_TEMP_SIZE 4096
47 // ============================================================================
49 // ============================================================================
51 // ----------------------------------------------------------------------------
53 // ----------------------------------------------------------------------------
55 void wxStreamBuffer::SetError(wxStreamError err
)
57 if ( m_stream
&& m_stream
->m_lasterror
== wxSTREAM_NO_ERROR
)
58 m_stream
->m_lasterror
= err
;
61 void wxStreamBuffer::InitBuffer()
68 // if we are going to allocate the buffer, we should free it later as well
72 void wxStreamBuffer::Init()
79 wxStreamBuffer::wxStreamBuffer(BufMode mode
)
89 wxStreamBuffer::wxStreamBuffer(wxStreamBase
& stream
, BufMode mode
)
99 wxStreamBuffer::wxStreamBuffer(const wxStreamBuffer
& buffer
)
101 // doing this has big chances to lead to a crash when the source buffer is
102 // destroyed (otherwise assume the caller knows what he does)
103 wxASSERT_MSG( !buffer
.m_destroybuf
,
104 _T("it's a bad idea to copy this buffer") );
106 m_buffer_start
= buffer
.m_buffer_start
;
107 m_buffer_end
= buffer
.m_buffer_end
;
108 m_buffer_pos
= buffer
.m_buffer_pos
;
109 m_buffer_size
= buffer
.m_buffer_size
;
110 m_fixed
= buffer
.m_fixed
;
111 m_flushable
= buffer
.m_flushable
;
112 m_stream
= buffer
.m_stream
;
113 m_mode
= buffer
.m_mode
;
114 m_destroybuf
= false;
117 void wxStreamBuffer::FreeBuffer()
120 free(m_buffer_start
);
123 wxStreamBuffer::~wxStreamBuffer()
128 wxInputStream
*wxStreamBuffer::GetInputStream() const
130 return m_mode
== write
? NULL
: (wxInputStream
*)m_stream
;
133 wxOutputStream
*wxStreamBuffer::GetOutputStream() const
135 return m_mode
== read
? NULL
: (wxOutputStream
*)m_stream
;
138 void wxStreamBuffer::SetBufferIO(void *buffer_start
,
142 SetBufferIO(buffer_start
, (char *)buffer_end
- (char *)buffer_start
,
146 void wxStreamBuffer::SetBufferIO(void *start
,
150 // start by freeing the old buffer
153 m_buffer_start
= (char *)start
;
154 m_buffer_end
= m_buffer_start
+ len
;
158 // if we own it, we free it
159 m_destroybuf
= takeOwnership
;
164 void wxStreamBuffer::SetBufferIO(size_t bufsize
)
166 // start by freeing the old buffer
171 SetBufferIO(malloc(bufsize
), bufsize
, true /* take ownership */);
173 else // no buffer size => no buffer
179 void wxStreamBuffer::ResetBuffer()
184 m_stream
->m_lastcount
= 0;
187 m_buffer_pos
= m_mode
== read
&& m_flushable
192 // fill the buffer with as much data as possible (only for read buffers)
193 bool wxStreamBuffer::FillBuffer()
195 wxInputStream
*inStream
= GetInputStream();
197 // It's legal to have no stream, so we don't complain about it just return false
201 size_t count
= inStream
->OnSysRead(m_buffer_start
, m_buffer_size
);
205 m_buffer_end
= m_buffer_start
+ count
;
206 m_buffer_pos
= m_buffer_start
;
211 // write the buffer contents to the stream (only for write buffers)
212 bool wxStreamBuffer::FlushBuffer()
214 wxCHECK_MSG( m_flushable
, false, _T("can't flush this buffer") );
216 // FIXME: what is this check for? (VZ)
217 if ( m_buffer_pos
== m_buffer_start
)
220 wxOutputStream
*outStream
= GetOutputStream();
222 wxCHECK_MSG( outStream
, false, _T("should have a stream in wxStreamBuffer") );
224 size_t current
= m_buffer_pos
- m_buffer_start
;
225 size_t count
= outStream
->OnSysWrite(m_buffer_start
, current
);
226 if ( count
!= current
)
229 m_buffer_pos
= m_buffer_start
;
234 size_t wxStreamBuffer::GetDataLeft()
236 /* Why is this done? RR. */
237 if ( m_buffer_pos
== m_buffer_end
&& m_flushable
)
240 return GetBytesLeft();
243 // copy up to size bytes from our buffer into the provided one
244 void wxStreamBuffer::GetFromBuffer(void *buffer
, size_t size
)
246 // don't get more bytes than left in the buffer
247 size_t left
= GetBytesLeft();
252 memcpy(buffer
, m_buffer_pos
, size
);
253 m_buffer_pos
+= size
;
256 // copy the contents of the provided buffer into this one
257 void wxStreamBuffer::PutToBuffer(const void *buffer
, size_t size
)
259 size_t left
= GetBytesLeft();
265 // we can't realloc the buffer, so just copy what we can
270 // realloc the buffer to have enough space for the data
271 size_t delta
= m_buffer_pos
- m_buffer_start
;
273 char *startOld
= m_buffer_start
;
274 m_buffer_size
+= size
;
275 m_buffer_start
= (char *)realloc(m_buffer_start
, m_buffer_size
);
276 if ( !m_buffer_start
)
278 // don't leak memory if realloc() failed
279 m_buffer_start
= startOld
;
280 m_buffer_size
-= size
;
282 // what else can we do?
286 // adjust the pointers invalidated by realloc()
287 m_buffer_pos
= m_buffer_start
+ delta
;
288 m_buffer_end
= m_buffer_start
+ m_buffer_size
;
292 memcpy(m_buffer_pos
, buffer
, size
);
293 m_buffer_pos
+= size
;
296 void wxStreamBuffer::PutChar(char c
)
298 wxOutputStream
*outStream
= GetOutputStream();
300 wxCHECK_RET( outStream
, _T("should have a stream in wxStreamBuffer") );
302 // if we don't have buffer at all, just forward this call to the stream,
305 outStream
->OnSysWrite(&c
, sizeof(c
));
309 // otherwise check we have enough space left
310 if ( !GetDataLeft() && !FlushBuffer() )
313 SetError(wxSTREAM_WRITE_ERROR
);
317 PutToBuffer(&c
, sizeof(c
));
318 m_stream
->m_lastcount
= 1;
323 char wxStreamBuffer::Peek()
325 wxCHECK_MSG( m_stream
&& HasBuffer(), 0,
326 _T("should have the stream and the buffer in wxStreamBuffer") );
328 if ( !GetDataLeft() )
330 SetError(wxSTREAM_READ_ERROR
);
335 GetFromBuffer(&c
, sizeof(c
));
341 char wxStreamBuffer::GetChar()
343 wxInputStream
*inStream
= GetInputStream();
345 wxCHECK_MSG( inStream
, 0, _T("should have a stream in wxStreamBuffer") );
350 inStream
->OnSysRead(&c
, sizeof(c
));
354 if ( !GetDataLeft() )
356 SetError(wxSTREAM_READ_ERROR
);
361 GetFromBuffer(&c
, sizeof(c
));
362 m_stream
->m_lastcount
= 1;
369 size_t wxStreamBuffer::Read(void *buffer
, size_t size
)
371 // lasterror is reset before all new IO calls
378 wxInputStream
*inStream
= GetInputStream();
380 wxCHECK_MSG( inStream
, 0, _T("should have a stream in wxStreamBuffer") );
382 readBytes
= inStream
->OnSysRead(buffer
, size
);
384 else // we have a buffer, use it
386 size_t orig_size
= size
;
390 size_t left
= GetDataLeft();
392 // if the requested number of bytes if greater than the buffer
393 // size, read data in chunks
396 GetFromBuffer(buffer
, left
);
398 buffer
= (char *)buffer
+ left
;
402 SetError(wxSTREAM_EOF
);
406 else // otherwise just do it in one gulp
408 GetFromBuffer(buffer
, size
);
413 readBytes
= orig_size
- size
;
417 m_stream
->m_lastcount
= readBytes
;
422 // this should really be called "Copy()"
423 size_t wxStreamBuffer::Read(wxStreamBuffer
*dbuf
)
425 wxCHECK_MSG( m_mode
!= write
, 0, _T("can't read from this buffer") );
427 char buf
[BUF_TEMP_SIZE
];
433 nRead
= Read(buf
, WXSIZEOF(buf
));
436 nRead
= dbuf
->Write(buf
, nRead
);
445 size_t wxStreamBuffer::Write(const void *buffer
, size_t size
)
449 // lasterror is reset before all new IO calls
455 if ( !HasBuffer() && m_fixed
)
457 wxOutputStream
*outStream
= GetOutputStream();
459 wxCHECK_MSG( outStream
, 0, _T("should have a stream in wxStreamBuffer") );
461 // no buffer, just forward the call to the stream
462 ret
= outStream
->OnSysWrite(buffer
, size
);
464 else // we [may] have a buffer, use it
466 size_t orig_size
= size
;
470 size_t left
= GetBytesLeft();
472 // if the buffer is too large to fit in the stream buffer, split
473 // it in smaller parts
475 // NB: If stream buffer isn't fixed (as for wxMemoryOutputStream),
476 // we always go to the second case.
478 // FIXME: fine, but if it fails we should (re)try writing it by
479 // chunks as this will (hopefully) always work (VZ)
481 if ( size
> left
&& m_fixed
)
483 PutToBuffer(buffer
, left
);
485 buffer
= (char *)buffer
+ left
;
487 if ( !FlushBuffer() )
489 SetError(wxSTREAM_WRITE_ERROR
);
494 m_buffer_pos
= m_buffer_start
;
496 else // we can do it in one gulp
498 PutToBuffer(buffer
, size
);
503 ret
= orig_size
- size
;
508 // i am not entirely sure what we do this for
509 m_stream
->m_lastcount
= ret
;
515 size_t wxStreamBuffer::Write(wxStreamBuffer
*sbuf
)
517 wxCHECK_MSG( m_mode
!= read
, 0, _T("can't write to this buffer") );
518 wxCHECK_MSG( sbuf
->m_mode
!= write
, 0, _T("can't read from that buffer") );
520 char buf
[BUF_TEMP_SIZE
];
526 size_t nRead
= sbuf
->Read(buf
, WXSIZEOF(buf
));
529 nWrite
= Write(buf
, nRead
);
530 if ( nWrite
< nRead
)
532 // put back data we couldn't copy
533 wxInputStream
*in_stream
= (wxInputStream
*)sbuf
->GetStream();
535 in_stream
->Ungetch(buf
+ nWrite
, nRead
- nWrite
);
545 while ( nWrite
== WXSIZEOF(buf
) );
550 wxFileOffset
wxStreamBuffer::Seek(wxFileOffset pos
, wxSeekMode mode
)
552 wxFileOffset ret_off
, diff
;
554 wxFileOffset last_access
= GetLastAccess();
565 diff
= pos
+ GetIntPosition();
569 diff
= pos
+ last_access
;
573 wxFAIL_MSG( _T("invalid seek mode") );
575 return wxInvalidOffset
;
577 if (diff
< 0 || diff
> last_access
)
578 return wxInvalidOffset
;
579 size_t int_diff
= wx_truncate_cast(size_t, diff
);
580 wxCHECK_MSG( (wxFileOffset
)int_diff
== diff
, wxInvalidOffset
, wxT("huge file not supported") );
581 SetIntPosition(int_diff
);
588 // We'll try to compute an internal position later ...
589 ret_off
= m_stream
->OnSysSeek(pos
, wxFromStart
);
594 diff
= pos
+ GetIntPosition();
596 if ( (diff
> last_access
) || (diff
< 0) )
598 // We must take into account the fact that we have read
599 // something previously.
600 ret_off
= m_stream
->OnSysSeek(diff
-last_access
, wxFromCurrent
);
606 size_t int_diff
= wx_truncate_cast(size_t, diff
);
607 wxCHECK_MSG( (wxFileOffset
)int_diff
== diff
, wxInvalidOffset
, wxT("huge file not supported") );
608 SetIntPosition(int_diff
);
613 // Hard to compute: always seek to the requested position.
614 ret_off
= m_stream
->OnSysSeek(pos
, wxFromEnd
);
619 return wxInvalidOffset
;
622 wxFileOffset
wxStreamBuffer::Tell() const
626 // ask the stream for position if we have a real one
629 pos
= m_stream
->OnSysTell();
630 if ( pos
== wxInvalidOffset
)
631 return wxInvalidOffset
;
633 else // no associated stream
638 pos
+= GetIntPosition();
640 if ( m_mode
== read
&& m_flushable
)
641 pos
-= GetLastAccess();
646 // ----------------------------------------------------------------------------
648 // ----------------------------------------------------------------------------
650 wxStreamBase::wxStreamBase()
652 m_lasterror
= wxSTREAM_NO_ERROR
;
656 wxStreamBase::~wxStreamBase()
660 size_t wxStreamBase::GetSize() const
662 wxFileOffset length
= GetLength();
663 if ( length
== wxInvalidOffset
)
666 const size_t len
= wx_truncate_cast(size_t, length
);
667 wxASSERT_MSG( len
== length
+ size_t(0), _T("large files not supported") );
672 wxFileOffset
wxStreamBase::OnSysSeek(wxFileOffset
WXUNUSED(seek
), wxSeekMode
WXUNUSED(mode
))
674 return wxInvalidOffset
;
677 wxFileOffset
wxStreamBase::OnSysTell() const
679 return wxInvalidOffset
;
682 // ----------------------------------------------------------------------------
684 // ----------------------------------------------------------------------------
686 wxInputStream::wxInputStream()
693 wxInputStream::~wxInputStream()
698 bool wxInputStream::CanRead() const
700 // we don't know if there is anything to read or not and by default we
701 // prefer to be optimistic and try to read data unless we know for sure
702 // there is no more of it
703 return m_lasterror
!= wxSTREAM_EOF
;
706 bool wxInputStream::Eof() const
708 // the only way the base class can know we're at EOF is when we'd already
709 // tried to read beyond it in which case last error is set accordingly
710 return GetLastError() == wxSTREAM_EOF
;
713 char *wxInputStream::AllocSpaceWBack(size_t needed_size
)
715 // get number of bytes left from previous wback buffer
716 size_t toget
= m_wbacksize
- m_wbackcur
;
718 // allocate a buffer large enough to hold prev + new data
719 char *temp_b
= (char *)malloc(needed_size
+ toget
);
724 // copy previous data (and free old buffer) if needed
727 memmove(temp_b
+ needed_size
, m_wback
+ m_wbackcur
, toget
);
734 m_wbacksize
= needed_size
+ toget
;
739 size_t wxInputStream::GetWBack(void *buf
, size_t size
)
744 // how many bytes do we have in the buffer?
745 size_t toget
= m_wbacksize
- m_wbackcur
;
749 // we won't read everything
753 // copy the data from the cache
754 memcpy(buf
, m_wback
+ m_wbackcur
, toget
);
757 if ( m_wbackcur
== m_wbacksize
)
759 // TODO: should we really free it here all the time? maybe keep it?
766 // return the number of bytes copied
770 size_t wxInputStream::Ungetch(const void *buf
, size_t bufsize
)
772 if ( m_lasterror
!= wxSTREAM_NO_ERROR
&& m_lasterror
!= wxSTREAM_EOF
)
774 // can't operate on this stream until the error is cleared
778 char *ptrback
= AllocSpaceWBack(bufsize
);
782 // Eof() shouldn't return true any longer
783 if ( m_lasterror
== wxSTREAM_EOF
)
784 m_lasterror
= wxSTREAM_NO_ERROR
;
786 memcpy(ptrback
, buf
, bufsize
);
790 bool wxInputStream::Ungetch(char c
)
792 return Ungetch(&c
, sizeof(c
)) != 0;
795 char wxInputStream::GetC()
802 wxInputStream
& wxInputStream::Read(void *buf
, size_t size
)
804 char *p
= (char *)buf
;
807 size_t read
= GetWBack(buf
, size
);
816 // we read the requested amount of data
820 if ( p
!= buf
&& !CanRead() )
822 // we have already read something and we would block in OnSysRead()
823 // now: don't do it but return immediately
827 read
= OnSysRead(p
, size
);
830 // no more data available
838 char wxInputStream::Peek()
842 if (m_lasterror
== wxSTREAM_NO_ERROR
)
851 wxInputStream
& wxInputStream::Read(wxOutputStream
& stream_out
)
853 char buf
[BUF_TEMP_SIZE
];
857 size_t bytes_read
= Read(buf
, WXSIZEOF(buf
)).LastRead();
861 if ( stream_out
.Write(buf
, bytes_read
).LastWrite() != bytes_read
)
868 wxFileOffset
wxInputStream::SeekI(wxFileOffset pos
, wxSeekMode mode
)
870 // RR: This code is duplicated in wxBufferedInputStream. This is
871 // not really a good design, but buffered stream are different
872 // from all other in that they handle two stream-related objects,
873 // the stream buffer and parent stream.
875 // I don't know whether it should be put as well in wxFileInputStream::OnSysSeek
876 if (m_lasterror
==wxSTREAM_EOF
)
877 m_lasterror
=wxSTREAM_NO_ERROR
;
879 /* RR: A call to SeekI() will automatically invalidate any previous
880 call to Ungetch(), otherwise it would be possible to SeekI() to
881 one position, unread some bytes there, SeekI() to another position
882 and the data would be corrupted.
884 GRG: Could add code here to try to navigate within the wback
885 buffer if possible, but is it really needed? It would only work
886 when seeking in wxFromCurrent mode, else it would invalidate
891 wxLogDebug( wxT("Seeking in stream which has data written back to it.") );
899 return OnSysSeek(pos
, mode
);
902 wxFileOffset
wxInputStream::TellI() const
904 wxFileOffset pos
= OnSysTell();
906 if (pos
!= wxInvalidOffset
)
907 pos
-= (m_wbacksize
- m_wbackcur
);
913 // ----------------------------------------------------------------------------
915 // ----------------------------------------------------------------------------
917 wxOutputStream::wxOutputStream()
921 wxOutputStream::~wxOutputStream()
925 size_t wxOutputStream::OnSysWrite(const void * WXUNUSED(buffer
),
926 size_t WXUNUSED(bufsize
))
931 void wxOutputStream::PutC(char c
)
933 Write(&c
, sizeof(c
));
936 wxOutputStream
& wxOutputStream::Write(const void *buffer
, size_t size
)
938 m_lastcount
= OnSysWrite(buffer
, size
);
942 wxOutputStream
& wxOutputStream::Write(wxInputStream
& stream_in
)
944 stream_in
.Read(*this);
948 wxFileOffset
wxOutputStream::TellO() const
953 wxFileOffset
wxOutputStream::SeekO(wxFileOffset pos
, wxSeekMode mode
)
955 return OnSysSeek(pos
, mode
);
958 void wxOutputStream::Sync()
963 // ----------------------------------------------------------------------------
964 // wxCountingOutputStream
965 // ----------------------------------------------------------------------------
967 wxCountingOutputStream::wxCountingOutputStream ()
972 wxFileOffset
wxCountingOutputStream::GetLength() const
977 size_t wxCountingOutputStream::OnSysWrite(const void *WXUNUSED(buffer
),
980 m_currentPos
+= size
;
981 if (m_currentPos
> m_lastcount
)
982 m_lastcount
= m_currentPos
;
987 wxFileOffset
wxCountingOutputStream::OnSysSeek(wxFileOffset pos
, wxSeekMode mode
)
989 ssize_t new_pos
= wx_truncate_cast(ssize_t
, pos
);
994 wxCHECK_MSG( (wxFileOffset
)new_pos
== pos
, wxInvalidOffset
, wxT("huge position not supported") );
998 new_pos
= m_lastcount
+ new_pos
;
999 wxCHECK_MSG( (wxFileOffset
)new_pos
== (wxFileOffset
)(m_lastcount
+ pos
), wxInvalidOffset
, wxT("huge position not supported") );
1003 new_pos
= m_currentPos
+ new_pos
;
1004 wxCHECK_MSG( (wxFileOffset
)new_pos
== (wxFileOffset
)(m_currentPos
+ pos
), wxInvalidOffset
, wxT("huge position not supported") );
1008 wxFAIL_MSG( _T("invalid seek mode") );
1009 return wxInvalidOffset
;
1012 m_currentPos
= new_pos
;
1014 if (m_currentPos
> m_lastcount
)
1015 m_lastcount
= m_currentPos
;
1017 return m_currentPos
;
1020 wxFileOffset
wxCountingOutputStream::OnSysTell() const
1022 return m_currentPos
;
1025 // ----------------------------------------------------------------------------
1026 // wxFilterInputStream
1027 // ----------------------------------------------------------------------------
1029 wxFilterInputStream::wxFilterInputStream()
1031 m_parent_i_stream
= NULL
;
1034 wxFilterInputStream::wxFilterInputStream(wxInputStream
& stream
)
1036 m_parent_i_stream
= &stream
;
1039 wxFilterInputStream::~wxFilterInputStream()
1043 // ----------------------------------------------------------------------------
1044 // wxFilterOutputStream
1045 // ----------------------------------------------------------------------------
1047 wxFilterOutputStream::wxFilterOutputStream()
1049 m_parent_o_stream
= NULL
;
1052 wxFilterOutputStream::wxFilterOutputStream(wxOutputStream
& stream
)
1054 m_parent_o_stream
= &stream
;
1057 wxFilterOutputStream::~wxFilterOutputStream()
1061 // ----------------------------------------------------------------------------
1062 // wxBufferedInputStream
1063 // ----------------------------------------------------------------------------
1065 wxBufferedInputStream::wxBufferedInputStream(wxInputStream
& s
,
1066 wxStreamBuffer
*buffer
)
1067 : wxFilterInputStream(s
)
1071 // use the buffer provided by the user
1072 m_i_streambuf
= buffer
;
1074 else // create a default buffer
1076 m_i_streambuf
= new wxStreamBuffer(*this, wxStreamBuffer::read
);
1078 m_i_streambuf
->SetBufferIO(1024);
1082 wxBufferedInputStream::~wxBufferedInputStream()
1084 m_parent_i_stream
->SeekI(-(wxFileOffset
)m_i_streambuf
->GetBytesLeft(),
1087 delete m_i_streambuf
;
1090 char wxBufferedInputStream::Peek()
1092 return m_i_streambuf
->Peek();
1095 wxInputStream
& wxBufferedInputStream::Read(void *buf
, size_t size
)
1097 // reset the error flag
1100 // first read from the already cached data
1101 m_lastcount
= GetWBack(buf
, size
);
1103 // do we have to read anything more?
1104 if ( m_lastcount
< size
)
1106 size
-= m_lastcount
;
1107 buf
= (char *)buf
+ m_lastcount
;
1109 // the call to wxStreamBuffer::Read() below will reset our m_lastcount,
1111 size_t countOld
= m_lastcount
;
1113 m_i_streambuf
->Read(buf
, size
);
1115 m_lastcount
+= countOld
;
1121 wxFileOffset
wxBufferedInputStream::SeekI(wxFileOffset pos
, wxSeekMode mode
)
1123 // RR: Look at wxInputStream for comments.
1125 if (m_lasterror
==wxSTREAM_EOF
)
1130 wxLogDebug( wxT("Seeking in stream which has data written back to it.") );
1138 return m_i_streambuf
->Seek(pos
, mode
);
1141 wxFileOffset
wxBufferedInputStream::TellI() const
1143 wxFileOffset pos
= m_i_streambuf
->Tell();
1145 if (pos
!= wxInvalidOffset
)
1146 pos
-= (m_wbacksize
- m_wbackcur
);
1151 size_t wxBufferedInputStream::OnSysRead(void *buffer
, size_t bufsize
)
1153 return m_parent_i_stream
->Read(buffer
, bufsize
).LastRead();
1156 wxFileOffset
wxBufferedInputStream::OnSysSeek(wxFileOffset seek
, wxSeekMode mode
)
1158 return m_parent_i_stream
->SeekI(seek
, mode
);
1161 wxFileOffset
wxBufferedInputStream::OnSysTell() const
1163 return m_parent_i_stream
->TellI();
1166 void wxBufferedInputStream::SetInputStreamBuffer(wxStreamBuffer
*buffer
)
1168 wxCHECK_RET( buffer
, _T("wxBufferedInputStream needs buffer") );
1170 delete m_i_streambuf
;
1171 m_i_streambuf
= buffer
;
1174 // ----------------------------------------------------------------------------
1175 // wxBufferedOutputStream
1176 // ----------------------------------------------------------------------------
1178 wxBufferedOutputStream::wxBufferedOutputStream(wxOutputStream
& s
,
1179 wxStreamBuffer
*buffer
)
1180 : wxFilterOutputStream(s
)
1184 m_o_streambuf
= buffer
;
1186 else // create a default one
1188 m_o_streambuf
= new wxStreamBuffer(*this, wxStreamBuffer::write
);
1190 m_o_streambuf
->SetBufferIO(1024);
1194 wxBufferedOutputStream::~wxBufferedOutputStream()
1197 delete m_o_streambuf
;
1200 bool wxBufferedOutputStream::Close()
1207 wxOutputStream
& wxBufferedOutputStream::Write(const void *buffer
, size_t size
)
1210 m_o_streambuf
->Write(buffer
, size
);
1214 wxFileOffset
wxBufferedOutputStream::SeekO(wxFileOffset pos
, wxSeekMode mode
)
1217 return m_o_streambuf
->Seek(pos
, mode
);
1220 wxFileOffset
wxBufferedOutputStream::TellO() const
1222 return m_o_streambuf
->Tell();
1225 void wxBufferedOutputStream::Sync()
1227 m_o_streambuf
->FlushBuffer();
1228 m_parent_o_stream
->Sync();
1231 size_t wxBufferedOutputStream::OnSysWrite(const void *buffer
, size_t bufsize
)
1233 return m_parent_o_stream
->Write(buffer
, bufsize
).LastWrite();
1236 wxFileOffset
wxBufferedOutputStream::OnSysSeek(wxFileOffset seek
, wxSeekMode mode
)
1238 return m_parent_o_stream
->SeekO(seek
, mode
);
1241 wxFileOffset
wxBufferedOutputStream::OnSysTell() const
1243 return m_parent_o_stream
->TellO();
1246 wxFileOffset
wxBufferedOutputStream::GetLength() const
1248 return m_parent_o_stream
->GetLength() + m_o_streambuf
->GetIntPosition();
1251 void wxBufferedOutputStream::SetOutputStreamBuffer(wxStreamBuffer
*buffer
)
1253 wxCHECK_RET( buffer
, _T("wxBufferedOutputStream needs buffer") );
1255 delete m_o_streambuf
;
1256 m_o_streambuf
= buffer
;
1259 // ----------------------------------------------------------------------------
1260 // Some IOManip function
1261 // ----------------------------------------------------------------------------
1263 wxOutputStream
& wxEndL(wxOutputStream
& stream
)
1265 static const wxChar
*eol
= wxTextFile::GetEOL();
1267 return stream
.Write(eol
, wxStrlen(eol
));
1270 #endif // wxUSE_STREAMS