]>
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