]>
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()/vsscanf() exists in the system libraries but 
  35    not in the 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 #if defined(HAVE_VSSCANF) && !defined(HAVE_VSSCANF_DECL) 
  61     int vsscanf(const char *str
, const char *format
, va_list ap
); 
  62 #endif /* !HAVE_SNPRINTF_DECL */ 
  64 /* Wrapper for vsnprintf if it's 3rd parameter is non-const. Note: the 
  65  * same isn't done for snprintf below, the builtin wxSnprintf_ is used 
  66  * instead since it's already a simple wrapper */ 
  67 #if defined __cplusplus && defined HAVE_BROKEN_VSNPRINTF_DECL 
  68     inline int wx_fixed_vsnprintf(char *str
, size_t size
, const char *format
, va_list ap
) 
  70         return vsnprintf(str
, size
, (char*)format
, ap
); 
  75    mingw32 normally uses MSVCRT which has non-standard vswprintf() and so 
  76    normally _vsnwprintf() is used instead, the only exception is when mingw32 
  77    is used with STLPort which does have a standard vswprintf() starting from 
  78    version 5.1 which we can use. 
  81     #if defined(_STLPORT_VERSION) && _STLPORT_VERSION >= 0x510 
  82         #ifndef HAVE_VSWPRINTF 
  83             #define HAVE_VSWPRINTF 
  85     #elif defined(HAVE_VSWPRINTF) 
  86         /* can't use non-standard vswprintf() */ 
  89 #endif /* __MINGW32__ */ 
  91 #if defined(__WATCOMC__) 
  92     #define HAVE_VSWPRINTF 1 
  95 #if wxUSE_PRINTF_POS_PARAMS 
  97         The systems where vsnprintf() supports positional parameters should 
  98         define the HAVE_UNIX98_PRINTF symbol. 
 100         On systems which don't (e.g. Windows) we are forced to use 
 101         our wxVsnprintf() implementation. 
 103     #if defined(HAVE_UNIX98_PRINTF) 
 104         #ifdef HAVE_VSWPRINTF 
 105             #define wxCRT_VsnprintfW   vswprintf 
 107         #ifdef HAVE_BROKEN_VSNPRINTF_DECL 
 108             #define wxCRT_VsnprintfA    wx_fixed_vsnprintf 
 110             #define wxCRT_VsnprintfA    vsnprintf 
 112     #else /* !HAVE_UNIX98_PRINTF */ 
 114             The only compiler with positional parameters support under Windows 
 115             is VC++ 8.0 which provides a new xxprintf_p() functions family. 
 116             The 2003 PSDK includes a slightly earlier version of VC8 than the 
 117             main release and does not have the printf_p functions. 
 119         #if defined _MSC_FULL_VER && _MSC_FULL_VER >= 140050727 && !defined __WXWINCE__ 
 120             #define wxCRT_VsnprintfA    _vsprintf_p 
 121             #define wxCRT_VsnprintfW    _vswprintf_p 
 123     #endif /* HAVE_UNIX98_PRINTF/!HAVE_UNIX98_PRINTF */ 
 124 #else /* !wxUSE_PRINTF_POS_PARAMS */ 
 126        We always want to define safe snprintf() function to be used instead of 
 127        sprintf(). Some compilers already have it (or rather vsnprintf() which 
 128        we really need...), otherwise we implement it using our own printf() 
 131        We define function with a trailing underscore here because the real one 
 132        is a wrapper around it as explained below 
 135     #if defined(__VISUALC__) || \ 
 136             (defined(__BORLANDC__) && __BORLANDC__ >= 0x540) 
 137         #define wxCRT_VsnprintfA    _vsnprintf 
 138         #define wxCRT_VsnprintfW    _vsnwprintf 
 140         #if defined(HAVE__VSNWPRINTF) 
 141             #define wxCRT_VsnprintfW    _vsnwprintf 
 142         #elif defined(HAVE_VSWPRINTF) 
 143             #define wxCRT_VsnprintfW     vswprintf 
 144         #elif defined(__WATCOMC__) 
 145             #define wxCRT_VsnprintfW    _vsnwprintf 
 149            All versions of CodeWarrior supported by wxWidgets apparently 
 150            have both snprintf() and vsnprintf() 
 152         #if defined(HAVE_VSNPRINTF) \ 
 153             || defined(__MWERKS__) || defined(__WATCOMC__) 
 154             #ifdef HAVE_BROKEN_VSNPRINTF_DECL 
 155                 #define wxCRT_VsnprintfA    wx_fixed_vsnprintf 
 157                 #define wxCRT_VsnprintfA    vsnprintf 
 161 #endif /* wxUSE_PRINTF_POS_PARAMS/!wxUSE_PRINTF_POS_PARAMS */ 
 163 #ifndef wxCRT_VsnprintfW 
 164     /* no (suitable) vsnprintf(), cook our own */ 
 166     wxCRT_VsnprintfW(wchar_t *buf
, size_t len
, const wchar_t *format
, va_list argptr
); 
 167     #define wxUSE_WXVSNPRINTFW 1 
 169     #define wxUSE_WXVSNPRINTFW 0 
 172 #ifndef wxCRT_VsnprintfA 
 173         /* no (suitable) vsnprintf(), cook our own */ 
 175         wxCRT_VsnprintfA(char *buf
, size_t len
, const char *format
, va_list argptr
); 
 176         #define wxUSE_WXVSNPRINTFA 1 
 178         #define wxUSE_WXVSNPRINTFA 0 
 181 // for wxString code, define wxUSE_WXVSNPRINTF to indicate that wx 
 182 // implementation is used no matter what (in UTF-8 build, either *A or *W 
 183 // version may be called): 
 185     #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFA 
 186 #elif wxUSE_UNICODE_WCHAR 
 187     #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFW 
 188 #elif wxUSE_UTF8_LOCALE_ONLY 
 189     #define wxUSE_WXVSNPRINTF wxUSE_WXVSNPRINTFA 
 190 #else // UTF-8 under any locale 
 191     #define wxUSE_WXVSNPRINTF (wxUSE_WXVSNPRINTFA && wxUSE_WXVSNPRINTFW) 
 194 #define wxCRT_FprintfA       fprintf 
 195 #define wxCRT_PrintfA        printf 
 196 #define wxCRT_VfprintfA      vfprintf 
 197 #define wxCRT_VprintfA       vprintf 
 198 #define wxCRT_VsprintfA      vsprintf 
 201    In Unicode mode we need to have all standard functions such as wprintf() and 
 202    so on but not all systems have them so use our own implementations in this 
 205 #if wxUSE_UNICODE && !defined(wxHAVE_TCHAR_SUPPORT) && !defined(HAVE_WPRINTF) 
 206     #define wxNEED_WPRINTF 
 208 #if wxUSE_UNICODE && !defined(wxHAVE_TCHAR_SUPPORT) && !defined(HAVE_VSWSCANF) 
 209     #define wxNEED_VSWSCANF 
 213 #if defined(wxNEED_WPRINTF) 
 215         we need to implement all wide character printf functions either because 
 216         we don't have them at all or because they don't have the semantics we 
 219     int wxCRT_PrintfW( const wchar_t *format
, ... ); 
 220     int wxCRT_FprintfW( FILE *stream
, const wchar_t *format
, ... ); 
 221     int wxCRT_VfprintfW( FILE *stream
, const wchar_t *format
, va_list ap 
); 
 222     int wxCRT_VprintfW( const wchar_t *format
, va_list ap 
); 
 223     int wxCRT_VsprintfW( wchar_t *str
, const wchar_t *format
, va_list ap 
); 
 224 #else /* !wxNEED_WPRINTF */ 
 225     #define wxCRT_FprintfW      fwprintf 
 226     #define wxCRT_PrintfW       wprintf 
 227     #define wxCRT_VfprintfW     vfwprintf 
 228     #define wxCRT_VprintfW      vwprintf 
 230     #if defined(__WINDOWS__) && !defined(HAVE_VSWPRINTF) 
 231         // only non-standard vswprintf() without buffer size argument can be used here 
 232         #define  wxCRT_VsprintfW     vswprintf 
 234 #endif /* wxNEED_WPRINTF */ 
 237 /* Required for wxScanf() etc. */ 
 238 #define wxCRT_ScanfA     scanf 
 239 #define wxCRT_SscanfA    sscanf 
 240 #define wxCRT_FscanfA    fscanf 
 241 #define wxCRT_VsscanfA   vsscanf 
 243 #if defined(wxNEED_WPRINTF) 
 244     int wxCRT_ScanfW(const wchar_t *format
, ...); 
 245     int wxCRT_SscanfW(const wchar_t *str
, const wchar_t *format
, ...); 
 246     int wxCRT_FscanfW(FILE *stream
, const wchar_t *format
, ...); 
 248     #define wxCRT_ScanfW     wxVMS_USE_STD wscanf 
 249     #define wxCRT_SscanfW    wxVMS_USE_STD swscanf 
 250     #define wxCRT_FscanfW    wxVMS_USE_STD fwscanf 
 252 #ifdef wxNEED_VSWSCANF 
 253     int wxCRT_VsscanfW(const wchar_t *str
, const wchar_t *format
, va_list ap
); 
 255     #define wxCRT_VsscanfW   wxVMS_USE_STD vswscanf 
 258 // ---------------------------------------------------------------------------- 
 259 // user-friendly wrappers to CRT functions 
 260 // ---------------------------------------------------------------------------- 
 263     // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351 
 264     #define wxPrintf    wxPrintf_Impl 
 265     #define wxFprintf   wxFprintf_Impl 
 266     #define wxSprintf   wxSprintf_Impl 
 267     #define wxSnprintf  wxSnprintf_Impl 
 270     // FIXME-UTF8: remove this 
 272     #define wxCRT_PrintfNative wxCRT_PrintfW 
 273     #define wxCRT_FprintfNative wxCRT_FprintfW 
 275     #define wxCRT_PrintfNative wxCRT_PrintfA 
 276     #define wxCRT_FprintfNative wxCRT_FprintfA 
 279 WX_DEFINE_VARARG_FUNC(int, wxPrintf
, 1, (const wxFormatString
&), 
 280                       wxCRT_PrintfNative
, wxCRT_PrintfA
) 
 281 WX_DEFINE_VARARG_FUNC(int, wxFprintf
, 2, (FILE*, const wxFormatString
&), 
 282                       wxCRT_FprintfNative
, wxCRT_FprintfA
) 
 284 // va_list versions of printf functions simply forward to the respective 
 285 // CRT function; note that they assume that va_list was created using 
 286 // wxArgNormalizer<T>! 
 287 #if wxUSE_UNICODE_UTF8 
 288     #if wxUSE_UTF8_LOCALE_ONLY 
 289         #define WX_VARARG_VFOO_IMPL(args, implW, implA)              \ 
 292         #define WX_VARARG_VFOO_IMPL(args, implW, implA)              \ 
 293             if ( wxLocaleIsUtf8 ) return implA args;                 \ 
 294             else return implW args 
 296 #elif wxUSE_UNICODE_WCHAR 
 297     #define WX_VARARG_VFOO_IMPL(args, implW, implA)                  \ 
 300     #define WX_VARARG_VFOO_IMPL(args, implW, implA)                  \ 
 305 wxVprintf(const wxString
& format
, va_list ap
) 
 307     WX_VARARG_VFOO_IMPL((wxFormatString(format
), ap
), 
 308                         wxCRT_VprintfW
, wxCRT_VprintfA
); 
 312 wxVfprintf(FILE *f
, const wxString
& format
, va_list ap
) 
 314     WX_VARARG_VFOO_IMPL((f
, wxFormatString(format
), ap
), 
 315                         wxCRT_VfprintfW
, wxCRT_VfprintfA
); 
 318 #undef WX_VARARG_VFOO_IMPL 
 321 // wxSprintf() and friends have to be implemented in two forms, one for 
 322 // writing to char* buffer and one for writing to wchar_t*: 
 324 #if !wxUSE_UTF8_LOCALE_ONLY 
 325 int WXDLLIMPEXP_BASE 
