1 /******************************************************************************
2 *
3 * Copyright (C) 2010-2013 Broadcom Corporation
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at:
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 ******************************************************************************/
18
19
20 /******************************************************************************
21 *
22 * This file contains source code for some utility functions to help parse
23 * and build NFC Data Exchange Format (NDEF) messages
24 *
25 ******************************************************************************/
26 #include <string.h>
27 #include "ndef_utils.h"
28
29 /*******************************************************************************
30 **
31 ** Static Local Functions
32 **
33 *******************************************************************************/
34
35
36 /*******************************************************************************
37 **
38 ** Function shiftdown
39 **
40 ** Description shift memory down (to make space to insert a record)
41 **
42 *******************************************************************************/
shiftdown(UINT8 * p_mem,UINT32 len,UINT32 shift_amount)43 static void shiftdown (UINT8 *p_mem, UINT32 len, UINT32 shift_amount)
44 {
45 register UINT8 *ps = p_mem + len - 1;
46 register UINT8 *pd = ps + shift_amount;
47 register UINT32 xx;
48
49 for (xx = 0; xx < len; xx++)
50 *pd-- = *ps--;
51 }
52
53 /*******************************************************************************
54 **
55 ** Function shiftup
56 **
57 ** Description shift memory up (to delete a record)
58 **
59 *******************************************************************************/
shiftup(UINT8 * p_dest,UINT8 * p_src,UINT32 len)60 static void shiftup (UINT8 *p_dest, UINT8 *p_src, UINT32 len)
61 {
62 register UINT8 *ps = p_src;
63 register UINT8 *pd = p_dest;
64 register UINT32 xx;
65
66 for (xx = 0; xx < len; xx++)
67 *pd++ = *ps++;
68 }
69
70 /*******************************************************************************
71 **
72 ** Function NDEF_MsgValidate
73 **
74 ** Description This function validates an NDEF message.
75 **
76 ** Returns TRUE if all OK, or FALSE if the message is invalid.
77 **
78 *******************************************************************************/
NDEF_MsgValidate(UINT8 * p_msg,UINT32 msg_len,BOOLEAN b_allow_chunks)79 tNDEF_STATUS NDEF_MsgValidate (UINT8 *p_msg, UINT32 msg_len, BOOLEAN b_allow_chunks)
80 {
81 UINT8 *p_rec = p_msg;
82 UINT8 *p_end = p_msg + msg_len;
83 UINT8 rec_hdr=0, type_len, id_len;
84 int count;
85 UINT32 payload_len;
86 BOOLEAN bInChunk = FALSE;
87
88 if ( (p_msg == NULL) || (msg_len < 3) )
89 return (NDEF_MSG_TOO_SHORT);
90
91 /* The first record must have the MB bit set */
92 if ((*p_msg & NDEF_MB_MASK) == 0)
93 return (NDEF_MSG_NO_MSG_BEGIN);
94
95 /* The first record cannot be a chunk */
96 if ((*p_msg & NDEF_TNF_MASK) == NDEF_TNF_UNCHANGED)
97 return (NDEF_MSG_UNEXPECTED_CHUNK);
98
99 for (count = 0; p_rec < p_end; count++)
100 {
101 /* if less than short record header */
102 if (p_rec + 3 > p_end)
103 return (NDEF_MSG_TOO_SHORT);
104
105 rec_hdr = *p_rec++;
106
107 /* The second and all subsequent records must NOT have the MB bit set */
108 if ( (count > 0) && (rec_hdr & NDEF_MB_MASK) )
109 return (NDEF_MSG_EXTRA_MSG_BEGIN);
110
111 /* Type field length */
112 type_len = *p_rec++;
113
114 /* Payload length - can be 1 or 4 bytes */
115 if (rec_hdr & NDEF_SR_MASK)
116 payload_len = *p_rec++;
117 else
118 {
119 /* if less than 4 bytes payload length */
120 if (p_rec + 4 > p_end)
121 return (NDEF_MSG_TOO_SHORT);
122
123 BE_STREAM_TO_UINT32 (payload_len, p_rec);
124 }
125
126 /* ID field Length */
127 if (rec_hdr & NDEF_IL_MASK)
128 {
129 /* if less than 1 byte ID field length */
130 if (p_rec + 1 > p_end)
131 return (NDEF_MSG_TOO_SHORT);
132
133 id_len = *p_rec++;
134 }
135 else
136 id_len = 0;
137
138 /* A chunk must have type "unchanged", and no type or ID fields */
139 if (rec_hdr & NDEF_CF_MASK)
140 {
141 if (!b_allow_chunks)
142 return (NDEF_MSG_UNEXPECTED_CHUNK);
143
144 /* Inside a chunk, the type must be unchanged and no type or ID field i sallowed */
145 if (bInChunk)
146 {
147 if ( (type_len != 0) || (id_len != 0) || ((rec_hdr & NDEF_TNF_MASK) != NDEF_TNF_UNCHANGED) )
148 return (NDEF_MSG_INVALID_CHUNK);
149 }
150 else
151 {
152 /* First record of a chunk must NOT have type "unchanged" */
153 if ((rec_hdr & NDEF_TNF_MASK) == NDEF_TNF_UNCHANGED)
154 return (NDEF_MSG_INVALID_CHUNK);
155
156 bInChunk = TRUE;
157 }
158 }
159 else
160 {
161 /* This may be the last guy in a chunk. */
162 if (bInChunk)
163 {
164 if ( (type_len != 0) || (id_len != 0) || ((rec_hdr & NDEF_TNF_MASK) != NDEF_TNF_UNCHANGED) )
165 return (NDEF_MSG_INVALID_CHUNK);
166
167 bInChunk = FALSE;
168 }
169 else
170 {
171 /* If not in a chunk, the record must NOT have type "unchanged" */
172 if ((rec_hdr & NDEF_TNF_MASK) == NDEF_TNF_UNCHANGED)
173 return (NDEF_MSG_INVALID_CHUNK);
174 }
175 }
176
177 /* An empty record must NOT have a type, ID or payload */
178 if ((rec_hdr & NDEF_TNF_MASK) == NDEF_TNF_EMPTY)
179 {
180 if ( (type_len != 0) || (id_len != 0) || (payload_len != 0) )
181 return (NDEF_MSG_INVALID_EMPTY_REC);
182 }
183
184 if ((rec_hdr & NDEF_TNF_MASK) == NDEF_TNF_UNKNOWN)
185 {
186 if (type_len != 0)
187 return (NDEF_MSG_LENGTH_MISMATCH);
188 }
189
190 /* Point to next record */
191 p_rec += (payload_len + type_len + id_len);
192
193 if (rec_hdr & NDEF_ME_MASK)
194 break;
195
196 rec_hdr = 0;
197 }
198
199 /* The last record should have the ME bit set */
200 if ((rec_hdr & NDEF_ME_MASK) == 0)
201 return (NDEF_MSG_NO_MSG_END);
202
203 /* p_rec should equal p_end if all the length fields were correct */
204 if (p_rec != p_end)
205 return (NDEF_MSG_LENGTH_MISMATCH);
206
207 return (NDEF_OK);
208 }
209
210 /*******************************************************************************
211 **
212 ** Function NDEF_MsgGetNumRecs
213 **
214 ** Description This function gets the number of records in the given NDEF
215 ** message.
216 **
217 ** Returns The record count, or 0 if the message is invalid.
218 **
219 *******************************************************************************/
NDEF_MsgGetNumRecs(UINT8 * p_msg)220 INT32 NDEF_MsgGetNumRecs (UINT8 *p_msg)
221 {
222 UINT8 *p_rec = p_msg;
223 UINT8 rec_hdr, type_len, id_len;
224 int count;
225 UINT32 payload_len;
226
227 for (count = 0; ; )
228 {
229 count++;
230
231 rec_hdr = *p_rec++;
232
233 if (rec_hdr & NDEF_ME_MASK)
234 break;
235
236 /* Type field length */
237 type_len = *p_rec++;
238
239 /* Payload length - can be 1 or 4 bytes */
240 if (rec_hdr & NDEF_SR_MASK)
241 payload_len = *p_rec++;
242 else
243 BE_STREAM_TO_UINT32 (payload_len, p_rec);
244
245 /* ID field Length */
246 if (rec_hdr & NDEF_IL_MASK)
247 id_len = *p_rec++;
248 else
249 id_len = 0;
250
251 /* Point to next record */
252 p_rec += (payload_len + type_len + id_len);
253 }
254
255 /* Return the number of records found */
256 return (count);
257 }
258
259 /*******************************************************************************
260 **
261 ** Function NDEF_MsgGetRecLength
262 **
263 ** Description This function returns length of the current record in the given
264 ** NDEF message.
265 **
266 ** Returns Length of record
267 **
268 *******************************************************************************/
NDEF_MsgGetRecLength(UINT8 * p_cur_rec)269 UINT32 NDEF_MsgGetRecLength (UINT8 *p_cur_rec)
270 {
271 UINT8 rec_hdr, type_len, id_len;
272 UINT32 rec_len = 0;
273 UINT32 payload_len;
274
275 /* Get the current record's header */
276 rec_hdr = *p_cur_rec++;
277 rec_len++;
278
279 /* Type field length */
280 type_len = *p_cur_rec++;
281 rec_len++;
282
283 /* Payload length - can be 1 or 4 bytes */
284 if (rec_hdr & NDEF_SR_MASK)
285 {
286 payload_len = *p_cur_rec++;
287 rec_len++;
288 }
289 else
290 {
291 BE_STREAM_TO_UINT32 (payload_len, p_cur_rec);
292 rec_len += 4;
293 }
294
295 /* ID field Length */
296 if (rec_hdr & NDEF_IL_MASK)
297 {
298 id_len = *p_cur_rec++;
299 rec_len++;
300 }
301 else
302 id_len = 0;
303
304 /* Total length of record */
305 rec_len += (payload_len + type_len + id_len);
306
307 return (rec_len);
308 }
309
310 /*******************************************************************************
311 **
312 ** Function NDEF_MsgGetNextRec
313 **
314 ** Description This function gets a pointer to the next record in the given
315 ** NDEF message. If the current record pointer is NULL, a pointer
316 ** to the first record is returned.
317 **
318 ** Returns Pointer to the start of the record, or NULL if no more
319 **
320 *******************************************************************************/
NDEF_MsgGetNextRec(UINT8 * p_cur_rec)321 UINT8 *NDEF_MsgGetNextRec (UINT8 *p_cur_rec)
322 {
323 UINT8 rec_hdr, type_len, id_len;
324 UINT32 payload_len;
325
326 /* Get the current record's header */
327 rec_hdr = *p_cur_rec++;
328
329 /* If this is the last record, return NULL */
330 if (rec_hdr & NDEF_ME_MASK)
331 return (NULL);
332
333 /* Type field length */
334 type_len = *p_cur_rec++;
335
336 /* Payload length - can be 1 or 4 bytes */
337 if (rec_hdr & NDEF_SR_MASK)
338 payload_len = *p_cur_rec++;
339 else
340 BE_STREAM_TO_UINT32 (payload_len, p_cur_rec);
341
342 /* ID field Length */
343 if (rec_hdr & NDEF_IL_MASK)
344 id_len = *p_cur_rec++;
345 else
346 id_len = 0;
347
348 /* Point to next record */
349 p_cur_rec += (payload_len + type_len + id_len);
350
351 return (p_cur_rec);
352 }
353
354 /*******************************************************************************
355 **
356 ** Function NDEF_MsgGetRecByIndex
357 **
358 ** Description This function gets a pointer to the record with the given
359 ** index (0-based index) in the given NDEF message.
360 **
361 ** Returns Pointer to the start of the record, or NULL
362 **
363 *******************************************************************************/
NDEF_MsgGetRecByIndex(UINT8 * p_msg,INT32 index)364 UINT8 *NDEF_MsgGetRecByIndex (UINT8 *p_msg, INT32 index)
365 {
366 UINT8 *p_rec = p_msg;
367 UINT8 rec_hdr, type_len, id_len;
368 INT32 count;
369 UINT32 payload_len;
370
371 for (count = 0; ; count++)
372 {
373 if (count == index)
374 return (p_rec);
375
376 rec_hdr = *p_rec++;
377
378 if (rec_hdr & NDEF_ME_MASK)
379 return (NULL);
380
381 /* Type field length */
382 type_len = *p_rec++;
383
384 /* Payload length - can be 1 or 4 bytes */
385 if (rec_hdr & NDEF_SR_MASK)
386 payload_len = *p_rec++;
387 else
388 BE_STREAM_TO_UINT32 (payload_len, p_rec);
389
390 /* ID field Length */
391 if (rec_hdr & NDEF_IL_MASK)
392 id_len = *p_rec++;
393 else
394 id_len = 0;
395
396 /* Point to next record */
397 p_rec += (payload_len + type_len + id_len);
398 }
399
400 /* If here, there is no record of that index */
401 return (NULL);
402 }
403
404
405 /*******************************************************************************
406 **
407 ** Function NDEF_MsgGetLastRecInMsg
408 **
409 ** Description This function gets a pointer to the last record in the
410 ** given NDEF message.
411 **
412 ** Returns Pointer to the start of the last record, or NULL if some problem
413 **
414 *******************************************************************************/
NDEF_MsgGetLastRecInMsg(UINT8 * p_msg)415 UINT8 *NDEF_MsgGetLastRecInMsg (UINT8 *p_msg)
416 {
417 UINT8 *p_rec = p_msg;
418 UINT8 *pRecStart;
419 UINT8 rec_hdr, type_len, id_len;
420 UINT32 payload_len;
421
422 for ( ; ; )
423 {
424 pRecStart = p_rec;
425 rec_hdr = *p_rec++;
426
427 if (rec_hdr & NDEF_ME_MASK)
428 break;
429
430 /* Type field length */
431 type_len = *p_rec++;
432
433 /* Payload length - can be 1 or 4 bytes */
434 if (rec_hdr & NDEF_SR_MASK)
435 payload_len = *p_rec++;
436 else
437 BE_STREAM_TO_UINT32 (payload_len, p_rec);
438
439 /* ID field Length */
440 if (rec_hdr & NDEF_IL_MASK)
441 id_len = *p_rec++;
442 else
443 id_len = 0;
444
445 /* Point to next record */
446 p_rec += (payload_len + type_len + id_len);
447 }
448
449 return (pRecStart);
450 }
451
452
453 /*******************************************************************************
454 **
455 ** Function NDEF_MsgGetFirstRecByType
456 **
457 ** Description This function gets a pointer to the first record with the given
458 ** record type in the given NDEF message.
459 **
460 ** Returns Pointer to the start of the record, or NULL
461 **
462 *******************************************************************************/
NDEF_MsgGetFirstRecByType(UINT8 * p_msg,UINT8 tnf,UINT8 * p_type,UINT8 tlen)463 UINT8 *NDEF_MsgGetFirstRecByType (UINT8 *p_msg, UINT8 tnf, UINT8 *p_type, UINT8 tlen)
464 {
465 UINT8 *p_rec = p_msg;
466 UINT8 *pRecStart;
467 UINT8 rec_hdr, type_len, id_len;
468 UINT32 payload_len;
469
470 for ( ; ; )
471 {
472 pRecStart = p_rec;
473
474 rec_hdr = *p_rec++;
475
476 /* Type field length */
477 type_len = *p_rec++;
478
479 /* Payload length - can be 1 or 4 bytes */
480 if (rec_hdr & NDEF_SR_MASK)
481 payload_len = *p_rec++;
482 else
483 BE_STREAM_TO_UINT32 (payload_len, p_rec);
484
485 /* ID field Length */
486 if (rec_hdr & NDEF_IL_MASK)
487 id_len = *p_rec++;
488 else
489 id_len = 0;
490
491 /* At this point, p_rec points to the start of the type field. We need to */
492 /* compare the type of the type, the length of the type and the data */
493 if ( ((rec_hdr & NDEF_TNF_MASK) == tnf)
494 && (type_len == tlen)
495 && (!memcmp (p_rec, p_type, tlen)) )
496 return (pRecStart);
497
498 /* If this was the last record, return NULL */
499 if (rec_hdr & NDEF_ME_MASK)
500 return (NULL);
501
502 /* Point to next record */
503 p_rec += (payload_len + type_len + id_len);
504 }
505
506 /* If here, there is no record of that type */
507 return (NULL);
508 }
509
510 /*******************************************************************************
511 **
512 ** Function NDEF_MsgGetNextRecByType
513 **
514 ** Description This function gets a pointer to the next record with the given
515 ** record type in the given NDEF message.
516 **
517 ** Returns Pointer to the start of the record, or NULL
518 **
519 *******************************************************************************/
NDEF_MsgGetNextRecByType(UINT8 * p_cur_rec,UINT8 tnf,UINT8 * p_type,UINT8 tlen)520 UINT8 *NDEF_MsgGetNextRecByType (UINT8 *p_cur_rec, UINT8 tnf, UINT8 *p_type, UINT8 tlen)
521 {
522 UINT8 *p_rec;
523 UINT8 *pRecStart;
524 UINT8 rec_hdr, type_len, id_len;
525 UINT32 payload_len;
526
527 /* If this is the last record in the message, return NULL */
528 if ((p_rec = NDEF_MsgGetNextRec (p_cur_rec)) == NULL)
529 return (NULL);
530
531 for ( ; ; )
532 {
533 pRecStart = p_rec;
534
535 rec_hdr = *p_rec++;
536
537 /* Type field length */
538 type_len = *p_rec++;
539
540 /* Payload length - can be 1 or 4 bytes */
541 if (rec_hdr & NDEF_SR_MASK)
542 payload_len = *p_rec++;
543 else
544 BE_STREAM_TO_UINT32 (payload_len, p_rec);
545
546 /* ID field Length */
547 if (rec_hdr & NDEF_IL_MASK)
548 id_len = *p_rec++;
549 else
550 id_len = 0;
551
552 /* At this point, p_rec points to the start of the type field. We need to */
553 /* compare the type of the type, the length of the type and the data */
554 if ( ((rec_hdr & NDEF_TNF_MASK) == tnf)
555 && (type_len == tlen)
556 && (!memcmp (p_rec, p_type, tlen)) )
557 return (pRecStart);
558
559 /* If this was the last record, return NULL */
560 if (rec_hdr & NDEF_ME_MASK)
561 break;
562
563 /* Point to next record */
564 p_rec += (payload_len + type_len + id_len);
565 }
566
567 /* If here, there is no record of that type */
568 return (NULL);
569 }
570
571
572 /*******************************************************************************
573 **
574 ** Function NDEF_MsgGetFirstRecById
575 **
576 ** Description This function gets a pointer to the first record with the given
577 ** record id in the given NDEF message.
578 **
579 ** Returns Pointer to the start of the record, or NULL
580 **
581 *******************************************************************************/
NDEF_MsgGetFirstRecById(UINT8 * p_msg,UINT8 * p_id,UINT8 ilen)582 UINT8 *NDEF_MsgGetFirstRecById (UINT8 *p_msg, UINT8 *p_id, UINT8 ilen)
583 {
584 UINT8 *p_rec = p_msg;
585 UINT8 *pRecStart;
586 UINT8 rec_hdr, type_len, id_len;
587 UINT32 payload_len;
588
589 for ( ; ; )
590 {
591 pRecStart = p_rec;
592
593 rec_hdr = *p_rec++;
594
595 /* Type field length */
596 type_len = *p_rec++;
597
598 /* Payload length - can be 1 or 4 bytes */
599 if (rec_hdr & NDEF_SR_MASK)
600 payload_len = *p_rec++;
601 else
602 BE_STREAM_TO_UINT32 (payload_len, p_rec);
603
604 /* ID field Length */
605 if (rec_hdr & NDEF_IL_MASK)
606 id_len = *p_rec++;
607 else
608 id_len = 0;
609
610 /* At this point, p_rec points to the start of the type field. Skip it */
611 p_rec += type_len;
612
613 /* At this point, p_rec points to the start of the ID field. Compare length and data */
614 if ( (id_len == ilen) && (!memcmp (p_rec, p_id, ilen)) )
615 return (pRecStart);
616
617 /* If this was the last record, return NULL */
618 if (rec_hdr & NDEF_ME_MASK)
619 return (NULL);
620
621 /* Point to next record */
622 p_rec += (id_len + payload_len);
623 }
624
625 /* If here, there is no record of that ID */
626 return (NULL);
627 }
628
629 /*******************************************************************************
630 **
631 ** Function NDEF_MsgGetNextRecById
632 **
633 ** Description This function gets a pointer to the next record with the given
634 ** record id in the given NDEF message.
635 **
636 ** Returns Pointer to the start of the record, or NULL
637 **
638 *******************************************************************************/
NDEF_MsgGetNextRecById(UINT8 * p_cur_rec,UINT8 * p_id,UINT8 ilen)639 UINT8 *NDEF_MsgGetNextRecById (UINT8 *p_cur_rec, UINT8 *p_id, UINT8 ilen)
640 {
641 UINT8 *p_rec;
642 UINT8 *pRecStart;
643 UINT8 rec_hdr, type_len, id_len;
644 UINT32 payload_len;
645
646 /* If this is the last record in the message, return NULL */
647 if ((p_rec = NDEF_MsgGetNextRec (p_cur_rec)) == NULL)
648 return (NULL);
649
650 for ( ; ; )
651 {
652 pRecStart = p_rec;
653
654 rec_hdr = *p_rec++;
655
656 /* Type field length */
657 type_len = *p_rec++;
658
659 /* Payload length - can be 1 or 4 bytes */
660 if (rec_hdr & NDEF_SR_MASK)
661 payload_len = *p_rec++;
662 else
663 BE_STREAM_TO_UINT32 (payload_len, p_rec);
664
665 /* ID field Length */
666 if (rec_hdr & NDEF_IL_MASK)
667 id_len = *p_rec++;
668 else
669 id_len = 0;
670
671 /* At this point, p_rec points to the start of the type field. Skip it */
672 p_rec += type_len;
673
674 /* At this point, p_rec points to the start of the ID field. Compare length and data */
675 if ( (id_len == ilen) && (!memcmp (p_rec, p_id, ilen)) )
676 return (pRecStart);
677
678 /* If this was the last record, return NULL */
679 if (rec_hdr & NDEF_ME_MASK)
680 break;
681
682 /* Point to next record */
683 p_rec += (id_len + payload_len);
684 }
685
686 /* If here, there is no record of that ID */
687 return (NULL);
688 }
689
690 /*******************************************************************************
691 **
692 ** Function NDEF_RecGetType
693 **
694 ** Description This function gets a pointer to the record type for the given NDEF record.
695 **
696 ** Returns Pointer to Type (NULL if none). TNF and len are filled in.
697 **
698 *******************************************************************************/
NDEF_RecGetType(UINT8 * p_rec,UINT8 * p_tnf,UINT8 * p_type_len)699 UINT8 *NDEF_RecGetType (UINT8 *p_rec, UINT8 *p_tnf, UINT8 *p_type_len)
700 {
701 UINT8 rec_hdr, type_len;
702
703 /* First byte is the record header */
704 rec_hdr = *p_rec++;
705
706 /* Next byte is the type field length */
707 type_len = *p_rec++;
708
709 /* Skip the payload length */
710 if (rec_hdr & NDEF_SR_MASK)
711 p_rec += 1;
712 else
713 p_rec += 4;
714
715 /* Skip ID field Length, if present */
716 if (rec_hdr & NDEF_IL_MASK)
717 p_rec++;
718
719 /* At this point, p_rec points to the start of the type field. */
720 *p_type_len = type_len;
721 *p_tnf = rec_hdr & NDEF_TNF_MASK;
722
723 if (type_len == 0)
724 return (NULL);
725 else
726 return (p_rec);
727 }
728
729 /*******************************************************************************
730 **
731 ** Function NDEF_RecGetId
732 **
733 ** Description This function gets a pointer to the record id for the given NDEF record.
734 **
735 ** Returns Pointer to Id (NULL if none). ID Len is filled in.
736 **
737 *******************************************************************************/
NDEF_RecGetId(UINT8 * p_rec,UINT8 * p_id_len)738 UINT8 *NDEF_RecGetId (UINT8 *p_rec, UINT8 *p_id_len)
739 {
740 UINT8 rec_hdr, type_len;
741
742 /* First byte is the record header */
743 rec_hdr = *p_rec++;
744
745 /* Next byte is the type field length */
746 type_len = *p_rec++;
747
748 /* Skip the payload length */
749 if (rec_hdr & NDEF_SR_MASK)
750 p_rec++;
751 else
752 p_rec += 4;
753
754 /* ID field Length */
755 if (rec_hdr & NDEF_IL_MASK)
756 *p_id_len = *p_rec++;
757 else
758 *p_id_len = 0;
759
760 /* p_rec now points to the start of the type field. The ID field follows it */
761 if (*p_id_len == 0)
762 return (NULL);
763 else
764 return (p_rec + type_len);
765 }
766
767
768 /*******************************************************************************
769 **
770 ** Function NDEF_RecGetPayload
771 **
772 ** Description This function gets a pointer to the payload for the given NDEF record.
773 **
774 ** Returns a pointer to the payload (or NULL none). Payload len filled in.
775 **
776 *******************************************************************************/
NDEF_RecGetPayload(UINT8 * p_rec,UINT32 * p_payload_len)777 UINT8 *NDEF_RecGetPayload (UINT8 *p_rec, UINT32 *p_payload_len)
778 {
779 UINT8 rec_hdr, type_len, id_len;
780 UINT32 payload_len;
781
782 /* First byte is the record header */
783 rec_hdr = *p_rec++;
784
785 /* Next byte is the type field length */
786 type_len = *p_rec++;
787
788 /* Next is the payload length (1 or 4 bytes) */
789 if (rec_hdr & NDEF_SR_MASK)
790 payload_len = *p_rec++;
791 else
792 BE_STREAM_TO_UINT32 (payload_len, p_rec);
793
794 *p_payload_len = payload_len;
795
796 /* ID field Length */
797 if (rec_hdr & NDEF_IL_MASK)
798 id_len = *p_rec++;
799 else
800 id_len = 0;
801
802 /* p_rec now points to the start of the type field. The ID field follows it, then the payload */
803 if (payload_len == 0)
804 return (NULL);
805 else
806 return (p_rec + type_len + id_len);
807 }
808
809
810 /*******************************************************************************
811 **
812 ** Function NDEF_MsgInit
813 **
814 ** Description This function initializes an NDEF message.
815 **
816 ** Returns void
817 ** *p_cur_size is initialized to 0
818 **
819 *******************************************************************************/
NDEF_MsgInit(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size)820 void NDEF_MsgInit (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size)
821 {
822 *p_cur_size = 0;
823 memset (p_msg, 0, max_size);
824 }
825
826 /*******************************************************************************
827 **
828 ** Function NDEF_MsgAddRec
829 **
830 ** Description This function adds an NDEF record to the end of an NDEF message.
831 **
832 ** Returns OK, or error if the record did not fit
833 ** *p_cur_size is updated
834 **
835 *******************************************************************************/
NDEF_MsgAddRec(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size,UINT8 tnf,UINT8 * p_type,UINT8 type_len,UINT8 * p_id,UINT8 id_len,UINT8 * p_payload,UINT32 payload_len)836 extern tNDEF_STATUS NDEF_MsgAddRec (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size,
837 UINT8 tnf, UINT8 *p_type, UINT8 type_len,
838 UINT8 *p_id, UINT8 id_len,
839 UINT8 *p_payload, UINT32 payload_len)
840 {
841 UINT8 *p_rec = p_msg + *p_cur_size;
842 UINT32 recSize;
843 int plen = (payload_len < 256) ? 1 : 4;
844 int ilen = (id_len == 0) ? 0 : 1;
845
846 if (tnf > NDEF_TNF_RESERVED)
847 {
848 tnf = NDEF_TNF_UNKNOWN;
849 type_len = 0;
850 }
851
852 /* First, make sure the record will fit. we need at least 2 bytes for header and type length */
853 recSize = payload_len + 2 + type_len + plen + ilen + id_len;
854
855 if ((*p_cur_size + recSize) > max_size)
856 return (NDEF_MSG_INSUFFICIENT_MEM);
857
858 /* Construct the record header. For the first record, set both begin and end bits */
859 if (*p_cur_size == 0)
860 *p_rec = tnf | NDEF_MB_MASK | NDEF_ME_MASK;
861 else
862 {
863 /* Find the previous last and clear his 'Message End' bit */
864 UINT8 *pLast = NDEF_MsgGetLastRecInMsg (p_msg);
865
866 if (!pLast)
867 return (FALSE);
868
869 *pLast &= ~NDEF_ME_MASK;
870 *p_rec = tnf | NDEF_ME_MASK;
871 }
872
873 if (plen == 1)
874 *p_rec |= NDEF_SR_MASK;
875
876 if (ilen != 0)
877 *p_rec |= NDEF_IL_MASK;
878
879 p_rec++;
880
881 /* The next byte is the type field length */
882 *p_rec++ = type_len;
883
884 /* Payload length - can be 1 or 4 bytes */
885 if (plen == 1)
886 *p_rec++ = (UINT8)payload_len;
887 else
888 UINT32_TO_BE_STREAM (p_rec, payload_len);
889
890 /* ID field Length (optional) */
891 if (ilen > 0)
892 *p_rec++ = id_len;
893
894 /* Next comes the type */
895 if (type_len)
896 {
897 if (p_type)
898 memcpy (p_rec, p_type, type_len);
899
900 p_rec += type_len;
901 }
902
903 /* Next comes the ID */
904 if (id_len)
905 {
906 if (p_id)
907 memcpy (p_rec, p_id, id_len);
908
909 p_rec += id_len;
910 }
911
912 /* And lastly the payload. If NULL, the app just wants to reserve memory */
913 if (p_payload)
914 memcpy (p_rec, p_payload, payload_len);
915
916 *p_cur_size += recSize;
917
918 return (NDEF_OK);
919 }
920
921 /*******************************************************************************
922 **
923 ** Function NDEF_MsgInsertRec
924 **
925 ** Description This function inserts a record at a specific index into the
926 ** given NDEF message
927 **
928 ** Returns OK, or error if the record did not fit
929 ** *p_cur_size is updated
930 **
931 *******************************************************************************/
NDEF_MsgInsertRec(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size,INT32 index,UINT8 tnf,UINT8 * p_type,UINT8 type_len,UINT8 * p_id,UINT8 id_len,UINT8 * p_payload,UINT32 payload_len)932 extern tNDEF_STATUS NDEF_MsgInsertRec (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size, INT32 index,
933 UINT8 tnf, UINT8 *p_type, UINT8 type_len,
934 UINT8 *p_id, UINT8 id_len,
935 UINT8 *p_payload, UINT32 payload_len)
936 {
937 UINT8 *p_rec;
938 UINT32 recSize;
939 INT32 plen = (payload_len < 256) ? 1 : 4;
940 INT32 ilen = (id_len == 0) ? 0 : 1;
941
942 /* First, make sure the record will fit. we need at least 2 bytes for header and type length */
943 recSize = payload_len + 2 + type_len + plen + ilen + id_len;
944
945 if ((*p_cur_size + recSize) > max_size)
946 return (NDEF_MSG_INSUFFICIENT_MEM);
947
948 /* See where the new record goes. If at the end, call the 'AddRec' function */
949 if ( (index >= NDEF_MsgGetNumRecs (p_msg))
950 || ((p_rec = NDEF_MsgGetRecByIndex(p_msg, index)) == NULL) )
951 {
952 return NDEF_MsgAddRec (p_msg, max_size, p_cur_size, tnf, p_type, type_len,
953 p_id, id_len, p_payload, payload_len);
954 }
955
956 /* If we are inserting at the beginning, remove the MB bit from the current first */
957 if (index == 0)
958 *p_msg &= ~NDEF_MB_MASK;
959
960 /* Make space for the new record */
961 shiftdown (p_rec, (UINT32)(*p_cur_size - (p_rec - p_msg)), recSize);
962
963 /* If adding at the beginning, set begin bit */
964 if (index == 0)
965 *p_rec = tnf | NDEF_MB_MASK;
966 else
967 *p_rec = tnf;
968
969 if (plen == 1)
970 *p_rec |= NDEF_SR_MASK;
971
972 if (ilen != 0)
973 *p_rec |= NDEF_IL_MASK;
974
975 p_rec++;
976
977 /* The next byte is the type field length */
978 *p_rec++ = type_len;
979
980 /* Payload length - can be 1 or 4 bytes */
981 if (plen == 1)
982 *p_rec++ = (UINT8)payload_len;
983 else
984 UINT32_TO_BE_STREAM (p_rec, payload_len);
985
986 /* ID field Length (optional) */
987 if (ilen != 0)
988 *p_rec++ = id_len;
989
990 /* Next comes the type */
991 if (type_len)
992 {
993 if (p_type)
994 memcpy (p_rec, p_type, type_len);
995
996 p_rec += type_len;
997 }
998
999 /* Next comes the ID */
1000 if (ilen != 0)
1001 {
1002 if (p_id)
1003 memcpy (p_rec, p_id, id_len);
1004
1005 p_rec += id_len;
1006 }
1007
1008 /* And lastly the payload. If NULL, the app just wants to reserve memory */
1009 if (p_payload)
1010 memcpy (p_rec, p_payload, payload_len);
1011
1012 *p_cur_size += recSize;
1013
1014 return (NDEF_OK);
1015 }
1016
1017 /*******************************************************************************
1018 **
1019 ** Function NDEF_MsgAppendRec
1020 **
1021 ** Description This function adds NDEF records to the end of an NDEF message.
1022 **
1023 ** Returns OK, or error if the record did not fit
1024 ** *p_cur_size is updated
1025 **
1026 *******************************************************************************/
NDEF_MsgAppendRec(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size,UINT8 * p_new_rec,UINT32 new_rec_len)1027 extern tNDEF_STATUS NDEF_MsgAppendRec (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size,
1028 UINT8 *p_new_rec, UINT32 new_rec_len)
1029 {
1030 UINT8 *p_rec;
1031 tNDEF_STATUS status;
1032
1033 /* First, validate new records */
1034 if ((status = NDEF_MsgValidate(p_new_rec, new_rec_len, FALSE)) != NDEF_OK)
1035 return (status);
1036
1037 /* First, make sure the record will fit */
1038 if ((*p_cur_size + new_rec_len) > max_size)
1039 return (NDEF_MSG_INSUFFICIENT_MEM);
1040
1041 /* Find where to copy new record */
1042 if (*p_cur_size == 0)
1043 p_rec = p_msg;
1044 else
1045 {
1046 /* Find the previous last and clear his 'Message End' bit */
1047 UINT8 *pLast = NDEF_MsgGetLastRecInMsg (p_msg);
1048
1049 if (!pLast)
1050 return (NDEF_MSG_NO_MSG_END);
1051
1052 *pLast &= ~NDEF_ME_MASK;
1053 p_rec = p_msg + *p_cur_size;
1054
1055 /* clear 'Message Begin' bit of new record */
1056 *p_new_rec &= ~NDEF_MB_MASK;
1057 }
1058
1059 /* append new records */
1060 memcpy (p_rec, p_new_rec, new_rec_len);
1061
1062 *p_cur_size += new_rec_len;
1063
1064 return (NDEF_OK);
1065 }
1066
1067 /*******************************************************************************
1068 **
1069 ** Function NDEF_MsgAppendPayload
1070 **
1071 ** Description This function appends extra payload to a specific record in the
1072 ** given NDEF message
1073 **
1074 ** Returns OK, or error if the extra payload did not fit
1075 ** *p_cur_size is updated
1076 **
1077 *******************************************************************************/
NDEF_MsgAppendPayload(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size,UINT8 * p_rec,UINT8 * p_add_pl,UINT32 add_pl_len)1078 tNDEF_STATUS NDEF_MsgAppendPayload (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size,
1079 UINT8 *p_rec, UINT8 *p_add_pl, UINT32 add_pl_len)
1080 {
1081 UINT32 prev_paylen, new_paylen;
1082 UINT8 *p_prev_pl, *pp;
1083 UINT8 incr_lenfld = 0;
1084 UINT8 type_len, id_len;
1085
1086 /* Skip header */
1087 pp = p_rec + 1;
1088
1089 /* Next byte is the type field length */
1090 type_len = *pp++;
1091
1092 /* Next is the payload length (1 or 4 bytes) */
1093 if (*p_rec & NDEF_SR_MASK)
1094 prev_paylen = *pp++;
1095 else
1096 BE_STREAM_TO_UINT32 (prev_paylen, pp);
1097
1098 /* ID field Length */
1099 if (*p_rec & NDEF_IL_MASK)
1100 id_len = *pp++;
1101 else
1102 id_len = 0;
1103
1104 p_prev_pl = pp + type_len + id_len;
1105
1106 new_paylen = prev_paylen + add_pl_len;
1107
1108 /* Previous payload may be < 256, and this addition may make it larger than 256 */
1109 /* If that were to happen, the payload length field goes from 1 byte to 4 bytes */
1110 if ( (prev_paylen < 256) && (new_paylen > 255) )
1111 incr_lenfld = 3;
1112
1113 /* Check that it all fits */
1114 if ((*p_cur_size + add_pl_len + incr_lenfld) > max_size)
1115 return (NDEF_MSG_INSUFFICIENT_MEM);
1116
1117 /* Point to payload length field */
1118 pp = p_rec + 2;
1119
1120 /* If we need to increase the length field from 1 to 4 bytes, do it first */
1121 if (incr_lenfld)
1122 {
1123 shiftdown (pp + 1, (UINT32)(*p_cur_size - (pp - p_msg) - 1), 3);
1124 p_prev_pl += 3;
1125 }
1126
1127 /* Store in the new length */
1128 if (new_paylen > 255)
1129 {
1130 *p_rec &= ~NDEF_SR_MASK;
1131 UINT32_TO_BE_STREAM (pp, new_paylen);
1132 }
1133 else
1134 *pp = (UINT8)new_paylen;
1135
1136 /* Point to the end of the previous payload */
1137 pp = p_prev_pl + prev_paylen;
1138
1139 /* If we are not the last record, make space for the extra payload */
1140 if ((*p_rec & NDEF_ME_MASK) == 0)
1141 shiftdown (pp, (UINT32)(*p_cur_size - (pp - p_msg)), add_pl_len);
1142
1143 /* Now copy in the additional payload data */
1144 memcpy (pp, p_add_pl, add_pl_len);
1145
1146 *p_cur_size += add_pl_len + incr_lenfld;
1147
1148 return (NDEF_OK);
1149 }
1150
1151 /*******************************************************************************
1152 **
1153 ** Function NDEF_MsgReplacePayload
1154 **
1155 ** Description This function replaces the payload of a specific record in the
1156 ** given NDEF message
1157 **
1158 ** Returns OK, or error if the new payload did not fit
1159 ** *p_cur_size is updated
1160 **
1161 *******************************************************************************/
NDEF_MsgReplacePayload(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size,UINT8 * p_rec,UINT8 * p_new_pl,UINT32 new_pl_len)1162 tNDEF_STATUS NDEF_MsgReplacePayload (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size,
1163 UINT8 *p_rec, UINT8 *p_new_pl, UINT32 new_pl_len)
1164 {
1165 UINT32 prev_paylen;
1166 UINT8 *p_prev_pl, *pp;
1167 UINT32 paylen_delta;
1168 UINT8 type_len, id_len;
1169
1170 /* Skip header */
1171 pp = p_rec + 1;
1172
1173 /* Next byte is the type field length */
1174 type_len = *pp++;
1175
1176 /* Next is the payload length (1 or 4 bytes) */
1177 if (*p_rec & NDEF_SR_MASK)
1178 prev_paylen = *pp++;
1179 else
1180 BE_STREAM_TO_UINT32 (prev_paylen, pp);
1181
1182 /* ID field Length */
1183 if (*p_rec & NDEF_IL_MASK)
1184 id_len = *pp++;
1185 else
1186 id_len = 0;
1187
1188 p_prev_pl = pp + type_len + id_len;
1189
1190 /* Point to payload length field again */
1191 pp = p_rec + 2;
1192
1193 if (new_pl_len > prev_paylen)
1194 {
1195 /* New payload is larger than the previous */
1196 paylen_delta = new_pl_len - prev_paylen;
1197
1198 /* If the previous payload length was < 256, and new is > 255 */
1199 /* the payload length field goes from 1 byte to 4 bytes */
1200 if ( (prev_paylen < 256) && (new_pl_len > 255) )
1201 {
1202 if ((*p_cur_size + paylen_delta + 3) > max_size)
1203 return (NDEF_MSG_INSUFFICIENT_MEM);
1204
1205 shiftdown (pp + 1, (UINT32)(*p_cur_size - (pp - p_msg) - 1), 3);
1206 p_prev_pl += 3;
1207 *p_cur_size += 3;
1208 *p_rec &= ~NDEF_SR_MASK;
1209 }
1210 else if ((*p_cur_size + paylen_delta) > max_size)
1211 return (NDEF_MSG_INSUFFICIENT_MEM);
1212
1213 /* Store in the new length */
1214 if (new_pl_len > 255)
1215 {
1216 UINT32_TO_BE_STREAM (pp, new_pl_len);
1217 }
1218 else
1219 *pp = (UINT8)new_pl_len;
1220
1221 /* Point to the end of the previous payload */
1222 pp = p_prev_pl + prev_paylen;
1223
1224 /* If we are not the last record, make space for the extra payload */
1225 if ((*p_rec & NDEF_ME_MASK) == 0)
1226 shiftdown (pp, (UINT32)(*p_cur_size - (pp - p_msg)), paylen_delta);
1227
1228 *p_cur_size += paylen_delta;
1229 }
1230 else if (new_pl_len < prev_paylen)
1231 {
1232 /* New payload is smaller than the previous */
1233 paylen_delta = prev_paylen - new_pl_len;
1234
1235 /* If the previous payload was > 256, and new is less than 256 */
1236 /* the payload length field goes from 4 bytes to 1 byte */
1237 if ( (prev_paylen > 255) && (new_pl_len < 256) )
1238 {
1239 shiftup (pp + 1, pp + 4, (UINT32)(*p_cur_size - (pp - p_msg) - 3));
1240 p_prev_pl -= 3;
1241 *p_cur_size -= 3;
1242 *p_rec |= NDEF_SR_MASK;
1243 }
1244
1245 /* Store in the new length */
1246 if (new_pl_len > 255)
1247 {
1248 UINT32_TO_BE_STREAM (pp, new_pl_len);
1249 }
1250 else
1251 *pp = (UINT8)new_pl_len;
1252
1253 /* Point to the end of the previous payload */
1254 pp = p_prev_pl + prev_paylen;
1255
1256 /* If we are not the last record, remove the extra space from the previous payload */
1257 if ((*p_rec & NDEF_ME_MASK) == 0)
1258 shiftup (pp - paylen_delta, pp, (UINT32)(*p_cur_size - (pp - p_msg)));
1259
1260 *p_cur_size -= paylen_delta;
1261 }
1262
1263 /* Now copy in the new payload data */
1264 if (p_new_pl)
1265 memcpy (p_prev_pl, p_new_pl, new_pl_len);
1266
1267 return (NDEF_OK);
1268 }
1269
1270 /*******************************************************************************
1271 **
1272 ** Function NDEF_MsgReplaceType
1273 **
1274 ** Description This function replaces the type field of a specific record in the
1275 ** given NDEF message
1276 **
1277 ** Returns OK, or error if the new type field did not fit
1278 ** *p_cur_size is updated
1279 **
1280 *******************************************************************************/
NDEF_MsgReplaceType(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size,UINT8 * p_rec,UINT8 * p_new_type,UINT8 new_type_len)1281 tNDEF_STATUS NDEF_MsgReplaceType (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size,
1282 UINT8 *p_rec, UINT8 *p_new_type, UINT8 new_type_len)
1283 {
1284 UINT8 typelen_delta;
1285 UINT8 *p_prev_type, prev_type_len;
1286 UINT8 *pp;
1287
1288 /* Skip header */
1289 pp = p_rec + 1;
1290
1291 /* Next byte is the type field length */
1292 prev_type_len = *pp++;
1293
1294 /* Skip the payload length */
1295 if (*p_rec & NDEF_SR_MASK)
1296 pp += 1;
1297 else
1298 pp += 4;
1299
1300 if (*p_rec & NDEF_IL_MASK)
1301 pp++;
1302
1303 /* Save pointer to the start of the type field */
1304 p_prev_type = pp;
1305
1306 if (new_type_len > prev_type_len)
1307 {
1308 /* New type is larger than the previous */
1309 typelen_delta = new_type_len - prev_type_len;
1310
1311 if ((*p_cur_size + typelen_delta) > max_size)
1312 return (NDEF_MSG_INSUFFICIENT_MEM);
1313
1314 /* Point to the end of the previous type, and make space for the extra data */
1315 pp = p_prev_type + prev_type_len;
1316 shiftdown (pp, (UINT32)(*p_cur_size - (pp - p_msg)), typelen_delta);
1317
1318 *p_cur_size += typelen_delta;
1319 }
1320 else if (new_type_len < prev_type_len)
1321 {
1322 /* New type field is smaller than the previous */
1323 typelen_delta = prev_type_len - new_type_len;
1324
1325 /* Point to the end of the previous type, and shift up to fill the the unused space */
1326 pp = p_prev_type + prev_type_len;
1327 shiftup (pp - typelen_delta, pp, (UINT32)(*p_cur_size - (pp - p_msg)));
1328
1329 *p_cur_size -= typelen_delta;
1330 }
1331
1332 /* Save in new type length */
1333 p_rec[1] = new_type_len;
1334
1335 /* Now copy in the new type field data */
1336 if (p_new_type)
1337 memcpy (p_prev_type, p_new_type, new_type_len);
1338
1339 return (NDEF_OK);
1340 }
1341
1342 /*******************************************************************************
1343 **
1344 ** Function NDEF_MsgReplaceId
1345 **
1346 ** Description This function replaces the ID field of a specific record in the
1347 ** given NDEF message
1348 **
1349 ** Returns OK, or error if the new ID field did not fit
1350 ** *p_cur_size is updated
1351 **
1352 *******************************************************************************/
NDEF_MsgReplaceId(UINT8 * p_msg,UINT32 max_size,UINT32 * p_cur_size,UINT8 * p_rec,UINT8 * p_new_id,UINT8 new_id_len)1353 tNDEF_STATUS NDEF_MsgReplaceId (UINT8 *p_msg, UINT32 max_size, UINT32 *p_cur_size,
1354 UINT8 *p_rec, UINT8 *p_new_id, UINT8 new_id_len)
1355 {
1356 UINT8 idlen_delta;
1357 UINT8 *p_prev_id, *p_idlen_field;
1358 UINT8 prev_id_len, type_len;
1359 UINT8 *pp;
1360
1361 /* Skip header */
1362 pp = p_rec + 1;
1363
1364 /* Next byte is the type field length */
1365 type_len = *pp++;
1366
1367 /* Skip the payload length */
1368 if (*p_rec & NDEF_SR_MASK)
1369 pp += 1;
1370 else
1371 pp += 4;
1372
1373 p_idlen_field = pp;
1374
1375 if (*p_rec & NDEF_IL_MASK)
1376 prev_id_len = *pp++;
1377 else
1378 prev_id_len = 0;
1379
1380 /* Save pointer to the start of the ID field (right after the type field) */
1381 p_prev_id = pp + type_len;
1382
1383 if (new_id_len > prev_id_len)
1384 {
1385 /* New ID field is larger than the previous */
1386 idlen_delta = new_id_len - prev_id_len;
1387
1388 /* If the previous ID length was 0, we need to add a 1-byte ID length */
1389 if (prev_id_len == 0)
1390 {
1391 if ((*p_cur_size + idlen_delta + 1) > max_size)
1392 return (NDEF_MSG_INSUFFICIENT_MEM);
1393
1394 shiftdown (p_idlen_field, (UINT32)(*p_cur_size - (p_idlen_field - p_msg)), 1);
1395 p_prev_id += 1;
1396 *p_cur_size += 1;
1397 *p_rec |= NDEF_IL_MASK;
1398 }
1399 else if ((*p_cur_size + idlen_delta) > max_size)
1400 return (NDEF_MSG_INSUFFICIENT_MEM);
1401
1402 /* Point to the end of the previous ID field, and make space for the extra data */
1403 pp = p_prev_id + prev_id_len;
1404 shiftdown (pp, (UINT32)(*p_cur_size - (pp - p_msg)), idlen_delta);
1405
1406 *p_cur_size += idlen_delta;
1407 }
1408 else if (new_id_len < prev_id_len)
1409 {
1410 /* New ID field is smaller than the previous */
1411 idlen_delta = prev_id_len - new_id_len;
1412
1413 /* Point to the end of the previous ID, and shift up to fill the the unused space */
1414 pp = p_prev_id + prev_id_len;
1415 shiftup (pp - idlen_delta, pp, (UINT32)(*p_cur_size - (pp - p_msg)));
1416
1417 *p_cur_size -= idlen_delta;
1418
1419 /* If removing the ID, make sure that length field is also removed */
1420 if (new_id_len == 0)
1421 {
1422 shiftup (p_idlen_field, p_idlen_field + 1, (UINT32)(*p_cur_size - (p_idlen_field - p_msg - (UINT32)1)));
1423 *p_rec &= ~NDEF_IL_MASK;
1424 *p_cur_size -= 1;
1425 }
1426 }
1427
1428 /* Save in new ID length and data */
1429 if (new_id_len)
1430 {
1431 *p_idlen_field = new_id_len;
1432
1433 if (p_new_id)
1434 memcpy (p_prev_id, p_new_id, new_id_len);
1435 }
1436
1437 return (NDEF_OK);
1438 }
1439
1440 /*******************************************************************************
1441 **
1442 ** Function NDEF_MsgRemoveRec
1443 **
1444 ** Description This function removes the record at the given
1445 ** index in the given NDEF message.
1446 **
1447 ** Returns TRUE if OK, FALSE if the index was invalid
1448 ** *p_cur_size is updated
1449 **
1450 *******************************************************************************/
NDEF_MsgRemoveRec(UINT8 * p_msg,UINT32 * p_cur_size,INT32 index)1451 tNDEF_STATUS NDEF_MsgRemoveRec (UINT8 *p_msg, UINT32 *p_cur_size, INT32 index)
1452 {
1453 UINT8 *p_rec = NDEF_MsgGetRecByIndex (p_msg, index);
1454 UINT8 *pNext, *pPrev;
1455
1456 if (!p_rec)
1457 return (NDEF_REC_NOT_FOUND);
1458
1459 /* If this is the first record in the message... */
1460 if (*p_rec & NDEF_MB_MASK)
1461 {
1462 /* Find the second record (if any) and set his 'Message Begin' bit */
1463 if ((pNext = NDEF_MsgGetRecByIndex(p_msg, 1)) != NULL)
1464 {
1465 *pNext |= NDEF_MB_MASK;
1466
1467 *p_cur_size -= (UINT32)(pNext - p_msg);
1468
1469 shiftup (p_msg, pNext, *p_cur_size);
1470 }
1471 else
1472 *p_cur_size = 0; /* No more records, lenght must be zero */
1473
1474 return (NDEF_OK);
1475 }
1476
1477 /* If this is the last record in the message... */
1478 if (*p_rec & NDEF_ME_MASK)
1479 {
1480 if (index > 0)
1481 {
1482 /* Find the previous record and set his 'Message End' bit */
1483 if ((pPrev = NDEF_MsgGetRecByIndex(p_msg, index - 1)) == NULL)
1484 return (FALSE);
1485
1486 *pPrev |= NDEF_ME_MASK;
1487 }
1488 *p_cur_size = (UINT32)(p_rec - p_msg);
1489
1490 return (NDEF_OK);
1491 }
1492
1493 /* Not the first or the last... get the address of the next record */
1494 if ((pNext = NDEF_MsgGetNextRec (p_rec)) == NULL)
1495 return (FALSE);
1496
1497 /* We are removing p_rec, so shift from pNext to the end */
1498 shiftup (p_rec, pNext, (UINT32)(*p_cur_size - (pNext - p_msg)));
1499
1500 *p_cur_size -= (UINT32)(pNext - p_rec);
1501
1502 return (NDEF_OK);
1503 }
1504
1505
1506 /*******************************************************************************
1507 **
1508 ** Function NDEF_MsgCopyAndDechunk
1509 **
1510 ** Description This function copies and de-chunks an NDEF message.
1511 ** It is assumed that the destination is at least as large
1512 ** as the source, since the source may not actually contain
1513 ** any chunks.
1514 **
1515 ** Returns The output byte count
1516 **
1517 *******************************************************************************/
NDEF_MsgCopyAndDechunk(UINT8 * p_src,UINT32 src_len,UINT8 * p_dest,UINT32 * p_out_len)1518 tNDEF_STATUS NDEF_MsgCopyAndDechunk (UINT8 *p_src, UINT32 src_len, UINT8 *p_dest, UINT32 *p_out_len)
1519 {
1520 UINT32 out_len, max_out_len;
1521 UINT8 *p_rec;
1522 UINT8 *p_prev_rec = p_dest;
1523 UINT8 *p_type, *p_id, *p_pay;
1524 UINT8 type_len, id_len, tnf;
1525 UINT32 pay_len;
1526 tNDEF_STATUS status;
1527
1528 /* First, validate the source */
1529 if ((status = NDEF_MsgValidate(p_src, src_len, TRUE)) != NDEF_OK)
1530 return (status);
1531
1532 /* The output buffer must be at least as large as the input buffer */
1533 max_out_len = src_len;
1534
1535 /* Initialize output */
1536 NDEF_MsgInit (p_dest, max_out_len, &out_len);
1537
1538 p_rec = p_src;
1539
1540 /* Now, copy record by record */
1541 while (p_rec != NULL)
1542 {
1543 p_type = NDEF_RecGetType (p_rec, &tnf, &type_len);
1544 p_id = NDEF_RecGetId (p_rec, &id_len);
1545 p_pay = NDEF_RecGetPayload (p_rec, &pay_len);
1546
1547 /* If this is the continuation of a chunk, append the payload to the previous */
1548 if (tnf == NDEF_TNF_UNCHANGED)
1549 {
1550 NDEF_MsgAppendPayload (p_dest, max_out_len, &out_len, p_prev_rec, p_pay, pay_len);
1551 }
1552 else
1553 {
1554 p_prev_rec = p_dest + out_len;
1555
1556 NDEF_MsgAddRec (p_dest, max_out_len, &out_len, tnf, p_type, type_len,
1557 p_id, id_len, p_pay, pay_len);
1558 }
1559
1560 p_rec = NDEF_MsgGetNextRec (p_rec);
1561 }
1562
1563 *p_out_len = out_len;
1564
1565 return (NDEF_OK);
1566 }
1567
1568