///////////////////////////////////////////////////////////////////////////// // Name: src/common/ffile.cpp // Purpose: wxFFile encapsulates "FILE *" IO stream // Author: Vadim Zeitlin // Modified by: // Created: 14.07.99 // Copyright: (c) 1998 Vadim Zeitlin // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// // ============================================================================ // declarations // ============================================================================ // ---------------------------------------------------------------------------- // headers // ---------------------------------------------------------------------------- // For compilers that support precompilation, includes "wx.h". #include "wx/wxprec.h" #ifdef __BORLANDC__ #pragma hdrstop #endif #if wxUSE_FFILE #ifndef WX_PRECOMP #include "wx/intl.h" #include "wx/log.h" #include "wx/crt.h" #endif #ifdef __WINDOWS__ #include "wx/msw/mslu.h" #endif #include "wx/ffile.h" // ============================================================================ // implementation // ============================================================================ // ---------------------------------------------------------------------------- // opening the file // ---------------------------------------------------------------------------- wxFFile::wxFFile(const wxString& filename, const wxString& mode) { m_fp = NULL; (void)Open(filename, mode); } bool wxFFile::Open(const wxString& filename, const wxString& mode) { wxASSERT_MSG( !m_fp, wxT("should close or detach the old file first") ); FILE* const fp = wxFopen(filename, mode); if ( !fp ) { wxLogSysError(_("can't open file '%s'"), filename); return false; } Attach(fp, filename); return true; } bool wxFFile::Close() { if ( IsOpened() ) { if ( fclose(m_fp) != 0 ) { wxLogSysError(_("can't close file '%s'"), m_name.c_str()); return false; } m_fp = NULL; } return true; } // ---------------------------------------------------------------------------- // read/write // ---------------------------------------------------------------------------- bool wxFFile::ReadAll(wxString *str, const wxMBConv& conv) { wxCHECK_MSG( str, false, wxT("invalid parameter") ); wxCHECK_MSG( IsOpened(), false, wxT("can't read from closed file") ); wxCHECK_MSG( Length() >= 0, false, wxT("invalid length") ); size_t length = wx_truncate_cast(size_t, Length()); wxCHECK_MSG( (wxFileOffset)length == Length(), false, wxT("huge file not supported") ); clearerr(m_fp); wxCharBuffer buf(length); // note that real length may be less than file length for text files with DOS EOLs // ('\r's get dropped by CRT when reading which means that we have // realLen = fileLen - numOfLinesInTheFile) length = fread(buf.data(), 1, length, m_fp); if ( Error() ) { wxLogSysError(_("Read error on file '%s'"), m_name.c_str()); return false; } buf.data()[length] = 0; wxString strTmp(buf, conv); str->swap(strTmp); return true; } size_t wxFFile::Read(void *pBuf, size_t nCount) { if ( !nCount ) return 0; wxCHECK_MSG( pBuf, 0, wxT("invalid parameter") ); wxCHECK_MSG( IsOpened(), 0, wxT("can't read from closed file") ); size_t nRead = fread(pBuf, 1, nCount, m_fp); if ( (nRead < nCount) && Error() ) { wxLogSysError(_("Read error on file '%s'"), m_name.c_str()); } return nRead; } size_t wxFFile::Write(const void *pBuf, size_t nCount) { if ( !nCount ) return 0; wxCHECK_MSG( pBuf, 0, wxT("invalid parameter") ); wxCHECK_MSG( IsOpened(), 0, wxT("can't write to closed file") ); size_t nWritten = fwrite(pBuf, 1, nCount, m_fp); if ( nWritten < nCount ) { wxLogSysError(_("Write error on file '%s'"), m_name.c_str()); } return nWritten; } bool wxFFile::Write(const wxString& s, const wxMBConv& conv) { const wxWX2MBbuf buf = s.mb_str(conv); if ( !buf ) return false; const size_t size = strlen(buf); // FIXME: use buf.length() when available return Write(buf, size) == size; } bool wxFFile::Flush() { if ( IsOpened() ) { if ( fflush(m_fp) != 0 ) { wxLogSysError(_("failed to flush the file '%s'"), m_name.c_str()); return false; } } return true; } // ---------------------------------------------------------------------------- // seeking // ---------------------------------------------------------------------------- bool wxFFile::Seek(wxFileOffset ofs, wxSeekMode mode) { wxCHECK_MSG( IsOpened(), false, wxT("can't seek on closed file") ); int origin; switch ( mode ) { default: wxFAIL_MSG(wxT("unknown seek mode")); // still fall through case wxFromStart: origin = SEEK_SET; break; case wxFromCurrent: origin = SEEK_CUR; break; case wxFromEnd: origin = SEEK_END; break; } #ifndef wxHAS_LARGE_FFILES if ((long)ofs != ofs) { wxLogError(_("Seek error on file '%s' (large files not supported by stdio)"), m_name.c_str()); return false; } if ( wxFseek(m_fp, (long)ofs, origin) != 0 ) #else if ( wxFseek(m_fp, ofs, origin) != 0 ) #endif { wxLogSysError(_("Seek error on file '%s'"), m_name.c_str()); return false; } return true; } wxFileOffset wxFFile::Tell() const { wxCHECK_MSG( IsOpened(), wxInvalidOffset, wxT("wxFFile::Tell(): file is closed!") ); wxFileOffset rc = wxFtell(m_fp); if ( rc == wxInvalidOffset ) { wxLogSysError(_("Can't find current position in file '%s'"), m_name.c_str()); } return rc; } wxFileOffset wxFFile::Length() const { wxCHECK_MSG( IsOpened(), wxInvalidOffset, wxT("wxFFile::Length(): file is closed!") ); wxFFile& self = *const_cast(this); wxFileOffset posOld = Tell(); if ( posOld != wxInvalidOffset ) { if ( self.SeekEnd() ) { wxFileOffset len = Tell(); (void)self.Seek(posOld); return len; } } return wxInvalidOffset; } #endif // wxUSE_FFILE