wxDoSprintfWchar(char *str
, const wxChar 
*format
, ...); 
 327 #if wxUSE_UNICODE_UTF8 
 328 int WXDLLIMPEXP_BASE 
wxDoSprintfUtf8(char *str
, const char *format
, ...); 
 330 WX_DEFINE_VARARG_FUNC(int, wxSprintf
, 2, (char*, const wxFormatString
&), 
 331                       wxDoSprintfWchar
, wxDoSprintfUtf8
) 
 334 wxVsprintf(char *str
, const wxString
& format
, va_list argptr
); 
 336 #if !wxUSE_UTF8_LOCALE_ONLY 
 337 int WXDLLIMPEXP_BASE 
wxDoSnprintfWchar(char *str
, size_t size
, const wxChar 
*format
, ...); 
 339 #if wxUSE_UNICODE_UTF8 
 340 int WXDLLIMPEXP_BASE 
wxDoSnprintfUtf8(char *str
, size_t size
, const char *format
, ...); 
 342 WX_DEFINE_VARARG_FUNC(int, wxSnprintf
, 3, (char*, size_t, const wxFormatString
&), 
 343                       wxDoSnprintfWchar
, wxDoSnprintfUtf8
) 
 346 wxVsnprintf(char *str
, size_t size
, const wxString
& format
, va_list argptr
); 
 350 #if !wxUSE_UTF8_LOCALE_ONLY 
 351 int WXDLLIMPEXP_BASE 
