vlc_fixups.h 10.8 KB
Newer Older
1
2
3
/*****************************************************************************
 * fixups.h: portability fixups included from config.h
 *****************************************************************************
4
 * Copyright © 1998-2008 the VideoLAN project
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
 *****************************************************************************/

/**
 * \file
 * This file is a collection of portability fixes
 */

26
27
28
#ifndef LIBVLC_FIXUPS_H
# define LIBVLC_FIXUPS_H 1

Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
29
#ifndef HAVE_STRDUP
30
31
32
33
34
# include <string.h>
# include <stdlib.h>
static inline char *strdup (const char *str)
{
    size_t len = strlen (str) + 1;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
35
    char *res = (char *)malloc (len);
36
37
38
    if (res) memcpy (res, str, len);
    return res;
}
39
40
#endif

41
42
#ifdef WIN32
# include <string.h>
43
# include <stdlib.h>
44
45
46
47
48
49
50
51
52
53
54
/**
 * vlc_fix_format_string:
 * @format: address of format string to fix (format string is not modified)
 *
 * Windows' printf doesn't support %z size modifiers.
 * Fix a *printf format string to make it safe for mingw/MSVCRT run times:
 *  %z* (not supported in MSVCRT) -> either %I64* or %I32.
 *
 * Returns: 1 if *format must be free()d; 0 otherwise
 */
static inline int vlc_fix_format_string (const char **format)
55
{
56
57
58
    int n = 0;
    const char *tmp = *format;
    while ((tmp = strstr (tmp, "%z")) != NULL)
59
60
    {
        n++;
61
        tmp += 2;
62
    }
63
64
    if (!n)
        return 0;
65

66
67
68
69
70
71
    char *dst = (char*)malloc (strlen (*format) + 2*n + 1);
    if (!dst)
    {
        *format = "vlc_fix_format_string: due to malloc failure, unable to fix unsafe string";
        return 0;
    }
72

73
74
    const char *src = *format;
    *format = dst;
75
76
    while ((tmp = strstr (src, "%z")) != NULL)
    {
77
        /* NB, don't use %l*, as this is buggy in mingw*/
78
79
80
        size_t d = tmp - src;
        memcpy (dst, src, d);
        dst += d;
81
82
83
84
85
86
87
88
89
        *dst++ = '%';
# ifdef WIN64
        *dst++ = 'I';
        *dst++ = '6';
        *dst++ = '4';
# else /* ie: WIN32 */
        /* on win32, since the default size is 32bit, dont specify
         * a modifer.  (I32 isn't on wince, l doesn't work on mingw) */
# endif
90
91
92
        src = tmp + 2;
    }
    strcpy (dst, src);
93
    return 1;
94
95
96
97
98
99
100
}

# include <stdio.h>
# include <stdarg.h>

static inline int vlc_vprintf (const char *format, va_list ap)
{
101
102
103
    int must_free = vlc_fix_format_string (&format);
    int ret = vprintf (format, ap);
    if (must_free) free ((char *)format);
104
105
106
107
108
109
    return ret;
}
# define vprintf vlc_vprintf

static inline int vlc_vfprintf (FILE *stream, const char *format, va_list ap)
{
110
111
112
    int must_free = vlc_fix_format_string (&format);
    int ret = vfprintf (stream, format, ap);
    if (must_free) free ((char *)format);
113
114
115
116
117
118
    return ret;
}
# define vfprintf vlc_vfprintf

static inline int vlc_vsprintf (char *str, const char *format, va_list ap)
{
119
120
121
    int must_free = vlc_fix_format_string (&format);
    int ret = vsprintf (str, format, ap);
    if (must_free) free ((char *)format);
122
123
124
125
126
127
    return ret;
}
# define vsprintf vlc_vsprintf

static inline int vlc_vsnprintf (char *str, size_t size, const char *format, va_list ap)
{
128
    int must_free = vlc_fix_format_string (&format);
129
130
131
132
    /* traditionally, MSVCRT has provided vsnprintf as _vsnprintf;
     * to 'aid' portability/standards compliance, mingw provides a
     * static version of vsnprintf that is buggy.  Be sure to use
     * MSVCRT version, at least it behaves as expected */
133
134
    int ret = _vsnprintf (str, size, format, ap);
    if (must_free) free ((char *)format);
135
136
137
138
139
140
141
142
143
    return ret;
}
# define vsnprintf vlc_vsnprintf

static inline int vlc_printf (const char *format, ...)
{
    va_list ap;
    int ret;
    va_start (ap, format);
144
    ret = vlc_vprintf (format, ap);
145
146
147
148
149
150
151
152
153
154
    va_end (ap);
    return ret;
}
# define printf(...) vlc_printf(__VA_ARGS__)

