diff options
Diffstat (limited to 'linden/indra/llmessage/lldatapacker.cpp')
-rw-r--r-- | linden/indra/llmessage/lldatapacker.cpp | 1886 |
1 files changed, 1886 insertions, 0 deletions
diff --git a/linden/indra/llmessage/lldatapacker.cpp b/linden/indra/llmessage/lldatapacker.cpp new file mode 100644 index 0000000..4a15d8b --- /dev/null +++ b/linden/indra/llmessage/lldatapacker.cpp | |||
@@ -0,0 +1,1886 @@ | |||
1 | /** | ||
2 | * @file lldatapacker.cpp | ||
3 | * @brief Data packer implementation. | ||
4 | * | ||
5 | * Copyright (c) 2006-2007, Linden Research, Inc. | ||
6 | * | ||
7 | * The source code in this file ("Source Code") is provided by Linden Lab | ||
8 | * to you under the terms of the GNU General Public License, version 2.0 | ||
9 | * ("GPL"), unless you have obtained a separate licensing agreement | ||
10 | * ("Other License"), formally executed by you and Linden Lab. Terms of | ||
11 | * the GPL can be found in doc/GPL-license.txt in this distribution, or | ||
12 | * online at http://secondlife.com/developers/opensource/gplv2 | ||
13 | * | ||
14 | * There are special exceptions to the terms and conditions of the GPL as | ||
15 | * it is applied to this Source Code. View the full text of the exception | ||
16 | * in the file doc/FLOSS-exception.txt in this software distribution, or | ||
17 | * online at http://secondlife.com/developers/opensource/flossexception | ||
18 | * | ||
19 | * By copying, modifying or distributing this software, you acknowledge | ||
20 | * that you have read and understood your obligations described above, | ||
21 | * and agree to abide by those obligations. | ||
22 | * | ||
23 | * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO | ||
24 | * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, | ||
25 | * COMPLETENESS OR PERFORMANCE. | ||
26 | */ | ||
27 | |||
28 | #include <string.h> | ||
29 | |||
30 | #include "linden_common.h" | ||
31 | |||
32 | #include "lldatapacker.h" | ||
33 | #include "llerror.h" | ||
34 | |||
35 | #include "message.h" | ||
36 | |||
37 | #include "v4color.h" | ||
38 | #include "v4coloru.h" | ||
39 | #include "v2math.h" | ||
40 | #include "v3math.h" | ||
41 | #include "v4math.h" | ||
42 | #include "lluuid.h" | ||
43 | |||
44 | // *NOTE: there are functions below which use sscanf and rely on this | ||
45 | // particular value of DP_BUFSIZE. Search for '511' (DP_BUFSIZE - 1) | ||
46 | // to find them if you change this number. | ||
47 | const S32 DP_BUFSIZE = 512; | ||
48 | |||
49 | static char DUMMY_BUFFER[128]; /*Flawfinder: ignore*/ | ||
50 | |||
51 | LLDataPacker::LLDataPacker() : mPassFlags(0), mWriteEnabled(FALSE) | ||
52 | { | ||
53 | } | ||
54 | |||
55 | BOOL LLDataPacker::packFixed(const F32 value, const char *name, | ||
56 | const BOOL is_signed, const U32 int_bits, const U32 frac_bits) | ||
57 | { | ||
58 | BOOL success = TRUE; | ||
59 | S32 unsigned_bits = int_bits + frac_bits; | ||
60 | S32 total_bits = unsigned_bits; | ||
61 | |||
62 | if (is_signed) | ||
63 | { | ||
64 | total_bits++; | ||
65 | } | ||
66 | |||
67 | S32 min_val; | ||
68 | U32 max_val; | ||
69 | if (is_signed) | ||
70 | { | ||
71 | min_val = 1 << int_bits; | ||
72 | min_val *= -1; | ||
73 | } | ||
74 | else | ||
75 | { | ||
76 | min_val = 0; | ||
77 | } | ||
78 | max_val = 1 << int_bits; | ||
79 | |||
80 | // Clamp to be within range | ||
81 | F32 fixed_val = llclamp(value, (F32)min_val, (F32)max_val); | ||
82 | if (is_signed) | ||
83 | { | ||
84 | fixed_val += max_val; | ||
85 | } | ||
86 | fixed_val *= 1 << frac_bits; | ||
87 | |||
88 | if (total_bits <= 8) | ||
89 | { | ||
90 | packU8((U8)fixed_val, name); | ||
91 | } | ||
92 | else if (total_bits <= 16) | ||
93 | { | ||
94 | packU16((U16)fixed_val, name); | ||
95 | } | ||
96 | else if (total_bits <= 31) | ||
97 | { | ||
98 | packU32((U32)fixed_val, name); | ||
99 | } | ||
100 | else | ||
101 | { | ||
102 | llerrs << "Using fixed-point packing of " << total_bits << " bits, why?!" << llendl; | ||
103 | } | ||
104 | return success; | ||
105 | } | ||
106 | |||
107 | BOOL LLDataPacker::unpackFixed(F32 &value, const char *name, | ||
108 | const BOOL is_signed, const U32 int_bits, const U32 frac_bits) | ||
109 | { | ||
110 | //BOOL success = TRUE; | ||
111 | //llinfos << "unpackFixed:" << name << " int:" << int_bits << " frac:" << frac_bits << llendl; | ||
112 | BOOL ok = FALSE; | ||
113 | S32 unsigned_bits = int_bits + frac_bits; | ||
114 | S32 total_bits = unsigned_bits; | ||
115 | |||
116 | if (is_signed) | ||
117 | { | ||
118 | total_bits++; | ||
119 | } | ||
120 | |||
121 | S32 min_val; | ||
122 | U32 max_val; | ||
123 | if (is_signed) | ||
124 | { | ||
125 | min_val = 1 << int_bits; | ||
126 | min_val *= -1; | ||
127 | } | ||
128 | max_val = 1 << int_bits; | ||
129 | |||
130 | F32 fixed_val; | ||
131 | if (total_bits <= 8) | ||
132 | { | ||
133 | U8 fixed_8; | ||
134 | ok = unpackU8(fixed_8, name); | ||
135 | fixed_val = (F32)fixed_8; | ||
136 | } | ||
137 | else if (total_bits <= 16) | ||
138 | { | ||
139 | U16 fixed_16; | ||
140 | ok = unpackU16(fixed_16, name); | ||
141 | fixed_val = (F32)fixed_16; | ||
142 | } | ||
143 | else if (total_bits <= 31) | ||
144 | { | ||
145 | U32 fixed_32; | ||
146 | ok = unpackU32(fixed_32, name); | ||
147 | fixed_val = (F32)fixed_32; | ||
148 | } | ||
149 | else | ||
150 | { | ||
151 | fixed_val = 0; | ||
152 | llerrs << "Bad bit count: " << total_bits << llendl; | ||
153 | } | ||
154 | |||
155 | //llinfos << "Fixed_val:" << fixed_val << llendl; | ||
156 | |||
157 | fixed_val /= (F32)(1 << frac_bits); | ||
158 | if (is_signed) | ||
159 | { | ||
160 | fixed_val -= max_val; | ||
161 | } | ||
162 | value = fixed_val; | ||
163 | //llinfos << "Value: " << value << llendl; | ||
164 | return ok; | ||
165 | } | ||
166 | |||
167 | //--------------------------------------------------------------------------- | ||
168 | // LLDataPackerBinaryBuffer implementation | ||
169 | //--------------------------------------------------------------------------- | ||
170 | |||
171 | BOOL LLDataPackerBinaryBuffer::packString(const char *value, const char *name) | ||
172 | { | ||
173 | BOOL success = TRUE; | ||
174 | S32 length = (S32)strlen(value) + 1; /*Flawfinder: ignore*/ | ||
175 | |||
176 | success &= verifyLength(length, name); | ||
177 | |||
178 | if (mWriteEnabled) | ||
179 | { | ||
180 | htonmemcpy(mCurBufferp, value, MVT_VARIABLE, length); | ||
181 | } | ||
182 | mCurBufferp += length; | ||
183 | return success; | ||
184 | } | ||
185 | |||
186 | |||
187 | BOOL LLDataPackerBinaryBuffer::unpackString(std::string& value, const char *name) | ||
188 | { | ||
189 | BOOL success = TRUE; | ||
190 | S32 length = (S32)strlen((char *)mCurBufferp) + 1; /*Flawfinder: ignore*/ | ||
191 | |||
192 | success &= verifyLength(length, name); | ||
193 | |||
194 | value = std::string((char*)mCurBufferp); // We already assume NULL termination calling strlen() | ||
195 | |||
196 | mCurBufferp += length; | ||
197 | return success; | ||
198 | } | ||
199 | |||
200 | BOOL LLDataPackerBinaryBuffer::packBinaryData(const U8 *value, S32 size, const char *name) | ||
201 | { | ||
202 | BOOL success = TRUE; | ||
203 | success &= verifyLength(size + 4, name); | ||
204 | |||
205 | if (mWriteEnabled) | ||
206 | { | ||
207 | htonmemcpy(mCurBufferp, &size, MVT_S32, 4); | ||
208 | } | ||
209 | mCurBufferp += 4; | ||
210 | if (mWriteEnabled) | ||
211 | { | ||
212 | htonmemcpy(mCurBufferp, value, MVT_VARIABLE, size); | ||
213 | } | ||
214 | mCurBufferp += size; | ||
215 | return success; | ||
216 | } | ||
217 | |||
218 | |||
219 | BOOL LLDataPackerBinaryBuffer::unpackBinaryData(U8 *value, S32 &size, const char *name) | ||
220 | { | ||
221 | BOOL success = TRUE; | ||
222 | success &= verifyLength(4, name); | ||
223 | htonmemcpy(&size, mCurBufferp, MVT_S32, 4); | ||
224 | mCurBufferp += 4; | ||
225 | success &= verifyLength(size, name); | ||
226 | if (success) | ||
227 | { | ||
228 | htonmemcpy(value, mCurBufferp, MVT_VARIABLE, size); | ||
229 | mCurBufferp += size; | ||
230 | } | ||
231 | else | ||
232 | { | ||
233 | llwarns << "LLDataPackerBinaryBuffer::unpackBinaryData would unpack invalid data, aborting!" << llendl; | ||
234 | success = FALSE; | ||
235 | } | ||
236 | return success; | ||
237 | } | ||
238 | |||
239 | |||
240 | BOOL LLDataPackerBinaryBuffer::packBinaryDataFixed(const U8 *value, S32 size, const char *name) | ||
241 | { | ||
242 | BOOL success = TRUE; | ||
243 | success &= verifyLength(size, name); | ||
244 | |||
245 | if (mWriteEnabled) | ||
246 | { | ||
247 | htonmemcpy(mCurBufferp, value, MVT_VARIABLE, size); | ||
248 | } | ||
249 | mCurBufferp += size; | ||
250 | return success; | ||
251 | } | ||
252 | |||
253 | |||
254 | BOOL LLDataPackerBinaryBuffer::unpackBinaryDataFixed(U8 *value, S32 size, const char *name) | ||
255 | { | ||
256 | BOOL success = TRUE; | ||
257 | success &= verifyLength(size, name); | ||
258 | htonmemcpy(value, mCurBufferp, MVT_VARIABLE, size); | ||
259 | mCurBufferp += size; | ||
260 | return success; | ||
261 | } | ||
262 | |||
263 | |||
264 | BOOL LLDataPackerBinaryBuffer::packU8(const U8 value, const char *name) | ||
265 | { | ||
266 | BOOL success = TRUE; | ||
267 | success &= verifyLength(sizeof(U8), name); | ||
268 | |||
269 | if (mWriteEnabled) | ||
270 | { | ||
271 | *mCurBufferp = value; | ||
272 | } | ||
273 | mCurBufferp++; | ||
274 | return success; | ||
275 | } | ||
276 | |||
277 | |||
278 | BOOL LLDataPackerBinaryBuffer::unpackU8(U8 &value, const char *name) | ||
279 | { | ||
280 | BOOL success = TRUE; | ||
281 | success &= verifyLength(sizeof(U8), name); | ||
282 | |||
283 | value = *mCurBufferp; | ||
284 | mCurBufferp++; | ||
285 | return success; | ||
286 | } | ||
287 | |||
288 | |||
289 | BOOL LLDataPackerBinaryBuffer::packU16(const U16 value, const char *name) | ||
290 | { | ||
291 | BOOL success = TRUE; | ||
292 | success &= verifyLength(sizeof(U16), name); | ||
293 | |||
294 | if (mWriteEnabled) | ||
295 | { | ||
296 | htonmemcpy(mCurBufferp, &value, MVT_U16, 2); | ||
297 | } | ||
298 | mCurBufferp += 2; | ||
299 | return success; | ||
300 | } | ||
301 | |||
302 | |||
303 | BOOL LLDataPackerBinaryBuffer::unpackU16(U16 &value, const char *name) | ||
304 | { | ||
305 | BOOL success = TRUE; | ||
306 | success &= verifyLength(sizeof(U16), name); | ||
307 | |||
308 | htonmemcpy(&value, mCurBufferp, MVT_U16, 2); | ||
309 | mCurBufferp += 2; | ||
310 | return success; | ||
311 | } | ||
312 | |||
313 | |||
314 | BOOL LLDataPackerBinaryBuffer::packU32(const U32 value, const char *name) | ||
315 | { | ||
316 | BOOL success = TRUE; | ||
317 | success &= verifyLength(sizeof(U32), name); | ||
318 | |||
319 | if (mWriteEnabled) | ||
320 | { | ||
321 | htonmemcpy(mCurBufferp, &value, MVT_U32, 4); | ||
322 | } | ||
323 | mCurBufferp += 4; | ||
324 | return success; | ||
325 | } | ||
326 | |||
327 | |||
328 | BOOL LLDataPackerBinaryBuffer::unpackU32(U32 &value, const char *name) | ||
329 | { | ||
330 | BOOL success = TRUE; | ||
331 | success &= verifyLength(sizeof(U32), name); | ||
332 | |||
333 | htonmemcpy(&value, mCurBufferp, MVT_U32, 4); | ||
334 | mCurBufferp += 4; | ||
335 | return success; | ||
336 | } | ||
337 | |||
338 | |||
339 | BOOL LLDataPackerBinaryBuffer::packS32(const S32 value, const char *name) | ||
340 | { | ||
341 | BOOL success = TRUE; | ||
342 | success &= verifyLength(sizeof(S32), name); | ||
343 | |||
344 | if (mWriteEnabled) | ||
345 | { | ||
346 | htonmemcpy(mCurBufferp, &value, MVT_S32, 4); | ||
347 | } | ||
348 | mCurBufferp += 4; | ||
349 | return success; | ||
350 | } | ||
351 | |||
352 | |||
353 | BOOL LLDataPackerBinaryBuffer::unpackS32(S32 &value, const char *name) | ||
354 | { | ||
355 | BOOL success = TRUE; | ||
356 | success &= verifyLength(sizeof(S32), name); | ||
357 | |||
358 | htonmemcpy(&value, mCurBufferp, MVT_S32, 4); | ||
359 | mCurBufferp += 4; | ||
360 | return success; | ||
361 | } | ||
362 | |||
363 | |||
364 | BOOL LLDataPackerBinaryBuffer::packF32(const F32 value, const char *name) | ||
365 | { | ||
366 | BOOL success = TRUE; | ||
367 | success &= verifyLength(sizeof(F32), name); | ||
368 | |||
369 | if (mWriteEnabled) | ||
370 | { | ||
371 | htonmemcpy(mCurBufferp, &value, MVT_F32, 4); | ||
372 | } | ||
373 | mCurBufferp += 4; | ||
374 | return success; | ||
375 | } | ||
376 | |||
377 | |||
378 | BOOL LLDataPackerBinaryBuffer::unpackF32(F32 &value, const char *name) | ||
379 | { | ||
380 | BOOL success = TRUE; | ||
381 | success &= verifyLength(sizeof(F32), name); | ||
382 | |||
383 | htonmemcpy(&value, mCurBufferp, MVT_F32, 4); | ||
384 | mCurBufferp += 4; | ||
385 | return success; | ||
386 | } | ||
387 | |||
388 | |||
389 | BOOL LLDataPackerBinaryBuffer::packColor4(const LLColor4 &value, const char *name) | ||
390 | { | ||
391 | BOOL success = TRUE; | ||
392 | success &= verifyLength(16, name); | ||
393 | |||
394 | if (mWriteEnabled) | ||
395 | { | ||
396 | htonmemcpy(mCurBufferp, value.mV, MVT_LLVector4, 16); | ||
397 | } | ||
398 | mCurBufferp += 16; | ||
399 | return success; | ||
400 | } | ||
401 | |||
402 | |||
403 | BOOL LLDataPackerBinaryBuffer::unpackColor4(LLColor4 &value, const char *name) | ||
404 | { | ||
405 | BOOL success = TRUE; | ||
406 | success &= verifyLength(16, name); | ||
407 | |||
408 | htonmemcpy(value.mV, mCurBufferp, MVT_LLVector4, 16); | ||
409 | mCurBufferp += 16; | ||
410 | return success; | ||
411 | } | ||
412 | |||
413 | |||
414 | BOOL LLDataPackerBinaryBuffer::packColor4U(const LLColor4U &value, const char *name) | ||
415 | { | ||
416 | BOOL success = TRUE; | ||
417 | success &= verifyLength(4, name); | ||
418 | |||
419 | if (mWriteEnabled) | ||
420 | { | ||
421 | htonmemcpy(mCurBufferp, value.mV, MVT_VARIABLE, 4); | ||
422 | } | ||
423 | mCurBufferp += 4; | ||
424 | return success; | ||
425 | } | ||
426 | |||
427 | |||
428 | BOOL LLDataPackerBinaryBuffer::unpackColor4U(LLColor4U &value, const char *name) | ||
429 | { | ||
430 | BOOL success = TRUE; | ||
431 | success &= verifyLength(4, name); | ||
432 | |||
433 | htonmemcpy(value.mV, mCurBufferp, MVT_VARIABLE, 4); | ||
434 | mCurBufferp += 4; | ||
435 | return success; | ||
436 | } | ||
437 | |||
438 | |||
439 | |||
440 | BOOL LLDataPackerBinaryBuffer::packVector2(const LLVector2 &value, const char *name) | ||
441 | { | ||
442 | BOOL success = TRUE; | ||
443 | success &= verifyLength(8, name); | ||
444 | |||
445 | if (mWriteEnabled) | ||
446 | { | ||
447 | htonmemcpy(mCurBufferp, &value.mV[0], MVT_F32, 4); | ||
448 | htonmemcpy(mCurBufferp+4, &value.mV[1], MVT_F32, 4); | ||
449 | } | ||
450 | mCurBufferp += 8; | ||
451 | return success; | ||
452 | } | ||
453 | |||
454 | |||
455 | BOOL LLDataPackerBinaryBuffer::unpackVector2(LLVector2 &value, const char *name) | ||
456 | { | ||
457 | BOOL success = TRUE; | ||
458 | success &= verifyLength(8, name); | ||
459 | |||
460 | htonmemcpy(&value.mV[0], mCurBufferp, MVT_F32, 4); | ||
461 | htonmemcpy(&value.mV[1], mCurBufferp+4, MVT_F32, 4); | ||
462 | mCurBufferp += 8; | ||
463 | return success; | ||
464 | } | ||
465 | |||
466 | |||
467 | BOOL LLDataPackerBinaryBuffer::packVector3(const LLVector3 &value, const char *name) | ||
468 | { | ||
469 | BOOL success = TRUE; | ||
470 | success &= verifyLength(12, name); | ||
471 | |||
472 | if (mWriteEnabled) | ||
473 | { | ||
474 | htonmemcpy(mCurBufferp, value.mV, MVT_LLVector3, 12); | ||
475 | } | ||
476 | mCurBufferp += 12; | ||
477 | return success; | ||
478 | } | ||
479 | |||
480 | |||
481 | BOOL LLDataPackerBinaryBuffer::unpackVector3(LLVector3 &value, const char *name) | ||
482 | { | ||
483 | BOOL success = TRUE; | ||
484 | success &= verifyLength(12, name); | ||
485 | |||
486 | htonmemcpy(value.mV, mCurBufferp, MVT_LLVector3, 12); | ||
487 | mCurBufferp += 12; | ||
488 | return success; | ||
489 | } | ||
490 | |||
491 | BOOL LLDataPackerBinaryBuffer::packVector4(const LLVector4 &value, const char *name) | ||
492 | { | ||
493 | BOOL success = TRUE; | ||
494 | success &= verifyLength(16, name); | ||
495 | |||
496 | if (mWriteEnabled) | ||
497 | { | ||
498 | htonmemcpy(mCurBufferp, value.mV, MVT_LLVector4, 16); | ||
499 | } | ||
500 | mCurBufferp += 16; | ||
501 | return success; | ||
502 | } | ||
503 | |||
504 | |||
505 | BOOL LLDataPackerBinaryBuffer::unpackVector4(LLVector4 &value, const char *name) | ||
506 | { | ||
507 | BOOL success = TRUE; | ||
508 | success &= verifyLength(16, name); | ||
509 | |||
510 | htonmemcpy(value.mV, mCurBufferp, MVT_LLVector4, 16); | ||
511 | mCurBufferp += 16; | ||
512 | return success; | ||
513 | } | ||
514 | |||
515 | BOOL LLDataPackerBinaryBuffer::packUUID(const LLUUID &value, const char *name) | ||
516 | { | ||
517 | BOOL success = TRUE; | ||
518 | success &= verifyLength(16, name); | ||
519 | |||
520 | if (mWriteEnabled) | ||
521 | { | ||
522 | htonmemcpy(mCurBufferp, value.mData, MVT_LLUUID, 16); | ||
523 | } | ||
524 | mCurBufferp += 16; | ||
525 | return success; | ||
526 | } | ||
527 | |||
528 | |||
529 | BOOL LLDataPackerBinaryBuffer::unpackUUID(LLUUID &value, const char *name) | ||
530 | { | ||
531 | BOOL success = TRUE; | ||
532 | success &= verifyLength(16, name); | ||
533 | |||
534 | htonmemcpy(value.mData, mCurBufferp, MVT_LLUUID, 16); | ||
535 | mCurBufferp += 16; | ||
536 | return success; | ||
537 | } | ||
538 | |||
539 | const LLDataPackerBinaryBuffer& LLDataPackerBinaryBuffer::operator=(const LLDataPackerBinaryBuffer &a) | ||
540 | { | ||
541 | if (a.getBufferSize() > getBufferSize()) | ||
542 | { | ||
543 | // We've got problems, ack! | ||
544 | llerrs << "Trying to do an assignment with not enough room in the target." << llendl; | ||
545 | } | ||
546 | memcpy(mBufferp, a.mBufferp, a.getBufferSize()); | ||
547 | return *this; | ||
548 | } | ||
549 | |||
550 | void LLDataPackerBinaryBuffer::dumpBufferToLog() | ||
551 | { | ||
552 | llwarns << "Binary Buffer Dump, size: " << mBufferSize << llendl; | ||
553 | char line_buffer[256]; /*Flawfinder: ignore*/ | ||
554 | S32 i; | ||
555 | S32 cur_line_pos = 0; | ||
556 | |||
557 | S32 cur_line = 0; | ||
558 | for (i = 0; i < mBufferSize; i++) | ||
559 | { | ||
560 | snprintf(line_buffer + cur_line_pos*3, sizeof(line_buffer) - cur_line_pos*3, "%02x ", mBufferp[i]); /*Flawfinder: ignore*/ | ||
561 | cur_line_pos++; | ||
562 | if (cur_line_pos >= 16) | ||
563 | { | ||
564 | cur_line_pos = 0; | ||
565 | llwarns << "Offset:" << std::hex << cur_line*16 << std::dec << " Data:" << line_buffer << llendl; | ||
566 | cur_line++; | ||
567 | } | ||
568 | } | ||
569 | if (cur_line_pos) | ||
570 | { | ||
571 | llwarns << "Offset:" << std::hex << cur_line*16 << std::dec << " Data:" << line_buffer << llendl; | ||
572 | } | ||
573 | } | ||
574 | |||
575 | //--------------------------------------------------------------------------- | ||
576 | // LLDataPackerAsciiBuffer implementation | ||
577 | //--------------------------------------------------------------------------- | ||
578 | BOOL LLDataPackerAsciiBuffer::packString(const char *value, const char *name) | ||
579 | { | ||
580 | BOOL success = TRUE; | ||
581 | writeIndentedName(name); | ||
582 | int numCopied = 0; | ||
583 | if (mWriteEnabled) | ||
584 | { | ||
585 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%s\n", value); /*Flawfinder: ignore*/ | ||
586 | } | ||
587 | else | ||
588 | { | ||
589 | numCopied = (S32)strlen(value) + 1; /*Flawfinder: ignore*/ | ||
590 | } | ||
591 | |||
592 | // snprintf returns number of bytes that would have been written | ||
593 | // had the output not being truncated. In that case, it will | ||
594 | // return >= passed in size value. so a check needs to be added | ||
595 | // to detect truncation, and if there is any, only account for the | ||
596 | // actual number of bytes written..and not what could have been | ||
597 | // written. | ||
598 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
599 | { | ||
600 | // *NOTE: I believe we need to mark a failure bit at this point. | ||
601 | numCopied = getBufferSize()-getCurrentSize(); | ||
602 | } | ||
603 | mCurBufferp += numCopied; | ||
604 | return success; | ||
605 | } | ||
606 | |||
607 | BOOL LLDataPackerAsciiBuffer::unpackString(std::string& value, const char *name) | ||
608 | { | ||
609 | BOOL success = TRUE; | ||
610 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore*/ | ||
611 | BOOL res = getValueStr(name, valuestr, DP_BUFSIZE); // NULL terminated | ||
612 | if (!res) // | ||
613 | { | ||
614 | return FALSE; | ||
615 | } | ||
616 | value = valuestr; | ||
617 | return success; | ||
618 | } | ||
619 | |||
620 | |||
621 | BOOL LLDataPackerAsciiBuffer::packBinaryData(const U8 *value, S32 size, const char *name) | ||
622 | { | ||
623 | BOOL success = TRUE; | ||
624 | writeIndentedName(name); | ||
625 | |||
626 | int numCopied = 0; | ||
627 | if (mWriteEnabled) | ||
628 | { | ||
629 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%010d ", size); /*Flawfinder: ignore*/ | ||
630 | |||
631 | // snprintf returns number of bytes that would have been | ||
632 | // written had the output not being truncated. In that case, | ||
633 | // it will retuen >= passed in size value. so a check needs | ||
634 | // to be added to detect truncation, and if there is any, only | ||
635 | // account for the actual number of bytes written..and not | ||
636 | // what could have been written. | ||
637 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
638 | { | ||
639 | numCopied = getBufferSize()-getCurrentSize(); | ||
640 | } | ||
641 | mCurBufferp += numCopied; | ||
642 | |||
643 | |||
644 | S32 i; | ||
645 | BOOL bBufferFull = FALSE; | ||
646 | for (i = 0; i < size && !bBufferFull; i++) | ||
647 | { | ||
648 | numCopied = snprintf(mCurBufferp, getBufferSize()-getCurrentSize(), "%02x ", value[i]); /* Flawfinder: ignore */ | ||
649 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
650 | { | ||
651 | numCopied = getBufferSize()-getCurrentSize(); | ||
652 | bBufferFull = TRUE; | ||
653 | } | ||
654 | mCurBufferp += numCopied; | ||
655 | } | ||
656 | |||
657 | if (!bBufferFull) | ||
658 | { | ||
659 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(), "\n"); /* Flawfinder: ignore */ | ||
660 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
661 | { | ||
662 | numCopied = getBufferSize()-getCurrentSize(); | ||
663 | } | ||
664 | mCurBufferp += numCopied; | ||
665 | } | ||
666 | } | ||
667 | else | ||
668 | { | ||
669 | // why +10 ?? XXXCHECK | ||
670 | numCopied = 10 + 1; // size plus newline | ||
671 | numCopied += size; | ||
672 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
673 | { | ||
674 | numCopied = getBufferSize()-getCurrentSize(); | ||
675 | } | ||
676 | mCurBufferp += numCopied; | ||
677 | } | ||
678 | |||
679 | return success; | ||
680 | } | ||
681 | |||
682 | |||
683 | BOOL LLDataPackerAsciiBuffer::unpackBinaryData(U8 *value, S32 &size, const char *name) | ||
684 | { | ||
685 | BOOL success = TRUE; | ||
686 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
687 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
688 | { | ||
689 | return FALSE; | ||
690 | } | ||
691 | |||
692 | char *cur_pos = &valuestr[0]; | ||
693 | sscanf(valuestr,"%010d", &size); | ||
694 | cur_pos += 11; | ||
695 | |||
696 | S32 i; | ||
697 | for (i = 0; i < size; i++) | ||
698 | { | ||
699 | S32 val; | ||
700 | sscanf(cur_pos,"%02x", &val); | ||
701 | value[i] = val; | ||
702 | cur_pos += 3; | ||
703 | } | ||
704 | return success; | ||
705 | } | ||
706 | |||
707 | |||
708 | BOOL LLDataPackerAsciiBuffer::packBinaryDataFixed(const U8 *value, S32 size, const char *name) | ||
709 | { | ||
710 | BOOL success = TRUE; | ||
711 | writeIndentedName(name); | ||
712 | |||
713 | if (mWriteEnabled) | ||
714 | { | ||
715 | S32 i; | ||
716 | int numCopied = 0; | ||
717 | BOOL bBufferFull = FALSE; | ||
718 | for (i = 0; i < size && !bBufferFull; i++) | ||
719 | { | ||
720 | numCopied = snprintf(mCurBufferp, getBufferSize()-getCurrentSize(), "%02x ", value[i]); /* Flawfinder: ignore */ | ||
721 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
722 | { | ||
723 | numCopied = getBufferSize()-getCurrentSize(); | ||
724 | bBufferFull = TRUE; | ||
725 | } | ||
726 | mCurBufferp += numCopied; | ||
727 | |||
728 | } | ||
729 | if (!bBufferFull) | ||
730 | { | ||
731 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(), "\n"); /* Flawfinder: ignore */ | ||
732 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
733 | { | ||
734 | numCopied = getBufferSize()-getCurrentSize(); | ||
735 | } | ||
736 | |||
737 | mCurBufferp += numCopied; | ||
738 | } | ||
739 | } | ||
740 | else | ||
741 | { | ||
742 | int numCopied = 2 * size + 1; //hex bytes plus newline | ||
743 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
744 | { | ||
745 | numCopied = getBufferSize()-getCurrentSize(); | ||
746 | } | ||
747 | mCurBufferp += numCopied; | ||
748 | } | ||
749 | return success; | ||
750 | } | ||
751 | |||
752 | |||
753 | BOOL LLDataPackerAsciiBuffer::unpackBinaryDataFixed(U8 *value, S32 size, const char *name) | ||
754 | { | ||
755 | BOOL success = TRUE; | ||
756 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
757 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
758 | { | ||
759 | return FALSE; | ||
760 | } | ||
761 | |||
762 | char *cur_pos = &valuestr[0]; | ||
763 | |||
764 | S32 i; | ||
765 | for (i = 0; i < size; i++) | ||
766 | { | ||
767 | S32 val; | ||
768 | sscanf(cur_pos,"%02x", &val); | ||
769 | value[i] = val; | ||
770 | cur_pos += 3; | ||
771 | } | ||
772 | return success; | ||
773 | } | ||
774 | |||
775 | |||
776 | |||
777 | BOOL LLDataPackerAsciiBuffer::packU8(const U8 value, const char *name) | ||
778 | { | ||
779 | BOOL success = TRUE; | ||
780 | writeIndentedName(name); | ||
781 | int numCopied = 0; | ||
782 | if (mWriteEnabled) | ||
783 | { | ||
784 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%d\n", value); /*Flawfinder: ignore*/ | ||
785 | } | ||
786 | else | ||
787 | { | ||
788 | // just do the write to a temp buffer to get the length | ||
789 | numCopied = snprintf(DUMMY_BUFFER, sizeof(DUMMY_BUFFER), "%d\n", value); /* Flawfinder: ignore */ | ||
790 | } | ||
791 | |||
792 | // snprintf returns number of bytes that would have been written had the | ||
793 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
794 | // so a check needs to be added to detect truncation, and if there is any, | ||
795 | // only account for the actual number of bytes written..and not what could have been written. | ||
796 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
797 | { | ||
798 | numCopied = getBufferSize()-getCurrentSize(); | ||
799 | } | ||
800 | |||
801 | mCurBufferp += numCopied; | ||
802 | |||
803 | return success; | ||
804 | } | ||
805 | |||
806 | |||
807 | BOOL LLDataPackerAsciiBuffer::unpackU8(U8 &value, const char *name) | ||
808 | { | ||
809 | BOOL success = TRUE; | ||
810 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
811 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
812 | { | ||
813 | return FALSE; | ||
814 | } | ||
815 | |||
816 | S32 in_val; | ||
817 | sscanf(valuestr,"%d", &in_val); | ||
818 | value = in_val; | ||
819 | return success; | ||
820 | } | ||
821 | |||
822 | BOOL LLDataPackerAsciiBuffer::packU16(const U16 value, const char *name) | ||
823 | { | ||
824 | BOOL success = TRUE; | ||
825 | writeIndentedName(name); | ||
826 | int numCopied = 0; | ||
827 | if (mWriteEnabled) | ||
828 | { | ||
829 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%d\n", value); /*Flawfinder: ignore*/ | ||
830 | } | ||
831 | else | ||
832 | { | ||
833 | numCopied = snprintf(DUMMY_BUFFER, sizeof(DUMMY_BUFFER), "%d\n", value); /* Flawfinder: ignore */ | ||
834 | } | ||
835 | |||
836 | // snprintf returns number of bytes that would have been written had the | ||
837 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
838 | // so a check needs to be added to detect truncation, and if there is any, | ||
839 | // only account for the actual number of bytes written..and not what could have been written. | ||
840 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
841 | { | ||
842 | numCopied = getBufferSize()-getCurrentSize(); | ||
843 | } | ||
844 | |||
845 | mCurBufferp += numCopied; | ||
846 | |||
847 | return success; | ||
848 | } | ||
849 | |||
850 | |||
851 | BOOL LLDataPackerAsciiBuffer::unpackU16(U16 &value, const char *name) | ||
852 | { | ||
853 | BOOL success = TRUE; | ||
854 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
855 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
856 | { | ||
857 | return FALSE; | ||
858 | } | ||
859 | |||
860 | S32 in_val; | ||
861 | sscanf(valuestr,"%d", &in_val); | ||
862 | value = in_val; | ||
863 | return success; | ||
864 | } | ||
865 | |||
866 | |||
867 | BOOL LLDataPackerAsciiBuffer::packU32(const U32 value, const char *name) | ||
868 | { | ||
869 | BOOL success = TRUE; | ||
870 | writeIndentedName(name); | ||
871 | int numCopied = 0; | ||
872 | if (mWriteEnabled) | ||
873 | { | ||
874 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%u\n", value); /* Flawfinder: ignore */ | ||
875 | } | ||
876 | else | ||
877 | { | ||
878 | numCopied = snprintf(DUMMY_BUFFER, sizeof(DUMMY_BUFFER), "%u\n", value); /* Flawfinder: ignore */ | ||
879 | } | ||
880 | // snprintf returns number of bytes that would have been written had the | ||
881 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
882 | // so a check needs to be added to detect truncation, and if there is any, | ||
883 | // only account for the actual number of bytes written..and not what could have been written. | ||
884 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
885 | { | ||
886 | numCopied = getBufferSize()-getCurrentSize(); | ||
887 | } | ||
888 | |||
889 | mCurBufferp += numCopied; | ||
890 | return success; | ||
891 | } | ||
892 | |||
893 | |||
894 | BOOL LLDataPackerAsciiBuffer::unpackU32(U32 &value, const char *name) | ||
895 | { | ||
896 | BOOL success = TRUE; | ||
897 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
898 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
899 | { | ||
900 | return FALSE; | ||
901 | } | ||
902 | |||
903 | sscanf(valuestr,"%u", &value); | ||
904 | return success; | ||
905 | } | ||
906 | |||
907 | |||
908 | BOOL LLDataPackerAsciiBuffer::packS32(const S32 value, const char *name) | ||
909 | { | ||
910 | BOOL success = TRUE; | ||
911 | writeIndentedName(name); | ||
912 | int numCopied = 0; | ||
913 | if (mWriteEnabled) | ||
914 | { | ||
915 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%d\n", value); /* Flawfinder: ignore */ | ||
916 | } | ||
917 | else | ||
918 | { | ||
919 | numCopied = snprintf(DUMMY_BUFFER, sizeof(DUMMY_BUFFER), "%d\n", value); /* Flawfinder: ignore */ | ||
920 | } | ||
921 | // snprintf returns number of bytes that would have been written had the | ||
922 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
923 | // so a check needs to be added to detect truncation, and if there is any, | ||
924 | // only account for the actual number of bytes written..and not what could have been written. | ||
925 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
926 | { | ||
927 | numCopied = getBufferSize()-getCurrentSize(); | ||
928 | } | ||
929 | |||
930 | mCurBufferp += numCopied; | ||
931 | return success; | ||
932 | } | ||
933 | |||
934 | |||
935 | BOOL LLDataPackerAsciiBuffer::unpackS32(S32 &value, const char *name) | ||
936 | { | ||
937 | BOOL success = TRUE; | ||
938 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
939 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
940 | { | ||
941 | return FALSE; | ||
942 | } | ||
943 | |||
944 | sscanf(valuestr,"%d", &value); | ||
945 | return success; | ||
946 | } | ||
947 | |||
948 | |||
949 | BOOL LLDataPackerAsciiBuffer::packF32(const F32 value, const char *name) | ||
950 | { | ||
951 | BOOL success = TRUE; | ||
952 | writeIndentedName(name); | ||
953 | int numCopied = 0; | ||
954 | if (mWriteEnabled) | ||
955 | { | ||
956 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%g\n", value); /* Flawfinder: ignore */ | ||
957 | } | ||
958 | else | ||
959 | { | ||
960 | numCopied = snprintf(DUMMY_BUFFER, sizeof(DUMMY_BUFFER), "%g\n", value); /* Flawfinder: ignore */ | ||
961 | } | ||
962 | // snprintf returns number of bytes that would have been written had the | ||
963 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
964 | // so a check needs to be added to detect truncation, and if there is any, | ||
965 | // only account for the actual number of bytes written..and not what could have been written. | ||
966 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
967 | { | ||
968 | numCopied = getBufferSize()-getCurrentSize(); | ||
969 | } | ||
970 | |||
971 | mCurBufferp += numCopied; | ||
972 | return success; | ||
973 | } | ||
974 | |||
975 | |||
976 | BOOL LLDataPackerAsciiBuffer::unpackF32(F32 &value, const char *name) | ||
977 | { | ||
978 | BOOL success = TRUE; | ||
979 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
980 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
981 | { | ||
982 | return FALSE; | ||
983 | } | ||
984 | |||
985 | sscanf(valuestr,"%g", &value); | ||
986 | return success; | ||
987 | } | ||
988 | |||
989 | |||
990 | BOOL LLDataPackerAsciiBuffer::packColor4(const LLColor4 &value, const char *name) | ||
991 | { | ||
992 | BOOL success = TRUE; | ||
993 | writeIndentedName(name); | ||
994 | int numCopied = 0; | ||
995 | if (mWriteEnabled) | ||
996 | { | ||
997 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%g %g %g %g\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); /* Flawfinder: ignore */ | ||
998 | } | ||
999 | else | ||
1000 | { | ||
1001 | numCopied = snprintf(DUMMY_BUFFER,sizeof(DUMMY_BUFFER),"%g %g %g %g\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); /* Flawfinder: ignore */ | ||
1002 | } | ||
1003 | // snprintf returns number of bytes that would have been written had the | ||
1004 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
1005 | // so a check needs to be added to detect truncation, and if there is any, | ||
1006 | // only account for the actual number of bytes written..and not what could have been written. | ||
1007 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
1008 | { | ||
1009 | numCopied = getBufferSize()-getCurrentSize(); | ||
1010 | } | ||
1011 | |||
1012 | mCurBufferp += numCopied; | ||
1013 | return success; | ||
1014 | } | ||
1015 | |||
1016 | |||
1017 | BOOL LLDataPackerAsciiBuffer::unpackColor4(LLColor4 &value, const char *name) | ||
1018 | { | ||
1019 | BOOL success = TRUE; | ||
1020 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1021 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1022 | { | ||
1023 | return FALSE; | ||
1024 | } | ||
1025 | |||
1026 | sscanf(valuestr,"%g %g %g %g", &value.mV[0], &value.mV[1], &value.mV[2], &value.mV[3]); | ||
1027 | return success; | ||
1028 | } | ||
1029 | |||
1030 | BOOL LLDataPackerAsciiBuffer::packColor4U(const LLColor4U &value, const char *name) | ||
1031 | { | ||
1032 | BOOL success = TRUE; | ||
1033 | writeIndentedName(name); | ||
1034 | int numCopied = 0; | ||
1035 | if (mWriteEnabled) | ||
1036 | { | ||
1037 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%d %d %d %d\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); /* Flawfinder: ignore */ | ||
1038 | } | ||
1039 | else | ||
1040 | { | ||
1041 | numCopied = snprintf(DUMMY_BUFFER,sizeof(DUMMY_BUFFER),"%d %d %d %d\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); /* Flawfinder: ignore */ | ||
1042 | } | ||
1043 | // snprintf returns number of bytes that would have been written had the | ||
1044 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
1045 | // so a check needs to be added to detect truncation, and if there is any, | ||
1046 | // only account for the actual number of bytes written..and not what could have been written. | ||
1047 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
1048 | { | ||
1049 | numCopied = getBufferSize()-getCurrentSize(); | ||
1050 | } | ||
1051 | |||
1052 | mCurBufferp += numCopied; | ||
1053 | return success; | ||
1054 | } | ||
1055 | |||
1056 | |||
1057 | BOOL LLDataPackerAsciiBuffer::unpackColor4U(LLColor4U &value, const char *name) | ||
1058 | { | ||
1059 | BOOL success = TRUE; | ||
1060 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1061 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1062 | { | ||
1063 | return FALSE; | ||
1064 | } | ||
1065 | |||
1066 | S32 r, g, b, a; | ||
1067 | |||
1068 | sscanf(valuestr,"%d %d %d %d", &r, &g, &b, &a); | ||
1069 | value.mV[0] = r; | ||
1070 | value.mV[1] = g; | ||
1071 | value.mV[2] = b; | ||
1072 | value.mV[3] = a; | ||
1073 | return success; | ||
1074 | } | ||
1075 | |||
1076 | |||
1077 | BOOL LLDataPackerAsciiBuffer::packVector2(const LLVector2 &value, const char *name) | ||
1078 | { | ||
1079 | BOOL success = TRUE; | ||
1080 | writeIndentedName(name); | ||
1081 | int numCopied = 0; | ||
1082 | if (mWriteEnabled) | ||
1083 | { | ||
1084 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%g %g\n", value.mV[0], value.mV[1]); /* Flawfinder: ignore */ | ||
1085 | } | ||
1086 | else | ||
1087 | { | ||
1088 | numCopied = snprintf(DUMMY_BUFFER,sizeof(DUMMY_BUFFER),"%g %g\n", value.mV[0], value.mV[1]); /* Flawfinder: ignore */ | ||
1089 | } | ||
1090 | // snprintf returns number of bytes that would have been written had the | ||
1091 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
1092 | // so a check needs to be added to detect truncation, and if there is any, | ||
1093 | // only account for the actual number of bytes written..and not what could have been written. | ||
1094 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
1095 | { | ||
1096 | numCopied = getBufferSize()-getCurrentSize(); | ||
1097 | } | ||
1098 | |||
1099 | mCurBufferp += numCopied; | ||
1100 | return success; | ||
1101 | } | ||
1102 | |||
1103 | |||
1104 | BOOL LLDataPackerAsciiBuffer::unpackVector2(LLVector2 &value, const char *name) | ||
1105 | { | ||
1106 | BOOL success = TRUE; | ||
1107 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1108 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1109 | { | ||
1110 | return FALSE; | ||
1111 | } | ||
1112 | |||
1113 | sscanf(valuestr,"%g %g", &value.mV[0], &value.mV[1]); | ||
1114 | return success; | ||
1115 | } | ||
1116 | |||
1117 | |||
1118 | BOOL LLDataPackerAsciiBuffer::packVector3(const LLVector3 &value, const char *name) | ||
1119 | { | ||
1120 | BOOL success = TRUE; | ||
1121 | writeIndentedName(name); | ||
1122 | int numCopied = 0; | ||
1123 | if (mWriteEnabled) | ||
1124 | { | ||
1125 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%g %g %g\n", value.mV[0], value.mV[1], value.mV[2]); /* Flawfinder: ignore */ | ||
1126 | } | ||
1127 | else | ||
1128 | { | ||
1129 | numCopied = snprintf(DUMMY_BUFFER,sizeof(DUMMY_BUFFER),"%g %g %g\n", value.mV[0], value.mV[1], value.mV[2]); /* Flawfinder: ignore */ | ||
1130 | } | ||
1131 | // snprintf returns number of bytes that would have been written had the | ||
1132 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
1133 | // so a check needs to be added to detect truncation, and if there is any, | ||
1134 | // only account for the actual number of bytes written..and not what could have been written. | ||
1135 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
1136 | { | ||
1137 | numCopied = getBufferSize()-getCurrentSize(); | ||
1138 | } | ||
1139 | |||
1140 | mCurBufferp += numCopied; | ||
1141 | return success; | ||
1142 | } | ||
1143 | |||
1144 | |||
1145 | BOOL LLDataPackerAsciiBuffer::unpackVector3(LLVector3 &value, const char *name) | ||
1146 | { | ||
1147 | BOOL success = TRUE; | ||
1148 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1149 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1150 | { | ||
1151 | return FALSE; | ||
1152 | } | ||
1153 | |||
1154 | sscanf(valuestr,"%g %g %g", &value.mV[0], &value.mV[1], &value.mV[2]); | ||
1155 | return success; | ||
1156 | } | ||
1157 | |||
1158 | BOOL LLDataPackerAsciiBuffer::packVector4(const LLVector4 &value, const char *name) | ||
1159 | { | ||
1160 | BOOL success = TRUE; | ||
1161 | writeIndentedName(name); | ||
1162 | int numCopied = 0; | ||
1163 | if (mWriteEnabled) | ||
1164 | { | ||
1165 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%g %g %g %g\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); /* Flawfinder: ignore */ | ||
1166 | } | ||
1167 | else | ||
1168 | { | ||
1169 | numCopied = snprintf(DUMMY_BUFFER,sizeof(DUMMY_BUFFER),"%g %g %g %g\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); /* Flawfinder: ignore */ | ||
1170 | } | ||
1171 | // snprintf returns number of bytes that would have been written had the | ||
1172 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
1173 | // so a check needs to be added to detect truncation, and if there is any, | ||
1174 | // only account for the actual number of bytes written..and not what could have been written. | ||
1175 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
1176 | { | ||
1177 | numCopied = getBufferSize()-getCurrentSize(); | ||
1178 | } | ||
1179 | |||
1180 | mCurBufferp += numCopied; | ||
1181 | return success; | ||
1182 | } | ||
1183 | |||
1184 | |||
1185 | BOOL LLDataPackerAsciiBuffer::unpackVector4(LLVector4 &value, const char *name) | ||
1186 | { | ||
1187 | BOOL success = TRUE; | ||
1188 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1189 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1190 | { | ||
1191 | return FALSE; | ||
1192 | } | ||
1193 | |||
1194 | sscanf(valuestr,"%g %g %g %g", &value.mV[0], &value.mV[1], &value.mV[2], &value.mV[3]); | ||
1195 | return success; | ||
1196 | } | ||
1197 | |||
1198 | |||
1199 | BOOL LLDataPackerAsciiBuffer::packUUID(const LLUUID &value, const char *name) | ||
1200 | { | ||
1201 | BOOL success = TRUE; | ||
1202 | writeIndentedName(name); | ||
1203 | |||
1204 | int numCopied = 0; | ||
1205 | if (mWriteEnabled) | ||
1206 | { | ||
1207 | char tmp_str[64]; /* Flawfinder: ignore */ | ||
1208 | value.toString(tmp_str); | ||
1209 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%s\n", tmp_str); /* Flawfinder: ignore */ | ||
1210 | } | ||
1211 | else | ||
1212 | { | ||
1213 | numCopied = 64 + 1; // UUID + newline | ||
1214 | } | ||
1215 | // snprintf returns number of bytes that would have been written had the | ||
1216 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
1217 | // so a check needs to be added to detect truncation, and if there is any, | ||
1218 | // only account for the actual number of bytes written..and not what could have been written. | ||
1219 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
1220 | { | ||
1221 | numCopied = getBufferSize()-getCurrentSize(); | ||
1222 | success = FALSE; | ||
1223 | } | ||
1224 | mCurBufferp += numCopied; | ||
1225 | return success; | ||
1226 | } | ||
1227 | |||
1228 | |||
1229 | BOOL LLDataPackerAsciiBuffer::unpackUUID(LLUUID &value, const char *name) | ||
1230 | { | ||
1231 | BOOL success = TRUE; | ||
1232 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1233 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1234 | { | ||
1235 | return FALSE; | ||
1236 | } | ||
1237 | |||
1238 | char tmp_str[64]; /* Flawfinder: ignore */ | ||
1239 | sscanf(valuestr, "%63s", tmp_str); | ||
1240 | value.set(tmp_str); | ||
1241 | |||
1242 | return success; | ||
1243 | } | ||
1244 | |||
1245 | void LLDataPackerAsciiBuffer::dump() | ||
1246 | { | ||
1247 | llinfos << "Buffer: " << mBufferp << llendl; | ||
1248 | } | ||
1249 | |||
1250 | void LLDataPackerAsciiBuffer::writeIndentedName(const char *name) | ||
1251 | { | ||
1252 | if (mIncludeNames) | ||
1253 | { | ||
1254 | int numCopied = 0; | ||
1255 | if (mWriteEnabled) | ||
1256 | { | ||
1257 | numCopied = snprintf(mCurBufferp,getBufferSize()-getCurrentSize(),"%s\t", name); /* Flawfinder: ignore */ | ||
1258 | } | ||
1259 | else | ||
1260 | { | ||
1261 | numCopied = (S32)strlen(name) + 1; //name + tab /* Flawfinder: ignore */ | ||
1262 | } | ||
1263 | |||
1264 | // snprintf returns number of bytes that would have been written had the | ||
1265 | // output not being truncated. In that case, it will retuen >= passed in size value. | ||
1266 | // so a check needs to be added to detect truncation, and if there is any, | ||
1267 | // only account for the actual number of bytes written..and not what could have been written. | ||
1268 | if (numCopied > getBufferSize()-getCurrentSize()) | ||
1269 | { | ||
1270 | numCopied = getBufferSize()-getCurrentSize(); | ||
1271 | } | ||
1272 | } | ||
1273 | } | ||
1274 | |||
1275 | BOOL LLDataPackerAsciiBuffer::getValueStr(const char *name, char *out_value, S32 value_len) | ||
1276 | { | ||
1277 | BOOL success = TRUE; | ||
1278 | char buffer[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1279 | char keyword[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1280 | char value[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1281 | |||
1282 | buffer[0] = '\0'; | ||
1283 | keyword[0] = '\0'; | ||
1284 | value[0] = '\0'; | ||
1285 | |||
1286 | if (mIncludeNames) | ||
1287 | { | ||
1288 | // Read both the name and the value, and validate the name. | ||
1289 | sscanf(mCurBufferp, "%511[^\n]", buffer); | ||
1290 | // Skip the \n | ||
1291 | mCurBufferp += (S32)strlen(buffer) + 1; | ||
1292 | |||
1293 | sscanf(buffer, "%511s %511[^\n]", keyword, value); | ||
1294 | |||
1295 | if (strcmp(keyword, name)) | ||
1296 | { | ||
1297 | llwarns << "Data packer expecting keyword of type " << name << ", got " << keyword << " instead!" << llendl; | ||
1298 | return FALSE; | ||
1299 | } | ||
1300 | } | ||
1301 | else | ||
1302 | { | ||
1303 | // Just the value exists | ||
1304 | sscanf(mCurBufferp, "%511[^\n]", value); | ||
1305 | // Skip the \n | ||
1306 | mCurBufferp += (S32)strlen(value) + 1; /* Flawfinder: ignore */ | ||
1307 | } | ||
1308 | |||
1309 | S32 in_value_len = (S32)strlen(value)+1; /* Flawfinder: ignore */ | ||
1310 | S32 min_len = llmin(in_value_len, value_len); | ||
1311 | memcpy(out_value, value, min_len); /* Flawfinder: ignore */ | ||
1312 | out_value[min_len-1] = 0; | ||
1313 | |||
1314 | return success; | ||
1315 | } | ||
1316 | |||
1317 | //--------------------------------------------------------------------------- | ||
1318 | // LLDataPackerAsciiFile implementation | ||
1319 | //--------------------------------------------------------------------------- | ||
1320 | BOOL LLDataPackerAsciiFile::packString(const char *value, const char *name) | ||
1321 | { | ||
1322 | BOOL success = TRUE; | ||
1323 | writeIndentedName(name); | ||
1324 | if (mFP) | ||
1325 | { | ||
1326 | fprintf(mFP,"%s\n", value); | ||
1327 | } | ||
1328 | else if (mOutputStream) | ||
1329 | { | ||
1330 | *mOutputStream << value << "\n"; | ||
1331 | } | ||
1332 | return success; | ||
1333 | } | ||
1334 | |||
1335 | BOOL LLDataPackerAsciiFile::unpackString(std::string& value, const char *name) | ||
1336 | { | ||
1337 | BOOL success = TRUE; | ||
1338 | char valuestr[DP_BUFSIZE]; /* Flawfinder: ignore */ | ||
1339 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1340 | { | ||
1341 | return FALSE; | ||
1342 | } | ||
1343 | value = valuestr; | ||
1344 | return success; | ||
1345 | } | ||
1346 | |||
1347 | |||
1348 | BOOL LLDataPackerAsciiFile::packBinaryData(const U8 *value, S32 size, const char *name) | ||
1349 | { | ||
1350 | BOOL success = TRUE; | ||
1351 | writeIndentedName(name); | ||
1352 | |||
1353 | if (mFP) | ||
1354 | { | ||
1355 | fprintf(mFP, "%010d ", size); | ||
1356 | |||
1357 | S32 i; | ||
1358 | for (i = 0; i < size; i++) | ||
1359 | { | ||
1360 | fprintf(mFP, "%02x ", value[i]); | ||
1361 | } | ||
1362 | fprintf(mFP, "\n"); | ||
1363 | } | ||
1364 | else if (mOutputStream) | ||
1365 | { | ||
1366 | char buffer[32]; /* Flawfinder: ignore */ | ||
1367 | snprintf(buffer,sizeof(buffer), "%010d ", size); /* Flawfinder: ignore */ | ||
1368 | *mOutputStream << buffer; | ||
1369 | |||
1370 | S32 i; | ||
1371 | for (i = 0; i < size; i++) | ||
1372 | { | ||
1373 | snprintf(buffer, sizeof(buffer), "%02x ", value[i]); /* Flawfinder: ignore */ | ||
1374 | *mOutputStream << buffer; | ||
1375 | } | ||
1376 | *mOutputStream << "\n"; | ||
1377 | } | ||
1378 | return success; | ||
1379 | } | ||
1380 | |||
1381 | |||
1382 | BOOL LLDataPackerAsciiFile::unpackBinaryData(U8 *value, S32 &size, const char *name) | ||
1383 | { | ||
1384 | BOOL success = TRUE; | ||
1385 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore*/ | ||
1386 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1387 | { | ||
1388 | return FALSE; | ||
1389 | } | ||
1390 | |||
1391 | char *cur_pos = &valuestr[0]; | ||
1392 | sscanf(valuestr,"%010d", &size); | ||
1393 | cur_pos += 11; | ||
1394 | |||
1395 | S32 i; | ||
1396 | for (i = 0; i < size; i++) | ||
1397 | { | ||
1398 | S32 val; | ||
1399 | sscanf(cur_pos,"%02x", &val); | ||
1400 | value[i] = val; | ||
1401 | cur_pos += 3; | ||
1402 | } | ||
1403 | return success; | ||
1404 | } | ||
1405 | |||
1406 | |||
1407 | BOOL LLDataPackerAsciiFile::packBinaryDataFixed(const U8 *value, S32 size, const char *name) | ||
1408 | { | ||
1409 | BOOL success = TRUE; | ||
1410 | writeIndentedName(name); | ||
1411 | |||
1412 | if (mFP) | ||
1413 | { | ||
1414 | S32 i; | ||
1415 | for (i = 0; i < size; i++) | ||
1416 | { | ||
1417 | fprintf(mFP, "%02x ", value[i]); | ||
1418 | } | ||
1419 | fprintf(mFP, "\n"); | ||
1420 | } | ||
1421 | else if (mOutputStream) | ||
1422 | { | ||
1423 | char buffer[32]; /*Flawfinder: ignore*/ | ||
1424 | S32 i; | ||
1425 | for (i = 0; i < size; i++) | ||
1426 | { | ||
1427 | snprintf(buffer, sizeof(buffer), "%02x ", value[i]); /*Flawfinder: ignore*/ | ||
1428 | *mOutputStream << buffer; | ||
1429 | } | ||
1430 | *mOutputStream << "\n"; | ||
1431 | } | ||
1432 | return success; | ||
1433 | } | ||
1434 | |||
1435 | |||
1436 | BOOL LLDataPackerAsciiFile::unpackBinaryDataFixed(U8 *value, S32 size, const char *name) | ||
1437 | { | ||
1438 | BOOL success = TRUE; | ||
1439 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore*/ | ||
1440 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1441 | { | ||
1442 | return FALSE; | ||
1443 | } | ||
1444 | |||
1445 | char *cur_pos = &valuestr[0]; | ||
1446 | |||
1447 | S32 i; | ||
1448 | for (i = 0; i < size; i++) | ||
1449 | { | ||
1450 | S32 val; | ||
1451 | sscanf(cur_pos,"%02x", &val); | ||
1452 | value[i] = val; | ||
1453 | cur_pos += 3; | ||
1454 | } | ||
1455 | return success; | ||
1456 | } | ||
1457 | |||
1458 | |||
1459 | |||
1460 | BOOL LLDataPackerAsciiFile::packU8(const U8 value, const char *name) | ||
1461 | { | ||
1462 | BOOL success = TRUE; | ||
1463 | writeIndentedName(name); | ||
1464 | if (mFP) | ||
1465 | { | ||
1466 | fprintf(mFP,"%d\n", value); | ||
1467 | } | ||
1468 | else if (mOutputStream) | ||
1469 | { | ||
1470 | // We have to cast this to an integer because streams serialize | ||
1471 | // bytes as bytes - not as text. | ||
1472 | *mOutputStream << (S32)value << "\n"; | ||
1473 | } | ||
1474 | return success; | ||
1475 | } | ||
1476 | |||
1477 | |||
1478 | BOOL LLDataPackerAsciiFile::unpackU8(U8 &value, const char *name) | ||
1479 | { | ||
1480 | BOOL success = TRUE; | ||
1481 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1482 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1483 | { | ||
1484 | return FALSE; | ||
1485 | } | ||
1486 | |||
1487 | S32 in_val; | ||
1488 | sscanf(valuestr,"%d", &in_val); | ||
1489 | value = in_val; | ||
1490 | return success; | ||
1491 | } | ||
1492 | |||
1493 | BOOL LLDataPackerAsciiFile::packU16(const U16 value, const char *name) | ||
1494 | { | ||
1495 | BOOL success = TRUE; | ||
1496 | writeIndentedName(name); | ||
1497 | if (mFP) | ||
1498 | { | ||
1499 | fprintf(mFP,"%d\n", value); | ||
1500 | } | ||
1501 | else if (mOutputStream) | ||
1502 | { | ||
1503 | *mOutputStream <<"" << value << "\n"; | ||
1504 | } | ||
1505 | return success; | ||
1506 | } | ||
1507 | |||
1508 | |||
1509 | BOOL LLDataPackerAsciiFile::unpackU16(U16 &value, const char *name) | ||
1510 | { | ||
1511 | BOOL success = TRUE; | ||
1512 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1513 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1514 | { | ||
1515 | return FALSE; | ||
1516 | } | ||
1517 | |||
1518 | S32 in_val; | ||
1519 | sscanf(valuestr,"%d", &in_val); | ||
1520 | value = in_val; | ||
1521 | return success; | ||
1522 | } | ||
1523 | |||
1524 | |||
1525 | BOOL LLDataPackerAsciiFile::packU32(const U32 value, const char *name) | ||
1526 | { | ||
1527 | BOOL success = TRUE; | ||
1528 | writeIndentedName(name); | ||
1529 | if (mFP) | ||
1530 | { | ||
1531 | fprintf(mFP,"%u\n", value); | ||
1532 | } | ||
1533 | else if (mOutputStream) | ||
1534 | { | ||
1535 | *mOutputStream <<"" << value << "\n"; | ||
1536 | } | ||
1537 | return success; | ||
1538 | } | ||
1539 | |||
1540 | |||
1541 | BOOL LLDataPackerAsciiFile::unpackU32(U32 &value, const char *name) | ||
1542 | { | ||
1543 | BOOL success = TRUE; | ||
1544 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1545 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1546 | { | ||
1547 | return FALSE; | ||
1548 | } | ||
1549 | |||
1550 | sscanf(valuestr,"%u", &value); | ||
1551 | return success; | ||
1552 | } | ||
1553 | |||
1554 | |||
1555 | BOOL LLDataPackerAsciiFile::packS32(const S32 value, const char *name) | ||
1556 | { | ||
1557 | BOOL success = TRUE; | ||
1558 | writeIndentedName(name); | ||
1559 | if (mFP) | ||
1560 | { | ||
1561 | fprintf(mFP,"%d\n", value); | ||
1562 | } | ||
1563 | else if (mOutputStream) | ||
1564 | { | ||
1565 | *mOutputStream <<"" << value << "\n"; | ||
1566 | } | ||
1567 | return success; | ||
1568 | } | ||
1569 | |||
1570 | |||
1571 | BOOL LLDataPackerAsciiFile::unpackS32(S32 &value, const char *name) | ||
1572 | { | ||
1573 | BOOL success = TRUE; | ||
1574 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1575 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1576 | { | ||
1577 | return FALSE; | ||
1578 | } | ||
1579 | |||
1580 | sscanf(valuestr,"%d", &value); | ||
1581 | return success; | ||
1582 | } | ||
1583 | |||
1584 | |||
1585 | BOOL LLDataPackerAsciiFile::packF32(const F32 value, const char *name) | ||
1586 | { | ||
1587 | BOOL success = TRUE; | ||
1588 | writeIndentedName(name); | ||
1589 | if (mFP) | ||
1590 | { | ||
1591 | fprintf(mFP,"%g\n", value); | ||
1592 | } | ||
1593 | else if (mOutputStream) | ||
1594 | { | ||
1595 | *mOutputStream <<"" << value << "\n"; | ||
1596 | } | ||
1597 | return success; | ||
1598 | } | ||
1599 | |||
1600 | |||
1601 | BOOL LLDataPackerAsciiFile::unpackF32(F32 &value, const char *name) | ||
1602 | { | ||
1603 | BOOL success = TRUE; | ||
1604 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1605 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1606 | { | ||
1607 | return FALSE; | ||
1608 | } | ||
1609 | |||
1610 | sscanf(valuestr,"%g", &value); | ||
1611 | return success; | ||
1612 | } | ||
1613 | |||
1614 | |||
1615 | BOOL LLDataPackerAsciiFile::packColor4(const LLColor4 &value, const char *name) | ||
1616 | { | ||
1617 | BOOL success = TRUE; | ||
1618 | writeIndentedName(name); | ||
1619 | if (mFP) | ||
1620 | { | ||
1621 | fprintf(mFP,"%g %g %g %g\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); | ||
1622 | } | ||
1623 | else if (mOutputStream) | ||
1624 | { | ||
1625 | *mOutputStream << value.mV[0] << " " << value.mV[1] << " " << value.mV[2] << " " << value.mV[3] << "\n"; | ||
1626 | } | ||
1627 | return success; | ||
1628 | } | ||
1629 | |||
1630 | |||
1631 | BOOL LLDataPackerAsciiFile::unpackColor4(LLColor4 &value, const char *name) | ||
1632 | { | ||
1633 | BOOL success = TRUE; | ||
1634 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1635 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1636 | { | ||
1637 | return FALSE; | ||
1638 | } | ||
1639 | |||
1640 | sscanf(valuestr,"%g %g %g %g", &value.mV[0], &value.mV[1], &value.mV[2], &value.mV[3]); | ||
1641 | return success; | ||
1642 | } | ||
1643 | |||
1644 | BOOL LLDataPackerAsciiFile::packColor4U(const LLColor4U &value, const char *name) | ||
1645 | { | ||
1646 | BOOL success = TRUE; | ||
1647 | writeIndentedName(name); | ||
1648 | if (mFP) | ||
1649 | { | ||
1650 | fprintf(mFP,"%d %d %d %d\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); | ||
1651 | } | ||
1652 | else if (mOutputStream) | ||
1653 | { | ||
1654 | *mOutputStream << (S32)(value.mV[0]) << " " << (S32)(value.mV[1]) << " " << (S32)(value.mV[2]) << " " << (S32)(value.mV[3]) << "\n"; | ||
1655 | } | ||
1656 | return success; | ||
1657 | } | ||
1658 | |||
1659 | |||
1660 | BOOL LLDataPackerAsciiFile::unpackColor4U(LLColor4U &value, const char *name) | ||
1661 | { | ||
1662 | BOOL success = TRUE; | ||
1663 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1664 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1665 | { | ||
1666 | return FALSE; | ||
1667 | } | ||
1668 | |||
1669 | S32 r, g, b, a; | ||
1670 | |||
1671 | sscanf(valuestr,"%d %d %d %d", &r, &g, &b, &a); | ||
1672 | value.mV[0] = r; | ||
1673 | value.mV[1] = g; | ||
1674 | value.mV[2] = b; | ||
1675 | value.mV[3] = a; | ||
1676 | return success; | ||
1677 | } | ||
1678 | |||
1679 | |||
1680 | BOOL LLDataPackerAsciiFile::packVector2(const LLVector2 &value, const char *name) | ||
1681 | { | ||
1682 | BOOL success = TRUE; | ||
1683 | writeIndentedName(name); | ||
1684 | if (mFP) | ||
1685 | { | ||
1686 | fprintf(mFP,"%g %g\n", value.mV[0], value.mV[1]); | ||
1687 | } | ||
1688 | else if (mOutputStream) | ||
1689 | { | ||
1690 | *mOutputStream << value.mV[0] << " " << value.mV[1] << "\n"; | ||
1691 | } | ||
1692 | return success; | ||
1693 | } | ||
1694 | |||
1695 | |||
1696 | BOOL LLDataPackerAsciiFile::unpackVector2(LLVector2 &value, const char *name) | ||
1697 | { | ||
1698 | BOOL success = TRUE; | ||
1699 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1700 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1701 | { | ||
1702 | return FALSE; | ||
1703 | } | ||
1704 | |||
1705 | sscanf(valuestr,"%g %g", &value.mV[0], &value.mV[1]); | ||
1706 | return success; | ||
1707 | } | ||
1708 | |||
1709 | |||
1710 | BOOL LLDataPackerAsciiFile::packVector3(const LLVector3 &value, const char *name) | ||
1711 | { | ||
1712 | BOOL success = TRUE; | ||
1713 | writeIndentedName(name); | ||
1714 | if (mFP) | ||
1715 | { | ||
1716 | fprintf(mFP,"%g %g %g\n", value.mV[0], value.mV[1], value.mV[2]); | ||
1717 | } | ||
1718 | else if (mOutputStream) | ||
1719 | { | ||
1720 | *mOutputStream << value.mV[0] << " " << value.mV[1] << " " << value.mV[2] << "\n"; | ||
1721 | } | ||
1722 | return success; | ||
1723 | } | ||
1724 | |||
1725 | |||
1726 | BOOL LLDataPackerAsciiFile::unpackVector3(LLVector3 &value, const char *name) | ||
1727 | { | ||
1728 | BOOL success = TRUE; | ||
1729 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1730 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1731 | { | ||
1732 | return FALSE; | ||
1733 | } | ||
1734 | |||
1735 | sscanf(valuestr,"%g %g %g", &value.mV[0], &value.mV[1], &value.mV[2]); | ||
1736 | return success; | ||
1737 | } | ||
1738 | |||
1739 | BOOL LLDataPackerAsciiFile::packVector4(const LLVector4 &value, const char *name) | ||
1740 | { | ||
1741 | BOOL success = TRUE; | ||
1742 | writeIndentedName(name); | ||
1743 | if (mFP) | ||
1744 | { | ||
1745 | fprintf(mFP,"%g %g %g %g\n", value.mV[0], value.mV[1], value.mV[2], value.mV[3]); | ||
1746 | } | ||
1747 | else if (mOutputStream) | ||
1748 | { | ||
1749 | *mOutputStream << value.mV[0] << " " << value.mV[1] << " " << value.mV[2] << " " << value.mV[3] << "\n"; | ||
1750 | } | ||
1751 | return success; | ||
1752 | } | ||
1753 | |||
1754 | |||
1755 | BOOL LLDataPackerAsciiFile::unpackVector4(LLVector4 &value, const char *name) | ||
1756 | { | ||
1757 | BOOL success = TRUE; | ||
1758 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1759 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1760 | { | ||
1761 | return FALSE; | ||
1762 | } | ||
1763 | |||
1764 | sscanf(valuestr,"%g %g %g %g", &value.mV[0], &value.mV[1], &value.mV[2], &value.mV[3]); | ||
1765 | return success; | ||
1766 | } | ||
1767 | |||
1768 | |||
1769 | BOOL LLDataPackerAsciiFile::packUUID(const LLUUID &value, const char *name) | ||
1770 | { | ||
1771 | BOOL success = TRUE; | ||
1772 | writeIndentedName(name); | ||
1773 | char tmp_str[64]; /*Flawfinder: ignore */ | ||
1774 | value.toString(tmp_str); | ||
1775 | if (mFP) | ||
1776 | { | ||
1777 | fprintf(mFP,"%s\n", tmp_str); | ||
1778 | } | ||
1779 | else if (mOutputStream) | ||
1780 | { | ||
1781 | *mOutputStream <<"" << tmp_str << "\n"; | ||
1782 | } | ||
1783 | return success; | ||
1784 | } | ||
1785 | |||
1786 | |||
1787 | BOOL LLDataPackerAsciiFile::unpackUUID(LLUUID &value, const char *name) | ||
1788 | { | ||
1789 | BOOL success = TRUE; | ||
1790 | char valuestr[DP_BUFSIZE]; /*Flawfinder: ignore */ | ||
1791 | if (!getValueStr(name, valuestr, DP_BUFSIZE)) | ||
1792 | { | ||
1793 | return FALSE; | ||
1794 | } | ||
1795 | |||
1796 | char tmp_str[64]; /*Flawfinder: ignore */ | ||
1797 | sscanf(valuestr,"%63s",tmp_str); | ||
1798 | value.set(tmp_str); | ||
1799 | |||
1800 | return success; | ||
1801 | } | ||
1802 | |||
1803 | |||
1804 | void LLDataPackerAsciiFile::writeIndentedName(const char *name) | ||
1805 | { | ||
1806 | char indent_buf[64]; /*Flawfinder: ignore*/ | ||
1807 | |||
1808 | S32 i; | ||
1809 | for(i = 0; i < mIndent; i++) | ||
1810 | { | ||
1811 | indent_buf[i] = '\t'; | ||
1812 | } | ||
1813 | indent_buf[i] = 0; | ||
1814 | if (mFP) | ||
1815 | { | ||
1816 | fprintf(mFP,"%s%s\t",indent_buf, name); | ||
1817 | } | ||
1818 | else if (mOutputStream) | ||
1819 | { | ||
1820 | *mOutputStream << indent_buf << name << "\t"; | ||
1821 | } | ||
1822 | } | ||
1823 | |||
1824 | BOOL LLDataPackerAsciiFile::getValueStr(const char *name, char *out_value, S32 value_len) | ||
1825 | { | ||
1826 | BOOL success = FALSE; | ||
1827 | char buffer[DP_BUFSIZE]; /*Flawfinder: ignore*/ | ||
1828 | char keyword[DP_BUFSIZE]; /*Flawfinder: ignore*/ | ||
1829 | char value[DP_BUFSIZE]; /*Flawfinder: ignore*/ | ||
1830 | |||
1831 | buffer[0] = '\0'; | ||
1832 | keyword[0] = '\0'; | ||
1833 | value[0] = '\0'; | ||
1834 | |||
1835 | if (mFP) | ||
1836 | { | ||
1837 | fpos_t last_pos; | ||
1838 | fgetpos(mFP, &last_pos); | ||
1839 | fgets(buffer, DP_BUFSIZE, mFP); | ||
1840 | |||
1841 | sscanf(buffer, "%511s %511[^\n]", keyword, value); | ||
1842 | |||
1843 | if (!keyword[0]) | ||
1844 | { | ||
1845 | llwarns << "Data packer could not get the keyword!" << llendl; | ||
1846 | fsetpos(mFP, &last_pos); | ||
1847 | return FALSE; | ||
1848 | } | ||
1849 | if (strcmp(keyword, name)) | ||
1850 | { | ||
1851 | llwarns << "Data packer expecting keyword of type " << name << ", got " << keyword << " instead!" << llendl; | ||
1852 | fsetpos(mFP, &last_pos); | ||
1853 | return FALSE; | ||
1854 | } | ||
1855 | |||
1856 | S32 in_value_len = (S32)strlen(value)+1; /*Flawfinder: ignore*/ | ||
1857 | S32 min_len = llmin(in_value_len, value_len); | ||
1858 | memcpy(out_value, value, min_len); /*Flawfinder: ignore*/ | ||
1859 | out_value[min_len-1] = 0; | ||
1860 | success = TRUE; | ||
1861 | } | ||
1862 | else if (mInputStream) | ||
1863 | { | ||
1864 | mInputStream->getline(buffer, DP_BUFSIZE); | ||
1865 | |||
1866 | sscanf(buffer, "%511s %511[^\n]", keyword, value); | ||
1867 | if (!keyword[0]) | ||
1868 | { | ||
1869 | llwarns << "Data packer could not get the keyword!" << llendl; | ||
1870 | return FALSE; | ||
1871 | } | ||
1872 | if (strcmp(keyword, name)) | ||
1873 | { | ||
1874 | llwarns << "Data packer expecting keyword of type " << name << ", got " << keyword << " instead!" << llendl; | ||
1875 | return FALSE; | ||
1876 | } | ||
1877 | |||
1878 | S32 in_value_len = (S32)strlen(value)+1; /*Flawfinder: ignore*/ | ||
1879 | S32 min_len = llmin(in_value_len, value_len); | ||
1880 | memcpy(out_value, value, min_len); /*Flawfinder: ignore*/ | ||
1881 | out_value[min_len-1] = 0; | ||
1882 | success = TRUE; | ||
1883 | } | ||
1884 | |||
1885 | return success; | ||
1886 | } | ||