wxDoSprintfWchar(wchar_t *str
, const wxChar 
*format
, ...); 
 353 #if wxUSE_UNICODE_UTF8 
 354 int WXDLLIMPEXP_BASE 
wxDoSprintfUtf8(wchar_t *str
, const char *format
, ...); 
 356 WX_DEFINE_VARARG_FUNC(int, wxSprintf
, 2, (wchar_t*, const wxFormatString
&), 
 357                       wxDoSprintfWchar
, wxDoSprintfUtf8
) 
 360 wxVsprintf(wchar_t *str
, const wxString
& format
, va_list argptr
); 
 362 #if !wxUSE_UTF8_LOCALE_ONLY 
 363 int WXDLLIMPEXP_BASE 
wxDoSnprintfWchar(wchar_t *str
, size_t size
, const wxChar 
*format
, ...); 
 365 #if wxUSE_UNICODE_UTF8 
 366 int WXDLLIMPEXP_BASE 
wxDoSnprintfUtf8(wchar_t *str
, size_t size
, const char *format
, ...); 
 368 WX_DEFINE_VARARG_FUNC(int, wxSnprintf
, 3, (wchar_t*, size_t, const wxFormatString
&), 
 369                       wxDoSnprintfWchar
, wxDoSnprintfUtf8
) 
 372 wxVsnprintf(wchar_t *str
, size_t size
, const wxString
& format
, va_list argptr
); 
 374 #endif // wxUSE_UNICODE 
 377     // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351 
 379     // fortunately, OpenWatcom implements __VA_ARGS__, so we can provide macros 
 380     // that cast the format argument to wxString: 
 386     #define wxPrintf(fmt, ...) \ 
 387             wxPrintf_Impl(wxFormatString(fmt), __VA_ARGS__) 
 388     #define wxFprintf(f, fmt, ...) \ 
 389             wxFprintf_Impl(f, wxFormatString(fmt), __VA_ARGS__) 
 390     #define wxSprintf(s, fmt, ...) \ 
 391             wxSprintf_Impl(s, wxFormatString(fmt), __VA_ARGS__) 
 392     #define wxSnprintf(s, n, fmt, ...) \ 
 393             wxSnprintf_Impl(s, n, wxFormatString(fmt), __VA_ARGS__) 
 394 #endif // __WATCOMC__ 
 397 // We can't use wxArgNormalizer<T> for variadic arguments to wxScanf() etc. 
 398 // because they are writable, so instead of providing friendly template 
 399 // vararg-like functions, we just provide both char* and wchar_t* variants 
 400 // of these functions. The type of output variadic arguments for %s must match 
 401 // the type of 'str' and 'format' arguments. 
 403 // For compatibility with earlier wx versions, we also provide wxSscanf() 
 404 // version with the first argument (input string) wxString; for this version, 
 405 // the type of output string values is determined by the type of format string 
 408 #define _WX_SCANFUNC_EXTRACT_ARGS_1(x)   x 
 409 #define _WX_SCANFUNC_EXTRACT_ARGS_2(x,y) x, y 
 410 #define _WX_SCANFUNC_EXTRACT_ARGS(N, args) _WX_SCANFUNC_EXTRACT_ARGS_##N args 
 412 #define _WX_VARARG_PASS_WRITABLE(i) a##i 
 414 #define _WX_DEFINE_SCANFUNC(N, dummy1, name, impl, passfixed, numfixed, fixed)\ 
 415     template<_WX_VARARG_JOIN(N, _WX_VARARG_TEMPL)>                            \ 
 416     int name(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, fixed),                      \ 
 417              _WX_VARARG_JOIN(N, _WX_VARARG_ARG))                              \ 
 419         return  impl(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, passfixed),          \ 
 420                      _WX_VARARG_JOIN(N, _WX_VARARG_PASS_WRITABLE));           \ 
 423 #define WX_DEFINE_SCANFUNC(name, numfixed, fixed, impl, passfixed)            \ 
 424     inline int name(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, fixed))               \ 
 426         return impl(_WX_SCANFUNC_EXTRACT_ARGS(numfixed, passfixed));          \ 
 428     _WX_VARARG_ITER(_WX_VARARG_MAX_ARGS,                                      \ 
 429                     _WX_DEFINE_SCANFUNC,                                      \ 
 430                     dummy1, name, impl, passfixed, numfixed, fixed) 
 432 // this is needed to normalize the format string, see src/common/strvararg.cpp 
 435     #define wxScanfConvertFormatW(fmt) fmt 
 438     WXDLLIMPEXP_BASE 
