Add function to filter a string of any invalid UTF-8 sequences. User must

free() the result.


git-svn-id: file:///home/lennart/svn/public/pulseaudio/trunk@872 fefdeb5f-60dc-0310-8127-8f9354f1896f
This commit is contained in:
Pierre Ossman 2006-05-15 13:04:13 +00:00
parent 7a92f361c6
commit 9c8661c675
2 changed files with 53 additions and 7 deletions

View file

@ -28,11 +28,15 @@
#include <config.h> #include <config.h>
#endif #endif
#include <assert.h>
#include <stdlib.h> #include <stdlib.h>
#include <inttypes.h> #include <inttypes.h>
#include <string.h>
#include "utf8.h" #include "utf8.h"
#define FILTER_CHAR '_'
static inline int is_unicode_valid(uint32_t ch) { static inline int is_unicode_valid(uint32_t ch) {
if (ch >= 0x110000) /* End of unicode space */ if (ch >= 0x110000) /* End of unicode space */
return 0; return 0;
@ -56,30 +60,39 @@ static inline void merge_continuation_char(uint32_t *u_ch, uint8_t ch) {
*u_ch |= ch & 0x3f; *u_ch |= ch & 0x3f;
} }
const char* pa_utf8_valid (const char *str) { static const char* utf8_validate (const char *str, char *output) {
uint32_t val = 0; uint32_t val = 0;
uint32_t min = 0; uint32_t min = 0;
const uint8_t *p, *last; const uint8_t *p, *last;
int size;
uint8_t *o;
for (p = (uint8_t*)str; *p; p++) { o = output;
if (*p < 128) for (p = (uint8_t*)str; *p; p++, o++) {
/* done */; if (*p < 128) {
else { if (o)
*output = *p;
} else {
last = p; last = p;
if ((*p & 0xe0) == 0xc0) { /* 110xxxxx two-char seq. */ if ((*p & 0xe0) == 0xc0) { /* 110xxxxx two-char seq. */
size = 2;
min = 128; min = 128;
val = *p & 0x1e; val = *p & 0x1e;
goto ONE_REMAINING; goto ONE_REMAINING;
} else if ((*p & 0xf0) == 0xe0) { /* 1110xxxx three-char seq.*/ } else if ((*p & 0xf0) == 0xe0) { /* 1110xxxx three-char seq.*/
size = 3;
min = (1 << 11); min = (1 << 11);
val = *p & 0x0f; val = *p & 0x0f;
goto TWO_REMAINING; goto TWO_REMAINING;
} else if ((*p & 0xf8) == 0xf0) { /* 11110xxx four-char seq */ } else if ((*p & 0xf8) == 0xf0) { /* 11110xxx four-char seq */
size = 4;
min = (1 << 16); min = (1 << 16);
val = *p & 0x07; val = *p & 0x07;
} else } else {
size = 1;
goto error; goto error;
}
p++; p++;
if (!is_continuation_char(*p)) if (!is_continuation_char(*p))
@ -103,11 +116,43 @@ ONE_REMAINING:
if (!is_unicode_valid(val)) if (!is_unicode_valid(val))
goto error; goto error;
if (o) {
memcpy(o, last, size);
o += size - 1;
}
continue;
error:
if (o) {
*o = FILTER_CHAR;
p = last + 1; /* We retry at the next character */
} else
goto failure;
} }
} }
if (o) {
*o = '\0';
return output;
}
return str; return str;
error: failure:
return NULL; return NULL;
} }
const char* pa_utf8_valid (const char *str) {
return utf8_validate(str, NULL);
}
const char* pa_utf8_filter (const char *str) {
char *new_str;
new_str = malloc(strlen(str) + 1);
assert(new_str);
return utf8_validate(str, new_str);
}

View file

@ -23,5 +23,6 @@
***/ ***/
const char *pa_utf8_valid(const char *str); const char *pa_utf8_valid(const char *str);
const char *pa_utf8_filter(const char *str);
#endif #endif