sdp_get.c revision 1.1.2.2 1 1.1.2.2 jym /* $NetBSD: sdp_get.c,v 1.1.2.2 2009/05/13 19:18:20 jym Exp $ */
2 1.1.2.2 jym
3 1.1.2.2 jym /*-
4 1.1.2.2 jym * Copyright (c) 2009 The NetBSD Foundation, Inc.
5 1.1.2.2 jym * All rights reserved.
6 1.1.2.2 jym *
7 1.1.2.2 jym * This code is derived from software contributed to The NetBSD Foundation
8 1.1.2.2 jym * by Iain Hibbert.
9 1.1.2.2 jym *
10 1.1.2.2 jym * Redistribution and use in source and binary forms, with or without
11 1.1.2.2 jym * modification, are permitted provided that the following conditions
12 1.1.2.2 jym * are met:
13 1.1.2.2 jym * 1. Redistributions of source code must retain the above copyright
14 1.1.2.2 jym * notice, this list of conditions and the following disclaimer.
15 1.1.2.2 jym * 2. Redistributions in binary form must reproduce the above copyright
16 1.1.2.2 jym * notice, this list of conditions and the following disclaimer in the
17 1.1.2.2 jym * documentation and/or other materials provided with the distribution.
18 1.1.2.2 jym *
19 1.1.2.2 jym * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 1.1.2.2 jym * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 1.1.2.2 jym * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 1.1.2.2 jym * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 1.1.2.2 jym * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 1.1.2.2 jym * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 1.1.2.2 jym * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 1.1.2.2 jym * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 1.1.2.2 jym * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 1.1.2.2 jym * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 1.1.2.2 jym * POSSIBILITY OF SUCH DAMAGE.
30 1.1.2.2 jym */
31 1.1.2.2 jym
32 1.1.2.2 jym #include <sys/cdefs.h>
33 1.1.2.2 jym __RCSID("$NetBSD: sdp_get.c,v 1.1.2.2 2009/05/13 19:18:20 jym Exp $");
34 1.1.2.2 jym
35 1.1.2.2 jym #include <sdp.h>
36 1.1.2.2 jym #include <limits.h>
37 1.1.2.2 jym
38 1.1.2.2 jym /******************************************************************************
39 1.1.2.2 jym * sdp_get_xxxx(data, value)
40 1.1.2.2 jym *
41 1.1.2.2 jym * examine first SDP data element in list for xxx type, extracting to given
42 1.1.2.2 jym * storage and advancing pointer if found.
43 1.1.2.2 jym * - these functions will not modify data pointer unless the value was
44 1.1.2.2 jym * extracted successfully
45 1.1.2.2 jym * - these functions always update the data pointer before the value pointer,
46 1.1.2.2 jym * so where the value is a sdp_data_t the data struct can be discarded.
47 1.1.2.2 jym */
48 1.1.2.2 jym
49 1.1.2.2 jym bool
50 1.1.2.2 jym sdp_get_data(sdp_data_t *data, sdp_data_t *value)
51 1.1.2.2 jym {
52 1.1.2.2 jym uint8_t *p = data->next;
53 1.1.2.2 jym ssize_t l = sdp_data_size(data);
54 1.1.2.2 jym
55 1.1.2.2 jym if (l == -1
56 1.1.2.2 jym || p + l > data->end)
57 1.1.2.2 jym return false;
58 1.1.2.2 jym
59 1.1.2.2 jym data->next = p + l;
60 1.1.2.2 jym value->next = p;
61 1.1.2.2 jym value->end = p + l;
62 1.1.2.2 jym return true;
63 1.1.2.2 jym }
64 1.1.2.2 jym
65 1.1.2.2 jym bool
66 1.1.2.2 jym sdp_get_attr(sdp_data_t *data, uint16_t *attr, sdp_data_t *value)
67 1.1.2.2 jym {
68 1.1.2.2 jym sdp_data_t v, d = *data;
69 1.1.2.2 jym uintmax_t a;
70 1.1.2.2 jym
71 1.1.2.2 jym if (sdp_data_type(&d) != SDP_DATA_UINT16
72 1.1.2.2 jym || !sdp_get_uint(&d, &a)
73 1.1.2.2 jym || !sdp_get_data(&d, &v))
74 1.1.2.2 jym return false;
75 1.1.2.2 jym
76 1.1.2.2 jym *attr = (uint16_t)a;
77 1.1.2.2 jym *data = d;
78 1.1.2.2 jym *value = v;
79 1.1.2.2 jym return true;
80 1.1.2.2 jym }
81 1.1.2.2 jym
82 1.1.2.2 jym bool
83 1.1.2.2 jym sdp_get_uuid(sdp_data_t *data, uuid_t *uuid)
84 1.1.2.2 jym {
85 1.1.2.2 jym uint8_t *p = data->next;
86 1.1.2.2 jym
87 1.1.2.2 jym if (p + 1 > data->end)
88 1.1.2.2 jym return false;
89 1.1.2.2 jym
90 1.1.2.2 jym switch (*p++) {
91 1.1.2.2 jym case SDP_DATA_UUID16:
92 1.1.2.2 jym if (p + 2 > data->end)
93 1.1.2.2 jym return false;
94 1.1.2.2 jym
95 1.1.2.2 jym *uuid = BLUETOOTH_BASE_UUID;
96 1.1.2.2 jym uuid->time_low = be16dec(p);
97 1.1.2.2 jym p += 2;
98 1.1.2.2 jym break;
99 1.1.2.2 jym
100 1.1.2.2 jym case SDP_DATA_UUID32:
101 1.1.2.2 jym if (p + 4 > data->end)
102 1.1.2.2 jym return false;
103 1.1.2.2 jym
104 1.1.2.2 jym *uuid = BLUETOOTH_BASE_UUID;
105 1.1.2.2 jym uuid->time_low = be32dec(p);
106 1.1.2.2 jym p += 4;
107 1.1.2.2 jym break;
108 1.1.2.2 jym
109 1.1.2.2 jym case SDP_DATA_UUID128:
110 1.1.2.2 jym if (p + 16 > data->end)
111 1.1.2.2 jym return false;
112 1.1.2.2 jym
113 1.1.2.2 jym uuid_dec_be(p, uuid);
114 1.1.2.2 jym p += 16;
115 1.1.2.2 jym break;
116 1.1.2.2 jym
117 1.1.2.2 jym default:
118 1.1.2.2 jym return false;
119 1.1.2.2 jym }
120 1.1.2.2 jym
121 1.1.2.2 jym data->next = p;
122 1.1.2.2 jym return true;
123 1.1.2.2 jym }
124 1.1.2.2 jym
125 1.1.2.2 jym bool
126 1.1.2.2 jym sdp_get_bool(sdp_data_t *data, bool *value)
127 1.1.2.2 jym {
128 1.1.2.2 jym uint8_t *p = data->next;
129 1.1.2.2 jym uint8_t v;
130 1.1.2.2 jym
131 1.1.2.2 jym if (p + 1 > data->end)
132 1.1.2.2 jym return false;
133 1.1.2.2 jym
134 1.1.2.2 jym switch (*p++) {
135 1.1.2.2 jym case SDP_DATA_BOOL:
136 1.1.2.2 jym if (p + 1 > data->end)
137 1.1.2.2 jym return false;
138 1.1.2.2 jym
139 1.1.2.2 jym v = *p;
140 1.1.2.2 jym p += 1;
141 1.1.2.2 jym break;
142 1.1.2.2 jym
143 1.1.2.2 jym default:
144 1.1.2.2 jym return false;
145 1.1.2.2 jym }
146 1.1.2.2 jym
147 1.1.2.2 jym data->next = p;
148 1.1.2.2 jym *value = ((v != 0) ? true : false);
149 1.1.2.2 jym return true;
150 1.1.2.2 jym }
151 1.1.2.2 jym
152 1.1.2.2 jym bool
153 1.1.2.2 jym sdp_get_uint(sdp_data_t *data, uintmax_t *value)
154 1.1.2.2 jym {
155 1.1.2.2 jym uint8_t *p = data->next;
156 1.1.2.2 jym uint64_t v, x;
157 1.1.2.2 jym
158 1.1.2.2 jym if (p + 1 > data->end)
159 1.1.2.2 jym return false;
160 1.1.2.2 jym
161 1.1.2.2 jym switch (*p++) {
162 1.1.2.2 jym case SDP_DATA_UINT8:
163 1.1.2.2 jym if (p + 1 > data->end)
164 1.1.2.2 jym return false;
165 1.1.2.2 jym
166 1.1.2.2 jym v = *p;
167 1.1.2.2 jym p += 1;
168 1.1.2.2 jym break;
169 1.1.2.2 jym
170 1.1.2.2 jym case SDP_DATA_UINT16:
171 1.1.2.2 jym if (p + 2 > data->end)
172 1.1.2.2 jym return false;
173 1.1.2.2 jym
174 1.1.2.2 jym v = be16dec(p);
175 1.1.2.2 jym p += 2;
176 1.1.2.2 jym break;
177 1.1.2.2 jym
178 1.1.2.2 jym case SDP_DATA_UINT32:
179 1.1.2.2 jym if (p + 4 > data->end)
180 1.1.2.2 jym return false;
181 1.1.2.2 jym
182 1.1.2.2 jym v = be32dec(p);
183 1.1.2.2 jym p += 4;
184 1.1.2.2 jym break;
185 1.1.2.2 jym
186 1.1.2.2 jym case SDP_DATA_UINT64:
187 1.1.2.2 jym if (p + 8 > data->end)
188 1.1.2.2 jym return false;
189 1.1.2.2 jym
190 1.1.2.2 jym v = be64dec(p);
191 1.1.2.2 jym if (v > UINTMAX_MAX)
192 1.1.2.2 jym return false;
193 1.1.2.2 jym
194 1.1.2.2 jym p += 8;
195 1.1.2.2 jym break;
196 1.1.2.2 jym
197 1.1.2.2 jym case SDP_DATA_UINT128:
198 1.1.2.2 jym if (p + 16 > data->end)
199 1.1.2.2 jym return false;
200 1.1.2.2 jym
201 1.1.2.2 jym x = be64dec(p);
202 1.1.2.2 jym v = be64dec(p + 8);
203 1.1.2.2 jym if (x != 0 || v > UINTMAX_MAX)
204 1.1.2.2 jym return false;
205 1.1.2.2 jym
206 1.1.2.2 jym p += 16;
207 1.1.2.2 jym break;
208 1.1.2.2 jym
209 1.1.2.2 jym default:
210 1.1.2.2 jym return false;
211 1.1.2.2 jym }
212 1.1.2.2 jym
213 1.1.2.2 jym data->next = p;
214 1.1.2.2 jym *value = (uintmax_t)v;
215 1.1.2.2 jym return true;
216 1.1.2.2 jym }
217 1.1.2.2 jym
218 1.1.2.2 jym bool
219 1.1.2.2 jym sdp_get_int(sdp_data_t *data, intmax_t *value)
220 1.1.2.2 jym {
221 1.1.2.2 jym uint8_t *p = data->next;
222 1.1.2.2 jym int64_t v, x;
223 1.1.2.2 jym
224 1.1.2.2 jym if (p + 1 > data->end)
225 1.1.2.2 jym return false;
226 1.1.2.2 jym
227 1.1.2.2 jym switch (*p++) {
228 1.1.2.2 jym case SDP_DATA_INT8:
229 1.1.2.2 jym if (p + 1 > data->end)
230 1.1.2.2 jym return false;
231 1.1.2.2 jym
232 1.1.2.2 jym v = *(int8_t *)p;
233 1.1.2.2 jym p += 1;
234 1.1.2.2 jym break;
235 1.1.2.2 jym
236 1.1.2.2 jym case SDP_DATA_INT16:
237 1.1.2.2 jym if (p + 2 > data->end)
238 1.1.2.2 jym return false;
239 1.1.2.2 jym
240 1.1.2.2 jym v = (int16_t)be16dec(p);
241 1.1.2.2 jym p += 2;
242 1.1.2.2 jym break;
243 1.1.2.2 jym
244 1.1.2.2 jym case SDP_DATA_INT32:
245 1.1.2.2 jym if (p + 4 > data->end)
246 1.1.2.2 jym return false;
247 1.1.2.2 jym
248 1.1.2.2 jym v = (int32_t)be32dec(p);
249 1.1.2.2 jym p += 4;
250 1.1.2.2 jym break;
251 1.1.2.2 jym
252 1.1.2.2 jym case SDP_DATA_INT64:
253 1.1.2.2 jym if (p + 8 > data->end)
254 1.1.2.2 jym return false;
255 1.1.2.2 jym
256 1.1.2.2 jym v = (int64_t)be64dec(p);
257 1.1.2.2 jym if (v > INTMAX_MAX || v < INTMAX_MIN)
258 1.1.2.2 jym return false;
259 1.1.2.2 jym
260 1.1.2.2 jym p += 8;
261 1.1.2.2 jym break;
262 1.1.2.2 jym
263 1.1.2.2 jym case SDP_DATA_INT128:
264 1.1.2.2 jym if (p + 16 > data->end)
265 1.1.2.2 jym return false;
266 1.1.2.2 jym
267 1.1.2.2 jym x = (int64_t)be64dec(p);
268 1.1.2.2 jym v = (int64_t)be64dec(p + 8);
269 1.1.2.2 jym if (x == 0) {
270 1.1.2.2 jym if (v > INTMAX_MAX)
271 1.1.2.2 jym return false;
272 1.1.2.2 jym } else if (x == -1) {
273 1.1.2.2 jym if (v < INTMAX_MIN)
274 1.1.2.2 jym return false;
275 1.1.2.2 jym } else {
276 1.1.2.2 jym return false;
277 1.1.2.2 jym }
278 1.1.2.2 jym
279 1.1.2.2 jym p += 16;
280 1.1.2.2 jym break;
281 1.1.2.2 jym
282 1.1.2.2 jym default:
283 1.1.2.2 jym return false;
284 1.1.2.2 jym }
285 1.1.2.2 jym
286 1.1.2.2 jym data->next = p;
287 1.1.2.2 jym *value = (intmax_t)v;
288 1.1.2.2 jym return true;
289 1.1.2.2 jym }
290 1.1.2.2 jym
291 1.1.2.2 jym static bool
292 1.1.2.2 jym _sdp_get_ext(uint8_t type, sdp_data_t *data, sdp_data_t *ext)
293 1.1.2.2 jym {
294 1.1.2.2 jym uint8_t *p = data->next;
295 1.1.2.2 jym uint32_t l;
296 1.1.2.2 jym
297 1.1.2.2 jym if (p + 1 > data->end
298 1.1.2.2 jym || SDP_DATA_TYPE(*p) != type)
299 1.1.2.2 jym return false;
300 1.1.2.2 jym
301 1.1.2.2 jym switch (SDP_DATA_SIZE(*p++)) {
302 1.1.2.2 jym case SDP_DATA_EXT8:
303 1.1.2.2 jym if (p + 1 > data->end)
304 1.1.2.2 jym return false;
305 1.1.2.2 jym
306 1.1.2.2 jym l = *p;
307 1.1.2.2 jym p += 1;
308 1.1.2.2 jym break;
309 1.1.2.2 jym
310 1.1.2.2 jym case SDP_DATA_EXT16:
311 1.1.2.2 jym if (p + 2 > data->end)
312 1.1.2.2 jym return false;
313 1.1.2.2 jym
314 1.1.2.2 jym l = be16dec(p);
315 1.1.2.2 jym p += 2;
316 1.1.2.2 jym break;
317 1.1.2.2 jym
318 1.1.2.2 jym case SDP_DATA_EXT32:
319 1.1.2.2 jym if (p + 4 > data->end)
320 1.1.2.2 jym return false;
321 1.1.2.2 jym
322 1.1.2.2 jym l = be32dec(p);
323 1.1.2.2 jym p += 4;
324 1.1.2.2 jym break;
325 1.1.2.2 jym
326 1.1.2.2 jym default:
327 1.1.2.2 jym return false;
328 1.1.2.2 jym }
329 1.1.2.2 jym
330 1.1.2.2 jym if (p + l > data->end)
331 1.1.2.2 jym return false;
332 1.1.2.2 jym
333 1.1.2.2 jym data->next = p + l;
334 1.1.2.2 jym ext->next = p;
335 1.1.2.2 jym ext->end = p + l;
336 1.1.2.2 jym return true;
337 1.1.2.2 jym }
338 1.1.2.2 jym
339 1.1.2.2 jym bool
340 1.1.2.2 jym sdp_get_seq(sdp_data_t *data, sdp_data_t *seq)
341 1.1.2.2 jym {
342 1.1.2.2 jym
343 1.1.2.2 jym return _sdp_get_ext(SDP_DATA_SEQ, data, seq);
344 1.1.2.2 jym }
345 1.1.2.2 jym
346 1.1.2.2 jym bool
347 1.1.2.2 jym sdp_get_alt(sdp_data_t *data, sdp_data_t *alt)
348 1.1.2.2 jym {
349 1.1.2.2 jym
350 1.1.2.2 jym return _sdp_get_ext(SDP_DATA_ALT, data, alt);
351 1.1.2.2 jym }
352 1.1.2.2 jym
353 1.1.2.2 jym bool
354 1.1.2.2 jym sdp_get_str(sdp_data_t *data, char **str, size_t *len)
355 1.1.2.2 jym {
356 1.1.2.2 jym sdp_data_t s;
357 1.1.2.2 jym
358 1.1.2.2 jym if (!_sdp_get_ext(SDP_DATA_STR, data, &s))
359 1.1.2.2 jym return false;
360 1.1.2.2 jym
361 1.1.2.2 jym *str = (char *)s.next;
362 1.1.2.2 jym *len = s.end - s.next;
363 1.1.2.2 jym return true;
364 1.1.2.2 jym }
365 1.1.2.2 jym
366 1.1.2.2 jym bool
367 1.1.2.2 jym sdp_get_url(sdp_data_t *data, char **url, size_t *len)
368 1.1.2.2 jym {
369 1.1.2.2 jym sdp_data_t u;
370 1.1.2.2 jym
371 1.1.2.2 jym if (!_sdp_get_ext(SDP_DATA_URL, data, &u))
372 1.1.2.2 jym return false;
373 1.1.2.2 jym
374 1.1.2.2 jym *url = (char *)u.next;
375 1.1.2.2 jym *len = u.end - u.next;
376 1.1.2.2 jym return true;
377 1.1.2.2 jym }
378