wxScanfConvertFormatW(const wchar_t *format
); 
 441 WX_DEFINE_SCANFUNC(wxScanf
, 1, (const char *format
), 
 442                    wxCRT_ScanfA
, (format
)) 
 443 WX_DEFINE_SCANFUNC(wxScanf
, 1, (const wchar_t *format
), 
 444                    wxCRT_ScanfW
, (wxScanfConvertFormatW(format
))) 
 446 WX_DEFINE_SCANFUNC(wxFscanf
, 2, (FILE *stream
, const char *format
), 
 447                    wxCRT_FscanfA
, (stream
, format
)) 
 448 WX_DEFINE_SCANFUNC(wxFscanf
, 2, (FILE *stream
, const wchar_t *format
), 
 449                    wxCRT_FscanfW
, (stream
, wxScanfConvertFormatW(format
))) 
 451 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const char *str
, const char *format
), 
 452                    wxCRT_SscanfA
, (str
, format
)) 
 453 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wchar_t *str
, const wchar_t *format
), 
 454                    wxCRT_SscanfW
, (str
, wxScanfConvertFormatW(format
))) 
 455 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCharBuffer
& str
, const char *format
), 
 456                    wxCRT_SscanfA
, (str
.data(), format
)) 
 457 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxWCharBuffer
& str
, const wchar_t *format
), 
 458                    wxCRT_SscanfW
, (str
.data(), wxScanfConvertFormatW(format
))) 
 459 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxString
& str
, const char *format
), 
 460                    wxCRT_SscanfA
, (str
.mb_str(), format
)) 
 461 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxString
& str
, const wchar_t *format
), 
 462                    wxCRT_SscanfW
, (str
.wc_str(), wxScanfConvertFormatW(format
))) 
 463 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCStrData
& str
, const char *format
), 
 464                    wxCRT_SscanfA
, (str
.AsCharBuf(), format
)) 
 465 WX_DEFINE_SCANFUNC(wxSscanf
, 2, (const wxCStrData
& str
, const wchar_t *format
), 
 466                    wxCRT_SscanfW
, (str
.AsWCharBuf(), wxScanfConvertFormatW(format
))) 
 468 // Visual C++ doesn't provide vsscanf() 
 470 int WXDLLIMPEXP_BASE 
wxVsscanf(const char *str
, const char *format
, va_list ap
); 
 471 int WXDLLIMPEXP_BASE 
wxVsscanf(const wchar_t *str
, const wchar_t *format
, va_list ap
); 
 472 int WXDLLIMPEXP_BASE 
wxVsscanf(const wxCharBuffer
& str
, const char *format
, va_list ap
); 
 473 int WXDLLIMPEXP_BASE 
wxVsscanf(const wxWCharBuffer
& str
, const wchar_t *format
, va_list ap
); 
 474 int WXDLLIMPEXP_BASE 
wxVsscanf(const wxString
& str
, const char *format
, va_list ap
); 
 475 int WXDLLIMPEXP_BASE 
wxVsscanf(const wxString
& str
, const wchar_t *format
, va_list ap
); 
 476 int WXDLLIMPEXP_BASE 
wxVsscanf(const wxCStrData
& str
, const char *format
, va_list ap
); 
 477 int WXDLLIMPEXP_BASE 
wxVsscanf(const wxCStrData
& str
, const wchar_t *format
, va_list ap
); 
 478 #endif // !__VISUALC__ 
 480 #endif /* _WX_WXCRTVARARG_H_ */