]> git.sesse.net Git - vlc/blob - src/extras/libc.c
* ALL: added vlc_asprintf() to our libc.
[vlc] / src / extras / libc.c
1 /*****************************************************************************
2  * libc.c: Extra libc function for some systems.
3  *****************************************************************************
4  * Copyright (C) 2002 VideoLAN
5  * $Id: libc.c,v 1.14 2004/01/09 18:32:04 gbazin Exp $
6  *
7  * Authors: Jon Lech Johansen <jon-vl@nanocrew.net>
8  *          Samuel Hocevar <sam@zoy.org>
9  *
10  * This program is free software; you can redistribute it and/or modify
11  * it under the terms of the GNU General Public License as published by
12  * the Free Software Foundation; either version 2 of the License, or
13  * (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program; if not, write to the Free Software
22  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
23  *****************************************************************************/
24 #include <string.h>                                              /* strdup() */
25 #include <stdlib.h>
26
27 #include <vlc/vlc.h>
28
29 /*****************************************************************************
30  * getenv: just in case, but it should never be called
31  *****************************************************************************/
32 #if !defined( HAVE_GETENV )
33 char *vlc_getenv( const char *name )
34 {
35     return NULL;
36 }
37 #endif
38
39 /*****************************************************************************
40  * strdup: returns a malloc'd copy of a string
41  *****************************************************************************/
42 #if !defined( HAVE_STRDUP )
43 char *vlc_strdup( const char *string )
44 {
45     return strndup( string, strlen( string ) );
46 }
47 #endif
48
49 /*****************************************************************************
50  * strndup: returns a malloc'd copy of at most n bytes of string
51  * Does anyone know whether or not it will be present in Jaguar?
52  *****************************************************************************/
53 #if !defined( HAVE_STRNDUP )
54 char *vlc_strndup( const char *string, size_t n )
55 {
56     char *psz;
57     size_t len = strlen( string );
58
59     len = __MIN( len, n );
60     psz = (char*)malloc( len + 1 );
61
62     if( psz != NULL )
63     {
64         memcpy( (void*)psz, (const void*)string, len );
65         psz[ len ] = 0;
66     }
67
68     return psz;
69 }
70 #endif
71
72 /*****************************************************************************
73  * strcasecmp: compare two strings ignoring case
74  *****************************************************************************/
75 #if !defined( HAVE_STRCASECMP ) && !defined( HAVE_STRICMP )
76 int vlc_strcasecmp( const char *s1, const char *s2 )
77 {
78     int i_delta = 0;
79
80     while( !i_delta && *s1 && *s2 )
81     {
82         i_delta = *s1 - *s2;
83
84         if( *s1 >= 'A' && *s1 <= 'Z' )
85         {
86             i_delta -= 'A' - 'a';
87         }
88
89         if( *s2 >= 'A' && *s2 <= 'Z' )
90         {
91             i_delta += 'A' - 'a';
92         }
93
94         s1++; s2++;
95     }
96
97     return i_delta;
98 }
99 #endif
100
101 /*****************************************************************************
102  * strncasecmp: compare n chars from two strings ignoring case
103  *****************************************************************************/
104 #if !defined( HAVE_STRNCASECMP ) && !defined( HAVE_STRNICMP )
105 int vlc_strncasecmp( const char *s1, const char *s2, size_t n )
106 {
107     int i_delta = 0;
108
109     while( n-- && !i_delta && *s1 )
110     {
111         i_delta = *s1 - *s2;
112
113         if( *s1 >= 'A' && *s1 <= 'Z' )
114         {
115             i_delta -= 'A' - 'a';
116         }
117
118         if( *s2 >= 'A' && *s2 <= 'Z' )
119         {
120             i_delta += 'A' - 'a';
121         }
122
123         s1++; s2++;
124     }
125
126     return i_delta;
127 }
128 #endif
129
130 /*****************************************************************************
131  * vasprintf:
132  *****************************************************************************/
133 #if !defined(HAVE_VASPRINTF) || defined(SYS_DARWIN) || defined(SYS_BEOS)
134 int vlc_vasprintf(char **strp, const char *fmt, va_list ap)
135 {
136     /* Guess we need no more than 100 bytes. */
137     int     i_size = 100;
138     char    *p = malloc( i_size );
139     int     n;
140
141     if( p == NULL )
142     {
143         *strp = NULL;
144         return -1;
145     }
146
147     for( ;; )
148     {
149         /* Try to print in the allocated space. */
150         n = vsnprintf( p, i_size, fmt, ap );
151
152         /* If that worked, return the string. */
153         if (n > -1 && n < i_size)
154         {
155             *strp = p;
156             return strlen( p );
157         }
158         /* Else try again with more space. */
159         if (n > -1)    /* glibc 2.1 */
160         {
161            i_size = n+1; /* precisely what is needed */
162         }
163         else           /* glibc 2.0 */
164         {
165            i_size *= 2;  /* twice the old size */
166         }
167         if( (p = realloc( p, i_size ) ) == NULL)
168         {
169             *strp = NULL;
170             return -1;
171         }
172     }
173 }
174 #endif
175
176 /*****************************************************************************
177  * asprintf:
178  *****************************************************************************/
179 #if !defined(HAVE_ASPRINTF) || defined(SYS_DARWIN) || defined(SYS_BEOS)
180 int vlc_asprintf( char **strp, const char *fmt, ... )
181 {
182     va_list args;
183     va_start( args, fmt );
184     vasprintf( strp, fmt, args );
185     va_end( args );
186 }
187 #endif
188
189 /*****************************************************************************
190  * atof: convert a string to a double.
191  *****************************************************************************/
192 #if !defined( HAVE_ATOF )
193 double vlc_atof( const char *nptr )
194 {
195     double f_result;
196     wchar_t *psz_tmp;
197     int i_len = strlen( nptr ) + 1;
198
199     psz_tmp = malloc( i_len * sizeof(wchar_t) );
200     MultiByteToWideChar( CP_ACP, 0, nptr, -1, psz_tmp, i_len );
201     f_result = wcstod( psz_tmp, NULL );
202     free( psz_tmp );
203
204     return f_result;
205 }
206 #endif
207
208 /*****************************************************************************
209  * atoll: convert a string to a 64 bits int.
210  *****************************************************************************/
211 #if !defined( HAVE_ATOLL )
212 int64_t vlc_atoll( const char *str )
213 {
214     int64_t i_value = 0;
215     int sign = 1;
216
217     if( *str == '-' )
218     {
219         sign = -1;
220     }
221
222     while( *str >= '0' && *str <= '9' )
223     {
224         i_value = i_value * 10 + ( *str - '0' );
225     }
226
227     return i_value * sign;
228 }
229 #endif
230
231 /*****************************************************************************
232  * lseek: reposition read/write file offset.
233  *****************************************************************************
234  * FIXME: this cast sucks!
235  *****************************************************************************/
236 #if !defined( HAVE_LSEEK )
237 off_t vlc_lseek( int fildes, off_t offset, int whence )
238 {
239     return SetFilePointer( (HANDLE)fildes, (long)offset, NULL, whence );
240 }
241 #endif
242
243 /*****************************************************************************
244  * dgettext: gettext for plugins.
245  *****************************************************************************/
246 char *vlc_dgettext( const char *package, const char *msgid )
247 {
248 #if defined( ENABLE_NLS ) \
249      && ( defined(HAVE_GETTEXT) || defined(HAVE_INCLUDED_GETTEXT) )
250     return dgettext( package, msgid );
251 #else
252     return (char *)msgid;
253 #endif
254 }
255
256 /*****************************************************************************
257  * count_utf8_string: returns the number of characters in the string.
258  *****************************************************************************/
259 static int count_utf8_string( const char *psz_string )
260 {
261     int i = 0, i_count = 0;
262     while( psz_string[ i ] != 0 )
263     {
264         if( ((unsigned char *)psz_string)[ i ] <  0x80UL ) i_count++;
265         i++;
266     }
267     return i_count;
268 }
269
270 /*****************************************************************************
271  * wraptext: inserts \n at convenient places to wrap the text.
272  *           Returns the modified string in a new buffer.
273  *****************************************************************************/
274 char *vlc_wraptext( const char *psz_text, int i_line, vlc_bool_t b_utf8 )
275 {
276     int i_len;
277     char *psz_line, *psz_new_text;
278
279     psz_line = psz_new_text = strdup( psz_text );
280
281     if( b_utf8 )
282         i_len = count_utf8_string( psz_text );
283     else
284         i_len = strlen( psz_text );
285
286     while( i_len > i_line )
287     {
288         /* Look if there is a newline somewhere. */
289         char *psz_parser = psz_line;
290         int i_count = 0;
291         while( i_count <= i_line && *psz_parser != '\n' )
292         {
293             if( b_utf8 )
294             {
295                 while( *((unsigned char *)psz_parser) >= 0x80UL ) psz_parser++;
296             }
297             psz_parser++;
298             i_count++;
299         }
300         if( *psz_parser == '\n' )
301         {
302             i_len -= (i_count + 1);
303             psz_line = psz_parser + 1;
304             continue;
305         }
306
307         /* Find the furthest space. */
308         while( psz_parser > psz_line && *psz_parser != ' ' )
309         {
310             if( b_utf8 )
311             {
312                 while( *((unsigned char *)psz_parser) >= 0x80UL ) psz_parser--;
313             }
314             psz_parser--;
315             i_count--;
316         }
317         if( *psz_parser == ' ' )
318         {
319             *psz_parser = '\n';
320             i_len -= (i_count + 1);
321             psz_line = psz_parser + 1;
322             continue;
323         }
324
325         /* Wrapping has failed. Find the first space or newline */
326         while( i_count < i_len && *psz_parser != ' ' && *psz_parser != '\n' )
327         {
328             if( b_utf8 )
329             {
330                 while( *((unsigned char *)psz_parser) >= 0x80UL ) psz_parser++;
331             }
332             psz_parser++;
333             i_count++;
334         }
335         if( i_count < i_len ) *psz_parser = '\n';
336         i_len -= (i_count + 1);
337         psz_line = psz_parser + 1;
338     }
339
340     return psz_new_text;
341 }