]> git.saurik.com Git - wxWidgets.git/blame - src/common/ffile.cpp
fixed hash collisions in ZIP files (patch 1082026)
[wxWidgets.git] / src / common / ffile.cpp
CommitLineData
a1b82138
VZ
1/////////////////////////////////////////////////////////////////////////////
2// Name: ffile.cpp
90e2cbf7 3// Purpose: wxFFile encapsulates "FILE *" IO stream
a1b82138
VZ
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 14.07.99
7// RCS-ID: $Id$
8// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
65571936 9// Licence: wxWindows licence
a1b82138
VZ
10/////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
14f355c2 20#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
a1b82138
VZ
21 #pragma implementation "ffile.h"
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#ifdef __BORLANDC__
28 #pragma hdrstop
29#endif
30
1e6feb95 31#if wxUSE_FFILE
a1b82138
VZ
32
33#ifndef WX_PRECOMP
0efe5ba7
VZ
34 #include "wx/intl.h"
35 #include "wx/log.h"
a1b82138
VZ
36#endif
37
38#include "wx/ffile.h"
39
40// ============================================================================
41// implementation
42// ============================================================================
43
70a7bd90
VZ
44// ----------------------------------------------------------------------------
45// seek and tell with large file support if available
46// ----------------------------------------------------------------------------
47
48#ifdef HAVE_FSEEKO
49# define wxFseek fseeko
50# define wxFtell ftello
51#else
52# define wxFseek fseek
53# define wxFtell ftell
54#endif
55
a1b82138
VZ
56// ----------------------------------------------------------------------------
57// opening the file
58// ----------------------------------------------------------------------------
59
90e2cbf7 60wxFFile::wxFFile(const wxChar *filename, const wxChar *mode)
a1b82138
VZ
61{
62 Detach();
63
64 (void)Open(filename, mode);
65}
66
90e2cbf7 67bool wxFFile::Open(const wxChar *filename, const wxChar *mode)
a1b82138 68{
223d09f6 69 wxASSERT_MSG( !m_fp, wxT("should close or detach the old file first") );
a1b82138 70
90e2cbf7 71 m_fp = wxFopen(filename, mode);
a1b82138
VZ
72
73 if ( !m_fp )
74 {
75 wxLogSysError(_("can't open file '%s'"), filename);
76
a62848fd 77 return false;
a1b82138
VZ
78 }
79
80 m_name = filename;
81
a62848fd 82 return true;
a1b82138
VZ
83}
84
85bool wxFFile::Close()
86{
87 if ( IsOpened() )
88 {
d2e1ef19 89 if ( fclose(m_fp) != 0 )
a1b82138
VZ
90 {
91 wxLogSysError(_("can't close file '%s'"), m_name.c_str());
92
a62848fd 93 return false;
a1b82138
VZ
94 }
95
96 Detach();
97 }
98
a62848fd 99 return true;
a1b82138
VZ
100}
101
102// ----------------------------------------------------------------------------
103// read/write
104// ----------------------------------------------------------------------------
105
3e15dde3 106bool wxFFile::ReadAll(wxString *str, wxMBConv& conv)
a1b82138 107{
a62848fd
WS
108 wxCHECK_MSG( str, false, wxT("invalid parameter") );
109 wxCHECK_MSG( IsOpened(), false, wxT("can't read from closed file") );
11846af3
WS
110 wxCHECK_MSG( Length() >= 0, false, wxT("invalid length") );
111 size_t length = (size_t)Length();
c3816108 112 wxCHECK_MSG( (wxFileOffset)length == Length(), false, wxT("huge file not supported") );
a1b82138
VZ
113
114 clearerr(m_fp);
115
3e15dde3
VZ
116 const size_t fileLen = Length();
117 wxCharBuffer buf(fileLen + 1);
118 if ( (fread(buf.data(), sizeof(char), fileLen, m_fp) < fileLen) || Error() )
a1b82138 119 {
3e15dde3 120 wxLogSysError(_("Read error on file '%s'"), m_name.c_str());
a1b82138 121
3e15dde3 122 return false;
a1b82138
VZ
123 }
124
3e15dde3
VZ
125 buf.data()[fileLen] = 0;
126 *str = wxString(buf, conv);
127
a62848fd 128 return true;
a1b82138
VZ
129}
130
131size_t wxFFile::Read(void *pBuf, size_t nCount)
132{
223d09f6
KB
133 wxCHECK_MSG( pBuf, FALSE, wxT("invalid parameter") );
134 wxCHECK_MSG( IsOpened(), FALSE, wxT("can't read from closed file") );
a1b82138
VZ
135
136 size_t nRead = fread(pBuf, 1, nCount, m_fp);
137 if ( (nRead < nCount) && Error() )
138 {
139 wxLogSysError(_("Read error on file '%s'"), m_name.c_str());
140 }
141
142 return nRead;
143}
144
145size_t wxFFile::Write(const void *pBuf, size_t nCount)
146{
223d09f6
KB
147 wxCHECK_MSG( pBuf, FALSE, wxT("invalid parameter") );
148 wxCHECK_MSG( IsOpened(), FALSE, wxT("can't write to closed file") );
a1b82138
VZ
149
150 size_t nWritten = fwrite(pBuf, 1, nCount, m_fp);
151 if ( nWritten < nCount )
152 {
153 wxLogSysError(_("Write error on file '%s'"), m_name.c_str());
154 }
155
156 return nWritten;
157}
158
159bool wxFFile::Flush()
160{
161 if ( IsOpened() )
162 {
5b96c684
MB
163 // fflush returns non-zero on error
164 //
165 if ( fflush(m_fp) )
a1b82138
VZ
166 {
167 wxLogSysError(_("failed to flush the file '%s'"), m_name.c_str());
168
a62848fd 169 return false;
a1b82138
VZ
170 }
171 }
172
a62848fd 173 return true;
a1b82138
VZ
174}
175
176// ----------------------------------------------------------------------------
177// seeking
178// ----------------------------------------------------------------------------
179
70a7bd90 180bool wxFFile::Seek(wxFileOffset ofs, wxSeekMode mode)
a1b82138 181{
a62848fd 182 wxCHECK_MSG( IsOpened(), false, wxT("can't seek on closed file") );
a1b82138
VZ
183
184 int origin;
185 switch ( mode )
186 {
187 default:
223d09f6 188 wxFAIL_MSG(wxT("unknown seek mode"));
a1b82138
VZ
189 // still fall through
190
191 case wxFromStart:
192 origin = SEEK_SET;
193 break;
194
195 case wxFromCurrent:
196 origin = SEEK_CUR;
197 break;
198
199 case wxFromEnd:
200 origin = SEEK_END;
201 break;
202 }
203
11846af3 204#ifndef HAVE_FSEEKO
70a7bd90
VZ
205 if ((long)ofs != ofs)
206 {
207 wxLogError(_("Seek error on file '%s' (large files not supported by stdio)"), m_name.c_str());
208
209 return false;
210 }
70a7bd90 211
11846af3
WS
212 if ( wxFseek(m_fp, (long)ofs, origin) != 0 )
213#else
70a7bd90 214 if ( wxFseek(m_fp, ofs, origin) != 0 )
11846af3 215#endif
a1b82138
VZ
216 {
217 wxLogSysError(_("Seek error on file '%s'"), m_name.c_str());
218
a62848fd 219 return false;
a1b82138
VZ
220 }
221
a62848fd 222 return true;
a1b82138
VZ
223}
224
70a7bd90 225wxFileOffset wxFFile::Tell() const
a1b82138 226{
70a7bd90 227 wxCHECK_MSG( IsOpened(), wxInvalidOffset,
de2ce07c
VZ
228 _T("wxFFile::Tell(): file is closed!") );
229
70a7bd90
VZ
230 wxFileOffset rc = wxFtell(m_fp);
231 if ( rc == wxInvalidOffset )
a1b82138
VZ
232 {
233 wxLogSysError(_("Can't find current position in file '%s'"),
234 m_name.c_str());
235 }
236
70a7bd90 237 return rc;
a1b82138
VZ
238}
239
70a7bd90 240wxFileOffset wxFFile::Length() const
a1b82138 241{
70a7bd90 242 wxCHECK_MSG( IsOpened(), wxInvalidOffset,
de2ce07c
VZ
243 _T("wxFFile::Length(): file is closed!") );
244
a1b82138
VZ
245 wxFFile& self = *(wxFFile *)this; // const_cast
246
70a7bd90
VZ
247 wxFileOffset posOld = Tell();
248 if ( posOld != wxInvalidOffset )
a1b82138
VZ
249 {
250 if ( self.SeekEnd() )
251 {
70a7bd90 252 wxFileOffset len = Tell();
a1b82138
VZ
253
254 (void)self.Seek(posOld);
255
256 return len;
257 }
258 }
259
11846af3 260 return wxInvalidOffset;
a1b82138
VZ
261}
262
1e6feb95 263#endif // wxUSE_FFILE