/*** This file is part of avahi. avahi 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. avahi 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 avahi; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. ***/ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include <stdlib.h> #include <sys/types.h> #include <assert.h> #include <string.h> #include "avahi-common/avahi-malloc.h" #include "dns_sd.h" #include "warn.h" typedef struct TXTRecordInternal { uint8_t *buffer, *malloc_buffer; size_t size, max_size; } TXTRecordInternal; #define INTERNAL_PTR(txtref) (* (TXTRecordInternal**) (txtref)) #define INTERNAL_PTR_CONST(txtref) (* (const TXTRecordInternal* const *) (txtref)) void DNSSD_API TXTRecordCreate( TXTRecordRef *txtref, uint16_t length, void *buffer) { TXTRecordInternal *t; AVAHI_WARN_LINKAGE; assert(txtref); /* Apple's API design is flawed in so many ways, including the * fact that it isn't compatible with 64 bit processors. To work * around this we need some magic here which involves allocating * our own memory. Please, Apple, do your homework next time * before designing an API! */ if ((t = avahi_new(TXTRecordInternal, 1))) { t->buffer = buffer; t->max_size = buffer ? length : (size_t)0; t->size = 0; t->malloc_buffer = NULL; } /* If we were unable to allocate memory, we store a NULL pointer * and return a NoMemory error later, is somewhat unclean, but * should work. */ INTERNAL_PTR(txtref) = t; } void DNSSD_API TXTRecordDeallocate(TXTRecordRef *txtref) { TXTRecordInternal *t; AVAHI_WARN_LINKAGE; assert(txtref); t = INTERNAL_PTR(txtref); if (!t) return; avahi_free(t->malloc_buffer); avahi_free(t); /* Just in case ... */ INTERNAL_PTR(txtref) = NULL; } static int make_sure_fits_in(TXTRecordInternal *t, size_t size) { uint8_t *n; size_t nsize; assert(t); if (t->size + size <= t->max_size) return 0; nsize = t->size + size + 100; if (nsize > 0xFFFF) return -1; if (!(n = avahi_realloc(t->malloc_buffer, nsize))) return -1; if (!t->malloc_buffer && t->size) memcpy(n, t->buffer, t->size); t->buffer = t->malloc_buffer = n; t->max_size = nsize; return 0; } static int remove_key(TXTRecordInternal *t, const char *key) { size_t i; uint8_t *p; size_t key_len; int found = 0; key_len = strlen(key); assert(key_len <= 0xFF); p = t->buffer; i = 0; while (i < t->size) { /* Does the item fit in? */ assert(*p <= t->size - i - 1); /* Key longer than buffer */ if (key_len > t->size - i - 1) break; if (key_len <= *p && strncmp(key, (char*) p+1, key_len) == 0 && (key_len == *p || p[1+key_len] == '=')) { uint8_t s; /* Key matches, so let's remove it */ s = *p; memmove(p, p + 1 + *p, t->size - i - *p -1); t->size -= s + 1; found = 1; } else { /* Skip to next */ i += *p +1; p += *p +1; } } return found; } DNSServiceErrorType DNSSD_API TXTRecordSetValue( TXTRecordRef *txtref, const char *key, uint8_t length, const void *value) { TXTRecordInternal *t; uint8_t *p; size_t l, n; AVAHI_WARN_LINKAGE; assert(key); assert(txtref); l = strlen(key); if (*key == 0 || strchr(key, '=') || l > 0xFF) /* Empty or invalid key */ return kDNSServiceErr_Invalid; if (!(t = INTERNAL_PTR(txtref))) return kDNSServiceErr_NoMemory; n = l + (value ? length + 1 : 0); if (n > 0xFF) return kDNSServiceErr_Invalid; if (make_sure_fits_in(t, 1 + n) < 0) return kDNSServiceErr_NoMemory; remove_key(t, key); p = t->buffer + t->size; *(p++) = (uint8_t) n; t->size ++; memcpy(p, key, l); p += l; t->size += l; if (value) { *(p++) = '='; memcpy(p, value, length); t->size += length + 1; } assert(t->size <= t->max_size); return kDNSServiceErr_NoError; } DNSServiceErrorType DNSSD_API TXTRecordRemoveValue(TXTRecordRef *txtref, const char *key) { TXTRecordInternal *t; int found; AVAHI_WARN_LINKAGE; assert(key); assert(txtref); if (*key == 0 || strchr(key, '=') || strlen(key) > 0xFF) /* Empty or invalid key */ return kDNSServiceErr_Invalid; if (!(t = INTERNAL_PTR(txtref))) return kDNSServiceErr_NoError; found = remove_key(t, key); return found ? kDNSServiceErr_NoError : kDNSServiceErr_NoSuchKey; } uint16_t DNSSD_API TXTRecordGetLength(const TXTRecordRef *txtref) { const TXTRecordInternal *t; AVAHI_WARN_LINKAGE; assert(txtref); if (!(t = INTERNAL_PTR_CONST(txtref))) return 0; assert(t->size <= 0xFFFF); return (uint16_t) t->size; } const void * DNSSD_API TXTRecordGetBytesPtr(const TXTRecordRef *txtref) { const TXTRecordInternal *t; AVAHI_WARN_LINKAGE; assert(txtref); if (!(t = INTERNAL_PTR_CONST(txtref)) || !t->buffer) return ""; return t->buffer; } static const uint8_t *find_key(const uint8_t *buffer, size_t size, const char *key) { size_t i; const uint8_t *p; size_t key_len; key_len = strlen(key); assert(key_len <= 0xFF); p = buffer; i = 0; while (i < size) { /* Does the item fit in? */ if (*p > size - i - 1) return NULL; /* Key longer than buffer */ if (key_len > size - i - 1) return NULL; if (key_len <= *p && strncmp(key, (const char*) p+1, key_len) == 0 && (key_len == *p || p[1+key_len] == '=')) { /* Key matches, so let's return it */ return p; } /* Skip to next */ i += *p +1; p += *p +1; } return NULL; } int DNSSD_API TXTRecordContainsKey ( uint16_t size, const void *buffer, const char *key) { AVAHI_WARN_LINKAGE; assert(key); if (!size) return 0; assert(buffer); if (!(find_key(buffer, size, key))) return 0; return 1; } const void * DNSSD_API TXTRecordGetValuePtr( uint16_t size, const void *buffer, const char *key, uint8_t *value_len) { const uint8_t *p; size_t n, l; AVAHI_WARN_LINKAGE; assert(key); if (!size) goto fail; if (*key == 0 || strchr(key, '=') || strlen(key) > 0xFF) /* Empty or invalid key */ return NULL; assert(buffer); if (!(p = find_key(buffer, size, key))) goto fail; n = *p; l = strlen(key); assert(n >= l); p += 1 + l; n -= l; if (n <= 0) goto fail; assert(*p == '='); p++; n--; if (value_len) *value_len = n; return p; fail: if (value_len) *value_len = 0; return NULL; } uint16_t DNSSD_API TXTRecordGetCount( uint16_t size, const void *buffer) { const uint8_t *p; unsigned n = 0; size_t i; AVAHI_WARN_LINKAGE; if (!size) return 0; assert(buffer); p = buffer; i = 0; while (i < size) { /* Does the item fit in? */ if (*p > size - i - 1) break; n++; /* Skip to next */ i += *p +1; p += *p +1; } assert(n <= 0xFFFF); return (uint16_t) n; } DNSServiceErrorType DNSSD_API TXTRecordGetItemAtIndex( uint16_t size, const void *buffer, uint16_t idx, uint16_t key_len, char *key, uint8_t *value_len, const void **value) { const uint8_t *p; size_t i; unsigned n = 0; DNSServiceErrorType ret = kDNSServiceErr_Invalid; AVAHI_WARN_LINKAGE; if (!size) goto fail; assert(buffer); p = buffer; i = 0; while (i < size) { /* Does the item fit in? */ if (*p > size - i - 1) goto fail; if (n >= idx) { size_t l; const uint8_t *d; d = memchr(p+1, '=', *p); /* Length of key */ l = d ? d - p - 1 : *p; if (key_len < l+1) { ret = kDNSServiceErr_NoMemory; goto fail; } strncpy(key, (const char*) p + 1, l); key[l] = 0; if (d) { if (value_len) *value_len = *p - l - 1; if (value) *value = d + 1; } else { if (value_len) *value_len = 0; if (value) *value = NULL; } return kDNSServiceErr_NoError; } n++; /* Skip to next */ i += *p +1; p += *p +1; } fail: if (value) *value = NULL; if (value_len) *value_len = 0; return ret; }