static inline int vlc_fprintf (FILE *stream, const char *format, ...)
{
    va_list ap;
    int ret;
    va_start (ap, format);
155
    ret = vlc_vfprintf (stream, format, ap);
156
157
158
159
160
    va_end (ap);
    return ret;
}
# define fprintf vlc_fprintf

161
#if 0
162
163
164
165
166
static inline int vlc_sprintf (char *str, const char *format, ...)
{
    va_list ap;
    int ret;
    va_start (ap, format);
167
    ret = vlc_vsprintf (str, format, ap);
168
169
170
171
    va_end (ap);
    return ret;
}
# define sprintf vlc_sprintf
172
#endif
173
174
175
176
177
178

static inline int vlc_snprintf (char *str, size_t size, const char *format, ...)
{
    va_list ap;
    int ret;
    va_start (ap, format);
179
    ret = vlc_vsnprintf (str, size, format, ap);
180
181
182
    va_end (ap);
    return ret;
}
183
184
/* win32: snprintf must always be vlc_snprintf or _snprintf,
 * see comment in vlc_vsnprintf */
185
186
187
188
189
190
191
# define snprintf vlc_snprintf

/* Make sure we don't use flawed vasprintf or asprintf either */
# undef HAVE_VASPRINTF
# undef HAVE_ASPRINTF
#endif

192
#ifndef HAVE_VASPRINTF
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
193
194
# include <stdio.h>
# include <stdlib.h>
195
196
197
# include <stdarg.h>
static inline int vasprintf (char **strp, const char *fmt, va_list ap)
{
geal's avatar
geal committed
198
#ifndef UNDER_CE
199
    int len = vsnprintf (NULL, 0, fmt, ap) + 1;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
200
    char *res = (char *)malloc (len);
201
202
203
    if (res == NULL)
        return -1;
    *strp = res;
Ken Self's avatar
Ken Self committed
204
    return vsnprintf (res, len, fmt, ap);
geal's avatar
geal committed
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
#else
    /* HACK: vsnprintf in the WinCE API behaves like
     * the one in glibc 2.0 and doesn't return the number of characters
     * it needed to copy the string.
     * cf http://msdn.microsoft.com/en-us/library/1kt27hek.aspx
     * and cf the man page of vsnprintf
     *
     Guess we need no more than 50 bytes. */
    int n, size = 50;
    char *res, *np;

    if ((res = (char *) malloc (size)) == NULL)
        return -1;

    while (1)
    {
        n = vsnprintf (res, size, fmt, ap);

        /* If that worked, return the string. */
        if (n > -1 && n < size)
        {
            *strp = res;
            return n;
        }

        /* Else try again with more space. */
231
        size *= 2;  /* twice the old size */
geal's avatar
geal committed
232
233
234
235
236
237
238
239
240
241
242
243
244

        if ((np = (char *) realloc (res, size)) == NULL)
        {
            free(res);
            return -1;
        }
        else
        {
            res = np;
        }

    }
#endif /* UNDER_CE */
245
}
246
247
248
#endif

#ifndef HAVE_ASPRINTF
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
249
# include <stdio.h>
250
251
252
253
254
# include <stdarg.h>
static inline int asprintf (char **strp, const char *fmt, ...)
{
    va_list ap;
    int ret;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
255
    va_start (ap, fmt);
256
257
258
259
    ret = vasprintf (strp, fmt, ap);
    va_end (ap);
    return ret;
}
260
261
#endif

262
#ifndef HAVE_STRNLEN
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
263
# include <string.h>
264
265
static inline size_t strnlen (const char *str, size_t max)
{
Rafaël Carré's avatar
Rafaël Carré committed
266
    const char *end = (const char *) memchr (str, 0, max);
267
268
269
270
    return end ? (size_t)(end - str) : max;
}
#endif

271
#ifndef HAVE_STRNDUP
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
272
273
# include <string.h>
# include <stdlib.h>
274
275
static inline char *strndup (const char *str, size_t max)
{
276
    size_t len = strnlen (str, max);
Rafaël Carré's avatar
Rafaël Carré committed
277
    char *res = (char *) malloc (len + 1);
278
279
280
281
282
283
284
    if (res)
    {
        memcpy (res, str, len);
        res[len] = '\0';
    }
    return res;
}
285
286
287
288
289
290
291
#endif

#ifndef HAVE_STRLCPY
# define strlcpy vlc_strlcpy
#endif

#ifndef HAVE_STRTOF
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
292
# define strtof( a, b ) ((float)strtod (a, b))
293
294
#endif

Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
295
296
#ifndef HAVE_ATOF
# define atof( str ) (strtod ((str), (char **)NULL, 10))
297
298
299
300
301
302
#endif

#ifndef HAVE_STRTOLL
# define strtoll vlc_strtoll
#endif

