153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2008 Colin Guthrie 553a5a1b3Sopenharmony_ci Copyright 2007 Lennart Poettering 653a5a1b3Sopenharmony_ci 753a5a1b3Sopenharmony_ci PulseAudio is free software; you can redistribute it and/or modify 853a5a1b3Sopenharmony_ci it under the terms of the GNU Lesser General Public License as 953a5a1b3Sopenharmony_ci published by the Free Software Foundation; either version 2.1 of the 1053a5a1b3Sopenharmony_ci License, or (at your option) any later version. 1153a5a1b3Sopenharmony_ci 1253a5a1b3Sopenharmony_ci PulseAudio is distributed in the hope that it will be useful, but 1353a5a1b3Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 1453a5a1b3Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 1553a5a1b3Sopenharmony_ci Lesser General Public License for more details. 1653a5a1b3Sopenharmony_ci 1753a5a1b3Sopenharmony_ci You should have received a copy of the GNU Lesser General Public 1853a5a1b3Sopenharmony_ci License along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 1953a5a1b3Sopenharmony_ci***/ 2053a5a1b3Sopenharmony_ci 2153a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H 2253a5a1b3Sopenharmony_ci#include <config.h> 2353a5a1b3Sopenharmony_ci#endif 2453a5a1b3Sopenharmony_ci 2553a5a1b3Sopenharmony_ci#include <string.h> 2653a5a1b3Sopenharmony_ci 2753a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h> 2853a5a1b3Sopenharmony_ci 2953a5a1b3Sopenharmony_ci#include <pulsecore/hashmap.h> 3053a5a1b3Sopenharmony_ci#include <pulsecore/strbuf.h> 3153a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h> 3253a5a1b3Sopenharmony_ci 3353a5a1b3Sopenharmony_ci#include "headerlist.h" 3453a5a1b3Sopenharmony_ci 3553a5a1b3Sopenharmony_cistruct header { 3653a5a1b3Sopenharmony_ci char *key; 3753a5a1b3Sopenharmony_ci void *value; 3853a5a1b3Sopenharmony_ci size_t nbytes; 3953a5a1b3Sopenharmony_ci}; 4053a5a1b3Sopenharmony_ci 4153a5a1b3Sopenharmony_ci#define MAKE_HASHMAP(p) ((pa_hashmap*) (p)) 4253a5a1b3Sopenharmony_ci#define MAKE_HEADERLIST(p) ((pa_headerlist*) (p)) 4353a5a1b3Sopenharmony_ci 4453a5a1b3Sopenharmony_cistatic void header_free(struct header *hdr) { 4553a5a1b3Sopenharmony_ci pa_assert(hdr); 4653a5a1b3Sopenharmony_ci 4753a5a1b3Sopenharmony_ci pa_xfree(hdr->key); 4853a5a1b3Sopenharmony_ci pa_xfree(hdr->value); 4953a5a1b3Sopenharmony_ci pa_xfree(hdr); 5053a5a1b3Sopenharmony_ci} 5153a5a1b3Sopenharmony_ci 5253a5a1b3Sopenharmony_cipa_headerlist* pa_headerlist_new(void) { 5353a5a1b3Sopenharmony_ci return MAKE_HEADERLIST(pa_hashmap_new_full(pa_idxset_string_hash_func, pa_idxset_string_compare_func, NULL, (pa_free_cb_t) header_free)); 5453a5a1b3Sopenharmony_ci} 5553a5a1b3Sopenharmony_ci 5653a5a1b3Sopenharmony_civoid pa_headerlist_free(pa_headerlist* p) { 5753a5a1b3Sopenharmony_ci pa_hashmap_free(MAKE_HASHMAP(p)); 5853a5a1b3Sopenharmony_ci} 5953a5a1b3Sopenharmony_ci 6053a5a1b3Sopenharmony_ciint pa_headerlist_puts(pa_headerlist *p, const char *key, const char *value) { 6153a5a1b3Sopenharmony_ci struct header *hdr; 6253a5a1b3Sopenharmony_ci bool add = false; 6353a5a1b3Sopenharmony_ci 6453a5a1b3Sopenharmony_ci pa_assert(p); 6553a5a1b3Sopenharmony_ci pa_assert(key); 6653a5a1b3Sopenharmony_ci 6753a5a1b3Sopenharmony_ci if (!(hdr = pa_hashmap_get(MAKE_HASHMAP(p), key))) { 6853a5a1b3Sopenharmony_ci hdr = pa_xnew(struct header, 1); 6953a5a1b3Sopenharmony_ci hdr->key = pa_xstrdup(key); 7053a5a1b3Sopenharmony_ci add = true; 7153a5a1b3Sopenharmony_ci } else 7253a5a1b3Sopenharmony_ci pa_xfree(hdr->value); 7353a5a1b3Sopenharmony_ci 7453a5a1b3Sopenharmony_ci hdr->value = pa_xstrdup(value); 7553a5a1b3Sopenharmony_ci hdr->nbytes = strlen(value)+1; 7653a5a1b3Sopenharmony_ci 7753a5a1b3Sopenharmony_ci if (add) 7853a5a1b3Sopenharmony_ci pa_hashmap_put(MAKE_HASHMAP(p), hdr->key, hdr); 7953a5a1b3Sopenharmony_ci 8053a5a1b3Sopenharmony_ci return 0; 8153a5a1b3Sopenharmony_ci} 8253a5a1b3Sopenharmony_ci 8353a5a1b3Sopenharmony_ciint pa_headerlist_putsappend(pa_headerlist *p, const char *key, const char *value) { 8453a5a1b3Sopenharmony_ci struct header *hdr; 8553a5a1b3Sopenharmony_ci bool add = false; 8653a5a1b3Sopenharmony_ci 8753a5a1b3Sopenharmony_ci pa_assert(p); 8853a5a1b3Sopenharmony_ci pa_assert(key); 8953a5a1b3Sopenharmony_ci 9053a5a1b3Sopenharmony_ci if (!(hdr = pa_hashmap_get(MAKE_HASHMAP(p), key))) { 9153a5a1b3Sopenharmony_ci hdr = pa_xnew(struct header, 1); 9253a5a1b3Sopenharmony_ci hdr->key = pa_xstrdup(key); 9353a5a1b3Sopenharmony_ci hdr->value = pa_xstrdup(value); 9453a5a1b3Sopenharmony_ci add = true; 9553a5a1b3Sopenharmony_ci } else { 9653a5a1b3Sopenharmony_ci void *newval = pa_sprintf_malloc("%s%s", (char*)hdr->value, value); 9753a5a1b3Sopenharmony_ci pa_xfree(hdr->value); 9853a5a1b3Sopenharmony_ci hdr->value = newval; 9953a5a1b3Sopenharmony_ci } 10053a5a1b3Sopenharmony_ci hdr->nbytes = strlen(hdr->value)+1; 10153a5a1b3Sopenharmony_ci 10253a5a1b3Sopenharmony_ci if (add) 10353a5a1b3Sopenharmony_ci pa_hashmap_put(MAKE_HASHMAP(p), hdr->key, hdr); 10453a5a1b3Sopenharmony_ci 10553a5a1b3Sopenharmony_ci return 0; 10653a5a1b3Sopenharmony_ci} 10753a5a1b3Sopenharmony_ci 10853a5a1b3Sopenharmony_ciconst char *pa_headerlist_gets(pa_headerlist *p, const char *key) { 10953a5a1b3Sopenharmony_ci struct header *hdr; 11053a5a1b3Sopenharmony_ci 11153a5a1b3Sopenharmony_ci pa_assert(p); 11253a5a1b3Sopenharmony_ci pa_assert(key); 11353a5a1b3Sopenharmony_ci 11453a5a1b3Sopenharmony_ci if (!(hdr = pa_hashmap_get(MAKE_HASHMAP(p), key))) 11553a5a1b3Sopenharmony_ci return NULL; 11653a5a1b3Sopenharmony_ci 11753a5a1b3Sopenharmony_ci if (hdr->nbytes <= 0) 11853a5a1b3Sopenharmony_ci return NULL; 11953a5a1b3Sopenharmony_ci 12053a5a1b3Sopenharmony_ci if (((char*) hdr->value)[hdr->nbytes-1] != 0) 12153a5a1b3Sopenharmony_ci return NULL; 12253a5a1b3Sopenharmony_ci 12353a5a1b3Sopenharmony_ci if (strlen((char*) hdr->value) != hdr->nbytes-1) 12453a5a1b3Sopenharmony_ci return NULL; 12553a5a1b3Sopenharmony_ci 12653a5a1b3Sopenharmony_ci return (char*) hdr->value; 12753a5a1b3Sopenharmony_ci} 12853a5a1b3Sopenharmony_ci 12953a5a1b3Sopenharmony_ciint pa_headerlist_remove(pa_headerlist *p, const char *key) { 13053a5a1b3Sopenharmony_ci pa_assert(p); 13153a5a1b3Sopenharmony_ci pa_assert(key); 13253a5a1b3Sopenharmony_ci 13353a5a1b3Sopenharmony_ci return pa_hashmap_remove_and_free(MAKE_HASHMAP(p), key); 13453a5a1b3Sopenharmony_ci} 13553a5a1b3Sopenharmony_ci 13653a5a1b3Sopenharmony_ciconst char *pa_headerlist_iterate(pa_headerlist *p, void **state) { 13753a5a1b3Sopenharmony_ci struct header *hdr; 13853a5a1b3Sopenharmony_ci 13953a5a1b3Sopenharmony_ci if (!(hdr = pa_hashmap_iterate(MAKE_HASHMAP(p), state, NULL))) 14053a5a1b3Sopenharmony_ci return NULL; 14153a5a1b3Sopenharmony_ci 14253a5a1b3Sopenharmony_ci return hdr->key; 14353a5a1b3Sopenharmony_ci} 14453a5a1b3Sopenharmony_ci 14553a5a1b3Sopenharmony_cichar *pa_headerlist_to_string(pa_headerlist *p) { 14653a5a1b3Sopenharmony_ci const char *key; 14753a5a1b3Sopenharmony_ci void *state = NULL; 14853a5a1b3Sopenharmony_ci pa_strbuf *buf; 14953a5a1b3Sopenharmony_ci 15053a5a1b3Sopenharmony_ci pa_assert(p); 15153a5a1b3Sopenharmony_ci 15253a5a1b3Sopenharmony_ci buf = pa_strbuf_new(); 15353a5a1b3Sopenharmony_ci 15453a5a1b3Sopenharmony_ci while ((key = pa_headerlist_iterate(p, &state))) { 15553a5a1b3Sopenharmony_ci 15653a5a1b3Sopenharmony_ci const char *v; 15753a5a1b3Sopenharmony_ci 15853a5a1b3Sopenharmony_ci if ((v = pa_headerlist_gets(p, key))) 15953a5a1b3Sopenharmony_ci pa_strbuf_printf(buf, "%s: %s\r\n", key, v); 16053a5a1b3Sopenharmony_ci } 16153a5a1b3Sopenharmony_ci 16253a5a1b3Sopenharmony_ci return pa_strbuf_to_string_free(buf); 16353a5a1b3Sopenharmony_ci} 16453a5a1b3Sopenharmony_ci 16553a5a1b3Sopenharmony_ciint pa_headerlist_contains(pa_headerlist *p, const char *key) { 16653a5a1b3Sopenharmony_ci pa_assert(p); 16753a5a1b3Sopenharmony_ci pa_assert(key); 16853a5a1b3Sopenharmony_ci 16953a5a1b3Sopenharmony_ci if (!(pa_hashmap_get(MAKE_HASHMAP(p), key))) 17053a5a1b3Sopenharmony_ci return 0; 17153a5a1b3Sopenharmony_ci 17253a5a1b3Sopenharmony_ci return 1; 17353a5a1b3Sopenharmony_ci} 174