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" 
  30 #include "wx/stream.h" 
  37 #include "wx/datstrm.h" 
  38 #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() 
  67     // if we are going to allocate the buffer, we should free it later as well 
  71 void wxStreamBuffer::Init() 
  78 void wxStreamBuffer::InitWithStream(wxStreamBase
& stream
, BufMode mode
) 
  88 wxStreamBuffer::wxStreamBuffer(BufMode mode
) 
  98 wxStreamBuffer::wxStreamBuffer(const wxStreamBuffer
& buffer
) 
 100     // doing this has big chances to lead to a crash when the source buffer is 
 101     // destroyed (otherwise assume the caller knows what he does) 
 102     wxASSERT_MSG( !buffer
.m_destroybuf
, 
 103                   _T("it's a bad idea to copy this buffer") ); 
 105     m_buffer_start 
= buffer
.m_buffer_start
; 
 106     m_buffer_end 
= buffer
.m_buffer_end
; 
 107     m_buffer_pos 
= buffer
.m_buffer_pos
; 
 108     m_fixed 
= buffer
.m_fixed
; 
 109     m_flushable 
= buffer
.m_flushable
; 
 110     m_stream 
= buffer
.m_stream
; 
 111     m_mode 
= buffer
.m_mode
; 
 112     m_destroybuf 
= false; 
 115 void wxStreamBuffer::FreeBuffer() 
 119         free(m_buffer_start
); 
 120         m_buffer_start 
= NULL
; 
 124 wxStreamBuffer::~wxStreamBuffer() 
 129 wxInputStream 
*wxStreamBuffer::GetInputStream() const 
 131     return m_mode 
== write 
? NULL 
: (wxInputStream 
*)m_stream
; 
 134 wxOutputStream 
*wxStreamBuffer::GetOutputStream() const 
 136     return m_mode 