303
#ifndef HAVE_STRSEP
Rémi Denis-Courmont's avatar
typo    
Rémi Denis-Courmont committed
304
static inline char *strsep( char **ppsz_string, const char *psz_delimiters )
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
{
    char *psz_string = *ppsz_string;
    if( !psz_string )
        return NULL;

    char *p = strpbrk( psz_string, psz_delimiters );
    if( !p )
    {
        *ppsz_string = NULL;
        return psz_string;
    }
    *p++ = '\0';

    *ppsz_string = p;
    return psz_string;
}
321
322
#endif

Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
323
324
325
326
#ifndef HAVE_ATOLL
# define atoll( str ) (strtoll ((str), (char **)NULL, 10))
#endif

327
#ifndef HAVE_LLDIV
328
329
330
331
332
333
334
335
336
337
typedef struct {
    long long quot; /* Quotient. */
    long long rem;  /* Remainder. */
} lldiv_t;

static inline lldiv_t lldiv (long long numer, long long denom)
{
    lldiv_t d = { .quot = numer / denom, .rem = numer % denom };
    return d;
}
338
339
340
341
342
343
344
345
#endif

#ifndef HAVE_SCANDIR
# define scandir vlc_scandir
# define alphasort vlc_alphasort
#endif

#ifndef HAVE_GETENV
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
346
static inline char *getenv (const char *name)
347
348
349
350
{
    (void)name;
    return NULL;
}
351
352
353
354
#endif

#ifndef HAVE_STRCASECMP
# ifndef HAVE_STRICMP
355
356
357
358
359
360
#  include <ctype.h>
static inline int strcasecmp (const char *s1, const char *s2)
{
    for (size_t i = 0;; i++)
    {
        int d = tolower (s1[i]) - tolower (s2[i]);
361
        if (d || !s1[i]) return d;
362
363
364
    }
    return 0;
}
365
366
367
368
369
370
371
# else
#  define strcasecmp stricmp
# endif
#endif

#ifndef HAVE_STRNCASECMP
# ifndef HAVE_STRNICMP
372
373
374
375
376
377
#  include <ctype.h>
static inline int strncasecmp (const char *s1, const char *s2, size_t n)
{
    for (size_t i = 0; i < n; i++)
    {
        int d = tolower (s1[i]) - tolower (s2[i]);
378
        if (d || !s1[i]) return d;
379
380
381
    }
    return 0;
}
382
383
384
385
386
387
388
389
390
391
392
393
394
# else
#  define strncasecmp strnicmp
# endif
#endif

#ifndef HAVE_STRCASESTR
# ifndef HAVE_STRISTR
#  define strcasestr vlc_strcasestr
# else
#  define strcasestr stristr
# endif
#endif

395
396
397
398
#ifndef HAVE_LOCALTIME_R
/* If localtime_r() is not provided, we assume localtime() uses
 * thread-specific storage. */
# include <time.h>
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
399
static inline struct tm *localtime_r (const time_t *timep, struct tm *result)
400
401
402
403
404
405
406
407
{
    struct tm *s = localtime (timep);
    if (s == NULL)
        return NULL;

    *result = *s;
    return result;
}
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
408
409
410
411
412
413
414
415
416
static inline struct tm *gmtime_r (const time_t *timep, struct tm *result)
{
    struct tm *s = gmtime (timep);
    if (s == NULL)
        return NULL;

    *result = *s;
    return result;
}
417
418
#endif

419
420
421
422
423
424
425
/* Alignment of critical static data structures */
#ifdef ATTRIBUTE_ALIGNED_MAX
#   define ATTR_ALIGN(align) __attribute__ ((__aligned__ ((ATTRIBUTE_ALIGNED_MAX < align) ? ATTRIBUTE_ALIGNED_MAX : align)))
#else
#   define ATTR_ALIGN(align)
#endif

426
427
428
429
430
431
432
#ifndef HAVE_USELOCALE
typedef void *locale_t;
# define newlocale( a, b, c ) ((locale_t)0)
# define uselocale( a ) ((locale_t)0)
# define freelocale( a ) (void)0
#endif

433
#ifdef WIN32
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
434
# include <dirent.h>
435
436
437
438
439
# define opendir Use_utf8_opendir_or_vlc_wopendir_instead!
# define readdir Use_utf8_readdir_or_vlc_wreaddir_instead!
# define closedir vlc_wclosedir
#endif

440
/* libintl support */
441
#define _(str) vlc_gettext (str)
442
443
444
445
446
447
448
449

#if defined (ENABLE_NLS)
# include <libintl.h>
#endif

#define N_(str) gettext_noop (str)
#define gettext_noop(str) (str)

geal's avatar
geal committed
450
451
452
453
454
455
456
457
#ifdef UNDER_CE
static inline void rewind ( FILE *stream )
{
    fseek(stream, 0L, SEEK_SET);
    clearerr(stream);
}
#endif

458
#endif /* !LIBVLC_FIXUPS_H */