/***************************************************************************** * strings.c: String related functions ***************************************************************************** * Copyright (C) 2006 VLC authors and VideoLAN * Copyright (C) 2008-2009 Rémi Denis-Courmont * * Authors: Antoine Cellerier * Daniel Stranger * Rémi Denis-Courmont * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser 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. *****************************************************************************/ /***************************************************************************** * Preamble *****************************************************************************/ #ifdef HAVE_CONFIG_H # include "config.h" #endif #include #include /* Needed by vlc_strftime */ #include #include #include #include #ifndef HAVE_STRCOLL # define strcoll strcasecmp #endif /* Needed by vlc_strfplayer */ #include #include #include #include #include #include #include #include #include #include "../input/input_internal.h" static const struct xml_entity_s { char psz_entity[8]; char psz_char[4]; } xml_entities[] = { /* Important: this list has to be in alphabetical order (psz_entity-wise) */ { "AElig;", "Æ" }, { "Aacute;", "Á" }, { "Acirc;", "Â" }, { "Agrave;", "À" }, { "Aring;", "Å" }, { "Atilde;", "Ã" }, { "Auml;", "Ä" }, { "Ccedil;", "Ç" }, { "Dagger;", "‡" }, { "ETH;", "Ð" }, { "Eacute;", "É" }, { "Ecirc;", "Ê" }, { "Egrave;", "È" }, { "Euml;", "Ë" }, { "Iacute;", "Í" }, { "Icirc;", "Î" }, { "Igrave;", "Ì" }, { "Iuml;", "Ï" }, { "Ntilde;", "Ñ" }, { "OElig;", "Œ" }, { "Oacute;", "Ó" }, { "Ocirc;", "Ô" }, { "Ograve;", "Ò" }, { "Oslash;", "Ø" }, { "Otilde;", "Õ" }, { "Ouml;", "Ö" }, { "Scaron;", "Š" }, { "THORN;", "Þ" }, { "Uacute;", "Ú" }, { "Ucirc;", "Û" }, { "Ugrave;", "Ù" }, { "Uuml;", "Ü" }, { "Yacute;", "Ý" }, { "Yuml;", "Ÿ" }, { "aacute;", "á" }, { "acirc;", "â" }, { "acute;", "´" }, { "aelig;", "æ" }, { "agrave;", "à" }, { "amp;", "&" }, { "apos;", "'" }, { "aring;", "å" }, { "atilde;", "ã" }, { "auml;", "ä" }, { "bdquo;", "„" }, { "brvbar;", "¦" }, { "ccedil;", "ç" }, { "cedil;", "¸" }, { "cent;", "¢" }, { "circ;", "ˆ" }, { "copy;", "©" }, { "curren;", "¤" }, { "dagger;", "†" }, { "deg;", "°" }, { "divide;", "÷" }, { "eacute;", "é" }, { "ecirc;", "ê" }, { "egrave;", "è" }, { "eth;", "ð" }, { "euml;", "ë" }, { "euro;", "€" }, { "frac12;", "½" }, { "frac14;", "¼" }, { "frac34;", "¾" }, { "gt;", ">" }, { "hellip;", "…" }, { "iacute;", "í" }, { "icirc;", "î" }, { "iexcl;", "¡" }, { "igrave;", "ì" }, { "iquest;", "¿" }, { "iuml;", "ï" }, { "laquo;", "«" }, { "ldquo;", "“" }, { "lsaquo;", "‹" }, { "lsquo;", "‘" }, { "lt;", "<" }, { "macr;", "¯" }, { "mdash;", "—" }, { "micro;", "µ" }, { "middot;", "·" }, { "nbsp;", "\xc2\xa0" }, { "ndash;", "–" }, { "not;", "¬" }, { "ntilde;", "ñ" }, { "oacute;", "ó" }, { "ocirc;", "ô" }, { "oelig;", "œ" }, { "ograve;", "ò" }, { "ordf;", "ª" }, { "ordm;", "º" }, { "oslash;", "ø" }, { "otilde;", "õ" }, { "ouml;", "ö" }, { "para;", "¶" }, { "permil;", "‰" }, { "plusmn;", "±" }, { "pound;", "£" }, { "quot;", "\"" }, { "raquo;", "»" }, { "rdquo;", "”" }, { "reg;", "®" }, { "rsaquo;", "›" }, { "rsquo;", "’" }, { "sbquo;", "‚" }, { "scaron;", "š" }, { "sect;", "§" }, { "shy;", "­" }, { "sup1;", "¹" }, { "sup2;", "²" }, { "sup3;", "³" }, { "szlig;", "ß" }, { "thorn;", "þ" }, { "tilde;", "˜" }, { "times;", "×" }, { "trade;", "™" }, { "uacute;", "ú" }, { "ucirc;", "û" }, { "ugrave;", "ù" }, { "uml;", "¨" }, { "uuml;", "ü" }, { "yacute;", "ý" }, { "yen;", "¥" }, { "yuml;", "ÿ" }, }; static int cmp_entity (const void *key, const void *elem) { const struct xml_entity_s *ent = elem; const char *name = key; return strncmp (name, ent->psz_entity, strlen (ent->psz_entity)); } void vlc_xml_decode( char *psz_value ) { char *p_pos = psz_value; while ( *psz_value ) { if( *psz_value == '&' ) { if( psz_value[1] == '#' ) { /* &#DDD; or &#xHHHH; Unicode code point */ char *psz_end; unsigned long cp; if( psz_value[2] == 'x' ) /* The x must be lower-case. */ cp = strtoul( psz_value + 3, &psz_end, 16 ); else cp = strtoul( psz_value + 2, &psz_end, 10 ); if( *psz_end == ';' ) { psz_value = psz_end + 1; if( cp == 0 ) (void)0; /* skip nulls */ else if( cp <= 0x7F ) { *p_pos = cp; } else /* Unicode code point outside ASCII. * &#xxx; representation is longer than UTF-8 :) */ if( cp <= 0x7FF ) { *p_pos++ = 0xC0 | (cp >> 6); *p_pos = 0x80 | (cp & 0x3F); } else if( cp <= 0xFFFF ) { *p_pos++ = 0xE0 | (cp >> 12); *p_pos++ = 0x80 | ((cp >> 6) & 0x3F); *p_pos = 0x80 | (cp & 0x3F); } else if( cp <= 0x1FFFFF ) /* Outside the BMP */ { /* Unicode stops at 10FFFF, but who cares? */ *p_pos++ = 0xF0 | (cp >> 18); *p_pos++ = 0x80 | ((cp >> 12) & 0x3F); *p_pos++ = 0x80 | ((cp >> 6) & 0x3F); *p_pos = 0x80 | (cp & 0x3F); } } else { /* Invalid entity number */ *p_pos = *psz_value; psz_value++; } } else { /* Well-known XML entity */ const struct xml_entity_s *ent; ent = bsearch (psz_value + 1, xml_entities, ARRAY_SIZE (xml_entities), sizeof (*ent), cmp_entity); if (ent != NULL) { size_t olen = strlen (ent->psz_char); memcpy (p_pos, ent->psz_char, olen); p_pos += olen - 1; psz_value += strlen (ent->psz_entity) + 1; } else { /* No match */ *p_pos = *psz_value; psz_value++; } } } else { *p_pos = *psz_value; psz_value++; } p_pos++; } *p_pos = '\0'; } char *vlc_xml_encode (const char *str) { struct vlc_memstream stream; size_t n; uint32_t cp; assert(str != NULL); vlc_memstream_open(&stream); while ((n = vlc_towc (str, &cp)) != 0) { if (unlikely(n == (size_t)-1)) { if (vlc_memstream_close(&stream) == 0) free(stream.ptr); errno = EILSEQ; return NULL; } switch (cp) { case '\"': vlc_memstream_puts(&stream, """); break; case '&': vlc_memstream_puts(&stream, "&"); break; case '\'': vlc_memstream_puts(&stream, "'"); break; case '<': vlc_memstream_puts(&stream, "<"); break; case '>': vlc_memstream_puts(&stream, ">"); break; default: if (cp < 32) /* C0 code not allowed (except 9, 10 and 13) */ break; if (cp >= 128 && cp < 160) /* C1 code encoded (except 133) */ { vlc_memstream_printf(&stream, "&#%"PRIu32";", cp); break; } /* fall through */ case 9: case 10: case 13: case 133: vlc_memstream_write(&stream, str, n); break; } str += n; } if (vlc_memstream_close(&stream)) return NULL; return stream.ptr; } /* Base64 encoding */ char *vlc_b64_encode_binary(const void *src, size_t length) { static const char b64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; const unsigned char *in = src; char *dst = malloc((((length + 2) / 3) * 4) + 1); char *out = dst; if (unlikely(dst == NULL)) return NULL; while (length >= 3) { /* pops (up to) 3 bytes of input, push 4 bytes */ uint_fast32_t v = (in[0] << 16) | (in[1] << 8) | in[2]; *(out++) = b64[(v >> 18)]; *(out++) = b64[(v >> 12) & 0x3f]; *(out++) = b64[(v >> 6) & 0x3f]; *(out++) = b64[(v >> 0) & 0x3f]; in += 3; length -= 3; } switch (length) { case 2: { uint_fast16_t v = (in[0] << 8) | in[1]; *(out++) = b64[(v >> 10)]; *(out++) = b64[(v >> 4) & 0x3f]; *(out++) = b64[(v << 2) & 0x3f]; *(out++) = '='; break; } case 1: { uint_fast8_t v = in[0]; *(out++) = b64[(v >> 2)]; *(out++) = b64[(v << 4) & 0x3f]; *(out++) = '='; *(out++) = '='; break; } } *out = '\0'; return dst; } char *vlc_b64_encode(const char *src) { if (src == NULL) src = ""; return vlc_b64_encode_binary(src, strlen(src)); } /* Base64 decoding */ size_t vlc_b64_decode_binary_to_buffer(void *dst, size_t size, const char *restrict src) { static const signed char b64[256] = { -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* 00-0F */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* 10-1F */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,62,-1,-1,-1,63, /* 20-2F */ 52,53,54,55,56,57,58,59,60,61,-1,-1,-1,-1,-1,-1, /* 30-3F */ -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,10,11,12,13,14, /* 40-4F */ 15,16,17,18,19,20,21,22,23,24,25,-1,-1,-1,-1,-1, /* 50-5F */ -1,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40, /* 60-6F */ 41,42,43,44,45,46,47,48,49,50,51,-1,-1,-1,-1,-1, /* 70-7F */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* 80-8F */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* 90-9F */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* A0-AF */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* B0-BF */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* C0-CF */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* D0-DF */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* E0-EF */ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, /* F0-FF */ }; const unsigned char *in = (const unsigned char *)src; unsigned char *out = dst; signed char prev; int shift = 0; static_assert (CHAR_BIT == 8, "Oops"); while (size > 0) { const signed char cur = b64[*(in++)]; if (cur < 0) break; if (shift != 0) { *(out++) = (prev << shift) | (cur >> (6 - shift)); size--; } prev = cur; shift = (shift + 2) & 7; } return out - (unsigned char *)dst; } size_t vlc_b64_decode_binary( uint8_t **pp_dst, const char *psz_src ) { const int i_src = strlen( psz_src ); uint8_t *p_dst; *pp_dst = p_dst = malloc( i_src ); if( !p_dst ) return 0; return vlc_b64_decode_binary_to_buffer( p_dst, i_src, psz_src ); } char *vlc_b64_decode( const char *psz_src ) { const int i_src = strlen( psz_src ); char *p_dst = malloc( i_src + 1 ); size_t i_dst; if( !p_dst ) return NULL; i_dst = vlc_b64_decode_binary_to_buffer( (uint8_t*)p_dst, i_src, psz_src ); p_dst[i_dst] = '\0'; return p_dst; } char *vlc_strftime( const char *tformat ) { time_t curtime; struct tm loctime; if (strcmp (tformat, "") == 0) return strdup (""); /* corner case w.r.t. strftime() return value */ /* Get the current time. */ time( &curtime ); /* Convert it to local time representation. */ localtime_r( &curtime, &loctime ); for (size_t buflen = strlen (tformat) + 32;; buflen += 32) { char *str = malloc (buflen); if (str == NULL) return NULL; size_t len = strftime (str, buflen, tformat, &loctime); if (len > 0) { char *ret = realloc (str, len + 1); return ret ? ret : str; /* <- this cannot fail */ } free (str); } vlc_assert_unreachable (); } static void write_duration(struct vlc_memstream *stream, vlc_tick_t duration) { lldiv_t d; long long sec; duration /= CLOCK_FREQ; d = lldiv(duration, 60); sec = d.rem; d = lldiv(d.quot, 60); vlc_memstream_printf(stream, "%02lld:%02lld:%02lld", d.quot, d.rem, sec); } static int write_meta(struct vlc_memstream *stream, input_item_t *item, vlc_meta_type_t type) { if (item == NULL) return EOF; char *value = input_item_GetMeta(item, type); if (value == NULL) return EOF; vlc_memstream_puts(stream, value); free(value); return 0; } char *vlc_strfplayer(vlc_player_t *player, input_item_t *item, const char *s) { struct vlc_memstream stream[1]; char c; bool b_is_format = false; bool b_empty_if_na = false; assert(s != NULL); if (!item && player) item = vlc_player_GetCurrentMedia(player); vlc_memstream_open(stream); while ((c = *s) != '\0') { s++; if (!b_is_format) { if (c == '$') { b_is_format = true; b_empty_if_na = false; continue; } vlc_memstream_putc(stream, c); continue; } b_is_format = false; switch (c) { case 'a': write_meta(stream, item, vlc_meta_Artist); break; case 'b': write_meta(stream, item, vlc_meta_Album); break; case 'c': write_meta(stream, item, vlc_meta_Copyright); break; case 'd': write_meta(stream, item, vlc_meta_Description); break; case 'e': write_meta(stream, item, vlc_meta_EncodedBy); break; case 'f': if (item != NULL) { vlc_mutex_lock(&item->lock); if (item->p_stats != NULL) vlc_memstream_printf(stream, "%"PRIi64, item->p_stats->i_displayed_pictures); else if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); vlc_mutex_unlock(&item->lock); } else if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; case 'g': write_meta(stream, item, vlc_meta_Genre); break; case 'l': write_meta(stream, item, vlc_meta_Language); break; case 'n': write_meta(stream, item, vlc_meta_TrackNumber); break; case 'o': write_meta(stream, item, vlc_meta_TrackTotal); break; case 'p': if (item == NULL) break; { char *value = input_item_GetNowPlayingFb(item); if (value == NULL) break; vlc_memstream_puts(stream, value); free(value); } break; case 'r': write_meta(stream, item, vlc_meta_Rating); break; case 's': { char *lang = NULL; if (player != NULL) lang = vlc_player_GetCategoryLanguage(player, SPU_ES); if (lang != NULL) { vlc_memstream_puts(stream, lang); free(lang); } else if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; } case 't': write_meta(stream, item, vlc_meta_Title); break; case 'u': write_meta(stream, item, vlc_meta_URL); break; case 'A': write_meta(stream, item, vlc_meta_Date); break; case 'B': { if (player) { const struct vlc_player_track *track = vlc_player_GetSelectedTrack(player, AUDIO_ES); if (track) { vlc_memstream_printf(stream, "%u", track->fmt.i_bitrate); break; } } if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; } case 'C': if (player) { ssize_t chapter = vlc_player_GetSelectedChapterIdx(player); if (chapter != -1) { vlc_memstream_printf(stream, "%zd", chapter); break; } } if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; case 'D': if (item != NULL) write_duration(stream, input_item_GetDuration(item)); else if (!b_empty_if_na) vlc_memstream_puts(stream, "--:--:--"); break; case 'F': if (item != NULL) { char *uri = input_item_GetURI(item); if (uri != NULL) { vlc_memstream_puts(stream, uri); free(uri); } } break; case 'I': if (player) { ssize_t title = vlc_player_GetSelectedTitleIdx(player); if (title != -1) { vlc_memstream_printf(stream, "%zd", title); break; } } if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; case 'L': if (player) { vlc_tick_t length = vlc_player_GetLength(player); vlc_tick_t time = vlc_player_GetTime(player); if (length != VLC_TICK_INVALID && time != VLC_TICK_INVALID) write_duration(stream, length - time); } if (!b_empty_if_na) vlc_memstream_puts(stream, "--:--:--"); break; case 'N': if (item != NULL) { char *name = input_item_GetName(item); if (name != NULL) { vlc_memstream_puts(stream, name); free(name); } } break; case 'O': { char *lang = NULL; if (player != NULL) lang = vlc_player_GetCategoryLanguage(player, AUDIO_ES); if (lang != NULL) { vlc_memstream_puts(stream, lang); free(lang); } else if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; } case 'P': if (player) { float pos = vlc_player_GetPosition(player); if (pos >= 0) { vlc_memstream_printf(stream, "%2.1f", pos); break; } } if (!b_empty_if_na) vlc_memstream_puts(stream, "--.-%"); break; case 'R': if (player) vlc_memstream_printf(stream, "%.3f", vlc_player_GetRate(player)); else if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; case 'S': if (player) { const struct vlc_player_track *track = vlc_player_GetSelectedTrack(player, AUDIO_ES); if (track) { div_t dr = div((track->fmt.audio.i_rate + 50) / 100, 10); vlc_memstream_printf(stream, "%d.%01d", dr.quot, dr.rem); break; } } if (!b_empty_if_na) vlc_memstream_putc(stream, '-'); break; case 'T': if (player) { vlc_tick_t time = vlc_player_GetTime(player); if (time != VLC_TICK_INVALID) { write_duration(stream, time); break; } } if (!b_empty_if_na) vlc_memstream_puts(stream, "--:--:--"); break; case 'U': write_meta(stream, item, vlc_meta_Publisher); break; case 'V': { float vol = 0.f; if (player) { audio_output_t *aout = vlc_player_aout_Hold(player); if (aout != NULL) { vol = aout_VolumeGet(aout); aout_Release(aout); } } if (vol >= 0.f) vlc_memstream_printf(stream, "%ld", lroundf(vol * 256.f)); else if (!b_empty_if_na) vlc_memstream_puts(stream, "---"); break; } case '_': vlc_memstream_putc(stream, '\n'); break; case 'Z': if (item == NULL) break; { char *value = input_item_GetNowPlayingFb(item); if (value != NULL) { vlc_memstream_puts(stream, value); free(value); } else { char *title = input_item_GetTitleFbName(item); if (write_meta(stream, item, vlc_meta_Artist) >= 0 && title != NULL) vlc_memstream_puts(stream, " - "); if (title != NULL) { vlc_memstream_puts(stream, title); free(title); } } } break; case ' ': b_empty_if_na = true; b_is_format = true; break; default: vlc_memstream_putc(stream, c); break; } } if (vlc_memstream_close(stream)) return NULL; return stream->ptr; } int vlc_filenamecmp(const char *a, const char *b) { size_t i; char ca, cb; /* Attempt to guess if the sorting algorithm should be alphabetic * (i.e. collation) or numeric: * - If the first mismatching characters are not both digits, * then collation is the only option. * - If one of the first mismatching characters is 0 and the other is also * a digit, the comparands are probably left-padded numerical values. * It does not matter which algorithm is used: the zero will be smaller * than non-zero either way. * - Otherwise, the comparands are numerical values, and might not be * aligned (i.e. not same order of magnitude). If so, collation would * fail. So numerical comparison is performed. */ for (i = 0; (ca = a[i]) == (cb = b[i]); i++) if (ca == '\0') return 0; /* strings are exactly identical */ if ((unsigned)(ca - '0') > 9 || (unsigned)(cb - '0') > 9) return strcoll(a, b); unsigned long long ua = strtoull(a + i, NULL, 10); unsigned long long ub = strtoull(b + i, NULL, 10); /* The number may be identical in two cases: * - leading zero (e.g. "012" and "12") * - overflow on both sides (#ULLONG_MAX) */ if (ua == ub) return strcoll(a, b); return (ua > ub) ? +1 : -1; } /** * Sanitize a file name. * * Remove forbidden, potentially forbidden and otherwise evil characters from * file names. That includes slashes, and popular characters like colon * (on Unix anyway). * * \warning This function should only be used for automatically generated * file names. Do not use this on full paths, only single file names without * any directory separator! */ void filename_sanitize( char *str ) { unsigned char c; /* Special file names, not allowed */ if( !strcmp( str, "." ) || !strcmp( str, ".." ) ) { while( *str ) *(str++) = '_'; return; } /* On platforms not using UTF-8, VLC cannot access non-Unicode paths. * Also, some file systems require Unicode file names. * NOTE: This may inserts '?' thus is done replacing '?' with '_'. */ EnsureUTF8( str ); /* Avoid leading spaces to please Windows. */ while( (c = *str) != '\0' ) { if( c != ' ' ) break; *(str++) = '_'; } char *start = str; while( (c = *str) != '\0' ) { /* Non-printable characters are not a good idea */ if( c < 32 ) *str = '_'; /* This is the list of characters not allowed by Microsoft. * We also black-list them on Unix as they may be confusing, and are * not supported by some file system types (notably CIFS). */ else if( strchr( "/:\\*\"?|<>", c ) != NULL ) *str = '_'; str++; } /* Avoid trailing spaces also to please Windows. */ while( str > start ) { if( *(--str) != ' ' ) break; *str = '_'; } }