== read 
? NULL 
: (wxOutputStream 
*)m_stream
; 
 139 void wxStreamBuffer::SetBufferIO(void *buffer_start
, 
 143     SetBufferIO(buffer_start
, (char *)buffer_end 
- (char *)buffer_start
, 
 147 void wxStreamBuffer::SetBufferIO(void *start
, 
 151     // start by freeing the old buffer 
 154     m_buffer_start 
= (char *)start
; 
 155     m_buffer_end   
= m_buffer_start 
+ len
; 
 157     // if we own it, we free it 
 158     m_destroybuf 
= takeOwnership
; 
 163 void wxStreamBuffer::SetBufferIO(size_t bufsize
) 
 167         // this will free the old buffer and allocate the new one 
 168         SetBufferIO(malloc(bufsize
), bufsize
, true /* take ownership */); 
 170     else // no buffer size => no buffer 
 172         // still free the old one 
 178 void wxStreamBuffer::ResetBuffer() 
 183         m_stream
->m_lastcount 
= 0; 
 186     m_buffer_pos 
= m_mode 
== read 
&& m_flushable
 
 191 void wxStreamBuffer::Truncate() 
 193     size_t new_size 
= m_buffer_pos 
- m_buffer_start
; 
 194     if ( m_buffer_pos 
== m_buffer_end 
) 
 204     char *new_start 
= (char *)realloc(m_buffer_start
, new_size
); 
 205     wxCHECK_RET( new_size
, _T("shrinking buffer shouldn't fail") ); 
 207     m_buffer_start 
= new_start
; 
 208     m_buffer_end 
= m_buffer_start 
+ new_size
; 
 209     m_buffer_pos 
= m_buffer_end
; 
 212 // fill the buffer with as much data as possible (only for read buffers) 
 213 bool wxStreamBuffer::FillBuffer() 
 215     wxInputStream 
*inStream 
= GetInputStream(); 
 217     // It's legal to have no stream, so we don't complain about it just return false 
 221     size_t count 
= inStream
->OnSysRead(GetBufferStart(), GetBufferSize()); 
 225     m_buffer_end 
= m_buffer_start 
+ count
; 
 226     m_buffer_pos 
= m_buffer_start
; 
 231 // write the buffer contents to the stream (only for write buffers) 
 232 bool wxStreamBuffer::FlushBuffer() 
 234     wxCHECK_MSG( m_flushable
, false, _T("can't flush this buffer") ); 
 236     // FIXME: what is this check for? (VZ) 
 237     if ( m_buffer_pos 
== m_buffer_start 
) 
 240     wxOutputStream 
*outStream 
= GetOutputStream(); 
 242     wxCHECK_MSG( outStream
, false, _T("should have a stream in wxStreamBuffer") ); 
 244     size_t current 
= m_buffer_pos 
- m_buffer_start
; 
 245     size_t count 
= outStream
->OnSysWrite(m_buffer_start
, current
); 
 246     if ( count 
!= current 
) 
 249     m_buffer_pos 
= m_buffer_start
; 
 254 size_t wxStreamBuffer::GetDataLeft() 
 256     /* Why is this done? RR. */ 
 257     if ( m_buffer_pos 
== m_buffer_end 
&& m_flushable
) 
 260     return GetBytesLeft(); 
 263 // copy up to size bytes from our buffer into the provided one 
 264 void wxStreamBuffer::GetFromBuffer(void *buffer
, size_t size
) 
 266     // don't get more bytes than left in the buffer 
 267     size_t left 
= GetBytesLeft(); 
 272     memcpy(buffer
, m_buffer_pos
, size
); 
 273     m_buffer_pos 
+= size
; 
 276 // copy the contents of the provided buffer into this one 
 277 void wxStreamBuffer::PutToBuffer(const void *buffer
, size_t size
) 
 279     size_t left 
= GetBytesLeft(); 
 285             // we can't realloc the buffer, so just copy what we can 
 290             // realloc the buffer to have enough space for the data 
 291             if ( m_buffer_pos 
+ size 
> m_buffer_end 
) 
 293                 size_t delta 
= m_buffer_pos 
- m_buffer_start
; 
 294                 size_t new_size 
= delta 
+ size
; 
 296                 char *startOld 
= m_buffer_start
; 
 297                 m_buffer_start 
= (char *)realloc(m_buffer_start
, new_size
); 
 298                 if ( !m_buffer_start 
) 
 300                     // don't leak memory if realloc() failed 
 301                     m_buffer_start 
= startOld
; 
 303                     // what else can we do? 
 307                 // adjust the pointers invalidated by realloc() 
 308                 m_buffer_pos 
= m_buffer_start 
+ delta
; 
 309                 m_buffer_end 
= m_buffer_start 
+ new_size
; 
 310             } // else: the buffer is big enough 
 314     memcpy(m_buffer_pos
, buffer
, size
); 
 315     m_buffer_pos 
+= size
; 
 318 void wxStreamBuffer::PutChar(char c
) 
 320     wxOutputStream 
*outStream 
= GetOutputStream(); 
 322     wxCHECK_RET( outStream
, _T("should have a stream in wxStreamBuffer") ); 
 324     // if we don't have buffer at all, just forward this call to the stream, 
 327         outStream
->OnSysWrite(&c
, sizeof(c
)); 
 331         // otherwise check we have enough space left 
 332         if ( !GetDataLeft() && !FlushBuffer() ) 
 335             SetError(wxSTREAM_WRITE_ERROR
); 
 339             PutToBuffer(&c
, sizeof(c
)); 
 340             m_stream
->m_lastcount 
= 1; 
 345 char wxStreamBuffer::Peek() 
 347     wxCHECK_MSG( m_stream 
&& HasBuffer(), 0, 
 348                  _T("should have the stream and the buffer in wxStreamBuffer") ); 
 350     if ( !GetDataLeft() ) 
 352         SetError(wxSTREAM_READ_ERROR
); 
 357     GetFromBuffer(&c
, sizeof(c
)); 
 363 char wxStreamBuffer::GetChar() 
 365     wxInputStream 
*inStream 
= GetInputStream(); 
 367     wxCHECK_MSG( inStream
, 0, _T("should have a stream in wxStreamBuffer") ); 
 372         inStream
->OnSysRead(&c
, sizeof(c
)); 
 376         if ( !GetDataLeft() ) 
 378             SetError(wxSTREAM_READ_ERROR
); 
 383             GetFromBuffer(&c
, sizeof(c
)); 
 384             m_stream
->m_lastcount 
= 1; 
 391 size_t wxStreamBuffer::Read(void *buffer
, size_t size
) 
 393     wxASSERT_MSG( buffer
, _T("Warning: Null pointer is about to be used") ); 
 395     /* Clear buffer first */ 
 396     memset(buffer
, 0x00, size
); 
 398     // lasterror is reset before all new IO calls 
 405         wxInputStream 
*inStream 
= GetInputStream(); 
 407         wxCHECK_MSG( inStream
, 0, _T("should have a stream in wxStreamBuffer") ); 
 409         readBytes 
= inStream
->OnSysRead(buffer
, size
); 
 411     else // we have a buffer, use it 
 413         size_t orig_size 
= size
; 
 417             size_t left 
= GetDataLeft(); 
 419             // if the requested number of bytes if greater than the buffer 
 420             // size, read data in chunks 
 423                 GetFromBuffer(buffer
, left
); 
 425                 buffer 
= (char *)buffer 
+ left
; 
 429                     SetError(wxSTREAM_EOF
); 
 433             else // otherwise just do it in one gulp 
 435                 GetFromBuffer(buffer
, size
); 
 440         readBytes 
= orig_size 
- size
; 
 444         m_stream
->m_lastcount 
= readBytes
; 
 449 // this should really be called "Copy()" 
 450 size_t wxStreamBuffer::Read(wxStreamBuffer 
*dbuf
) 
 452     wxCHECK_MSG( m_mode 
!= write
, 0, _T("can't read from this buffer") ); 
 454     char buf
[BUF_TEMP_SIZE
]; 
 460         nRead 
= Read(buf
, WXSIZEOF(buf
)); 
 463             nRead 
= dbuf
->Write(buf
, nRead
); 
 472 size_t wxStreamBuffer::Write(const void *buffer
, size_t size
) 
 474     wxASSERT_MSG( buffer
, _T("Warning: Null pointer is about to be send") ); 
 478         // lasterror is reset before all new IO calls 
 484     if ( !HasBuffer() && m_fixed 
) 
 486         wxOutputStream 
*outStream 
= GetOutputStream(); 
 488         wxCHECK_MSG( outStream
, 0, _T("should have a stream in wxStreamBuffer") ); 
 490         // no buffer, just forward the call to the stream 
 491         ret 
= outStream
->OnSysWrite(buffer
, size
); 
 493     else // we [may] have a buffer, use it 
 495         size_t orig_size 
= size
; 
 499             size_t left 
= GetBytesLeft(); 
 501             // if the buffer is too large to fit in the stream buffer, split 
 502             // it in smaller parts 
 504             // NB: If stream buffer isn't fixed (as for wxMemoryOutputStream), 
 505             //     we always go to the second case. 
 507             // FIXME: fine, but if it fails we should (re)try writing it by 
 508             //        chunks as this will (hopefully) always work (VZ) 
 510             if ( size 
> left 
&& m_fixed 
) 
 512                 PutToBuffer(buffer
, left
); 
 514                 buffer 
= (char *)buffer 
+ left
; 
 516                 if ( !FlushBuffer() ) 
 518                     SetError(wxSTREAM_WRITE_ERROR
); 
 523                 m_buffer_pos 
= m_buffer_start
; 
 525             else // we can do it in one gulp 
 527                 PutToBuffer(buffer
, size
); 
 532         ret 
= orig_size 
- size
; 
 537         // i am not entirely sure what we do this for 
 538         m_stream
->m_lastcount 
= ret
; 
 544 size_t wxStreamBuffer::Write(wxStreamBuffer 
*sbuf
) 
 546     wxCHECK_MSG( m_mode 
!= read
, 0, _T("can't write to this buffer") ); 
 547     wxCHECK_MSG( sbuf
->m_mode 
!= write
, 0, _T("can't read from that buffer") ); 
 549     char buf
[BUF_TEMP_SIZE
]; 
 555         size_t nRead 
= sbuf
->Read(buf
, WXSIZEOF(buf
)); 
 558             nWrite 
= Write(buf
, nRead
); 
 559             if ( nWrite 
< nRead 
) 
 561                 // put back data we couldn't copy 
 562                 wxInputStream 
*in_stream 
= (wxInputStream 
*)sbuf
->GetStream(); 
 564                 in_stream
->Ungetch(buf 
+ nWrite
, nRead 
- nWrite
); 
 574     while ( nWrite 
== WXSIZEOF(buf
) ); 
 579 wxFileOffset 
wxStreamBuffer::Seek(wxFileOffset pos
, wxSeekMode mode
) 
 581     wxFileOffset ret_off
, diff
; 
 583     wxFileOffset last_access 
= GetLastAccess(); 
 594                 diff 
= pos 
+ GetIntPosition(); 
 598                 diff 
= pos 
+ last_access
; 
 602                 wxFAIL_MSG( _T("invalid seek mode") ); 
 604                 return wxInvalidOffset
; 
 606         if (diff 
< 0 || diff 
> last_access
) 
 607             return wxInvalidOffset
; 
 608         size_t int_diff 
= wx_truncate_cast(size_t, diff
); 
 609         wxCHECK_MSG( (wxFileOffset
)int_diff 
== diff
, wxInvalidOffset
, wxT("huge file not supported") ); 
 610         SetIntPosition(int_diff
); 
 617             // We'll try to compute an internal position later ... 
 618             ret_off 
= m_stream
->OnSysSeek(pos
, wxFromStart
); 
 623             diff 
= pos 
+ GetIntPosition(); 
 625             if ( (diff 
> last_access
) || (diff 
< 0) ) 
 627                 // We must take into account the fact that we have read 
 628                 // something previously. 
 629                 ret_off 
= m_stream
->OnSysSeek(diff
-last_access
, wxFromCurrent
); 
 635                 size_t int_diff 
= wx_truncate_cast(size_t, diff
); 
 636                 wxCHECK_MSG( (wxFileOffset
)int_diff 
== diff
, wxInvalidOffset
, wxT("huge file not supported") ); 
 637                 SetIntPosition(int_diff
); 
 642             // Hard to compute: always seek to the requested position. 
 643             ret_off 
= m_stream
->OnSysSeek(pos
, wxFromEnd
); 
 648     return wxInvalidOffset
; 
 651 wxFileOffset 
wxStreamBuffer::Tell() const 
 655     // ask the stream for position if we have a real one 
 658         pos 
= m_stream
->OnSysTell(); 
 659         if ( pos 
== wxInvalidOffset 
) 
 660             return wxInvalidOffset
; 
 662     else // no associated stream 
 667     pos 
+= GetIntPosition(); 
 669     if ( m_mode 
== read 
&& m_flushable 
) 
 670         pos 
-= GetLastAccess(); 
 675 // ---------------------------------------------------------------------------- 
 677 // ---------------------------------------------------------------------------- 
 679 IMPLEMENT_ABSTRACT_CLASS(wxStreamBase
, wxObject
) 
 681 wxStreamBase::wxStreamBase() 
 683     m_lasterror 
= wxSTREAM_NO_ERROR
; 
 687 wxStreamBase::~wxStreamBase() 
 691 size_t wxStreamBase::GetSize() const 
 693     wxFileOffset length 
= GetLength(); 
 694     if ( length 
== (wxFileOffset
)wxInvalidOffset 
) 
 697     const size_t len 
= wx_truncate_cast(size_t, length
); 
 698     wxASSERT_MSG( len 
== length 
+ size_t(0), _T("large files not supported") ); 
 703 wxFileOffset 
wxStreamBase::OnSysSeek(wxFileOffset 
WXUNUSED(seek
), wxSeekMode 
WXUNUSED(mode
)) 
 705     return wxInvalidOffset
; 
 708 wxFileOffset 
wxStreamBase::OnSysTell() const 
 710     return wxInvalidOffset
; 
 713 // ---------------------------------------------------------------------------- 
 715 // ---------------------------------------------------------------------------- 
 717 IMPLEMENT_ABSTRACT_CLASS(wxInputStream
, wxStreamBase
) 
 719 wxInputStream::wxInputStream() 
 726 wxInputStream::~wxInputStream() 
 731 bool wxInputStream::CanRead() const 
 733     // we don't know if there is anything to read or not and by default we 
 734     // prefer to be optimistic and try to read data unless we know for sure 
 735     // there is no more of it 
 736     return m_lasterror 
!= wxSTREAM_EOF
; 
 739 bool wxInputStream::Eof() const 
 741     // the only way the base class can know we're at EOF is when we'd already 
 742     // tried to read beyond it in which case last error is set accordingly 
 743     return GetLastError() == wxSTREAM_EOF
; 
 746 char *wxInputStream::AllocSpaceWBack(size_t needed_size
) 
 748     // get number of bytes left from previous wback buffer 
 749     size_t toget 
= m_wbacksize 
- m_wbackcur
; 
 751     // allocate a buffer large enough to hold prev + new data 
 752     char *temp_b 
= (char *)malloc(needed_size 
+ toget
); 
 757     // copy previous data (and free old buffer) if needed 
 760         memmove(temp_b 
+ needed_size
, m_wback 
+ m_wbackcur
, toget
); 
 767     m_wbacksize 
= needed_size 
+ toget
; 
 772 size_t wxInputStream::GetWBack(void *buf
, size_t size
) 
 774     wxASSERT_MSG( buf
, _T("Warning: Null pointer is about to be used") ); 
 776     /* Clear buffer first */ 
 777     memset(buf
, 0x00, size
); 
 782     // how many bytes do we have in the buffer? 
 783     size_t toget 
= m_wbacksize 
- m_wbackcur
; 
 787         // we won't read everything 
 791     // copy the data from the cache 
 792     memcpy(buf
, m_wback 
+ m_wbackcur
, toget
); 
 795     if ( m_wbackcur 
== m_wbacksize 
) 
 797         // TODO: should we really free it here all the time? maybe keep it? 
 804     // return the number of bytes copied 
 808 size_t wxInputStream::Ungetch(const void *buf
, size_t bufsize
) 
 810     wxASSERT_MSG( buf
, _T("Warning: Null pointer is about to be used in Ungetch()") ); 
 812     if ( m_lasterror 
!= wxSTREAM_NO_ERROR 
&& m_lasterror 
!= wxSTREAM_EOF 
) 
 814         // can't operate on this stream until the error is cleared 
 818     char *ptrback 
= AllocSpaceWBack(bufsize
); 
 822     // Eof() shouldn't return true any longer 
 823     if ( m_lasterror 
== wxSTREAM_EOF 
) 
 824         m_lasterror 
= wxSTREAM_NO_ERROR
; 
 826     memcpy(ptrback
, buf
, bufsize
); 
 830 bool wxInputStream::Ungetch(char c
) 
 832     return Ungetch(&c
, sizeof(c
)) != 0; 
 835 int wxInputStream::GetC() 
 839     return LastRead() ? c 
: wxEOF
; 
 842 wxInputStream
& wxInputStream::Read(void *buf
, size_t size
) 
 844     wxASSERT_MSG( buf
, _T("Warning: Null pointer is about to be read") ); 
 846     char *p 
= (char *)buf
; 
 849     size_t read 
= GetWBack(buf
, size
); 
 858             // we read the requested amount of data 
 862         if ( p 
!= buf 
&& !CanRead() ) 
 864             // we have already read something and we would block in OnSysRead() 
 865             // now: don't do it but return immediately 
 869         read 
= OnSysRead(p
, size
); 
 872             // no more data available 
 880 char wxInputStream::Peek() 
 884     if (m_lasterror 
== wxSTREAM_NO_ERROR
) 
 893 wxInputStream
& wxInputStream::Read(wxOutputStream
& stream_out
) 
 895     size_t lastcount 
= 0; 
 896     char buf
[BUF_TEMP_SIZE
]; 
 900         size_t bytes_read 
= Read(buf
, WXSIZEOF(buf
)).LastRead(); 
 904         if ( stream_out
.Write(buf
, bytes_read
).LastWrite() != bytes_read 
) 
 907         lastcount 
+= bytes_read
; 
 910     m_lastcount 
= lastcount
; 
 915 wxFileOffset 
wxInputStream::SeekI(wxFileOffset pos
, wxSeekMode mode
) 
 917     // RR: This code is duplicated in wxBufferedInputStream. This is 
 918     // not really a good design, but buffered stream are different 
 919     // from all other in that they handle two stream-related objects, 
 920     // the stream buffer and parent stream. 
 922     // I don't know whether it should be put as well in wxFileInputStream::OnSysSeek 
 923     if (m_lasterror
==wxSTREAM_EOF
) 
 924         m_lasterror
=wxSTREAM_NO_ERROR
; 
 926     /* RR: A call to SeekI() will automatically invalidate any previous 
 927        call to Ungetch(), otherwise it would be possible to SeekI() to 
 928        one position, unread some bytes there, SeekI() to another position 
 929        and the data would be corrupted. 
 931        GRG: Could add code here to try to navigate within the wback 
 932        buffer if possible, but is it really needed? It would only work 
 933        when seeking in wxFromCurrent mode, else it would invalidate 
 938         wxLogDebug( wxT("Seeking in stream which has data written back to it.") ); 
 946     return OnSysSeek(pos
, mode
); 
 949 wxFileOffset 
wxInputStream::TellI() const 
 951     wxFileOffset pos 
= OnSysTell(); 
 953     if (pos 
!= wxInvalidOffset
) 
 954         pos 
-= (m_wbacksize 
- m_wbackcur
); 
 960 // ---------------------------------------------------------------------------- 
 962 // ---------------------------------------------------------------------------- 
 964 IMPLEMENT_ABSTRACT_CLASS(wxOutputStream
, wxStreamBase
) 
 966 wxOutputStream::wxOutputStream() 
 970 wxOutputStream::~wxOutputStream() 
 974 size_t wxOutputStream::OnSysWrite(const void * WXUNUSED(buffer
), 
 975                                   size_t WXUNUSED(bufsize
)) 
 980 void wxOutputStream::PutC(char c
) 
 982     Write(&c
, sizeof(c
)); 
 985 wxOutputStream
& wxOutputStream::Write(const void *buffer
, size_t size
) 
 987     m_lastcount 
= OnSysWrite(buffer
, size
); 
 991 wxOutputStream
& wxOutputStream::Write(wxInputStream
& stream_in
) 
 993     stream_in
.Read(*this); 
 997 wxFileOffset 
wxOutputStream::TellO() const 
1002 wxFileOffset 
wxOutputStream::SeekO(wxFileOffset pos
, wxSeekMode mode
) 
1004     return OnSysSeek(pos
, mode
); 
1007 void wxOutputStream::Sync() 
1012 // ---------------------------------------------------------------------------- 
1013 // wxCountingOutputStream 
1014 // ---------------------------------------------------------------------------- 
1016 IMPLEMENT_DYNAMIC_CLASS(wxCountingOutputStream
, wxOutputStream
) 
1018 wxCountingOutputStream::wxCountingOutputStream () 
1023 wxFileOffset 
wxCountingOutputStream::GetLength() const 
1028 size_t wxCountingOutputStream::OnSysWrite(const void *WXUNUSED(buffer
), 
1031     m_currentPos 
+= size
; 
1032     if (m_currentPos 
> m_lastcount
) 
1033         m_lastcount 
= m_currentPos
; 
1035     return m_currentPos
; 
1038 wxFileOffset 
wxCountingOutputStream::OnSysSeek(wxFileOffset pos
, wxSeekMode mode
) 
1040     ssize_t new_pos 
= wx_truncate_cast(ssize_t
, pos
); 
1045             wxCHECK_MSG( (wxFileOffset
)new_pos 
== pos
, wxInvalidOffset
, wxT("huge position not supported") ); 
1049             new_pos 
= m_lastcount 
+ new_pos
; 
1050             wxCHECK_MSG( (wxFileOffset
)new_pos 
== (wxFileOffset
)(m_lastcount 
+ pos
), wxInvalidOffset
, wxT("huge position not supported") ); 
1054             new_pos 
= m_currentPos 
+ new_pos
; 
1055             wxCHECK_MSG( (wxFileOffset
)new_pos 
== (wxFileOffset
)(m_currentPos 
+ pos
), wxInvalidOffset
, wxT("huge position not supported") ); 
1059             wxFAIL_MSG( _T("invalid seek mode") ); 
1060             return wxInvalidOffset
; 
1063     m_currentPos 
= new_pos
; 
1065     if (m_currentPos 
> m_lastcount
) 
1066         m_lastcount 
= m_currentPos
; 
1068     return m_currentPos
; 
1071 wxFileOffset 
wxCountingOutputStream::OnSysTell() const 
1073     return m_currentPos
; 
1076 // ---------------------------------------------------------------------------- 
1077 // wxFilterInputStream 
1078 // ---------------------------------------------------------------------------- 
1080 IMPLEMENT_ABSTRACT_CLASS(wxFilterInputStream
, wxInputStream
) 
1082 wxFilterInputStream::wxFilterInputStream() 
1083  :  m_parent_i_stream(NULL
), 
1088 wxFilterInputStream::wxFilterInputStream(wxInputStream
& stream
) 
1089  :  m_parent_i_stream(&stream
), 
1094 wxFilterInputStream::wxFilterInputStream(wxInputStream 
*stream
) 
1095  :  m_parent_i_stream(stream
), 
1100 wxFilterInputStream::~wxFilterInputStream() 
1103         delete m_parent_i_stream
; 
1106 // ---------------------------------------------------------------------------- 
1107 // wxFilterOutputStream 
1108 // ---------------------------------------------------------------------------- 
1110 IMPLEMENT_ABSTRACT_CLASS(wxFilterOutputStream
, wxOutputStream
) 
1112 wxFilterOutputStream::wxFilterOutputStream() 
1113  :  m_parent_o_stream(NULL
), 
1118 wxFilterOutputStream::wxFilterOutputStream(wxOutputStream
& stream
) 
1119  :  m_parent_o_stream(&stream
), 
1124 wxFilterOutputStream::wxFilterOutputStream(wxOutputStream 
*stream
) 
1125  :  m_parent_o_stream(stream
), 
1130 bool wxFilterOutputStream::Close() 
1132     if (m_parent_o_stream 
&& m_owns
) 
1133         return m_parent_o_stream
->Close(); 
1138 wxFilterOutputStream::~wxFilterOutputStream() 
1141         delete m_parent_o_stream
; 
1144 // ---------------------------------------------------------------------------- 
1145 // wxFilterClassFactoryBase 
1146 // ---------------------------------------------------------------------------- 
1148 IMPLEMENT_ABSTRACT_CLASS(wxFilterClassFactoryBase
, wxObject
) 
1150 wxString 
wxFilterClassFactoryBase::PopExtension(const wxString
& location
) const 
1152     return location
.substr(0, FindExtension(location
)); 
1155 wxString::size_type 
wxFilterClassFactoryBase::FindExtension( 
1156         const wxString
& location
) const 
1158     for (const wxChar 
*const *p 
= GetProtocols(wxSTREAM_FILEEXT
); *p
; p
++) 
1160         if ( location
.EndsWith(*p
) ) 
1161             return location
.length() - wxStrlen(*p
); 
1164     return wxString::npos
; 
1167 bool wxFilterClassFactoryBase::CanHandle(const wxString
& protocol
, 
1168                                          wxStreamProtocolType type
) const 
1170     if (type 
== wxSTREAM_FILEEXT
) 
1171         return FindExtension(protocol
) != wxString::npos
; 
1173         for (const wxChar 
*const *p 
= GetProtocols(type
); *p
; p
++) 
1180 // ---------------------------------------------------------------------------- 
1181 // wxFilterClassFactory 
1182 // ---------------------------------------------------------------------------- 
1184 IMPLEMENT_ABSTRACT_CLASS(wxFilterClassFactory
, wxFilterClassFactoryBase
) 
1186 wxFilterClassFactory 
*wxFilterClassFactory::sm_first 
= NULL
; 
1188 void wxFilterClassFactory::Remove() 
1192         wxFilterClassFactory 
**pp 
= &sm_first
; 
1195             pp 
= &(*pp
)->m_next
; 
1203 // ---------------------------------------------------------------------------- 
1204 // wxBufferedInputStream 
1205 // ---------------------------------------------------------------------------- 
1210 // helper function used for initializing the buffer used by 
1211 // wxBufferedInput/OutputStream: it simply returns the provided buffer if it's 
1212 // not NULL or creates a buffer of the given size otherwise 
1213 template <typename T
> 
1215 CreateBufferIfNeeded(T
& stream
, wxStreamBuffer 
*buffer
, size_t bufsize 
= 1024) 
1217     return buffer 
? buffer 
: new wxStreamBuffer(stream
, bufsize
); 
1220 } // anonymous namespace 
1222 wxBufferedInputStream::wxBufferedInputStream(wxInputStream
& stream
, 
1223                                              wxStreamBuffer 
*buffer
) 
1224                      : wxFilterInputStream(stream
) 
1226     m_i_streambuf 
= CreateBufferIfNeeded(*this, buffer
); 
1229 wxBufferedInputStream::wxBufferedInputStream(wxInputStream
& stream
, 
1231                      : wxFilterInputStream(stream
) 
1233     m_i_streambuf 
= CreateBufferIfNeeded(*this, NULL
, bufsize
); 
1236 wxBufferedInputStream::~wxBufferedInputStream() 
1238     m_parent_i_stream
->SeekI(-(wxFileOffset
)m_i_streambuf
->GetBytesLeft(), 
1241     delete m_i_streambuf
; 
1244 char wxBufferedInputStream::Peek() 
1246     return m_i_streambuf
->Peek(); 
1249 wxInputStream
& wxBufferedInputStream::Read(void *buf
, size_t size
) 
1251     // reset the error flag 
1254     // first read from the already cached data 
1255     m_lastcount 
= GetWBack(buf
, size
); 
1257     // do we have to read anything more? 
1258     if ( m_lastcount 
< size 
) 
1260         size 
-= m_lastcount
; 
1261         buf 
= (char *)buf 
+ m_lastcount
; 
1263         // the call to wxStreamBuffer::Read() below may reset our m_lastcount 
1264         // (but it also may not do it if the buffer is associated to another 
1265         // existing stream and wasn't created by us), so save it 
1266         size_t countOld 
= m_lastcount
; 
1268         // the new count of the bytes read is the count of bytes read this time 
1269         m_lastcount 
= m_i_streambuf
->Read(buf
, size
); 
1271         // plus those we had read before 
1272         m_lastcount 
+= countOld
; 
1278 wxFileOffset 
wxBufferedInputStream::SeekI(wxFileOffset pos
, wxSeekMode mode
) 
1280     // RR: Look at wxInputStream for comments. 
1282     if (m_lasterror
==wxSTREAM_EOF
) 
1287         wxLogDebug( wxT("Seeking in stream which has data written back to it.") ); 
1295     return m_i_streambuf
->Seek(pos
, mode
); 
1298 wxFileOffset 
wxBufferedInputStream::TellI() const 
1300     wxFileOffset pos 
= m_i_streambuf
->Tell(); 
1302     if (pos 
!= wxInvalidOffset
) 
1303         pos 
-= (m_wbacksize 
- m_wbackcur
); 
1308 size_t wxBufferedInputStream::OnSysRead(void *buffer
, size_t bufsize
) 
1310     return m_parent_i_stream
->Read(buffer
, bufsize
).LastRead(); 
1313 wxFileOffset 
wxBufferedInputStream::OnSysSeek(wxFileOffset seek
, wxSeekMode mode
) 
1315     return m_parent_i_stream
->SeekI(seek
, mode
); 
1318 wxFileOffset 
wxBufferedInputStream::OnSysTell() const 
1320     return m_parent_i_stream
->TellI(); 
1323 void wxBufferedInputStream::SetInputStreamBuffer(wxStreamBuffer 
*buffer
) 
1325     wxCHECK_RET( buffer
, _T("wxBufferedInputStream needs buffer") ); 
1327     delete m_i_streambuf
; 
1328     m_i_streambuf 
= buffer
; 
1331 // ---------------------------------------------------------------------------- 
1332 // wxBufferedOutputStream 
1333 // ---------------------------------------------------------------------------- 
1335 wxBufferedOutputStream::wxBufferedOutputStream(wxOutputStream
& stream
, 
1336                                                wxStreamBuffer 
*buffer
) 
1337                       : wxFilterOutputStream(stream
) 
1339     m_o_streambuf 
= CreateBufferIfNeeded(*this, buffer
); 
1342 wxBufferedOutputStream::wxBufferedOutputStream(wxOutputStream
& stream
, 
1344                       : wxFilterOutputStream(stream
) 
1346     m_o_streambuf 
= CreateBufferIfNeeded(*this, NULL
, bufsize
); 
1349 wxBufferedOutputStream::~wxBufferedOutputStream() 
1352     delete m_o_streambuf
; 
1355 bool wxBufferedOutputStream::Close() 
1362 wxOutputStream
& wxBufferedOutputStream::Write(const void *buffer
, size_t size
) 
1365     m_o_streambuf
->Write(buffer
, size
); 
1369 wxFileOffset 
wxBufferedOutputStream::SeekO(wxFileOffset pos
, wxSeekMode mode
) 
1372     return m_o_streambuf
->Seek(pos
, mode
); 
1375 wxFileOffset 
wxBufferedOutputStream::TellO() const 
1377     return m_o_streambuf
->Tell(); 
1380 void wxBufferedOutputStream::Sync() 
1382     m_o_streambuf
->FlushBuffer(); 
1383     m_parent_o_stream
->Sync(); 
1386 size_t wxBufferedOutputStream::OnSysWrite(const void *buffer
, size_t bufsize
) 
1388     return m_parent_o_stream
->Write(buffer
, bufsize
).LastWrite(); 
1391 wxFileOffset 
wxBufferedOutputStream::OnSysSeek(wxFileOffset seek
, wxSeekMode mode
) 
1393     return m_parent_o_stream
->SeekO(seek
, mode
); 
1396 wxFileOffset 
wxBufferedOutputStream::OnSysTell() const 
1398     return m_parent_o_stream
->TellO(); 
1401 wxFileOffset 
wxBufferedOutputStream::GetLength() const 
1403    return m_parent_o_stream
->GetLength() + m_o_streambuf
->GetIntPosition(); 
1406 void wxBufferedOutputStream::SetOutputStreamBuffer(wxStreamBuffer 
*buffer
) 
1408     wxCHECK_RET( buffer
, _T("wxBufferedOutputStream needs buffer") ); 
1410     delete m_o_streambuf
; 
1411     m_o_streambuf 
= buffer
; 
1414 // ---------------------------------------------------------------------------- 
1415 // Some IOManip function 
1416 // ---------------------------------------------------------------------------- 
1418 wxOutputStream
& wxEndL(wxOutputStream
& stream
) 
1420     static const wxChar 
*eol 
= wxTextFile::GetEOL(); 
1422     return stream
.Write(eol
, wxStrlen(eol
)); 
1425 #endif // wxUSE_STREAMS