]>
Commit | Line | Data |
---|---|---|
67c8f8a1 A |
1 | /* -*- Mode: C; tab-width: 4 -*- |
2 | * | |
9f221bca | 3 | * Copyright (c) 2004-2011 Apple Inc. All rights reserved. |
8e92c31c | 4 | * |
83fb1e36 | 5 | * Redistribution and use in source and binary forms, with or without |
7f0064bd A |
6 | * modification, are permitted provided that the following conditions are met: |
7 | * | |
83fb1e36 A |
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. | |
9f221bca | 13 | * 3. Neither the name of Apple Inc. ("Apple") nor the names of its |
83fb1e36 A |
14 | * contributors may be used to endorse or promote products derived from this |
15 | * software without specific prior written permission. | |
7f0064bd | 16 | * |
83fb1e36 A |
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 | |
7f0064bd | 26 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
8e92c31c A |
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 | ||
7f0064bd A |
38 | #if defined(_WIN32) |
39 | // disable warning "conversion from <data> to uint16_t" | |
40 | #pragma warning(disable:4244) | |
32bb7e43 | 41 | #define strncasecmp _strnicmp |
83fb1e36 | 42 | #define strcasecmp _stricmp |
7f0064bd A |
43 | #endif |
44 | ||
8e92c31c | 45 | /********************************************************************************************* |
83fb1e36 A |
46 | * |
47 | * Supporting Functions | |
48 | * | |
49 | *********************************************************************************************/ | |
8e92c31c | 50 | |
32bb7e43 A |
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) | |
8e92c31c A |
55 | |
56 | static int DomainEndsInDot(const char *dom) | |
83fb1e36 A |
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 | } | |
8e92c31c A |
70 | |
71 | static uint8_t *InternalTXTRecordSearch | |
83fb1e36 A |
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 | } | |
8e92c31c A |
91 | |
92 | /********************************************************************************************* | |
83fb1e36 A |
93 | * |
94 | * General Utility Functions | |
95 | * | |
96 | *********************************************************************************************/ | |
8e92c31c | 97 | |
32bb7e43 A |
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 | |
83fb1e36 A |
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; | |
32bb7e43 A |
154 | |
155 | fail: | |
83fb1e36 A |
156 | *fn = '\0'; |
157 | return kDNSServiceErr_BadParam; | |
158 | } | |
8e92c31c A |
159 | |
160 | /********************************************************************************************* | |
83fb1e36 A |
161 | * |
162 | * TXT Record Construction Functions | |
163 | * | |
164 | *********************************************************************************************/ | |
8e92c31c A |
165 | |
166 | typedef struct _TXTRecordRefRealType | |
83fb1e36 A |
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; | |
8e92c31c A |
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. | |
67c8f8a1 | 178 | struct CompileTimeAssertionCheck_dnssd_clientlib |
83fb1e36 A |
179 | { |
180 | char assert0[(sizeof(TXTRecordRefRealType) <= 16) ? 1 : -1]; | |
181 | }; | |
8e92c31c | 182 | |
4aea607d | 183 | void DNSSD_API TXTRecordCreate |
83fb1e36 A |
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 | } | |
8e92c31c | 195 | |
4aea607d | 196 | void DNSSD_API TXTRecordDeallocate(TXTRecordRef *txtRecord) |
83fb1e36 A |
197 | { |
198 | if (txtRec->malloced) free(txtRec->buffer); | |
199 | } | |
8e92c31c | 200 | |
4aea607d | 201 | DNSServiceErrorType DNSSD_API TXTRecordSetValue |
83fb1e36 A |
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 | } | |
8e92c31c | 245 | |
4aea607d | 246 | DNSServiceErrorType DNSSD_API TXTRecordRemoveValue |
83fb1e36 A |
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 | } | |
8e92c31c | 262 | |
4aea607d A |
263 | uint16_t DNSSD_API TXTRecordGetLength (const TXTRecordRef *txtRecord) { return(txtRec->datalen); } |
264 | const void * DNSSD_API TXTRecordGetBytesPtr(const TXTRecordRef *txtRecord) { return(txtRec->buffer); } | |
8e92c31c A |
265 | |
266 | /********************************************************************************************* | |
83fb1e36 A |
267 | * |
268 | * TXT Record Parsing Functions | |
269 | * | |
270 | *********************************************************************************************/ | |
8e92c31c | 271 | |
4aea607d | 272 | int DNSSD_API TXTRecordContainsKey |
83fb1e36 A |
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 | } | |
8e92c31c | 282 | |
4aea607d | 283 | const void * DNSSD_API TXTRecordGetValuePtr |
83fb1e36 A |
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 | } | |
8e92c31c | 297 | |
4aea607d | 298 | uint16_t DNSSD_API TXTRecordGetCount |
83fb1e36 A |
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 | } | |
8e92c31c | 310 | |
4aea607d | 311 | DNSServiceErrorType DNSSD_API TXTRecordGetItemAtIndex |
83fb1e36 A |
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 | } | |
67c8f8a1 A |
349 | |
350 | /********************************************************************************************* | |
83fb1e36 A |
351 | * |
352 | * SCCS-compatible version string | |
353 | * | |
354 | *********************************************************************************************/ | |
67c8f8a1 A |
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 | |
83fb1e36 | 361 | #define STRINGIFY_ARGUMENT_WITHOUT_EXPANSION(s) # s |
67c8f8a1 A |
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 | const char VersionString_SCCS_libdnssd[] = "@(#) libdns_sd " STRINGIFY(mDNSResponderVersion) " (" __DATE__ " " __TIME__ ")"; |