xref: /illumos-gate/usr/src/contrib/mDNSResponder/mDNSShared/dnssd_clientlib.c (revision 8a2b682e57a046b828f37bcde1776f131ef4629f)
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