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