]>
git.saurik.com Git - wxWidgets.git/blob - include/wx/wxcrtvararg.h
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: wx/wxcrtvararg.h
3 // Purpose: Type-safe ANSI and Unicode builds compatible wrappers for
4 // printf(), scanf() and related CRT functions
5 // Author: Joel Farley, Ove Kåven
6 // Modified by: Vadim Zeitlin, Robert Roebling, Ron Lee
9 // Copyright: (c) 2007 REA Elektronik GmbH
10 // Licence: wxWindows licence
11 ///////////////////////////////////////////////////////////////////////////////
13 #ifndef _WX_WXCRTVARARG_H_
14 #define _WX_WXCRTVARARG_H_
16 // NB: User code should include wx/crt.h instead of including this
20 #include "wx/strvararg.h"
22 #include "wx/string.h"
24 // ----------------------------------------------------------------------------
25 // CRT functions aliases
26 // ----------------------------------------------------------------------------
28 /* Required for wxPrintf() etc */
31 /* printf() family saga */
34 For some systems [v]snprintf() exists in the system libraries but not in the
35 headers, so we need to declare it ourselves to be able to use it.
37 #if defined(HAVE_VSNPRINTF) && !defined(HAVE_VSNPRINTF_DECL)
43 int vsnprintf(char *str
, size_t size
, const char *format
, va_list ap
);
44 #endif /* !HAVE_VSNPRINTF_DECL */
46 #if defined(HAVE_SNPRINTF) && !defined(HAVE_SNPRINTF_DECL)
52 int snprintf(char *str
, size_t size
, const char *format
, ...);
53 #endif /* !HAVE_SNPRINTF_DECL */
55 /* Wrapper for vsnprintf if it's 3rd parameter is non-const. Note: the
56 * same isn't done for snprintf below, the builtin wxSnprintf_ is used
57 * instead since it's already a simple wrapper */
58 #if defined __cplusplus && defined HAVE_BROKEN_VSNPRINTF_DECL
59 inline int wx_fixed_vsnprintf(char *str
, size_t size
, const char *format
, va_list ap
)
61 return vsnprintf(str
, size
, (char*)format
, ap
);
66 MinGW MSVCRT has non-standard vswprintf() (for MSVC compatibility
67 presumably) and normally _vsnwprintf() is used instead
69 #if defined(HAVE_VSWPRINTF) && defined(__MINGW32__)
73 #if defined(__WATCOMC__)
74 #define HAVE_VSWPRINTF 1
77 #if wxUSE_PRINTF_POS_PARAMS
79 The systems where vsnprintf() supports positional parameters should
80 define the HAVE_UNIX98_PRINTF symbol.
82 On systems which don't (e.g. Windows) we are forced to use
83 our wxVsnprintf() implementation.
85 #if defined(HAVE_UNIX98_PRINTF)
87 #define wxCRT_VsnprintfW vswprintf
89 #ifdef HAVE_BROKEN_VSNPRINTF_DECL
90 #define wxCRT_VsnprintfA wx_fixed_vsnprintf
92 #define wxCRT_VsnprintfA vsnprintf
94 #else /* !HAVE_UNIX98_PRINTF */
96 The only compiler with positional parameters support under Windows
97 is VC++ 8.0 which provides a new xxprintf_p() functions family.
98 The 2003 PSDK includes a slightly earlier version of VC8 than the
99 main release and does not have the printf_p functions.
101 #if defined _MSC_FULL_VER && _MSC_FULL_VER >= 140050727 && !defined __WXWINCE__
102 #define wxCRT_VsnprintfA _vsprintf_p
103 #define wxCRT_VsnprintfW _vswprintf_p
105 #endif /* HAVE_UNIX98_PRINTF/!HAVE_UNIX98_PRINTF */
106 #else /* !wxUSE_PRINTF_POS_PARAMS */
108 We always want to define safe snprintf() function to be used instead of
109 sprintf(). Some compilers already have it (or rather vsnprintf() which
110 we really need...), otherwise we implement it using our own printf()
113 We define function with a trailing underscore here because the real one
114 is a wrapper around it as explained below
117 #if defined(__VISUALC__) || \
118 (defined(__BORLANDC__) && __BORLANDC__ >= 0x540)
119 #define wxCRT_VsnprintfA _vsnprintf
120 #define wxCRT_VsnprintfW _vsnwprintf
122 #if defined(HAVE__VSNWPRINTF)
123 #define wxCRT_VsnprintfW _vsnwprintf
124 #elif defined(HAVE_VSWPRINTF)
125 #define wxCRT_VsnprintfW vswprintf
126 #elif defined(__WATCOMC__)
127 #define wxCRT_VsnprintfW _vsnwprintf
131 All versions of CodeWarrior supported by wxWidgets apparently
132 have both snprintf() and vsnprintf()
134 #if defined(HAVE_VSNPRINTF) \
135 || defined(__MWERKS__) || defined(__WATCOMC__)
136 #ifdef HAVE_BROKEN_VSNPRINTF_DECL
137 #define wxCRT_VsnprintfA wx_fixed_vsnprintf
139 #define wxCRT_VsnprintfA vsnprintf
143 #endif /* wxUSE_PRINTF_POS_PARAMS/!wxUSE_PRINTF_POS_PARAMS */
145 #ifndef wxCRT_VsnprintfW
146 /* no (suitable) vsnprintf(), cook our own */
148 wxCRT_VsnprintfW(wchar_t *buf
, size_t len
, const wchar_t *format
, va_list argptr
);
149 #define wxUSE_WXVSNPRINTFW 1
151 #define wxUSE_WXVSNPRINTFW 0
154 #ifndef wxCRT_VsnprintfA
155 /* no (suitable) vsnprintf(), cook our own */
157 wxCRT_VsnprintfA(char *buf
, size_t len
, const char *format
, va_list argptr
);
158 #define wxUSE_WXVSNPRINTFA 1
160 #define wxUSE_WXVSNPRINTFA 0
163 // for wxString code, define wxUSE_WXVSNPRINTF to indicate that wx
164 // implementation is used no matter what (in UTF-8 build, either *A or *W
165 // version may be called):
167 #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFA
168 #elif wxUSE_UNICODE_WCHAR
169 #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFW
170 #elif wxUSE_UTF8_LOCALE_ONLY
171 #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFA
172 #else // UTF-8 under any locale
173 #define wxUSE_WXVSNPRINTF (wxUSE_WXVSNPRINTFA && wxUSE_WXVSNPRINTFW)
176 #define wxCRT_FprintfA fprintf
177 #define wxCRT_PrintfA printf
178 #define wxCRT_VfprintfA vfprintf
179 #define wxCRT_VprintfA vprintf
180 #define wxCRT_VsprintfA vsprintf
183 In Unicode mode we need to have all standard functions such as wprintf() and
184 so on but not all systems have them so use our own implementations in this
187 #if wxUSE_UNICODE && !defined(wxHAVE_TCHAR_SUPPORT) && !defined(HAVE_WPRINTF)
188 #define wxNEED_WPRINTF
192 #if defined(wxNEED_WPRINTF)
194 we need to implement all wide character printf functions either because
195 we don't have them at all or because they don't have the semantics we
198 int wxCRT_PrintfW( const wchar_t *format
, ... );
199 int wxCRT_FprintfW( FILE *stream
, const wchar_t *format
, ... );
200 int wxCRT_VfprintfW( FILE *stream
, const wchar_t *format
, va_list ap
);
201 int wxCRT_VprintfW( const wchar_t *format
, va_list ap
);
202 int wxCRT_VsprintfW( wchar_t *str
, const wchar_t *format
, va_list ap
);
203 #else /* !wxNEED_WPRINTF */
204 #define wxCRT_FprintfW fwprintf
205 #define wxCRT_PrintfW wprintf
206 #define wxCRT_VfprintfW vfwprintf
207 #define wxCRT_VprintfW vwprintf
209 #if defined(__WINDOWS__) && !defined(HAVE_VSWPRINTF)
210 // only non-standard vswprintf() without buffer size argument can be used here
211 #define wxCRT_VsprintfW vswprintf
213 #endif /* wxNEED_WPRINTF */
216 /* Required for wxScanf() etc. */
217 #define wxCRT_ScanfA scanf
218 #define wxCRT_SscanfA sscanf
219 #define wxCRT_FscanfA fscanf
220 #define wxCRT_VsscanfA vsscanf
222 #if defined(wxNEED_WPRINTF)
223 int wxCRT_ScanfW(const wchar_t *format
, ...);
224 int wxCRT_SscanfW(const wchar_t *str
, const wchar_t *format
, ...);
225 int wxCRT_FscanfW(FILE *stream
, const wchar_t *format
, ...);
226 int wxCRT_VsscanfW(const wchar_t *str
, const wchar_t *format
, va_list ap
);
229 #if (__DECCXX_VER >= 70100000) && !defined(__STD_CFRONT) && !defined( __NONAMESPACE_STD )
230 # define wxCRT_ScanfW std::wscanf
231 # define wxCRT_SscanfW std::swscanf
232 # define wxCRT_FscanfW std::fwscanf
233 # define wxCRT_VsscanfW std::vswscanf
235 #define wxCRT_ScanfW wscanf
236 #define wxCRT_SscanfW swscanf
237 #define wxCRT_FscanfW fwscanf
238 #define wxCRT_VsscanfW vswscanf
241 #define wxCRT_ScanfW wscanf
242 #define wxCRT_SscanfW swscanf
243 #define wxCRT_FscanfW fwscanf
244 #define wxCRT_VsscanfW vswscanf
248 // ----------------------------------------------------------------------------
249 // user-friendly wrappers to CRT functions
250 // ----------------------------------------------------------------------------
253 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
254 #define wxPrintf wxPrintf_Impl
255 #define wxFprintf wxFprintf_Impl
256 #define wxSprintf wxSprintf_Impl
257 #define wxSnprintf wxSnprintf_Impl
260 // FIXME-UTF8: remove this
262 #define wxCRT_PrintfNative wxCRT_PrintfW
263 #define wxCRT_FprintfNative wxCRT_FprintfW
265 #define wxCRT_PrintfNative wxCRT_PrintfA
266 #define wxCRT_FprintfNative wxCRT_FprintfA
269 WX_DEFINE_VARARG_FUNC(int, wxPrintf
, 1, (const wxFormatString
&),
270 wxCRT_PrintfNative
, wxCRT_PrintfA
)
271 WX_DEFINE_VARARG_FUNC(int, wxFprintf
, 2, (FILE*, const wxFormatString
&),
272 wxCRT_FprintfNative
, wxCRT_FprintfA
)
274 // va_list versions of printf functions simply forward to the respective
275 // CRT function; note that they assume that va_list was created using
276 // wxArgNormalizer<T>!
277 #if wxUSE_UNICODE_UTF8
278 #if wxUSE_UTF8_LOCALE_ONLY
279 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
282 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
283 if ( wxLocaleIsUtf8 ) return implA args; \
284 else return implW args
286 #elif wxUSE_UNICODE_WCHAR
287 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
290 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
295 wxVprintf(const wxString
& format
, va_list ap
)
297 WX_VARARG_VFOO_IMPL((wxFormatString(format
), ap
),
298 wxCRT_VprintfW
, wxCRT_VprintfA
);
302 wxVfprintf(FILE *f
, const wxString
& format
, va_list ap
)
304 WX_VARARG_VFOO_IMPL((f
, wxFormatString(format
), ap
),
305 wxCRT_VfprintfW
, wxCRT_VfprintfA
);
308 #undef WX_VARARG_VFOO_IMPL
311 // wxSprintf() and friends have to be implemented in two forms, one for
312 // writing to char* buffer and one for writing to wchar_t*:
314 #if !wxUSE_UTF8_LOCALE_ONLY
315 int WXDLLIMPEXP_BASE
wxDoSprintfWchar(char *str
, const wxChar
*format
, ...);
317 #if wxUSE_UNICODE_UTF8
318 int WXDLLIMPEXP_BASE
wxDoSprintfUtf8(char *str
, const char *format
, ...);
320 WX_DEFINE_VARARG_FUNC(int, wxSprintf
, 2, (char*, const wxFormatString
&),
321 wxDoSprintfWchar
, wxDoSprintfUtf8
)
324 wxVsprintf(char *str
, const wxString
& format
, va_list argptr
);
326 #if !wxUSE_UTF8_LOCALE_ONLY
327 int WXDLLIMPEXP_BASE
wxDoSnprintfWchar(char *str
, size_t size
, const wxChar
*format
, ...);
329 #if wxUSE_UNICODE_UTF8
330 int WXDLLIMPEXP_BASE
wxDoSnprintfUtf8(char *str
, size_t size
, const char *format
, ...);
332 WX_DEFINE_VARARG_FUNC(int, wxSnprintf
, 3, (char*, size_t, const wxFormatString
&),
333 wxDoSnprintfWchar
, wxDoSnprintfUtf8
)
336 wxVsnprintf(char *str
, size_t size
, const wxString
& format
, va_list argptr
);
340 #if !wxUSE_UTF8_LOCALE_ONLY
341 int WXDLLIMPEXP_BASE
wxDoSprintfWchar(wchar_t *str
, const wxChar
*format
, ...);
343 #if wxUSE_UNICODE_UTF8
344 int WXDLLIMPEXP_BASE
wxDoSprintfUtf8(wchar_t *str
, const char *format
, ...);
346 WX_DEFINE_VARARG_FUNC(int, wxSprintf
, 2, (wchar_t*, const wxFormatString
&),
347 wxDoSprintfWchar
, wxDoSprintfUtf8
)
350 wxVsprintf(wchar_t *str
, const wxString
& format
, va_list argptr
);
352 #if !wxUSE_UTF8_LOCALE_ONLY
353 int WXDLLIMPEXP_BASE
wxDoSnprintfWchar(wchar_t *str
, size_t size
, const wxChar
*format
, ...);
355 #if wxUSE_UNICODE_UTF8
356 int WXDLLIMPEXP_BASE
wxDoSnprintfUtf8(wchar_t *str
, size_t size
, const char *format
, ...);
358 WX_DEFINE_VARARG_FUNC(int, wxSnprintf
, 3, (wchar_t*, size_t, const wxFormatString
&),
359 wxDoSnprintfWchar
, wxDoSnprintfUtf8
)
362 wxVsnprintf(wchar_t *str
, size_t size
, const wxString
& format
, va_list argptr
);
364 #endif // wxUSE_UNICODE
367 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
369 // fortunately, OpenWatcom implements __VA_ARGS__, so we can provide macros
370 // that cast the format argument to wxString:
376 #define wxPrintf(fmt, ...) \
377 wxPrintf_Impl(wxFormatString(fmt), __VA_ARGS__)
378 #define wxFprintf(f, fmt, ...) \
379 wxFprintf_Impl(f, wxFormatString(fmt), __VA_ARGS__)
380 #define wxSprintf(s, fmt, ...) \
381 wxSprintf_Impl(s, wxFormatString(fmt), __VA_ARGS__)
382 #define wxSnprintf(s, n, fmt, ...) \
383 wxSnprintf_Impl(s, n, wxFormatString(fmt), __VA_ARGS__)
384 #endif // __WATCOMC__
387 // We can't use wxArgNormalizer<T> for variadic arguments to wxScanf() etc.
388 // because they are writable, so instead of providing friendly template
389 // vararg-like functions, we just provide both char* and wchar_t* variants
390 // of these functions. The type of output variadic arguments for %s must match
391 // the type of 'str' and 'format' arguments.
393 // For compatibility with earlier wx versions, we also provide wxSscanf()
394 // version with the first argument (input string) wxString; for this version,
395 // the type of output string values is determined by the type of format string
398 #define _WX_SCANFUNC_EXTRACT_ARGS_1(x) x
399 #define _WX_SCANFUNC_EXTRACT_ARGS_2(x,y) x, y
400 #define _WX_SCANFUNC_EXTRACT_ARGS(N, args) _WX_SCANFUNC_EXTRACT_ARGS_##N args
402 #define _WX_VARARG_PASS_WRITABLE(i) a##i
404 #define _WX_DEFINE_SCANFUNC(N, dummy1, name, impl, passfixed, numfixed, fixed)\
405 template<_WX_VARARG_JOIN(N, _WX_VARARG_TEMPL)> \
406 int name(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, fixed), \
407 _WX_VARARG_JOIN(N, _WX_VARARG_ARG)) \
409 return impl(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, passfixed), \
410 _WX_VARARG_JOIN(N, _WX_VARARG_PASS_WRITABLE)); \
413 #define WX_DEFINE_SCANFUNC(name, numfixed, fixed, impl, passfixed) \
414 inline int name(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, fixed)) \
416 return impl(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, passfixed)); \
418 _WX_VARARG_ITER(_WX_VARARG_MAX_ARGS, \
419 _WX_DEFINE_SCANFUNC, \
420 dummy1, name, impl, passfixed, numfixed, fixed)
422 // this is needed to normalize the format string, see src/common/strvararg.cpp
425 #define wxScanfConvertFormatW(fmt) fmt
428 WXDLLIMPEXP_BASE
wxScanfConvertFormatW(const wchar_t *format
);
431 WX_DEFINE_SCANFUNC(wxScanf
, 1, (const char *format
),
432 wxCRT_ScanfA
, (format
))
433 WX_DEFINE_SCANFUNC(wxScanf
, 1, (const wchar_t *format
),
434 wxCRT_ScanfW
, (wxScanfConvertFormatW(format
)))
436 WX_DEFINE_SCANFUNC(wxFscanf
, 2, (FILE *stream
, const char *format
),
437 wxCRT_FscanfA
, (stream
, format
))
438 WX_DEFINE_SCANFUNC(wxFscanf
, 2, (FILE *stream
, const wchar_t *format
),
439 wxCRT_FscanfW
, (stream
, wxScanfConvertFormatW(format
)))
441 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const char *str
, const char *format
),
442 wxCRT_SscanfA
, (str
, format
))
443 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wchar_t *str
, const wchar_t *format
),
444 wxCRT_SscanfW
, (str
, wxScanfConvertFormatW(format
)))
445 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCharBuffer
& str
, const char *format
),
446 wxCRT_SscanfA
, (str
.data(), format
))
447 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxWCharBuffer
& str
, const wchar_t *format
),
448 wxCRT_SscanfW
, (str
.data(), wxScanfConvertFormatW(format
)))
449 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxString
& str
, const char *format
),
450 wxCRT_SscanfA
, (str
.mb_str(), format
))
451 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxString
& str
, const wchar_t *format
),
452 wxCRT_SscanfW
, (str
.wc_str(), wxScanfConvertFormatW(format
)))
453 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCStrData
& str
, const char *format
),
454 wxCRT_SscanfA
, (str
.AsCharBuf(), format
))
455 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCStrData
& str
, const wchar_t *format
),
456 wxCRT_SscanfW
, (str
.AsWCharBuf(), wxScanfConvertFormatW(format
)))
458 // Visual C++ doesn't provide vsscanf()
460 int WXDLLIMPEXP_BASE
wxVsscanf(const char *str
, const char *format
, va_list ap
);
461 int WXDLLIMPEXP_BASE
wxVsscanf(const wchar_t *str
, const wchar_t *format
, va_list ap
);
462 int WXDLLIMPEXP_BASE
wxVsscanf(const wxCharBuffer
& str
, const char *format
, va_list ap
);
463 int WXDLLIMPEXP_BASE
wxVsscanf(const wxWCharBuffer
& str
, const wchar_t *format
, va_list ap
);
464 int WXDLLIMPEXP_BASE
wxVsscanf(const wxString
& str
, const char *format
, va_list ap
);
465 int WXDLLIMPEXP_BASE
wxVsscanf(const wxString
& str
, const wchar_t *format
, va_list ap
);
466 int WXDLLIMPEXP_BASE
wxVsscanf(const wxCStrData
& str
, const char *format
, va_list ap
);
467 int WXDLLIMPEXP_BASE
wxVsscanf(const wxCStrData
& str
, const wchar_t *format
, va_list ap
);
468 #endif // !__VISUALC__
470 #endif /* _WX_WXCRTVARARG_H_ */