mirror of
https://gitlab.freedesktop.org/pulseaudio/pulseaudio.git
synced 2025-11-04 13:29:59 -05:00
fix up parser in pa_proplist_from_string() to handle escapes correctly; make pa_proplist_to_string() escape quotes properly
This commit is contained in:
parent
0fc59e4585
commit
b445741ddf
2 changed files with 314 additions and 80 deletions
|
|
@ -24,6 +24,7 @@
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
|
||||||
#include <pulse/xmalloc.h>
|
#include <pulse/xmalloc.h>
|
||||||
#include <pulse/utf8.h>
|
#include <pulse/utf8.h>
|
||||||
|
|
@ -46,7 +47,7 @@ struct property {
|
||||||
|
|
||||||
static pa_bool_t property_name_valid(const char *key) {
|
static pa_bool_t property_name_valid(const char *key) {
|
||||||
|
|
||||||
if (!pa_utf8_valid(key))
|
if (!pa_ascii_valid(key))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
if (strlen(key) <= 0)
|
if (strlen(key) <= 0)
|
||||||
|
|
@ -64,8 +65,6 @@ static void property_free(struct property *prop) {
|
||||||
}
|
}
|
||||||
|
|
||||||
pa_proplist* pa_proplist_new(void) {
|
pa_proplist* pa_proplist_new(void) {
|
||||||
pa_init_i18n();
|
|
||||||
|
|
||||||
return MAKE_PROPLIST(pa_hashmap_new(pa_idxset_string_hash_func, pa_idxset_string_compare_func));
|
return MAKE_PROPLIST(pa_hashmap_new(pa_idxset_string_hash_func, pa_idxset_string_compare_func));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -83,6 +82,7 @@ int pa_proplist_sets(pa_proplist *p, const char *key, const char *value) {
|
||||||
|
|
||||||
pa_assert(p);
|
pa_assert(p);
|
||||||
pa_assert(key);
|
pa_assert(key);
|
||||||
|
pa_assert(value);
|
||||||
|
|
||||||
if (!property_name_valid(key) || !pa_utf8_valid(value))
|
if (!property_name_valid(key) || !pa_utf8_valid(value))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
@ -104,25 +104,130 @@ int pa_proplist_sets(pa_proplist *p, const char *key, const char *value) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Will accept only valid UTF-8 */
|
/** Will accept only valid UTF-8 */
|
||||||
int pa_proplist_setf(pa_proplist *p, const char *key, const char *format, ...) {
|
static int proplist_setn(pa_proplist *p, const char *key, size_t key_length, const char *value, size_t value_length) {
|
||||||
va_list ap;
|
struct property *prop;
|
||||||
int r;
|
pa_bool_t add = FALSE;
|
||||||
char *t;
|
char *k, *v;
|
||||||
|
|
||||||
pa_assert(p);
|
pa_assert(p);
|
||||||
pa_assert(key);
|
pa_assert(key);
|
||||||
|
pa_assert(value);
|
||||||
|
|
||||||
|
k = pa_xstrndup(key, key_length);
|
||||||
|
v = pa_xstrndup(value, value_length);
|
||||||
|
|
||||||
|
if (!property_name_valid(k) || !pa_utf8_valid(v)) {
|
||||||
|
pa_xfree(k);
|
||||||
|
pa_xfree(v);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(prop = pa_hashmap_get(MAKE_HASHMAP(p), k))) {
|
||||||
|
prop = pa_xnew(struct property, 1);
|
||||||
|
prop->key = k;
|
||||||
|
add = TRUE;
|
||||||
|
} else {
|
||||||
|
pa_xfree(prop->value);
|
||||||
|
pa_xfree(k);
|
||||||
|
}
|
||||||
|
|
||||||
|
prop->value = v;
|
||||||
|
prop->nbytes = strlen(v)+1;
|
||||||
|
|
||||||
|
if (add)
|
||||||
|
pa_hashmap_put(MAKE_HASHMAP(p), prop->key, prop);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int proplist_sethex(pa_proplist *p, const char *key, size_t key_length, const char *value, size_t value_length) {
|
||||||
|
struct property *prop;
|
||||||
|
pa_bool_t add = FALSE;
|
||||||
|
char *k, *v;
|
||||||
|
uint8_t *d;
|
||||||
|
size_t dn;
|
||||||
|
|
||||||
|
pa_assert(p);
|
||||||
|
pa_assert(key);
|
||||||
|
pa_assert(value);
|
||||||
|
|
||||||
|
k = pa_xstrndup(key, key_length);
|
||||||
|
|
||||||
|
if (!property_name_valid(k)) {
|
||||||
|
pa_xfree(k);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
v = pa_xstrndup(value, value_length);
|
||||||
|
d = pa_xmalloc(value_length*2+1);
|
||||||
|
|
||||||
|
if ((dn = pa_parsehex(v, d, value_length*2)) == (size_t) -1) {
|
||||||
|
pa_xfree(k);
|
||||||
|
pa_xfree(v);
|
||||||
|
pa_xfree(d);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_xfree(v);
|
||||||
|
|
||||||
|
if (!(prop = pa_hashmap_get(MAKE_HASHMAP(p), k))) {
|
||||||
|
prop = pa_xnew(struct property, 1);
|
||||||
|
prop->key = k;
|
||||||
|
add = TRUE;
|
||||||
|
} else {
|
||||||
|
pa_xfree(prop->value);
|
||||||
|
pa_xfree(k);
|
||||||
|
}
|
||||||
|
|
||||||
|
d[dn] = 0;
|
||||||
|
prop->value = d;
|
||||||
|
prop->nbytes = dn;
|
||||||
|
|
||||||
|
if (add)
|
||||||
|
pa_hashmap_put(MAKE_HASHMAP(p), prop->key, prop);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Will accept only valid UTF-8 */
|
||||||
|
int pa_proplist_setf(pa_proplist *p, const char *key, const char *format, ...) {
|
||||||
|
struct property *prop;
|
||||||
|
pa_bool_t add = FALSE;
|
||||||
|
va_list ap;
|
||||||
|
char *v;
|
||||||
|
|
||||||
|
pa_assert(p);
|
||||||
|
pa_assert(key);
|
||||||
|
pa_assert(format);
|
||||||
|
|
||||||
if (!property_name_valid(key) || !pa_utf8_valid(format))
|
if (!property_name_valid(key) || !pa_utf8_valid(format))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
va_start(ap, format);
|
va_start(ap, format);
|
||||||
t = pa_vsprintf_malloc(format, ap);
|
v = pa_vsprintf_malloc(format, ap);
|
||||||
va_end(ap);
|
va_end(ap);
|
||||||
|
|
||||||
r = pa_proplist_sets(p, key, t);
|
if (!pa_utf8_valid(v))
|
||||||
|
goto fail;
|
||||||
|
|
||||||
pa_xfree(t);
|
if (!(prop = pa_hashmap_get(MAKE_HASHMAP(p), key))) {
|
||||||
return r;
|
prop = pa_xnew(struct property, 1);
|
||||||
|
prop->key = pa_xstrdup(key);
|
||||||
|
add = TRUE;
|
||||||
|
} else
|
||||||
|
pa_xfree(prop->value);
|
||||||
|
|
||||||
|
prop->value = v;
|
||||||
|
prop->nbytes = strlen(v)+1;
|
||||||
|
|
||||||
|
if (add)
|
||||||
|
pa_hashmap_put(MAKE_HASHMAP(p), prop->key, prop);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
pa_xfree(v);
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int pa_proplist_set(pa_proplist *p, const char *key, const void *data, size_t nbytes) {
|
int pa_proplist_set(pa_proplist *p, const char *key, const void *data, size_t nbytes) {
|
||||||
|
|
@ -131,6 +236,7 @@ int pa_proplist_set(pa_proplist *p, const char *key, const void *data, size_t nb
|
||||||
|
|
||||||
pa_assert(p);
|
pa_assert(p);
|
||||||
pa_assert(key);
|
pa_assert(key);
|
||||||
|
pa_assert(data);
|
||||||
|
|
||||||
if (!property_name_valid(key))
|
if (!property_name_valid(key))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
@ -142,7 +248,9 @@ int pa_proplist_set(pa_proplist *p, const char *key, const void *data, size_t nb
|
||||||
} else
|
} else
|
||||||
pa_xfree(prop->value);
|
pa_xfree(prop->value);
|
||||||
|
|
||||||
prop->value = pa_xmemdup(data, nbytes);
|
prop->value = pa_xmalloc(nbytes+1);
|
||||||
|
memcpy(prop->value, data, nbytes);
|
||||||
|
((char*) prop->value)[nbytes] = 0;
|
||||||
prop->nbytes = nbytes;
|
prop->nbytes = nbytes;
|
||||||
|
|
||||||
if (add)
|
if (add)
|
||||||
|
|
@ -183,6 +291,8 @@ int pa_proplist_get(pa_proplist *p, const char *key, const void **data, size_t *
|
||||||
|
|
||||||
pa_assert(p);
|
pa_assert(p);
|
||||||
pa_assert(key);
|
pa_assert(key);
|
||||||
|
pa_assert(data);
|
||||||
|
pa_assert(nbytes);
|
||||||
|
|
||||||
if (!property_name_valid(key))
|
if (!property_name_valid(key))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
@ -275,9 +385,32 @@ char *pa_proplist_to_string_sep(pa_proplist *p, const char *sep) {
|
||||||
if (!pa_strbuf_isempty(buf))
|
if (!pa_strbuf_isempty(buf))
|
||||||
pa_strbuf_puts(buf, sep);
|
pa_strbuf_puts(buf, sep);
|
||||||
|
|
||||||
if ((v = pa_proplist_gets(p, key)))
|
if ((v = pa_proplist_gets(p, key))) {
|
||||||
pa_strbuf_printf(buf, "%s = \"%s\"", key, v);
|
const char *t;
|
||||||
else {
|
|
||||||
|
pa_strbuf_printf(buf, "%s = \"", key);
|
||||||
|
|
||||||
|
for (t = v;;) {
|
||||||
|
size_t h;
|
||||||
|
|
||||||
|
h = strcspn(t, "\"");
|
||||||
|
|
||||||
|
if (h > 0)
|
||||||
|
pa_strbuf_putsn(buf, t, h);
|
||||||
|
|
||||||
|
t += h;
|
||||||
|
|
||||||
|
if (*t == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
pa_assert(*t == '"');
|
||||||
|
pa_strbuf_puts(buf, "\\\"");
|
||||||
|
|
||||||
|
t++;
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_strbuf_puts(buf, "\"");
|
||||||
|
} else {
|
||||||
const void *value;
|
const void *value;
|
||||||
size_t nbytes;
|
size_t nbytes;
|
||||||
char *c;
|
char *c;
|
||||||
|
|
@ -304,88 +437,189 @@ char *pa_proplist_to_string(pa_proplist *p) {
|
||||||
return t;
|
return t;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Remove all whitepsapce from the beginning and the end of *s. *s may
|
pa_proplist *pa_proplist_from_string(const char *s) {
|
||||||
* be modified. (from conf-parser.c) */
|
enum {
|
||||||
#define WHITESPACE " \t\n"
|
WHITESPACE,
|
||||||
#define in_string(c,s) (strchr(s,c) != NULL)
|
KEY,
|
||||||
|
AFTER_KEY,
|
||||||
|
VALUE_START,
|
||||||
|
VALUE_SIMPLE,
|
||||||
|
VALUE_DOUBLE_QUOTES,
|
||||||
|
VALUE_DOUBLE_QUOTES_ESCAPE,
|
||||||
|
VALUE_TICKS,
|
||||||
|
VALUE_TICKS_ESCAPED,
|
||||||
|
VALUE_HEX
|
||||||
|
} state;
|
||||||
|
|
||||||
static char *strip(char *s) {
|
pa_proplist *pl;
|
||||||
char *b = s+strspn(s, WHITESPACE);
|
const char *p, *key = NULL, *value = NULL;
|
||||||
char *e, *l = NULL;
|
size_t key_len = 0, value_len = 0;
|
||||||
|
|
||||||
for (e = b; *e; e++)
|
pa_assert(s);
|
||||||
if (!in_string(*e, WHITESPACE))
|
|
||||||
l = e;
|
|
||||||
|
|
||||||
if (l)
|
pl = pa_proplist_new();
|
||||||
*(l+1) = 0;
|
|
||||||
|
|
||||||
return b;
|
state = WHITESPACE;
|
||||||
|
|
||||||
|
for (p = s;; p++) {
|
||||||
|
switch (state) {
|
||||||
|
|
||||||
|
case WHITESPACE:
|
||||||
|
if (*p == 0)
|
||||||
|
goto success;
|
||||||
|
else if (*p == '=')
|
||||||
|
goto fail;
|
||||||
|
else if (!isspace(*p)) {
|
||||||
|
key = p;
|
||||||
|
state = KEY;
|
||||||
|
key_len = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
pa_proplist *pa_proplist_from_string(const char *str) {
|
|
||||||
pa_proplist *p;
|
|
||||||
char *s, *v, *k, *e;
|
|
||||||
|
|
||||||
pa_assert(str);
|
|
||||||
pa_assert_se(p = pa_proplist_new());
|
|
||||||
pa_assert_se(s = strdup(str));
|
|
||||||
|
|
||||||
for (k = s; *k; k = e) {
|
|
||||||
k = k+strspn(k, WHITESPACE);
|
|
||||||
|
|
||||||
if (!*k)
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
if (!(v = strchr(k, '='))) {
|
case KEY:
|
||||||
pa_log("Missing '='.");
|
if (*p == 0)
|
||||||
|
goto fail;
|
||||||
|
else if (*p == '=')
|
||||||
|
state = VALUE_START;
|
||||||
|
else if (isspace(*p))
|
||||||
|
state = AFTER_KEY;
|
||||||
|
else
|
||||||
|
key_len++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AFTER_KEY:
|
||||||
|
if (*p == 0)
|
||||||
|
goto fail;
|
||||||
|
else if (*p == '=')
|
||||||
|
state = VALUE_START;
|
||||||
|
else if (!isspace(*p))
|
||||||
|
goto fail;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VALUE_START:
|
||||||
|
if (*p == 0)
|
||||||
|
goto fail;
|
||||||
|
else if (strncmp(p, "hex:", 4) == 0) {
|
||||||
|
state = VALUE_HEX;
|
||||||
|
value = p+4;
|
||||||
|
value_len = 0;
|
||||||
|
p += 3;
|
||||||
|
} else if (*p == '\'') {
|
||||||
|
state = VALUE_TICKS;
|
||||||
|
value = p+1;
|
||||||
|
value_len = 0;
|
||||||
|
} else if (*p == '"') {
|
||||||
|
state = VALUE_DOUBLE_QUOTES;
|
||||||
|
value = p+1;
|
||||||
|
value_len = 0;
|
||||||
|
} else if (!isspace(*p)) {
|
||||||
|
state = VALUE_SIMPLE;
|
||||||
|
value = p;
|
||||||
|
value_len = 1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VALUE_SIMPLE:
|
||||||
|
if (*p == 0 || isspace(*p)) {
|
||||||
|
if (proplist_setn(pl, key, key_len, value, value_len) < 0)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
if (*p == 0)
|
||||||
|
goto success;
|
||||||
|
|
||||||
|
state = WHITESPACE;
|
||||||
|
} else
|
||||||
|
value_len++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VALUE_DOUBLE_QUOTES:
|
||||||
|
if (*p == 0)
|
||||||
|
goto fail;
|
||||||
|
else if (*p == '"') {
|
||||||
|
char *v;
|
||||||
|
|
||||||
|
v = pa_unescape(pa_xstrndup(value, value_len));
|
||||||
|
|
||||||
|
if (proplist_setn(pl, key, key_len, v, strlen(v)) < 0) {
|
||||||
|
pa_xfree(v);
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_xfree(v);
|
||||||
|
state = WHITESPACE;
|
||||||
|
} else if (*p == '\\') {
|
||||||
|
state = VALUE_DOUBLE_QUOTES_ESCAPE;
|
||||||
|
value_len++;
|
||||||
|
} else
|
||||||
|
value_len++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VALUE_DOUBLE_QUOTES_ESCAPE:
|
||||||
|
if (*p == 0)
|
||||||
|
goto fail;
|
||||||
|
else {
|
||||||
|
state = VALUE_DOUBLE_QUOTES;
|
||||||
|
value_len++;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VALUE_TICKS:
|
||||||
|
if (*p == 0)
|
||||||
|
goto fail;
|
||||||
|
else if (*p == '\'') {
|
||||||
|
char *v;
|
||||||
|
|
||||||
|
v = pa_unescape(pa_xstrndup(value, value_len));
|
||||||
|
|
||||||
|
if (proplist_setn(pl, key, key_len, v, strlen(v)) < 0) {
|
||||||
|
pa_xfree(v);
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_xfree(v);
|
||||||
|
state = WHITESPACE;
|
||||||
|
} else if (*p == '\\') {
|
||||||
|
state = VALUE_TICKS_ESCAPED;
|
||||||
|
value_len++;
|
||||||
|
} else
|
||||||
|
value_len++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VALUE_TICKS_ESCAPED:
|
||||||
|
if (*p == 0)
|
||||||
|
goto fail;
|
||||||
|
else {
|
||||||
|
state = VALUE_TICKS;
|
||||||
|
value_len++;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VALUE_HEX:
|
||||||
|
if ((*p >= '0' && *p <= '9') ||
|
||||||
|
(*p >= 'A' && *p <= 'F') ||
|
||||||
|
(*p >= 'a' && *p <= 'f')) {
|
||||||
|
value_len++;
|
||||||
|
} else if (*p == 0 || isspace(*p)) {
|
||||||
|
|
||||||
|
if (proplist_sethex(pl, key, key_len, value, value_len) < 0)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
if (*p == 0)
|
||||||
|
goto success;
|
||||||
|
|
||||||
|
state = WHITESPACE;
|
||||||
|
} else
|
||||||
|
goto fail;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
*v++ = '\0';
|
|
||||||
k = strip(k);
|
|
||||||
|
|
||||||
v = v+strspn(v, WHITESPACE);
|
|
||||||
if (*v == '"') {
|
|
||||||
v++;
|
|
||||||
if (!(e = strchr(v, '"'))) { /* FIXME: handle escape */
|
|
||||||
pa_log("Missing '\"' at end of string value.");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
*e++ = '\0';
|
|
||||||
pa_proplist_sets(p, k, v);
|
|
||||||
} else {
|
|
||||||
uint8_t *blob;
|
|
||||||
|
|
||||||
if (*v++ != 'h' || *v++ != 'e' || *v++ != 'x' || *v++ != ':') {
|
|
||||||
pa_log("Value must be a string or \"hex:\"");
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
e = v;
|
success:
|
||||||
while (in_string(*e, "0123456789abcdefABCDEF"))
|
return MAKE_PROPLIST(pl);
|
||||||
++e;
|
|
||||||
|
|
||||||
if ((e - v) % 2) {
|
fail:
|
||||||
pa_log("Invalid \"hex:\" value data");
|
pa_proplist_free(pl);
|
||||||
break;
|
return NULL;
|
||||||
}
|
|
||||||
|
|
||||||
blob = pa_xmalloc((size_t)(e-v)/2);
|
|
||||||
if (pa_parsehex(v, blob, (e-v)/2) != (size_t)((e-v)/2)) {
|
|
||||||
pa_log("Invalid \"hex:\" value data");
|
|
||||||
pa_xfree(blob);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
pa_proplist_set(p, k, blob, (e-v)/2);
|
|
||||||
pa_xfree(blob);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pa_xfree(s);
|
|
||||||
|
|
||||||
return p;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int pa_proplist_contains(pa_proplist *p, const char *key) {
|
int pa_proplist_contains(pa_proplist *p, const char *key) {
|
||||||
|
|
|
||||||
|
|
@ -136,7 +136,7 @@ PA_C_DECL_BEGIN
|
||||||
#define PA_PROP_MODULE_USAGE "module.usage"
|
#define PA_PROP_MODULE_USAGE "module.usage"
|
||||||
#define PA_PROP_MODULE_VERSION "module.version"
|
#define PA_PROP_MODULE_VERSION "module.version"
|
||||||
|
|
||||||
/** A property list object. Basically a dictionary with UTF-8 strings
|
/** A property list object. Basically a dictionary with ASCII strings
|
||||||
* as keys and arbitrary data as values. \since 0.9.11 */
|
* as keys and arbitrary data as values. \since 0.9.11 */
|
||||||
typedef struct pa_proplist pa_proplist;
|
typedef struct pa_proplist pa_proplist;
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue