diff options
author | Max Kellermann <max@duempel.org> | 2014-10-10 21:58:21 +0200 |
---|---|---|
committer | Max Kellermann <max@duempel.org> | 2014-10-10 22:11:38 +0200 |
commit | 666f700a93586096e26bf29e4d63851b11b0c400 (patch) | |
tree | aac9796af5fd5adcf50ee041889222801ad3332a | |
parent | b70bf938c230bb5e8aafe95094975a8e10109876 (diff) | |
download | mpd-666f700a93586096e26bf29e4d63851b11b0c400.tar.gz mpd-666f700a93586096e26bf29e4d63851b11b0c400.tar.xz mpd-666f700a93586096e26bf29e4d63851b11b0c400.zip |
TagString: implement fix_utf8() without GLib
Diffstat (limited to '')
-rw-r--r-- | src/tag/TagString.cxx | 56 |
1 files changed, 34 insertions, 22 deletions
diff --git a/src/tag/TagString.cxx b/src/tag/TagString.cxx index f4a170fbe..4f07cd62a 100644 --- a/src/tag/TagString.cxx +++ b/src/tag/TagString.cxx @@ -21,30 +21,53 @@ #include "TagString.hxx" #include "util/Alloc.hxx" #include "util/WritableBuffer.hxx" - -#ifdef HAVE_GLIB -#include <glib.h> -#endif +#include "util/UTF8.hxx" #include <assert.h> #include <string.h> #include <stdlib.h> -#ifdef HAVE_GLIB +gcc_pure +static const char * +FindInvalidUTF8(const char *p, const char *const end) +{ + while (p < end) { + const size_t s = SequenceLengthUTF8(*p); + if (p + s > end) + /* partial sequence at end of string */ + return p; + + /* now call the other SequenceLengthUTF8() overload + which also validates the continuations */ + const size_t t = SequenceLengthUTF8(p); + assert(s == t); + if (t == 0) + return p; + + p += s; + } + + return nullptr; +} /** * Replace invalid sequences with the question mark. */ static WritableBuffer<char> -patch_utf8(const char *src, size_t length, const gchar *end) +patch_utf8(const char *src, size_t length, const char *_invalid) { /* duplicate the string, and replace invalid bytes in that buffer */ char *dest = (char *)xmemdup(src, length); + char *const end = dest + length; + char *invalid = dest + (_invalid - src); do { - dest[end - src] = '?'; - } while (!g_utf8_validate(end + 1, (src + length) - (end + 1), &end)); + *invalid = '?'; + + const char *__invalid = FindInvalidUTF8(invalid + 1, end); + invalid = const_cast<char *>(__invalid); + } while (invalid != nullptr); return { dest, length }; } @@ -52,20 +75,15 @@ patch_utf8(const char *src, size_t length, const gchar *end) static WritableBuffer<char> fix_utf8(const char *str, size_t length) { - const gchar *end; - - assert(str != nullptr); - /* check if the string is already valid UTF-8 */ - if (g_utf8_validate(str, length, &end)) + const char *invalid = FindInvalidUTF8(str, str + length); + if (invalid == nullptr) return nullptr; /* no, broken - patch invalid sequences */ - return patch_utf8(str, length, end); + return patch_utf8(str, length, invalid); } -#endif - static bool char_is_non_printable(unsigned char ch) { @@ -105,23 +123,17 @@ clear_non_printable(const char *p, size_t length) WritableBuffer<char> FixTagString(const char *p, size_t length) { -#ifdef HAVE_GLIB - // TODO: implement without GLib - auto utf8 = fix_utf8(p, length); if (!utf8.IsNull()) { p = utf8.data; length = utf8.size; } -#endif WritableBuffer<char> cleared = clear_non_printable(p, length); -#ifdef HAVE_GLIB if (cleared.IsNull()) cleared = utf8; else free(utf8.data); -#endif return cleared; } |