1*c5f754b9Splunky /* $NetBSD: sdp_set.c,v 1.4 2011/04/05 18:58:43 plunky Exp $ */
2dfbf818aSplunky
3dfbf818aSplunky /*-
4dfbf818aSplunky * Copyright (c) 2009 The NetBSD Foundation, Inc.
5dfbf818aSplunky * All rights reserved.
6dfbf818aSplunky *
7dfbf818aSplunky * This code is derived from software contributed to The NetBSD Foundation
8dfbf818aSplunky * by Iain Hibbert.
9dfbf818aSplunky *
10dfbf818aSplunky * Redistribution and use in source and binary forms, with or without
11dfbf818aSplunky * modification, are permitted provided that the following conditions
12dfbf818aSplunky * are met:
13dfbf818aSplunky * 1. Redistributions of source code must retain the above copyright
14dfbf818aSplunky * notice, this list of conditions and the following disclaimer.
15dfbf818aSplunky * 2. Redistributions in binary form must reproduce the above copyright
16dfbf818aSplunky * notice, this list of conditions and the following disclaimer in the
17dfbf818aSplunky * documentation and/or other materials provided with the distribution.
18dfbf818aSplunky *
19dfbf818aSplunky * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20dfbf818aSplunky * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21dfbf818aSplunky * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22dfbf818aSplunky * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23dfbf818aSplunky * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24dfbf818aSplunky * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25dfbf818aSplunky * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26dfbf818aSplunky * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27dfbf818aSplunky * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28dfbf818aSplunky * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29dfbf818aSplunky * POSSIBILITY OF SUCH DAMAGE.
30dfbf818aSplunky */
31dfbf818aSplunky
32dfbf818aSplunky #include <sys/cdefs.h>
33*c5f754b9Splunky __RCSID("$NetBSD: sdp_set.c,v 1.4 2011/04/05 18:58:43 plunky Exp $");
34dfbf818aSplunky
35dfbf818aSplunky #include <bluetooth.h>
36dfbf818aSplunky #include <limits.h>
37dfbf818aSplunky #include <sdp.h>
38dfbf818aSplunky
39dfbf818aSplunky /******************************************************************************
40dfbf818aSplunky * sdp_set_xxxx(data, value)
41dfbf818aSplunky *
42dfbf818aSplunky * update a value, will fail if data element is not of the correct type
43dfbf818aSplunky */
44dfbf818aSplunky
45dfbf818aSplunky bool
sdp_set_bool(const sdp_data_t * data,bool value)46dfbf818aSplunky sdp_set_bool(const sdp_data_t *data, bool value)
47dfbf818aSplunky {
48dfbf818aSplunky uint8_t *p = data->next;
49dfbf818aSplunky
50dfbf818aSplunky if (p + 2 > data->end
51*c5f754b9Splunky || *p++ != SDP_DATA_BOOL)
52dfbf818aSplunky return false;
53dfbf818aSplunky
54dfbf818aSplunky *p = (value ? 0x01 : 0x00);
55dfbf818aSplunky return true;
56dfbf818aSplunky }
57dfbf818aSplunky
58dfbf818aSplunky bool
sdp_set_uint(const sdp_data_t * data,uintmax_t value)59dfbf818aSplunky sdp_set_uint(const sdp_data_t *data, uintmax_t value)
60dfbf818aSplunky {
61dfbf818aSplunky uint8_t *p = data->next;
62dfbf818aSplunky
63dfbf818aSplunky if (p + 1 > data->end)
64dfbf818aSplunky return false;
65dfbf818aSplunky
66dfbf818aSplunky switch (*p++) {
67dfbf818aSplunky case SDP_DATA_UINT8:
68dfbf818aSplunky if (p + 1 > data->end
69dfbf818aSplunky || value > UINT8_MAX)
70dfbf818aSplunky return false;
71dfbf818aSplunky
72dfbf818aSplunky *p = (uint8_t)value;
73dfbf818aSplunky break;
74dfbf818aSplunky
75dfbf818aSplunky case SDP_DATA_UINT16:
76dfbf818aSplunky if (p + 2 > data->end
77dfbf818aSplunky || value > UINT16_MAX)
78dfbf818aSplunky return false;
79dfbf818aSplunky
80dfbf818aSplunky be16enc(p, (uint16_t)value);
81dfbf818aSplunky break;
82dfbf818aSplunky
83dfbf818aSplunky case SDP_DATA_UINT32:
84dfbf818aSplunky if (p + 4 > data->end
85dfbf818aSplunky || value > UINT32_MAX)
86dfbf818aSplunky return false;
87dfbf818aSplunky
88dfbf818aSplunky be32enc(p, (uint32_t)value);
89dfbf818aSplunky break;
90dfbf818aSplunky
91dfbf818aSplunky case SDP_DATA_UINT64:
92dfbf818aSplunky if (p + 8 > data->end
93dfbf818aSplunky || value > UINT64_MAX)
94dfbf818aSplunky return false;
95dfbf818aSplunky
96dfbf818aSplunky be64enc(p, (uint64_t)value);
97dfbf818aSplunky break;
98dfbf818aSplunky
99dfbf818aSplunky case SDP_DATA_UINT128:
100dfbf818aSplunky if (p + 16 > data->end)
101dfbf818aSplunky return false;
102dfbf818aSplunky
103dfbf818aSplunky be64enc(p + 0, (uint64_t)0);
104dfbf818aSplunky be64enc(p + 8, (uint64_t)value);
105dfbf818aSplunky break;
106dfbf818aSplunky
107dfbf818aSplunky default:
108dfbf818aSplunky return false;
109dfbf818aSplunky }
110dfbf818aSplunky
111dfbf818aSplunky return true;
112dfbf818aSplunky }
113dfbf818aSplunky
114dfbf818aSplunky bool
sdp_set_int(const sdp_data_t * data,intmax_t value)115dfbf818aSplunky sdp_set_int(const sdp_data_t *data, intmax_t value)
116dfbf818aSplunky {
117dfbf818aSplunky uint8_t *p = data->next;
118dfbf818aSplunky
119dfbf818aSplunky if (p + 1 > data->end)
120dfbf818aSplunky return false;
121dfbf818aSplunky
122dfbf818aSplunky switch (*p++) {
123dfbf818aSplunky case SDP_DATA_INT8:
124dfbf818aSplunky if (p + 1 > data->end
125dfbf818aSplunky || value > INT8_MAX
126dfbf818aSplunky || value < INT8_MIN)
127dfbf818aSplunky return false;
128dfbf818aSplunky
129dfbf818aSplunky *p = (uint8_t)value;
130dfbf818aSplunky break;
131dfbf818aSplunky
132dfbf818aSplunky case SDP_DATA_INT16:
133dfbf818aSplunky if (p + 2 > data->end
134dfbf818aSplunky || value > INT16_MAX
135dfbf818aSplunky || value < INT16_MIN)
136dfbf818aSplunky return false;
137dfbf818aSplunky
138dfbf818aSplunky be16enc(p, (uint16_t)value);
139dfbf818aSplunky break;
140dfbf818aSplunky
141dfbf818aSplunky case SDP_DATA_INT32:
142dfbf818aSplunky if (p + 4 > data->end
143dfbf818aSplunky || value > INT32_MAX
144dfbf818aSplunky || value < INT32_MIN)
145dfbf818aSplunky return false;
146dfbf818aSplunky
147dfbf818aSplunky be32enc(p, (uint32_t)value);
148dfbf818aSplunky break;
149dfbf818aSplunky
150dfbf818aSplunky case SDP_DATA_INT64:
151dfbf818aSplunky if (p + 8 > data->end
152dfbf818aSplunky || value > INT64_MAX
153dfbf818aSplunky || value < INT64_MIN)
154dfbf818aSplunky return false;
155dfbf818aSplunky
156dfbf818aSplunky be64enc(p, (uint64_t)value);
157dfbf818aSplunky break;
158dfbf818aSplunky
159dfbf818aSplunky case SDP_DATA_INT128:
160dfbf818aSplunky if (p + 16 > data->end)
161dfbf818aSplunky return false;
162dfbf818aSplunky
163dfbf818aSplunky be64enc(p + 0, (uint64_t)0);
164dfbf818aSplunky be64enc(p + 8, (uint64_t)value);
165dfbf818aSplunky break;
166dfbf818aSplunky
167dfbf818aSplunky default:
168dfbf818aSplunky return false;
169dfbf818aSplunky }
170dfbf818aSplunky
171dfbf818aSplunky return true;
172dfbf818aSplunky }
173dfbf818aSplunky
174dfbf818aSplunky static bool
_sdp_set_ext(uint8_t type,const sdp_data_t * data,ssize_t len)175dfbf818aSplunky _sdp_set_ext(uint8_t type, const sdp_data_t *data, ssize_t len)
176dfbf818aSplunky {
177dfbf818aSplunky uint8_t *p = data->next;
178dfbf818aSplunky
179dfbf818aSplunky if (p + 1 > data->end
180dfbf818aSplunky || SDP_DATA_TYPE(*p) != type)
181dfbf818aSplunky return false;
182dfbf818aSplunky
183dfbf818aSplunky switch (SDP_DATA_SIZE(*p++)) {
184dfbf818aSplunky case SDP_DATA_EXT8:
185dfbf818aSplunky if (len == -1) {
186dfbf818aSplunky if (p + 1 > data->end)
187dfbf818aSplunky return false;
188dfbf818aSplunky
189dfbf818aSplunky len = data->end - p - 1;
19063f86cb4Splunky } else if (len > data->end - 1 - p)
191dfbf818aSplunky return false;
192dfbf818aSplunky
193dfbf818aSplunky if (len > UINT8_MAX)
194dfbf818aSplunky return false;
195dfbf818aSplunky
196dfbf818aSplunky *p = (uint8_t)len;
197dfbf818aSplunky break;
198dfbf818aSplunky
199dfbf818aSplunky case SDP_DATA_EXT16:
200dfbf818aSplunky if (len == -1) {
201dfbf818aSplunky if (p + 2 > data->end)
202dfbf818aSplunky return false;
203dfbf818aSplunky
204dfbf818aSplunky len = data->end - p - 2;
20563f86cb4Splunky } else if (len > data->end - 2 - p)
206dfbf818aSplunky return false;
207dfbf818aSplunky
208dfbf818aSplunky if (len > UINT16_MAX)
209dfbf818aSplunky return false;
210dfbf818aSplunky
211dfbf818aSplunky be16enc(p, (uint16_t)len);
212dfbf818aSplunky break;
213dfbf818aSplunky
214dfbf818aSplunky case SDP_DATA_EXT32:
215dfbf818aSplunky if (len == -1) {
216dfbf818aSplunky if (p + 4 > data->end)
217dfbf818aSplunky return false;
218dfbf818aSplunky
219dfbf818aSplunky len = data->end - p - 4;
22063f86cb4Splunky } else if (len > data->end - 4 - p)
221dfbf818aSplunky return false;
222dfbf818aSplunky
2238ebedbbbSplunky if ((size_t)len > UINT32_MAX)
224dfbf818aSplunky return false;
225dfbf818aSplunky
226dfbf818aSplunky be32enc(p, (uint32_t)len);
227dfbf818aSplunky break;
228dfbf818aSplunky
229dfbf818aSplunky default:
230dfbf818aSplunky return false;
231dfbf818aSplunky }
232dfbf818aSplunky
233dfbf818aSplunky return true;
234dfbf818aSplunky }
235dfbf818aSplunky
236dfbf818aSplunky bool
sdp_set_seq(const sdp_data_t * data,ssize_t len)237dfbf818aSplunky sdp_set_seq(const sdp_data_t *data, ssize_t len)
238dfbf818aSplunky {
239dfbf818aSplunky
240dfbf818aSplunky return _sdp_set_ext(SDP_DATA_SEQ, data, len);
241dfbf818aSplunky }
242dfbf818aSplunky
243dfbf818aSplunky bool
sdp_set_alt(const sdp_data_t * data,ssize_t len)244dfbf818aSplunky sdp_set_alt(const sdp_data_t *data, ssize_t len)
245dfbf818aSplunky {
246dfbf818aSplunky
247dfbf818aSplunky return _sdp_set_ext(SDP_DATA_ALT, data, len);
248dfbf818aSplunky }
249