xref: /freebsd/lib/libsdp/search.c (revision 97cb52fa9aefd90fad38790fded50905aeeb9b9e)
1 /*-
2  * search.c
3  *
4  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
5  *
6  * Copyright (c) 2001-2003 Maksim Yevmenkin <m_evmenkin@yahoo.com>
7  * All rights reserved.
8  *
9  * Redistribution and use in source and binary forms, with or without
10  * modification, are permitted provided that the following conditions
11  * are met:
12  * 1. Redistributions of source code must retain the above copyright
13  *    notice, this list of conditions and the following disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28  * SUCH DAMAGE.
29  *
30  * $Id: search.c,v 1.2 2003/09/04 22:12:13 max Exp $
31  * $FreeBSD$
32  */
33 
34 #include <sys/uio.h>
35 #include <netinet/in.h>
36 #include <arpa/inet.h>
37 #define L2CAP_SOCKET_CHECKED
38 #include <bluetooth.h>
39 #include <errno.h>
40 #include <stdio.h>
41 #include <stdlib.h>
42 #include <string.h>
43 #include <unistd.h>
44 
45 #include <sdp-int.h>
46 #include <sdp.h>
47 
48 int32_t
49 sdp_search(void *xss,
50 		uint32_t plen, uint16_t const *pp,
51 		uint32_t alen, uint32_t const *ap,
52 		uint32_t vlen, sdp_attr_t *vp)
53 {
54 	struct sdp_xpdu {
55 		sdp_pdu_t		 pdu;
56 		uint16_t		 len;
57 	} __attribute__ ((packed))	 xpdu;
58 
59 	sdp_session_p			 ss = (sdp_session_p) xss;
60 	uint8_t				*req = NULL, *rsp = NULL, *rsp_tmp = NULL;
61 	int32_t				 t, len;
62 	uint16_t			 lo, hi;
63 
64 	if (ss == NULL)
65 		return (-1);
66 
67 	if (ss->req == NULL || ss->rsp == NULL ||
68 	    plen == 0 || pp == NULL || alen == 0 || ap == NULL) {
69 		ss->error = EINVAL;
70 		return (-1);
71 	}
72 
73 	req = ss->req;
74 
75 	/* Calculate ServiceSearchPattern length */
76 	plen = plen * (sizeof(pp[0]) + 1);
77 
78 	/* Calculate AttributeIDList length */
79 	for (len = 0, t = 0; t < alen; t ++) {
80 		lo = (uint16_t) (ap[t] >> 16);
81 		hi = (uint16_t) (ap[t]);
82 
83 		if (lo > hi) {
84 			ss->error = EINVAL;
85 			return (-1);
86 		}
87 
88 		if (lo != hi)
89 			len += (sizeof(ap[t]) + 1);
90 		else
91 			len += (sizeof(lo) + 1);
92 	}
93 	alen = len;
94 
95 	/* Calculate length of the request */
96 	len =	plen + sizeof(uint8_t) + sizeof(uint16_t) +
97 			/* ServiceSearchPattern */
98 		sizeof(uint16_t) +
99 			/* MaximumAttributeByteCount */
100 		alen + sizeof(uint8_t) + sizeof(uint16_t);
101 			/* AttributeIDList */
102 
103 	if (ss->req_e - req < len) {
104 		ss->error = ENOBUFS;
105 		return (-1);
106 	}
107 
108 	/* Put ServiceSearchPattern */
109 	SDP_PUT8(SDP_DATA_SEQ16, req);
110 	SDP_PUT16(plen, req);
111 	for (; plen > 0; pp ++, plen -= (sizeof(pp[0]) + 1)) {
112 		SDP_PUT8(SDP_DATA_UUID16, req);
113 		SDP_PUT16(*pp, req);
114 	}
115 
116 	/* Put MaximumAttributeByteCount */
117 	SDP_PUT16(0xffff, req);
118 
119 	/* Put AttributeIDList */
120 	SDP_PUT8(SDP_DATA_SEQ16, req);
121 	SDP_PUT16(alen, req);
122 	for (; alen > 0; ap ++) {
123 		lo = (uint16_t) (*ap >> 16);
124 		hi = (uint16_t) (*ap);
125 
126 		if (lo != hi) {
127 			/* Put attribute range */
128 			SDP_PUT8(SDP_DATA_UINT32, req);
129 			SDP_PUT32(*ap, req);
130 			alen -= (sizeof(ap[0]) + 1);
131 		} else {
132 			/* Put attribute */
133 			SDP_PUT8(SDP_DATA_UINT16, req);
134 			SDP_PUT16(lo, req);
135 			alen -= (sizeof(lo) + 1);
136 		}
137 	}
138 
139 	/* Submit ServiceSearchAttributeRequest and wait for response */
140 	ss->cslen = 0;
141 	rsp = ss->rsp;
142 
143 	do {
144 		struct iovec	 iov[2];
145 		uint8_t		*req_cs = req;
146 
147 		/* Add continuation state (if any) */
148 		if (ss->req_e - req_cs < ss->cslen + 1) {
149 			ss->error = ENOBUFS;
150 			return (-1);
151 		}
152 
153 		SDP_PUT8(ss->cslen, req_cs);
154 		if (ss->cslen > 0) {
155 			memcpy(req_cs, ss->cs, ss->cslen);
156 			req_cs += ss->cslen;
157 		}
158 
159 		/* Prepare SDP PDU header */
160 		xpdu.pdu.pid = SDP_PDU_SERVICE_SEARCH_ATTRIBUTE_REQUEST;
161 		xpdu.pdu.tid = htons(ss->tid);
162 		xpdu.pdu.len = htons(req_cs - ss->req);
163 
164 		/* Submit request */
165 		iov[0].iov_base = (void *) &xpdu;
166 		iov[0].iov_len = sizeof(xpdu.pdu);
167 		iov[1].iov_base = (void *) ss->req;
168 		iov[1].iov_len = req_cs - ss->req;
169 
170 		do {
171 			len = writev(ss->s, iov, sizeof(iov)/sizeof(iov[0]));
172 		} while (len < 0 && errno == EINTR);
173 
174 		if (len < 0) {
175 			ss->error = errno;
176 			return (-1);
177 		}
178 
179 		/* Read response */
180 		iov[0].iov_base = (void *) &xpdu;
181 		iov[0].iov_len = sizeof(xpdu);
182 		iov[1].iov_base = (void *) rsp;
183 		iov[1].iov_len = ss->imtu;
184 
185 		do {
186 			len = readv(ss->s, iov, sizeof(iov)/sizeof(iov[0]));
187 		} while (len < 0 && errno == EINTR);
188 
189 		if (len < 0) {
190 			ss->error = errno;
191 			return (-1);
192 		}
193 		if (len < sizeof(xpdu)) {
194 			ss->error = ENOMSG;
195 			return (-1);
196 		}
197 
198 		xpdu.pdu.tid = ntohs(xpdu.pdu.tid);
199 		xpdu.pdu.len = ntohs(xpdu.pdu.len);
200 		xpdu.len = ntohs(xpdu.len);
201 
202 		if (xpdu.pdu.pid == SDP_PDU_ERROR_RESPONSE ||
203 		    xpdu.pdu.tid != ss->tid ||
204 		    xpdu.pdu.len > len ||
205 		    xpdu.len > xpdu.pdu.len) {
206 			ss->error = EIO;
207 			return (-1);
208 		}
209 
210 		rsp += xpdu.len;
211 		ss->tid ++;
212 
213 		/* Save continuation state (if any) */
214 		ss->cslen = rsp[0];
215 		if (ss->cslen > 0) {
216 			if (ss->cslen > sizeof(ss->cs)) {
217 				ss->error = ENOBUFS;
218 				return (-1);
219 			}
220 
221 			memcpy(ss->cs, rsp + 1, ss->cslen);
222 
223 			/*
224 			 * Ensure that we always have ss->imtu bytes
225 			 * available in the ss->rsp buffer
226 			 */
227 
228 			if (ss->rsp_e - rsp <= ss->imtu) {
229 				uint32_t	 size, offset;
230 
231 				size = ss->rsp_e - ss->rsp + ss->imtu;
232 				offset = rsp - ss->rsp;
233 
234 				rsp_tmp = realloc(ss->rsp, size);
235 				if (rsp_tmp == NULL) {
236 					ss->error = ENOMEM;
237 					return (-1);
238 				}
239 
240 				ss->rsp = rsp_tmp;
241 				ss->rsp_e = ss->rsp + size;
242 				rsp = ss->rsp + offset;
243 			}
244 		}
245 	} while (ss->cslen > 0);
246 
247 	/*
248 	 * If we got here then we have completed SDP transaction and now
249 	 * we must populate attribute values into vp array. At this point
250 	 * ss->rsp points to the beginning of the response and rsp points
251 	 * to the end of the response.
252 	 *
253 	 * From Bluetooth v1.1 spec page 364
254 	 *
255 	 * The AttributeLists is a data element sequence where each element
256 	 * in turn is a data element sequence representing an attribute list.
257 	 * Each attribute list contains attribute IDs and attribute values
258 	 * from one service record. The first element in each attribute list
259 	 * contains the attribute ID of the first attribute to be returned for
260 	 * that service record. The second element in each attribute list
261 	 * contains the corresponding attribute value. Successive pairs of
262 	 * elements in each attribute list contain additional attribute ID
263 	 * and value pairs. Only attributes that have non-null values within
264 	 * the service record and whose attribute IDs were specified in the
265 	 * SDP_ServiceSearchAttributeRequest are contained in the AttributeLists
266 	 * Neither an attribute ID nor attribute value is placed in
267 	 * AttributeLists for attributes in the service record that have no
268 	 * value. Within each attribute list, the attributes are listed in
269 	 * ascending order of attribute ID value.
270 	 */
271 
272 	if (vp == NULL)
273 		goto done;
274 
275 	rsp_tmp = ss->rsp;
276 
277 	/* Skip the first SEQ */
278 	SDP_GET8(t, rsp_tmp);
279 	switch (t) {
280 	case SDP_DATA_SEQ8:
281 		SDP_GET8(len, rsp_tmp);
282 		break;
283 
284 	case SDP_DATA_SEQ16:
285 		SDP_GET16(len, rsp_tmp);
286 		break;
287 
288 	case SDP_DATA_SEQ32:
289 		SDP_GET32(len, rsp_tmp);
290 		break;
291 
292 	default:
293 		ss->error = ENOATTR;
294 		return (-1);
295 		/* NOT REACHED */
296 	}
297 
298 	for (; rsp_tmp < rsp && vlen > 0; ) {
299 		/* Get set of attributes for the next record */
300 		SDP_GET8(t, rsp_tmp);
301 		switch (t) {
302 		case SDP_DATA_SEQ8:
303 			SDP_GET8(len, rsp_tmp);
304 			break;
305 
306 		case SDP_DATA_SEQ16:
307 			SDP_GET16(len, rsp_tmp);
308 			break;
309 
310 		case SDP_DATA_SEQ32:
311 			SDP_GET32(len, rsp_tmp);
312 			break;
313 
314 		default:
315 			ss->error = ENOATTR;
316 			return (-1);
317 			/* NOT REACHED */
318 		}
319 
320 		/* Now rsp_tmp points to list of (attr,value) pairs */
321 		for (; len > 0 && vlen > 0; vp ++, vlen --) {
322 			/* Attribute */
323 			SDP_GET8(t, rsp_tmp);
324 			if (t != SDP_DATA_UINT16) {
325 				ss->error = ENOATTR;
326 				return (-1);
327 			}
328 			SDP_GET16(vp->attr, rsp_tmp);
329 
330 			/* Attribute value */
331 			switch (rsp_tmp[0]) {
332 			case SDP_DATA_NIL:
333 				alen = 0;
334 				break;
335 
336 			case SDP_DATA_UINT8:
337 			case SDP_DATA_INT8:
338 			case SDP_DATA_BOOL:
339 				alen = sizeof(uint8_t);
340 				break;
341 
342 			case SDP_DATA_UINT16:
343 			case SDP_DATA_INT16:
344 			case SDP_DATA_UUID16:
345 				alen = sizeof(uint16_t);
346 				break;
347 
348 			case SDP_DATA_UINT32:
349 			case SDP_DATA_INT32:
350 			case SDP_DATA_UUID32:
351 				alen = sizeof(uint32_t);
352 				break;
353 
354 			case SDP_DATA_UINT64:
355 			case SDP_DATA_INT64:
356 				alen = sizeof(uint64_t);
357 				break;
358 
359 			case SDP_DATA_UINT128:
360 			case SDP_DATA_INT128:
361 			case SDP_DATA_UUID128:
362 				alen = sizeof(uint128_t);
363 				break;
364 
365 			case SDP_DATA_STR8:
366 			case SDP_DATA_URL8:
367 			case SDP_DATA_SEQ8:
368 			case SDP_DATA_ALT8:
369 				alen = rsp_tmp[1] + sizeof(uint8_t);
370 				break;
371 
372 			case SDP_DATA_STR16:
373 			case SDP_DATA_URL16:
374 			case SDP_DATA_SEQ16:
375 			case SDP_DATA_ALT16:
376 				alen =	  ((uint16_t)rsp_tmp[1] << 8)
377 					| ((uint16_t)rsp_tmp[2]);
378 				alen += sizeof(uint16_t);
379 				break;
380 
381 			case SDP_DATA_STR32:
382 			case SDP_DATA_URL32:
383 			case SDP_DATA_SEQ32:
384 			case SDP_DATA_ALT32:
385 				alen =    ((uint32_t)rsp_tmp[1] << 24)
386 					| ((uint32_t)rsp_tmp[2] << 16)
387 					| ((uint32_t)rsp_tmp[3] <<  8)
388 					| ((uint32_t)rsp_tmp[4]);
389 				alen += sizeof(uint32_t);
390 				break;
391 
392 			default:
393 				ss->error = ENOATTR;
394 				return (-1);
395 				/* NOT REACHED */
396 			}
397 
398 			alen += sizeof(uint8_t);
399 
400 			if (vp->value != NULL) {
401 				if (alen <= vp->vlen) {
402 					vp->flags = SDP_ATTR_OK;
403 					vp->vlen = alen;
404 				} else
405 					vp->flags = SDP_ATTR_TRUNCATED;
406 
407 				memcpy(vp->value, rsp_tmp, vp->vlen);
408 			} else
409 				vp->flags = SDP_ATTR_INVALID;
410 
411 			len -=	(
412 				sizeof(uint8_t) + sizeof(uint16_t) +
413 				alen
414 				);
415 
416 			rsp_tmp += alen;
417 		}
418 	}
419 done:
420 	ss->error = 0;
421 
422 	return (0);
423 }
424 
425