Mercurial > hg > beaglert
comparison include/oscpkt.hh @ 284:7bfb25a2e158 Doxy prerelease
Merge
author | Robert Jack <robert.h.jack@gmail.com> |
---|---|
date | Tue, 17 May 2016 15:53:24 +0100 |
parents | de37582ce6f3 |
children |
comparison
equal
deleted
inserted
replaced
269:ac8eb07afcf5 | 284:7bfb25a2e158 |
---|---|
1 /** @mainpage OSCPKT : a minimalistic OSC ( http://opensoundcontrol.org ) c++ library | |
2 | |
3 Before using this file please take the time to read the OSC spec, it | |
4 is short and not complicated: http://opensoundcontrol.org/spec-1_0 | |
5 | |
6 Features: | |
7 - handles basic OSC types: TFihfdsb | |
8 - handles bundles | |
9 - handles OSC pattern-matching rules (wildcards etc in message paths) | |
10 - portable on win / macos / linux | |
11 - robust wrt malformed packets | |
12 - optional udp transport for packets | |
13 - concise, all in a single .h file | |
14 - does not throw exceptions | |
15 | |
16 does not: | |
17 - take into account timestamp values. | |
18 - provide a cpu-scalable message dispatching. | |
19 - not suitable for use inside a realtime thread as it allocates memory when | |
20 building or reading messages. | |
21 | |
22 | |
23 There are basically 3 classes of interest: | |
24 - oscpkt::Message : read/write the content of an OSC message | |
25 - oscpkt::PacketReader : read the bundles/messages embedded in an OSC packet | |
26 - oscpkt::PacketWriter : write bundles/messages into an OSC packet | |
27 | |
28 And optionaly: | |
29 - oscpkt::UdpSocket : read/write OSC packets over UDP. | |
30 | |
31 @example: oscpkt_demo.cc | |
32 @example: oscpkt_test.cc | |
33 */ | |
34 | |
35 /* Copyright (C) 2010 Julien Pommier | |
36 | |
37 This software is provided 'as-is', without any express or implied | |
38 warranty. In no event will the authors be held liable for any damages | |
39 arising from the use of this software. | |
40 | |
41 Permission is granted to anyone to use this software for any purpose, | |
42 including commercial applications, and to alter it and redistribute it | |
43 freely, subject to the following restrictions: | |
44 | |
45 1. The origin of this software must not be misrepresented; you must not | |
46 claim that you wrote the original software. If you use this software | |
47 in a product, an acknowledgment in the product documentation would be | |
48 appreciated but is not required. | |
49 2. Altered source versions must be plainly marked as such, and must not be | |
50 misrepresented as being the original software. | |
51 3. This notice may not be removed or altered from any source distribution. | |
52 | |
53 (this is the zlib license) | |
54 */ | |
55 | |
56 #ifndef OSCPKT_HH | |
57 #define OSCPKT_HH | |
58 | |
59 #ifndef _MSC_VER | |
60 #include <stdint.h> | |
61 #else | |
62 namespace oscpkt { | |
63 typedef __int32 int32_t; | |
64 typedef unsigned __int32 uint32_t; | |
65 typedef __int64 int64_t; | |
66 typedef unsigned __int64 uint64_t; | |
67 } | |
68 #endif | |
69 #include <cstring> | |
70 #include <cassert> | |
71 #include <string> | |
72 #include <vector> | |
73 #include <list> | |
74 | |
75 #if defined(OSCPKT_OSTREAM_OUTPUT) || defined(OSCPKT_TEST) | |
76 #include <iostream> | |
77 #endif | |
78 | |
79 namespace oscpkt { | |
80 | |
81 /** | |
82 OSC timetag stuff, the highest 32-bit are seconds, the lowest are fraction of a second. | |
83 */ | |
84 class TimeTag { | |
85 uint64_t v; | |
86 public: | |
87 TimeTag() : v(1) {} | |
88 explicit TimeTag(uint64_t w): v(w) {} | |
89 operator uint64_t() const { return v; } | |
90 static TimeTag immediate() { return TimeTag(1); } | |
91 }; | |
92 | |
93 /* the various types that we handle (OSC 1.0 specifies that INT32/FLOAT/STRING/BLOB are the bare minimum) */ | |
94 enum { | |
95 TYPE_TAG_TRUE = 'T', | |
96 TYPE_TAG_FALSE = 'F', | |
97 TYPE_TAG_INT32 = 'i', | |
98 TYPE_TAG_INT64 = 'h', | |
99 TYPE_TAG_FLOAT = 'f', | |
100 TYPE_TAG_DOUBLE = 'd', | |
101 TYPE_TAG_STRING = 's', | |
102 TYPE_TAG_BLOB = 'b' | |
103 }; | |
104 | |
105 /* a few utility functions follow.. */ | |
106 | |
107 // round to the next multiple of 4, works for size_t and pointer arguments | |
108 template <typename Type> Type ceil4(Type p) { return (Type)((size_t(p) + 3)&(~size_t(3))); } | |
109 | |
110 // check that a memory area is zero padded until the next address which is a multiple of 4 | |
111 inline bool isZeroPaddingCorrect(const char *p) { | |
112 const char *q = ceil4(p); | |
113 for (;p < q; ++p) | |
114 if (*p != 0) { return false; } | |
115 return true; | |
116 } | |
117 | |
118 // stuff for reading / writing POD ("Plain Old Data") variables to unaligned bytes. | |
119 template <typename POD> union PodBytes { | |
120 char bytes[sizeof(POD)]; | |
121 POD value; | |
122 }; | |
123 | |
124 inline bool isBigEndian() { // a compile-time constant would certainly improve performances.. | |
125 PodBytes<int32_t> p; p.value = 0x12345678; | |
126 return p.bytes[0] == 0x12; | |
127 } | |
128 | |
129 /** read unaligned bytes into a POD type, assuming the bytes are a little endian representation */ | |
130 template <typename POD> POD bytes2pod(const char *bytes) { | |
131 PodBytes<POD> p; | |
132 for (size_t i=0; i < sizeof(POD); ++i) { | |
133 if (isBigEndian()) | |
134 p.bytes[i] = bytes[i]; | |
135 else | |
136 p.bytes[i] = bytes[sizeof(POD) - i - 1]; | |
137 } | |
138 return p.value; | |
139 } | |
140 | |
141 /** stored a POD type into an unaligned bytes array, using little endian representation */ | |
142 template <typename POD> void pod2bytes(const POD value, char *bytes) { | |
143 PodBytes<POD> p; p.value = value; | |
144 for (size_t i=0; i < sizeof(POD); ++i) { | |
145 if (isBigEndian()) | |
146 bytes[i] = p.bytes[i]; | |
147 else | |
148 bytes[i] = p.bytes[sizeof(POD) - i - 1]; | |
149 } | |
150 } | |
151 | |
152 /** internal stuff, handles the dynamic storage with correct alignments to 4 bytes */ | |
153 struct Storage { | |
154 std::vector<char> data; | |
155 Storage() { data.reserve(200); } | |
156 char *getBytes(size_t sz) { | |
157 assert((data.size() & 3) == 0); | |
158 if (data.size() + sz > data.capacity()) { data.reserve((data.size() + sz)*2); } | |
159 size_t sz4 = ceil4(sz); | |
160 size_t pos = data.size(); | |
161 data.resize(pos + sz4); // resize will fill with zeros, so the zero padding is OK | |
162 return &(data[pos]); | |
163 } | |
164 char *begin() { return data.size() ? &data.front() : 0; } | |
165 char *end() { return begin() + size(); } | |
166 const char *begin() const { return data.size() ? &data.front() : 0; } | |
167 const char *end() const { return begin() + size(); } | |
168 size_t size() const { return data.size(); } | |
169 void assign(const char *beg, const char *end) { data.assign(beg, end); } | |
170 void clear() { data.resize(0); } | |
171 }; | |
172 | |
173 /** check if the path matches the supplied path pattern , according to the OSC spec pattern | |
174 rules ('*' and '//' wildcards, '{}' alternatives, brackets etc) */ | |
175 bool fullPatternMatch(const std::string &pattern, const std::string &path); | |
176 /** check if the path matches the beginning of pattern */ | |
177 bool partialPatternMatch(const std::string &pattern, const std::string &path); | |
178 | |
179 #if defined(OSCPKT_DEBUG) | |
180 #define OSCPKT_SET_ERR(errcode) do { if (!err) { err = errcode; std::cerr << "set " #errcode << " at line " << __LINE__ << "\n"; } } while (0) | |
181 #else | |
182 #define OSCPKT_SET_ERR(errcode) do { if (!err) err = errcode; } while (0) | |
183 #endif | |
184 | |
185 typedef enum { OK_NO_ERROR=0, | |
186 // errors raised by the Message class: | |
187 MALFORMED_ADDRESS_PATTERN, MALFORMED_TYPE_TAGS, MALFORMED_ARGUMENTS, UNHANDLED_TYPE_TAGS, | |
188 // errors raised by ArgReader | |
189 TYPE_MISMATCH, NOT_ENOUGH_ARG, PATTERN_MISMATCH, | |
190 // errors raised by PacketReader/PacketWriter | |
191 INVALID_BUNDLE, INVALID_PACKET_SIZE, BUNDLE_REQUIRED_FOR_MULTI_MESSAGES } ErrorCode; | |
192 | |
193 /** | |
194 struct used to hold an OSC message that will be written or read. | |
195 | |
196 The list of arguments is exposed as a sort of queue. You "pop" | |
197 arguments from the front of the queue when reading, you push | |
198 arguments at the back of the queue when writing. | |
199 | |
200 Many functions return *this, so they can be chained: init("/foo").pushInt32(2).pushStr("kllk")... | |
201 | |
202 Example of use: | |
203 | |
204 creation of a message: | |
205 @code | |
206 msg.init("/foo").pushInt32(4).pushStr("bar"); | |
207 @endcode | |
208 reading a message, with error detection: | |
209 @code | |
210 if (msg.match("/foo/b*ar/plop")) { | |
211 int i; std::string s; std::vector<char> b; | |
212 if (msg.arg().popInt32(i).popStr(s).popBlob(b).isOkNoMoreArgs()) { | |
213 process message...; | |
214 } else arguments mismatch; | |
215 } | |
216 @endcode | |
217 */ | |
218 class Message { | |
219 TimeTag time_tag; | |
220 std::string address; | |
221 std::string type_tags; | |
222 std::vector<std::pair<size_t, size_t> > arguments; // array of pairs (pos,size), pos being an index into the 'storage' array. | |
223 Storage storage; // the arguments data is stored here | |
224 ErrorCode err; | |
225 public: | |
226 /** ArgReader is used for popping arguments from a Message, holds a | |
227 pointer to the original Message, and maintains a local error code */ | |
228 class ArgReader { | |
229 const Message *msg; | |
230 ErrorCode err; | |
231 size_t arg_idx; // arg index of the next arg that will be popped out. | |
232 public: | |
233 ArgReader(const Message &m, ErrorCode e = OK_NO_ERROR) : msg(&m), err(msg->getErr()), arg_idx(0) { | |
234 if (e != OK_NO_ERROR && err == OK_NO_ERROR) err=e; | |
235 } | |
236 ArgReader(const ArgReader &other) : msg(other.msg), err(other.err), arg_idx(other.arg_idx) {} | |
237 bool isBool() { return currentTypeTag() == TYPE_TAG_TRUE || currentTypeTag() == TYPE_TAG_FALSE; } | |
238 bool isInt32() { return currentTypeTag() == TYPE_TAG_INT32; } | |
239 bool isInt64() { return currentTypeTag() == TYPE_TAG_INT64; } | |
240 bool isFloat() { return currentTypeTag() == TYPE_TAG_FLOAT; } | |
241 bool isDouble() { return currentTypeTag() == TYPE_TAG_DOUBLE; } | |
242 bool isStr() { return currentTypeTag() == TYPE_TAG_STRING; } | |
243 bool isBlob() { return currentTypeTag() == TYPE_TAG_BLOB; } | |
244 | |
245 size_t nbArgRemaining() const { return msg->arguments.size() - arg_idx; } | |
246 bool isOk() const { return err == OK_NO_ERROR; } | |
247 operator bool() const { return isOk(); } // implicit bool conversion is handy here | |
248 /** call this at the end of the popXXX() chain to make sure everything is ok and | |
249 all arguments have been popped */ | |
250 bool isOkNoMoreArgs() const { return err == OK_NO_ERROR && nbArgRemaining() == 0; } | |
251 ErrorCode getErr() const { return err; } | |
252 | |
253 /** retrieve an int32 argument */ | |
254 ArgReader &popInt32(int32_t &i) { return popPod<int32_t>(TYPE_TAG_INT32, i); } | |
255 /** retrieve an int64 argument */ | |
256 ArgReader &popInt64(int64_t &i) { return popPod<int64_t>(TYPE_TAG_INT64, i); } | |
257 /** retrieve a single precision floating point argument */ | |
258 ArgReader &popFloat(float &f) { return popPod<float>(TYPE_TAG_FLOAT, f); } | |
259 /** retrieve a double precision floating point argument */ | |
260 ArgReader &popDouble(double &d) { return popPod<double>(TYPE_TAG_DOUBLE, d); } | |
261 /** retrieve a string argument (no check performed on its content, so it may contain any byte value except 0) */ | |
262 ArgReader &popStr(std::string &s) { | |
263 if (precheck(TYPE_TAG_STRING)) { | |
264 s = argBeg(arg_idx++); | |
265 } | |
266 return *this; | |
267 } | |
268 /** retrieve a binary blob */ | |
269 ArgReader &popBlob(std::vector<char> &b) { | |
270 if (precheck(TYPE_TAG_BLOB)) { | |
271 b.assign(argBeg(arg_idx)+4, argEnd(arg_idx)); | |
272 ++arg_idx; | |
273 } | |
274 return *this; | |
275 } | |
276 /** retrieve a boolean argument */ | |
277 ArgReader &popBool(bool &b) { | |
278 b = false; | |
279 if (arg_idx >= msg->arguments.size()) OSCPKT_SET_ERR(NOT_ENOUGH_ARG); | |
280 else if (currentTypeTag() == TYPE_TAG_TRUE) b = true; | |
281 else if (currentTypeTag() == TYPE_TAG_FALSE) b = false; | |
282 else OSCPKT_SET_ERR(TYPE_MISMATCH); | |
283 ++arg_idx; | |
284 return *this; | |
285 } | |
286 /** skip whatever comes next */ | |
287 ArgReader &pop() { | |
288 if (arg_idx >= msg->arguments.size()) OSCPKT_SET_ERR(NOT_ENOUGH_ARG); | |
289 else ++arg_idx; | |
290 return *this; | |
291 } | |
292 private: | |
293 const char *argBeg(size_t idx) { | |
294 if (err || idx >= msg->arguments.size()) return 0; | |
295 else return msg->storage.begin() + msg->arguments[idx].first; | |
296 } | |
297 const char *argEnd(size_t idx) { | |
298 if (err || idx >= msg->arguments.size()) return 0; | |
299 else return msg->storage.begin() + msg->arguments[idx].first + msg->arguments[idx].second; | |
300 } | |
301 int currentTypeTag() { | |
302 if (!err && arg_idx < msg->type_tags.size()) return msg->type_tags[arg_idx]; | |
303 else OSCPKT_SET_ERR(NOT_ENOUGH_ARG); | |
304 return -1; | |
305 } | |
306 template <typename POD> ArgReader &popPod(int tag, POD &v) { | |
307 if (precheck(tag)) { | |
308 v = bytes2pod<POD>(argBeg(arg_idx)); | |
309 ++arg_idx; | |
310 } else v = POD(0); | |
311 return *this; | |
312 } | |
313 /* pre-check stuff before popping an argument from the message */ | |
314 bool precheck(int tag) { | |
315 if (arg_idx >= msg->arguments.size()) OSCPKT_SET_ERR(NOT_ENOUGH_ARG); | |
316 else if (!err && currentTypeTag() != tag) OSCPKT_SET_ERR(TYPE_MISMATCH); | |
317 return err == OK_NO_ERROR; | |
318 } | |
319 }; | |
320 | |
321 Message() { clear(); } | |
322 Message(const std::string &s, TimeTag tt = TimeTag::immediate()) : time_tag(tt), address(s), err(OK_NO_ERROR) {} | |
323 Message(const void *ptr, size_t sz, TimeTag tt = TimeTag::immediate()) { buildFromRawData(ptr, sz); time_tag = tt; } | |
324 | |
325 bool isOk() const { return err == OK_NO_ERROR; } | |
326 ErrorCode getErr() const { return err; } | |
327 | |
328 /** return the type_tags string, with its initial ',' stripped. */ | |
329 const std::string &typeTags() const { return type_tags; } | |
330 /** retrieve the address pattern. If you want to follow to the whole OSC spec, you | |
331 have to handle its matching rules for address specifications -- this file does | |
332 not provide this functionality */ | |
333 const std::string &addressPattern() const { return address; } | |
334 TimeTag timeTag() const { return time_tag; } | |
335 /** clear the message and start a new message with the supplied address and time_tag. */ | |
336 Message &init(const std::string &addr, TimeTag tt = TimeTag::immediate()) { | |
337 clear(); | |
338 address = addr; time_tag = tt; | |
339 if (address.empty() || address[0] != '/') OSCPKT_SET_ERR(MALFORMED_ADDRESS_PATTERN); | |
340 return *this; | |
341 } | |
342 | |
343 /** start a matching test. The typical use-case is to follow this by | |
344 a sequence of calls to popXXX() and a final call to | |
345 isOkNoMoreArgs() which will allow to check that everything went | |
346 fine. For example: | |
347 @code | |
348 if (msg.match("/foo").popInt32(i).isOkNoMoreArgs()) { blah(i); } | |
349 else if (msg.match("/bar").popStr(s).popInt32(i).isOkNoMoreArgs()) { plop(s,i); } | |
350 else cerr << "unhandled message: " << msg << "\n"; | |
351 @endcode | |
352 */ | |
353 ArgReader match(const std::string &test) const { | |
354 return ArgReader(*this, fullPatternMatch(address.c_str(), test.c_str()) ? OK_NO_ERROR : PATTERN_MISMATCH); | |
355 } | |
356 /** return true if the 'test' path matched by the first characters of addressPattern(). | |
357 For ex. ("/foo/bar").partialMatch("/foo/") is true */ | |
358 ArgReader partialMatch(const std::string &test) const { | |
359 return ArgReader(*this, partialPatternMatch(address.c_str(), test.c_str()) ? OK_NO_ERROR : PATTERN_MISMATCH); | |
360 } | |
361 ArgReader arg() const { return ArgReader(*this, OK_NO_ERROR); } | |
362 | |
363 /** build the osc message for raw data (the message will keep a copy of that data) */ | |
364 void buildFromRawData(const void *ptr, size_t sz) { | |
365 clear(); | |
366 storage.assign((const char*)ptr, (const char*)ptr + sz); | |
367 const char *address_beg = storage.begin(); | |
368 const char *address_end = (const char*)memchr(address_beg, 0, storage.end()-address_beg); | |
369 if (!address_end || !isZeroPaddingCorrect(address_end+1) || address_beg[0] != '/') { | |
370 OSCPKT_SET_ERR(MALFORMED_ADDRESS_PATTERN); return; | |
371 } else address.assign(address_beg, address_end); | |
372 | |
373 const char *type_tags_beg = ceil4(address_end+1); | |
374 const char *type_tags_end = (const char*)memchr(type_tags_beg, 0, storage.end()-type_tags_beg); | |
375 if (!type_tags_end || !isZeroPaddingCorrect(type_tags_end+1) || type_tags_beg[0] != ',') { | |
376 OSCPKT_SET_ERR(MALFORMED_TYPE_TAGS); return; | |
377 } else type_tags.assign(type_tags_beg+1, type_tags_end); // we do not copy the initial ',' | |
378 | |
379 const char *arg = ceil4(type_tags_end+1); assert(arg <= storage.end()); | |
380 size_t iarg = 0; | |
381 while (isOk() && iarg < type_tags.size()) { | |
382 assert(arg <= storage.end()); | |
383 size_t len = getArgSize(type_tags[iarg], arg); | |
384 if (isOk()) arguments.push_back(std::make_pair(arg - storage.begin(), len)); | |
385 arg += ceil4(len); ++iarg; | |
386 } | |
387 if (iarg < type_tags.size() || arg != storage.end()) { | |
388 OSCPKT_SET_ERR(MALFORMED_ARGUMENTS); | |
389 } | |
390 } | |
391 | |
392 /* below are all the functions that serve when *writing* a message */ | |
393 Message &pushBool(bool b) { | |
394 type_tags += (b ? TYPE_TAG_TRUE : TYPE_TAG_FALSE); | |
395 arguments.push_back(std::make_pair(storage.size(), storage.size())); | |
396 return *this; | |
397 } | |
398 Message &pushInt32(int32_t i) { return pushPod(TYPE_TAG_INT32, i); } | |
399 Message &pushInt64(int64_t h) { return pushPod(TYPE_TAG_INT64, h); } | |
400 Message &pushFloat(float f) { return pushPod(TYPE_TAG_FLOAT, f); } | |
401 Message &pushDouble(double d) { return pushPod(TYPE_TAG_DOUBLE, d); } | |
402 Message &pushStr(const std::string &s) { | |
403 assert(s.size() < 2147483647); // insane values are not welcome | |
404 type_tags += TYPE_TAG_STRING; | |
405 arguments.push_back(std::make_pair(storage.size(), s.size() + 1)); | |
406 strcpy(storage.getBytes(s.size()+1), s.c_str()); | |
407 return *this; | |
408 } | |
409 Message &pushBlob(void *ptr, size_t num_bytes) { | |
410 assert(num_bytes < 2147483647); // insane values are not welcome | |
411 type_tags += TYPE_TAG_BLOB; | |
412 arguments.push_back(std::make_pair(storage.size(), num_bytes+4)); | |
413 pod2bytes<int32_t>((int32_t)num_bytes, storage.getBytes(4)); | |
414 if (num_bytes) | |
415 memcpy(storage.getBytes(num_bytes), ptr, num_bytes); | |
416 return *this; | |
417 } | |
418 | |
419 /** reset the message to a clean state */ | |
420 void clear() { | |
421 address.clear(); type_tags.clear(); storage.clear(); arguments.clear(); | |
422 err = OK_NO_ERROR; time_tag = TimeTag::immediate(); | |
423 } | |
424 | |
425 /** write the raw message data (used by PacketWriter) */ | |
426 void packMessage(Storage &s, bool write_size) const { | |
427 if (!isOk()) return; | |
428 size_t l_addr = address.size()+1, l_type = type_tags.size()+2; | |
429 if (write_size) | |
430 pod2bytes<uint32_t>(uint32_t(ceil4(l_addr) + ceil4(l_type) + ceil4(storage.size())), s.getBytes(4)); | |
431 strcpy(s.getBytes(l_addr), address.c_str()); | |
432 strcpy(s.getBytes(l_type), ("," + type_tags).c_str()); | |
433 if (storage.size()) | |
434 memcpy(s.getBytes(storage.size()), const_cast<Storage&>(storage).begin(), storage.size()); | |
435 } | |
436 | |
437 private: | |
438 | |
439 /* get the number of bytes occupied by the argument */ | |
440 size_t getArgSize(int type, const char *p) { | |
441 if (err) return 0; | |
442 size_t sz = 0; | |
443 assert(p >= storage.begin() && p <= storage.end()); | |
444 switch (type) { | |
445 case TYPE_TAG_TRUE: | |
446 case TYPE_TAG_FALSE: sz = 0; break; | |
447 case TYPE_TAG_INT32: | |
448 case TYPE_TAG_FLOAT: sz = 4; break; | |
449 case TYPE_TAG_INT64: | |
450 case TYPE_TAG_DOUBLE: sz = 8; break; | |
451 case TYPE_TAG_STRING: { | |
452 const char *q = (const char*)memchr(p, 0, storage.end()-p); | |
453 if (!q) OSCPKT_SET_ERR(MALFORMED_ARGUMENTS); | |
454 else sz = (q-p)+1; | |
455 } break; | |
456 case TYPE_TAG_BLOB: { | |
457 if (p == storage.end()) { OSCPKT_SET_ERR(MALFORMED_ARGUMENTS); return 0; } | |
458 sz = 4+bytes2pod<uint32_t>(p); | |
459 } break; | |
460 default: { | |
461 OSCPKT_SET_ERR(UNHANDLED_TYPE_TAGS); return 0; | |
462 } break; | |
463 } | |
464 if (p+sz > storage.end() || /* string or blob too large.. */ | |
465 p+sz < p /* or even blob so large that it did overflow */) { | |
466 OSCPKT_SET_ERR(MALFORMED_ARGUMENTS); return 0; | |
467 } | |
468 if (!isZeroPaddingCorrect(p+sz)) { OSCPKT_SET_ERR(MALFORMED_ARGUMENTS); return 0; } | |
469 return sz; | |
470 } | |
471 | |
472 template <typename POD> Message &pushPod(int tag, POD v) { | |
473 type_tags += (char)tag; | |
474 arguments.push_back(std::make_pair(storage.size(), sizeof(POD))); | |
475 pod2bytes(v, storage.getBytes(sizeof(POD))); | |
476 return *this; | |
477 } | |
478 | |
479 #ifdef OSCPKT_OSTREAM_OUTPUT | |
480 friend std::ostream &operator<<(std::ostream &os, const Message &msg) { | |
481 os << "osc_address: '" << msg.address << "', types: '" << msg.type_tags << "', timetag=" << msg.time_tag << ", args=["; | |
482 Message::ArgReader arg(msg); | |
483 while (arg.nbArgRemaining() && arg.isOk()) { | |
484 if (arg.isBool()) { bool b; arg.popBool(b); os << (b?"True":"False"); } | |
485 else if (arg.isInt32()) { int32_t i; arg.popInt32(i); os << i; } | |
486 else if (arg.isInt64()) { int64_t h; arg.popInt64(h); os << h << "ll"; } | |
487 else if (arg.isFloat()) { float f; arg.popFloat(f); os << f << "f"; } | |
488 else if (arg.isDouble()) { double d; arg.popDouble(d); os << d; } | |
489 else if (arg.isStr()) { std::string s; arg.popStr(s); os << "'" << s << "'"; } | |
490 else if (arg.isBlob()) { std::vector<char> b; arg.popBlob(b); os << "Blob " << b.size() << " bytes"; } | |
491 else { | |
492 assert(0); // I forgot a case.. | |
493 } | |
494 if (arg.nbArgRemaining()) os << ", "; | |
495 } | |
496 if (!arg.isOk()) { os << " ERROR#" << arg.getErr(); } | |
497 os << "]"; | |
498 return os; | |
499 } | |
500 #endif | |
501 }; | |
502 | |
503 /** | |
504 parse an OSC packet and extracts the embedded OSC messages. | |
505 */ | |
506 class PacketReader { | |
507 public: | |
508 PacketReader() { err = OK_NO_ERROR; } | |
509 /** pointer and size of the osc packet to be parsed. */ | |
510 PacketReader(const void *ptr, size_t sz) { init(ptr, sz); } | |
511 | |
512 void init(const void *ptr, size_t sz) { | |
513 err = OK_NO_ERROR; messages.clear(); | |
514 if ((sz%4) == 0) { | |
515 parse((const char*)ptr, (const char *)ptr+sz, TimeTag::immediate()); | |
516 } else OSCPKT_SET_ERR(INVALID_PACKET_SIZE); | |
517 it_messages = messages.begin(); | |
518 } | |
519 | |
520 /** extract the next osc message from the packet. return 0 when all messages have been read, or in case of error. */ | |
521 Message *popMessage() { | |
522 if (!err && !messages.empty() && it_messages != messages.end()) return &*it_messages++; | |
523 else return 0; | |
524 } | |
525 bool isOk() const { return err == OK_NO_ERROR; } | |
526 ErrorCode getErr() const { return err; } | |
527 | |
528 private: | |
529 std::list<Message> messages; | |
530 std::list<Message>::iterator it_messages; | |
531 ErrorCode err; | |
532 | |
533 void parse(const char *beg, const char *end, TimeTag time_tag) { | |
534 assert(beg <= end && !err); assert(((end-beg)%4)==0); | |
535 | |
536 if (beg == end) return; | |
537 if (*beg == '#') { | |
538 /* it's a bundle */ | |
539 if (end - beg >= 20 | |
540 && memcmp(beg, "#bundle\0", 8) == 0) { | |
541 TimeTag time_tag2(bytes2pod<uint64_t>(beg+8)); | |
542 const char *pos = beg + 16; | |
543 do { | |
544 uint32_t sz = bytes2pod<uint32_t>(pos); pos += 4; | |
545 if ((sz&3) != 0 || pos + sz > end || pos+sz < pos) { | |
546 OSCPKT_SET_ERR(INVALID_BUNDLE); | |
547 } else { | |
548 parse(pos, pos+sz, time_tag2); | |
549 pos += sz; | |
550 } | |
551 } while (!err && pos != end); | |
552 } else { | |
553 OSCPKT_SET_ERR(INVALID_BUNDLE); | |
554 } | |
555 } else { | |
556 messages.push_back(Message(beg, end-beg, time_tag)); | |
557 if (!messages.back().isOk()) OSCPKT_SET_ERR(messages.back().getErr()); | |
558 } | |
559 } | |
560 }; | |
561 | |
562 | |
563 /** | |
564 Assemble messages into an OSC packet. Example of use: | |
565 @code | |
566 PacketWriter pkt; | |
567 Message msg; | |
568 pkt.startBundle(); | |
569 pkt.addMessage(msg.init("/foo").pushBool(true).pushStr("plop").pushFloat(3.14f)); | |
570 pkt.addMessage(msg.init("/bar").pushBool(false)); | |
571 pkt.endBundle(); | |
572 if (pkt.isOk()) { | |
573 send(pkt.data(), pkt.size()); | |
574 } | |
575 @endcode | |
576 */ | |
577 class PacketWriter { | |
578 public: | |
579 PacketWriter() { init(); } | |
580 PacketWriter &init() { err = OK_NO_ERROR; storage.clear(); bundles.clear(); return *this; } | |
581 | |
582 /** begin a new bundle. If you plan to pack more than one message in the Osc packet, you have to | |
583 put them in a bundle. Nested bundles inside bundles are also allowed. */ | |
584 PacketWriter &startBundle(TimeTag ts = TimeTag::immediate()) { | |
585 char *p; | |
586 if (bundles.size()) storage.getBytes(4); // hold the bundle size | |
587 p = storage.getBytes(8); strcpy(p, "#bundle"); bundles.push_back(p - storage.begin()); | |
588 p = storage.getBytes(8); pod2bytes<uint64_t>(ts, p); | |
589 return *this; | |
590 } | |
591 /** close the current bundle. */ | |
592 PacketWriter &endBundle() { | |
593 if (bundles.size()) { | |
594 if (storage.size() - bundles.back() == 16) { | |
595 pod2bytes<uint32_t>(0, storage.getBytes(4)); // the 'empty bundle' case, not very elegant | |
596 } | |
597 if (bundles.size()>1) { // no size stored for the top-level bundle | |
598 pod2bytes<uint32_t>(uint32_t(storage.size() - bundles.back()), storage.begin() + bundles.back()-4); | |
599 } | |
600 bundles.pop_back(); | |
601 } else OSCPKT_SET_ERR(INVALID_BUNDLE); | |
602 return *this; | |
603 } | |
604 | |
605 /** insert an Osc message into the current bundle / packet. | |
606 */ | |
607 PacketWriter &addMessage(const Message &msg) { | |
608 if (storage.size() != 0 && bundles.empty()) OSCPKT_SET_ERR(BUNDLE_REQUIRED_FOR_MULTI_MESSAGES); | |
609 else msg.packMessage(storage, bundles.size()>0); | |
610 if (!msg.isOk()) OSCPKT_SET_ERR(msg.getErr()); | |
611 return *this; | |
612 } | |
613 | |
614 /** the error flag will be raised if an opened bundle is not closed, or if more than one message is | |
615 inserted in the packet without a bundle */ | |
616 bool isOk() { return err == OK_NO_ERROR; } | |
617 ErrorCode getErr() { return err; } | |
618 | |
619 /** return the number of bytes of the osc packet -- will always be a | |
620 multiple of 4 -- returns 0 if the construction of the packet has | |
621 failed. */ | |
622 uint32_t packetSize() { return err ? 0 : (uint32_t)storage.size(); } | |
623 | |
624 /** return the bytes of the osc packet (NULL if the construction of the packet has failed) */ | |
625 char *packetData() { return err ? 0 : storage.begin(); } | |
626 private: | |
627 std::vector<size_t> bundles; // hold the position in the storage array of the beginning marker of each bundle | |
628 Storage storage; | |
629 ErrorCode err; | |
630 }; | |
631 | |
632 // see the OSC spec for the precise pattern matching rules | |
633 inline const char *internalPatternMatch(const char *pattern, const char *path) { | |
634 while (*pattern) { | |
635 const char *p = pattern; | |
636 if (*p == '?' && *path) { ++p; ++path; } | |
637 else if (*p == '[' && *path) { // bracketted range, e.g. [a-zABC] | |
638 ++p; | |
639 bool reverse = false; | |
640 if (*p == '!') { reverse = true; ++p; } | |
641 bool match = reverse; | |
642 for (; *p && *p != ']'; ++p) { | |
643 char c0 = *p, c1 = c0; | |
644 if (p[1] == '-' && p[2] && p[2] != ']') { p += 2; c1 = *p; } | |
645 if (*path >= c0 && *path <= c1) { match = !reverse; } | |
646 } | |
647 if (!match || *p != ']') return pattern; | |
648 ++p; ++path; | |
649 } else if (*p == '*') { // wildcard '*' | |
650 while (*p == '*') ++p; | |
651 const char *best = 0; | |
652 while (true) { | |
653 const char *ret = internalPatternMatch(p, path); | |
654 if (ret && ret > best) best = ret; | |
655 if (*path == 0 || *path == '/') break; | |
656 else ++path; | |
657 } | |
658 return best; | |
659 } else if (*p == '/' && *(p+1) == '/') { // the super-wildcard '//' | |
660 while (*(p+1)=='/') ++p; | |
661 const char *best = 0; | |
662 while (true) { | |
663 const char *ret = internalPatternMatch(p, path); | |
664 if (ret && ret > best) best = ret; | |
665 if (*path == 0) break; | |
666 if (*path == 0 || (path = strchr(path+1, '/')) == 0) break; | |
667 } | |
668 return best; | |
669 } else if (*p == '{') { // braced list {foo,bar,baz} | |
670 const char *end = strchr(p, '}'), *q; | |
671 if (!end) return 0; // syntax error in brace list.. | |
672 bool match = false; | |
673 do { | |
674 ++p; | |
675 q = strchr(p, ','); | |
676 if (q == 0 || q > end) q = end; | |
677 if (strncmp(p, path, q-p)==0) { | |
678 path += (q-p); p = end+1; match = true; | |
679 } else p=q; | |
680 } while (q != end && !match); | |
681 if (!match) return pattern; | |
682 } else if (*p == *path) { ++p; ++path; } // any other character | |
683 else break; | |
684 pattern = p; | |
685 } | |
686 return (*path == 0 ? pattern : 0); | |
687 } | |
688 | |
689 inline bool partialPatternMatch(const std::string &pattern, const std::string &test) { | |
690 const char *q = internalPatternMatch(pattern.c_str(), test.c_str()); | |
691 return q != 0; | |
692 } | |
693 | |
694 inline bool fullPatternMatch(const std::string &pattern, const std::string &test) { | |
695 const char *q = internalPatternMatch(pattern.c_str(), test.c_str()); | |
696 return q && *q == 0; | |
697 } | |
698 | |
699 } // namespace oscpkt | |
700 | |
701 #endif // OSCPKT_HH |