]>
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 mingw32 normally uses MSVCRT which has non-standard vswprintf() and so
67 normally _vsnwprintf() is used instead, the only exception is when mingw32
68 is used with STLPort which does have a standard vswprintf() starting from
69 version 5.1 which we can use.
72 #if defined(_STLPORT_VERSION) && _STLPORT_VERSION >= 0x510
73 #ifndef HAVE_VSWPRINTF
74 #define HAVE_VSWPRINTF
76 #elif defined(HAVE_VSWPRINTF)
77 /* can't use non-standard vswprintf() */
80 #endif /* __MINGW32__ */
82 #if defined(__WATCOMC__)
83 #define HAVE_VSWPRINTF 1
86 #if wxUSE_PRINTF_POS_PARAMS
88 The systems where vsnprintf() supports positional parameters should
89 define the HAVE_UNIX98_PRINTF symbol.
91 On systems which don't (e.g. Windows) we are forced to use
92 our wxVsnprintf() implementation.
94 #if defined(HAVE_UNIX98_PRINTF)
96 #define wxCRT_VsnprintfW vswprintf
98 #ifdef HAVE_BROKEN_VSNPRINTF_DECL
99 #define wxCRT_VsnprintfA wx_fixed_vsnprintf
101 #define wxCRT_VsnprintfA vsnprintf
103 #else /* !HAVE_UNIX98_PRINTF */
105 The only compiler with positional parameters support under Windows
106 is VC++ 8.0 which provides a new xxprintf_p() functions family.
107 The 2003 PSDK includes a slightly earlier version of VC8 than the
108 main release and does not have the printf_p functions.
110 #if defined _MSC_FULL_VER && _MSC_FULL_VER >= 140050727 && !defined __WXWINCE__
111 #define wxCRT_VsnprintfA _vsprintf_p
112 #define wxCRT_VsnprintfW _vswprintf_p
114 #endif /* HAVE_UNIX98_PRINTF/!HAVE_UNIX98_PRINTF */
115 #else /* !wxUSE_PRINTF_POS_PARAMS */
117 We always want to define safe snprintf() function to be used instead of
118 sprintf(). Some compilers already have it (or rather vsnprintf() which
119 we really need...), otherwise we implement it using our own printf()
122 We define function with a trailing underscore here because the real one
123 is a wrapper around it as explained below
126 #if defined(__VISUALC__) || \
127 (defined(__BORLANDC__) && __BORLANDC__ >= 0x540)
128 #define wxCRT_VsnprintfA _vsnprintf
129 #define wxCRT_VsnprintfW _vsnwprintf
131 #if defined(HAVE__VSNWPRINTF)
132 #define wxCRT_VsnprintfW _vsnwprintf
133 #elif defined(HAVE_VSWPRINTF)
134 #define wxCRT_VsnprintfW vswprintf
135 #elif defined(__WATCOMC__)
136 #define wxCRT_VsnprintfW _vsnwprintf
140 All versions of CodeWarrior supported by wxWidgets apparently
141 have both snprintf() and vsnprintf()
143 #if defined(HAVE_VSNPRINTF) \
144 || defined(__MWERKS__) || defined(__WATCOMC__)
145 #ifdef HAVE_BROKEN_VSNPRINTF_DECL
146 #define wxCRT_VsnprintfA wx_fixed_vsnprintf
148 #define wxCRT_VsnprintfA vsnprintf
152 #endif /* wxUSE_PRINTF_POS_PARAMS/!wxUSE_PRINTF_POS_PARAMS */
154 #ifndef wxCRT_VsnprintfW
155 /* no (suitable) vsnprintf(), cook our own */
157 wxCRT_VsnprintfW(wchar_t *buf
, size_t len
, const wchar_t *format
, va_list argptr
);
158 #define wxUSE_WXVSNPRINTFW 1
160 #define wxUSE_WXVSNPRINTFW 0
163 #ifndef wxCRT_VsnprintfA
164 /* no (suitable) vsnprintf(), cook our own */
166 wxCRT_VsnprintfA(char *buf
, size_t len
, const char *format
, va_list argptr
);
167 #define wxUSE_WXVSNPRINTFA 1
169 #define wxUSE_WXVSNPRINTFA 0
172 // for wxString code, define wxUSE_WXVSNPRINTF to indicate that wx
173 // implementation is used no matter what (in UTF-8 build, either *A or *W
174 // version may be called):
176 #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFA
177 #elif wxUSE_UNICODE_WCHAR
178 #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFW
179 #elif wxUSE_UTF8_LOCALE_ONLY
180 #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFA
181 #else // UTF-8 under any locale
182 #define wxUSE_WXVSNPRINTF (wxUSE_WXVSNPRINTFA && wxUSE_WXVSNPRINTFW)
185 #define wxCRT_FprintfA fprintf
186 #define wxCRT_PrintfA printf
187 #define wxCRT_VfprintfA vfprintf
188 #define wxCRT_VprintfA vprintf
189 #define wxCRT_VsprintfA vsprintf
192 In Unicode mode we need to have all standard functions such as wprintf() and
193 so on but not all systems have them so use our own implementations in this
196 #if wxUSE_UNICODE && !defined(wxHAVE_TCHAR_SUPPORT) && !defined(HAVE_WPRINTF)
197 #define wxNEED_WPRINTF
201 #if defined(wxNEED_WPRINTF)
203 we need to implement all wide character printf functions either because
204 we don't have them at all or because they don't have the semantics we
207 int wxCRT_PrintfW( const wchar_t *format
, ... );
208 int wxCRT_FprintfW( FILE *stream
, const wchar_t *format
, ... );
209 int wxCRT_VfprintfW( FILE *stream
, const wchar_t *format
, va_list ap
);
210 int wxCRT_VprintfW( const wchar_t *format
, va_list ap
);
211 int wxCRT_VsprintfW( wchar_t *str
, const wchar_t *format
, va_list ap
);
212 #else /* !wxNEED_WPRINTF */
213 #define wxCRT_FprintfW fwprintf
214 #define wxCRT_PrintfW wprintf
215 #define wxCRT_VfprintfW vfwprintf
216 #define wxCRT_VprintfW vwprintf
218 #if defined(__WINDOWS__) && !defined(HAVE_VSWPRINTF)
219 // only non-standard vswprintf() without buffer size argument can be used here
220 #define wxCRT_VsprintfW vswprintf
222 #endif /* wxNEED_WPRINTF */
225 /* Required for wxScanf() etc. */
226 #define wxCRT_ScanfA scanf
227 #define wxCRT_SscanfA sscanf
228 #define wxCRT_FscanfA fscanf
229 #define wxCRT_VsscanfA vsscanf
231 #if defined(wxNEED_WPRINTF)
232 int wxCRT_ScanfW(const wchar_t *format
, ...);
233 int wxCRT_SscanfW(const wchar_t *str
, const wchar_t *format
, ...);
234 int wxCRT_FscanfW(FILE *stream
, const wchar_t *format
, ...);
235 int wxCRT_VsscanfW(const wchar_t *str
, const wchar_t *format
, va_list ap
);
237 #define wxCRT_ScanfW wxVMS_USE_STD wscanf
238 #define wxCRT_SscanfW wxVMS_USE_STD swscanf
239 #define wxCRT_FscanfW wxVMS_USE_STD fwscanf
240 #define wxCRT_VsscanfW wxVMS_USE_STD vswscanf
243 // ----------------------------------------------------------------------------
244 // user-friendly wrappers to CRT functions
245 // ----------------------------------------------------------------------------
248 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
249 #define wxPrintf wxPrintf_Impl
250 #define wxFprintf wxFprintf_Impl
251 #define wxSprintf wxSprintf_Impl
252 #define wxSnprintf wxSnprintf_Impl
255 // FIXME-UTF8: remove this
257 #define wxCRT_PrintfNative wxCRT_PrintfW
258 #define wxCRT_FprintfNative wxCRT_FprintfW
260 #define wxCRT_PrintfNative wxCRT_PrintfA
261 #define wxCRT_FprintfNative wxCRT_FprintfA
264 WX_DEFINE_VARARG_FUNC(int, wxPrintf
, 1, (const wxFormatString
&),
265 wxCRT_PrintfNative
, wxCRT_PrintfA
)
266 WX_DEFINE_VARARG_FUNC(int, wxFprintf
, 2, (FILE*, const wxFormatString
&),
267 wxCRT_FprintfNative
, wxCRT_FprintfA
)
269 // va_list versions of printf functions simply forward to the respective
270 // CRT function; note that they assume that va_list was created using
271 // wxArgNormalizer<T>!
272 #if wxUSE_UNICODE_UTF8
273 #if wxUSE_UTF8_LOCALE_ONLY
274 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
277 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
278 if ( wxLocaleIsUtf8 ) return implA args; \
279 else return implW args
281 #elif wxUSE_UNICODE_WCHAR
282 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
285 #define WX_VARARG_VFOO_IMPL(args, implW, implA) \
290 wxVprintf(const wxString
& format
, va_list ap
)
292 WX_VARARG_VFOO_IMPL((wxFormatString(format
), ap
),
293 wxCRT_VprintfW
, wxCRT_VprintfA
);
297 wxVfprintf(FILE *f
, const wxString
& format
, va_list ap
)
299 WX_VARARG_VFOO_IMPL((f
, wxFormatString(format
), ap
),
300 wxCRT_VfprintfW
, wxCRT_VfprintfA
);
303 #undef WX_VARARG_VFOO_IMPL
306 // wxSprintf() and friends have to be implemented in two forms, one for
307 // writing to char* buffer and one for writing to wchar_t*:
309 #if !wxUSE_UTF8_LOCALE_ONLY
310 int WXDLLIMPEXP_BASE
wxDoSprintfWchar(char *str
, const wxChar
*format
, ...);
312 #if wxUSE_UNICODE_UTF8
313 int WXDLLIMPEXP_BASE
wxDoSprintfUtf8(char *str
, const char *format
, ...);
315 WX_DEFINE_VARARG_FUNC(int, wxSprintf
, 2, (char*, const wxFormatString
&),
316 wxDoSprintfWchar
, wxDoSprintfUtf8
)
319 wxVsprintf(char *str
, const wxString
& format
, va_list argptr
);
321 #if !wxUSE_UTF8_LOCALE_ONLY
322 int WXDLLIMPEXP_BASE
wxDoSnprintfWchar(char *str
, size_t size
, const wxChar
*format
, ...);
324 #if wxUSE_UNICODE_UTF8
325 int WXDLLIMPEXP_BASE
wxDoSnprintfUtf8(char *str
, size_t size
, const char *format
, ...);
327 WX_DEFINE_VARARG_FUNC(int, wxSnprintf
, 3, (char*, size_t, const wxFormatString
&),
328 wxDoSnprintfWchar
, wxDoSnprintfUtf8
)
331 wxVsnprintf(char *str
, size_t size
, const wxString
& format
, va_list argptr
);
335 #if !wxUSE_UTF8_LOCALE_ONLY
336 int WXDLLIMPEXP_BASE
wxDoSprintfWchar(wchar_t *str
, const wxChar
*format
, ...);
338 #if wxUSE_UNICODE_UTF8
339 int WXDLLIMPEXP_BASE
wxDoSprintfUtf8(wchar_t *str
, const char *format
, ...);
341 WX_DEFINE_VARARG_FUNC(int, wxSprintf
, 2, (wchar_t*, const wxFormatString
&),
342 wxDoSprintfWchar
, wxDoSprintfUtf8
)
345 wxVsprintf(wchar_t *str
, const wxString
& format
, va_list argptr
);
347 #if !wxUSE_UTF8_LOCALE_ONLY
348 int WXDLLIMPEXP_BASE
wxDoSnprintfWchar(wchar_t *str
, size_t size
, const wxChar
*format
, ...);
350 #if wxUSE_UNICODE_UTF8
351 int WXDLLIMPEXP_BASE
wxDoSnprintfUtf8(wchar_t *str
, size_t size
, const char *format
, ...);
353 WX_DEFINE_VARARG_FUNC(int, wxSnprintf
, 3, (wchar_t*, size_t, const wxFormatString
&),
354 wxDoSnprintfWchar
, wxDoSnprintfUtf8
)
357 wxVsnprintf(wchar_t *str
, size_t size
, const wxString
& format
, va_list argptr
);
359 #endif // wxUSE_UNICODE
362 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
364 // fortunately, OpenWatcom implements __VA_ARGS__, so we can provide macros
365 // that cast the format argument to wxString:
371 #define wxPrintf(fmt, ...) \
372 wxPrintf_Impl(wxFormatString(fmt), __VA_ARGS__)
373 #define wxFprintf(f, fmt, ...) \
374 wxFprintf_Impl(f, wxFormatString(fmt), __VA_ARGS__)
375 #define wxSprintf(s, fmt, ...) \
376 wxSprintf_Impl(s, wxFormatString(fmt), __VA_ARGS__)
377 #define wxSnprintf(s, n, fmt, ...) \
378 wxSnprintf_Impl(s, n, wxFormatString(fmt), __VA_ARGS__)
379 #endif // __WATCOMC__
382 // We can't use wxArgNormalizer<T> for variadic arguments to wxScanf() etc.
383 // because they are writable, so instead of providing friendly template
384 // vararg-like functions, we just provide both char* and wchar_t* variants
385 // of these functions. The type of output variadic arguments for %s must match
386 // the type of 'str' and 'format' arguments.
388 // For compatibility with earlier wx versions, we also provide wxSscanf()
389 // version with the first argument (input string) wxString; for this version,
390 // the type of output string values is determined by the type of format string
393 #define _WX_SCANFUNC_EXTRACT_ARGS_1(x) x
394 #define _WX_SCANFUNC_EXTRACT_ARGS_2(x,y) x, y
395 #define _WX_SCANFUNC_EXTRACT_ARGS(N, args) _WX_SCANFUNC_EXTRACT_ARGS_##N args
397 #define _WX_VARARG_PASS_WRITABLE(i) a##i
399 #define _WX_DEFINE_SCANFUNC(N, dummy1, name, impl, passfixed, numfixed, fixed)\
400 template<_WX_VARARG_JOIN(N, _WX_VARARG_TEMPL)> \
401 int name(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, fixed), \
402 _WX_VARARG_JOIN(N, _WX_VARARG_ARG)) \
404 return impl(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, passfixed), \
405 _WX_VARARG_JOIN(N, _WX_VARARG_PASS_WRITABLE)); \
408 #define WX_DEFINE_SCANFUNC(name, numfixed, fixed, impl, passfixed) \
409 inline int name(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, fixed)) \
411 return impl(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, passfixed)); \
413 _WX_VARARG_ITER(_WX_VARARG_MAX_ARGS, \
414 _WX_DEFINE_SCANFUNC, \
415 dummy1, name, impl, passfixed, numfixed, fixed)
417 // this is needed to normalize the format string, see src/common/strvararg.cpp
420 #define wxScanfConvertFormatW(fmt) fmt
423 WXDLLIMPEXP_BASE
wxScanfConvertFormatW(const wchar_t *format
);
426 WX_DEFINE_SCANFUNC(wxScanf
, 1, (const char *format
),
427 wxCRT_ScanfA
, (format
))
428 WX_DEFINE_SCANFUNC(wxScanf
, 1, (const wchar_t *format
),
429 wxCRT_ScanfW
, (wxScanfConvertFormatW(format
)))
431 WX_DEFINE_SCANFUNC(wxFscanf
, 2, (FILE *stream
, const char *format
),
432 wxCRT_FscanfA
, (stream
, format
))
433 WX_DEFINE_SCANFUNC(wxFscanf
, 2, (FILE *stream
, const wchar_t *format
),
434 wxCRT_FscanfW
, (stream
, wxScanfConvertFormatW(format
)))
436 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const char *str
, const char *format
),
437 wxCRT_SscanfA
, (str
, format
))
438 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wchar_t *str
, const wchar_t *format
),
439 wxCRT_SscanfW
, (str
, wxScanfConvertFormatW(format
)))
440 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCharBuffer
& str
, const char *format
),
441 wxCRT_SscanfA
, (str
.data(), format
))
442 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxWCharBuffer
& str
, const wchar_t *format
),
443 wxCRT_SscanfW
, (str
.data(), wxScanfConvertFormatW(format
)))
444 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxString
& str
, const char *format
),
445 wxCRT_SscanfA
, (str
.mb_str(), format
))
446 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxString
& str
, const wchar_t *format
),
447 wxCRT_SscanfW
, (str
.wc_str(), wxScanfConvertFormatW(format
)))
448 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCStrData
& str
, const char *format
),
449 wxCRT_SscanfA
, (str
.AsCharBuf(), format
))
450 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCStrData
& str
, const wchar_t *format
),
451 wxCRT_SscanfW
, (str
.AsWCharBuf(), wxScanfConvertFormatW(format
)))
453 // Visual C++ doesn't provide vsscanf()
455 int WXDLLIMPEXP_BASE
wxVsscanf(const char *str
, const char *format
, va_list ap
);
456 int WXDLLIMPEXP_BASE
wxVsscanf(const wchar_t *str
, const wchar_t *format
, va_list ap
);
457 int WXDLLIMPEXP_BASE
wxVsscanf(const wxCharBuffer
& str
, const char *format
, va_list ap
);
458 int WXDLLIMPEXP_BASE
wxVsscanf(const wxWCharBuffer
& str
, const wchar_t *format
, va_list ap
);
459 int WXDLLIMPEXP_BASE
wxVsscanf(const wxString
& str
, const char *format
, va_list ap
);
460 int WXDLLIMPEXP_BASE
wxVsscanf(const wxString
& str
, const wchar_t *format
, va_list ap
);
461 int WXDLLIMPEXP_BASE
wxVsscanf(const wxCStrData
& str
, const char *format
, va_list ap
);
462 int WXDLLIMPEXP_BASE
wxVsscanf(const wxCStrData
& str
, const wchar_t *format
, va_list ap
);
463 #endif // !__VISUALC__
465 #endif /* _WX_WXCRTVARARG_H_ */