1 /* -*- Mode: C; tab-width: 4 -*- 2 * 3 * Copyright (c) 2004-2011 Apple Inc. All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions are met: 7 * 8 * 1. Redistributions of source code must retain the above copyright notice, 9 * this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright notice, 11 * this list of conditions and the following disclaimer in the documentation 12 * and/or other materials provided with the distribution. 13 * 3. Neither the name of Apple Inc. ("Apple") nor the names of its 14 * contributors may be used to endorse or promote products derived from this 15 * software without specific prior written permission. 16 * 17 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY 18 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 20 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY 21 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 23 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 24 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 26 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 27 */ 28 29 #include <stdlib.h> 30 #include <string.h> 31 32 #include "dns_sd.h" 33 34 #if MDNS_BUILDINGSHAREDLIBRARY || MDNS_BUILDINGSTUBLIBRARY 35 #pragma export on 36 #endif 37 38 #if defined(_WIN32) 39 // disable warning "conversion from <data> to uint16_t" 40 #pragma warning(disable:4244) 41 #define strncasecmp _strnicmp 42 #define strcasecmp _stricmp 43 #endif 44 45 /********************************************************************************************* 46 * 47 * Supporting Functions 48 * 49 *********************************************************************************************/ 50 51 #define mDNSIsDigit(X) ((X) >= '0' && (X) <= '9') 52 53 // DomainEndsInDot returns 1 if name ends with a dot, 0 otherwise 54 // (DNSServiceConstructFullName depends this returning 1 for true, rather than any non-zero value meaning true) 55 56 static int DomainEndsInDot(const char *dom) 57 { 58 while (dom[0] && dom[1]) 59 { 60 if (dom[0] == '\\') // advance past escaped byte sequence 61 { 62 if (mDNSIsDigit(dom[1]) && mDNSIsDigit(dom[2]) && mDNSIsDigit(dom[3])) 63 dom += 4; // If "\ddd" then skip four 64 else dom += 2; // else if "\x" then skip two 65 } 66 else dom++; // else goto next character 67 } 68 return (dom[0] == '.'); 69 } 70 71 static uint8_t *InternalTXTRecordSearch 72 ( 73 uint16_t txtLen, 74 const void *txtRecord, 75 const char *key, 76 unsigned long *keylen 77 ) 78 { 79 uint8_t *p = (uint8_t*)txtRecord; 80 uint8_t *e = p + txtLen; 81 *keylen = (unsigned long) strlen(key); 82 while (p<e) 83 { 84 uint8_t *x = p; 85 p += 1 + p[0]; 86 if (p <= e && *keylen <= x[0] && !strncasecmp(key, (char*)x+1, *keylen)) 87 if (*keylen == x[0] || x[1+*keylen] == '=') return(x); 88 } 89 return(NULL); 90 } 91 92 /********************************************************************************************* 93 * 94 * General Utility Functions 95 * 96 *********************************************************************************************/ 97 98 // Note: Need to make sure we don't write more than kDNSServiceMaxDomainName (1009) bytes to fullName 99 // In earlier builds this constant was defined to be 1005, so to avoid buffer overruns on clients 100 // compiled with that constant we'll actually limit the output to 1005 bytes. 101 102 DNSServiceErrorType DNSSD_API DNSServiceConstructFullName 103 ( 104 char *const fullName, 105 const char *const service, // May be NULL 106 const char *const regtype, 107 const char *const domain 108 ) 109 { 110 const size_t len = !regtype ? 0 : strlen(regtype) - DomainEndsInDot(regtype); 111 char *fn = fullName; 112 char *const lim = fullName + 1005; 113 const char *s = service; 114 const char *r = regtype; 115 const char *d = domain; 116 117 // regtype must be at least "x._udp" or "x._tcp" 118 if (len < 6 || !domain || !domain[0]) return kDNSServiceErr_BadParam; 119 if (strncasecmp((regtype + len - 4), "_tcp", 4) && strncasecmp((regtype + len - 4), "_udp", 4)) return kDNSServiceErr_BadParam; 120 121 if (service && *service) 122 { 123 while (*s) 124 { 125 unsigned char c = *s++; // Needs to be unsigned, or values like 0xFF will be interpreted as < 32 126 if (c <= ' ') // Escape non-printable characters 127 { 128 if (fn+4 >= lim) goto fail; 129 *fn++ = '\\'; 130 *fn++ = '0' + (c / 100); 131 *fn++ = '0' + (c / 10) % 10; 132 c = '0' + (c ) % 10; 133 } 134 else if (c == '.' || (c == '\\')) // Escape dot and backslash literals 135 { 136 if (fn+2 >= lim) goto fail; 137 *fn++ = '\\'; 138 } 139 else 140 if (fn+1 >= lim) goto fail; 141 *fn++ = (char)c; 142 } 143 *fn++ = '.'; 144 } 145 146 while (*r) if (fn+1 >= lim) goto fail;else *fn++ = *r++; 147 if (!DomainEndsInDot(regtype)) { if (fn+1 >= lim) goto fail;else *fn++ = '.';} 148 149 while (*d) if (fn+1 >= lim) goto fail;else *fn++ = *d++; 150 if (!DomainEndsInDot(domain)) { if (fn+1 >= lim) goto fail;else *fn++ = '.';} 151 152 *fn = '\0'; 153 return kDNSServiceErr_NoError; 154 155 fail: 156 *fn = '\0'; 157 return kDNSServiceErr_BadParam; 158 } 159 160 /********************************************************************************************* 161 * 162 * TXT Record Construction Functions 163 * 164 *********************************************************************************************/ 165 166 typedef struct _TXTRecordRefRealType 167 { 168 uint8_t *buffer; // Pointer to data 169 uint16_t buflen; // Length of buffer 170 uint16_t datalen; // Length currently in use 171 uint16_t malloced; // Non-zero if buffer was allocated via malloc() 172 } TXTRecordRefRealType; 173 174 #define txtRec ((TXTRecordRefRealType*)txtRecord) 175 176 // The opaque storage defined in the public dns_sd.h header is 16 bytes; 177 // make sure we don't exceed that. 178 struct CompileTimeAssertionCheck_dnssd_clientlib 179 { 180 char assert0[(sizeof(TXTRecordRefRealType) <= 16) ? 1 : -1]; 181 }; 182 183 void DNSSD_API TXTRecordCreate 184 ( 185 TXTRecordRef *txtRecord, 186 uint16_t bufferLen, 187 void *buffer 188 ) 189 { 190 txtRec->buffer = buffer; 191 txtRec->buflen = buffer ? bufferLen : (uint16_t)0; 192 txtRec->datalen = 0; 193 txtRec->malloced = 0; 194 } 195 196 void DNSSD_API TXTRecordDeallocate(TXTRecordRef *txtRecord) 197 { 198 if (txtRec->malloced) free(txtRec->buffer); 199 } 200 201 DNSServiceErrorType DNSSD_API TXTRecordSetValue 202 ( 203 TXTRecordRef *txtRecord, 204 const char *key, 205 uint8_t valueSize, 206 const void *value 207 ) 208 { 209 uint8_t *start, *p; 210 const char *k; 211 unsigned long keysize, keyvalsize; 212 213 for (k = key; *k; k++) if (*k < 0x20 || *k > 0x7E || *k == '=') return(kDNSServiceErr_Invalid); 214 keysize = (unsigned long)(k - key); 215 keyvalsize = 1 + keysize + (value ? (1 + valueSize) : 0); 216 if (keysize < 1 || keyvalsize > 255) return(kDNSServiceErr_Invalid); 217 (void)TXTRecordRemoveValue(txtRecord, key); 218 if (txtRec->datalen + keyvalsize > txtRec->buflen) 219 { 220 unsigned char *newbuf; 221 unsigned long newlen = txtRec->datalen + keyvalsize; 222 if (newlen > 0xFFFF) return(kDNSServiceErr_Invalid); 223 newbuf = malloc((size_t)newlen); 224 if (!newbuf) return(kDNSServiceErr_NoMemory); 225 memcpy(newbuf, txtRec->buffer, txtRec->datalen); 226 if (txtRec->malloced) free(txtRec->buffer); 227 txtRec->buffer = newbuf; 228 txtRec->buflen = (uint16_t)(newlen); 229 txtRec->malloced = 1; 230 } 231 start = txtRec->buffer + txtRec->datalen; 232 p = start + 1; 233 memcpy(p, key, keysize); 234 p += keysize; 235 if (value) 236 { 237 *p++ = '='; 238 memcpy(p, value, valueSize); 239 p += valueSize; 240 } 241 *start = (uint8_t)(p - start - 1); 242 txtRec->datalen += p - start; 243 return(kDNSServiceErr_NoError); 244 } 245 246 DNSServiceErrorType DNSSD_API TXTRecordRemoveValue 247 ( 248 TXTRecordRef *txtRecord, 249 const char *key 250 ) 251 { 252 unsigned long keylen, itemlen, remainder; 253 uint8_t *item = InternalTXTRecordSearch(txtRec->datalen, txtRec->buffer, key, &keylen); 254 if (!item) return(kDNSServiceErr_NoSuchKey); 255 itemlen = (unsigned long)(1 + item[0]); 256 remainder = (unsigned long)((txtRec->buffer + txtRec->datalen) - (item + itemlen)); 257 // Use memmove because memcpy behaviour is undefined for overlapping regions 258 memmove(item, item + itemlen, remainder); 259 txtRec->datalen -= itemlen; 260 return(kDNSServiceErr_NoError); 261 } 262 263 uint16_t DNSSD_API TXTRecordGetLength (const TXTRecordRef *txtRecord) { return(txtRec->datalen); } 264 const void * DNSSD_API TXTRecordGetBytesPtr(const TXTRecordRef *txtRecord) { return(txtRec->buffer); } 265 266 /********************************************************************************************* 267 * 268 * TXT Record Parsing Functions 269 * 270 *********************************************************************************************/ 271 272 int DNSSD_API TXTRecordContainsKey 273 ( 274 uint16_t txtLen, 275 const void *txtRecord, 276 const char *key 277 ) 278 { 279 unsigned long keylen; 280 return (InternalTXTRecordSearch(txtLen, txtRecord, key, &keylen) ? 1 : 0); 281 } 282 283 const void * DNSSD_API TXTRecordGetValuePtr 284 ( 285 uint16_t txtLen, 286 const void *txtRecord, 287 const char *key, 288 uint8_t *valueLen 289 ) 290 { 291 unsigned long keylen; 292 uint8_t *item = InternalTXTRecordSearch(txtLen, txtRecord, key, &keylen); 293 if (!item || item[0] <= keylen) return(NULL); // If key not found, or found with no value, return NULL 294 *valueLen = (uint8_t)(item[0] - (keylen + 1)); 295 return (item + 1 + keylen + 1); 296 } 297 298 uint16_t DNSSD_API TXTRecordGetCount 299 ( 300 uint16_t txtLen, 301 const void *txtRecord 302 ) 303 { 304 uint16_t count = 0; 305 uint8_t *p = (uint8_t*)txtRecord; 306 uint8_t *e = p + txtLen; 307 while (p<e) { p += 1 + p[0]; count++; } 308 return((p>e) ? (uint16_t)0 : count); 309 } 310 311 DNSServiceErrorType DNSSD_API TXTRecordGetItemAtIndex 312 ( 313 uint16_t txtLen, 314 const void *txtRecord, 315 uint16_t itemIndex, 316 uint16_t keyBufLen, 317 char *key, 318 uint8_t *valueLen, 319 const void **value 320 ) 321 { 322 uint16_t count = 0; 323 uint8_t *p = (uint8_t*)txtRecord; 324 uint8_t *e = p + txtLen; 325 while (p<e && count<itemIndex) { p += 1 + p[0]; count++; } // Find requested item 326 if (p<e && p + 1 + p[0] <= e) // If valid 327 { 328 uint8_t *x = p+1; 329 unsigned long len = 0; 330 e = p + 1 + p[0]; 331 while (x+len<e && x[len] != '=') len++; 332 if (len >= keyBufLen) return(kDNSServiceErr_NoMemory); 333 memcpy(key, x, len); 334 key[len] = 0; 335 if (x+len<e) // If we found '=' 336 { 337 *value = x + len + 1; 338 *valueLen = (uint8_t)(p[0] - (len + 1)); 339 } 340 else 341 { 342 *value = NULL; 343 *valueLen = 0; 344 } 345 return(kDNSServiceErr_NoError); 346 } 347 return(kDNSServiceErr_Invalid); 348 } 349 350 /********************************************************************************************* 351 * 352 * SCCS-compatible version string 353 * 354 *********************************************************************************************/ 355 356 // For convenience when using the "strings" command, this is the last thing in the file 357 358 // Note: The C preprocessor stringify operator ('#') makes a string from its argument, without macro expansion 359 // e.g. If "version" is #define'd to be "4", then STRINGIFY_AWE(version) will return the string "version", not "4" 360 // To expand "version" to its value before making the string, use STRINGIFY(version) instead 361 #define STRINGIFY_ARGUMENT_WITHOUT_EXPANSION(s) # s 362 #define STRINGIFY(s) STRINGIFY_ARGUMENT_WITHOUT_EXPANSION(s) 363 364 // NOT static -- otherwise the compiler may optimize it out 365 // The "@(#) " pattern is a special prefix the "what" command looks for 366 #ifndef MDNS_VERSIONSTR_NODTS 367 const char VersionString_SCCS_libdnssd[] = "@(#) libdns_sd " STRINGIFY(mDNSResponderVersion) " (" __DATE__ " " __TIME__ ")"; 368 #else 369 const char VersionString_SCCS_libdnssd[] = "@(#) libdns_sd " STRINGIFY(mDNSResponderVersion); 370 #endif 371