• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 the implementation for Type 3 tag in Reader/Writer
23  *  mode.
24  *
25  ******************************************************************************/
26 #include <string.h>
27 #include "nfc_target.h"
28 #include "bt_types.h"
29 #include "trace_api.h"
30 
31 #include "nfc_api.h"
32 #include "nfc_int.h"
33 #include "nci_hmsgs.h"
34 #include "rw_api.h"
35 #include "rw_int.h"
36 #include "tags_int.h"
37 #include "gki.h"
38 
39 /* Definitions for constructing t3t command messages */
40 #define RW_T3T_FL_PADDING   0x01        /* Padding needed for last NDEF block */
41 #define RW_T3T_MAX_NDEF_BLOCKS_PER_UPDATE_1_BYTE_FORMAT (13)    /* Maximum number of NDEF blocks updates that can fit into one command (when all block-numbers are < 256) */
42 #define RW_T3T_MAX_NDEF_BLOCKS_PER_UPDATE_2_BYTE_FORMAT (12)    /* Maximum number of NDEF blocks updates that can fit into one command (when all block-numbers are >= 256) */
43 
44 /* Definitions for SENSF_RES */
45 #define RW_T3T_SENSF_RES_RD_OFFSET      17  /* Offset of RD in SENSF_RES from NCI_POLL NTF (includes 1 byte SENSF_RES length) */
46 #define RW_T3T_SENSF_RES_RD_LEN         2   /* Size of RD in SENSF_RES   */
47 
48 /* Timeout definitions for commands */
49 #define RW_T3T_POLL_CMD_TIMEOUT_TICKS                               ((RW_T3T_TOUT_RESP*2*QUICK_TIMER_TICKS_PER_SEC) / 1000)
50 #define RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS                            ((RW_T3T_TOUT_RESP*QUICK_TIMER_TICKS_PER_SEC) / 1000)
51 #define RW_T3T_RAW_FRAME_CMD_TIMEOUT_TICKS                          (RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS * 4)
52 
53 /* Macro to extract major version from NDEF version byte */
54 #define T3T_GET_MAJOR_VERSION(ver)      (ver>>4)
55 
56 /* Enumeration of API commands */
57 enum
58 {
59     RW_T3T_CMD_DETECT_NDEF,
60     RW_T3T_CMD_CHECK_NDEF,
61     RW_T3T_CMD_UPDATE_NDEF,
62     RW_T3T_CMD_CHECK,
63     RW_T3T_CMD_UPDATE,
64     RW_T3T_CMD_SEND_RAW_FRAME,
65     RW_T3T_CMD_GET_SYSTEM_CODES,
66     RW_T3T_CMD_FORMAT,
67     RW_T3T_CMD_SET_READ_ONLY_SOFT,
68     RW_T3T_CMD_SET_READ_ONLY_HARD,
69 
70     RW_T3T_CMD_MAX
71 };
72 
73 /* RW_CBACK events corresponding to API comands */
74 const UINT8 rw_t3t_api_res_evt[RW_T3T_CMD_MAX] =
75 {
76     RW_T3T_NDEF_DETECT_EVT,         /* RW_T3T_CMD_DETECT_NDEF */
77     RW_T3T_CHECK_CPLT_EVT,          /* RW_T3T_CMD_CHECK_NDEF  */
78     RW_T3T_UPDATE_CPLT_EVT,         /* RW_T3T_CMD_UPDATE_NDEF */
79     RW_T3T_CHECK_CPLT_EVT,          /* RW_T3T_CMD_CHECK */
80     RW_T3T_UPDATE_CPLT_EVT,         /* RW_T3T_CMD_UPDATE */
81     RW_T3T_RAW_FRAME_EVT,           /* RW_T3T_CMD_SEND_RAW_FRAME */
82     RW_T3T_GET_SYSTEM_CODES_EVT,    /* RW_T3T_CMD_GET_SYSTEM_CODES */
83     RW_T3T_FORMAT_CPLT_EVT,         /* RW_T3T_CMD_FORMAT */
84     RW_T3T_SET_READ_ONLY_CPLT_EVT   /* RW_T3T_CMD_SET_READ_ONLY */
85 };
86 
87 /* States */
88 enum
89 {
90     RW_T3T_STATE_NOT_ACTIVATED,
91     RW_T3T_STATE_IDLE,
92     RW_T3T_STATE_COMMAND_PENDING
93 };
94 
95 /* Sub-states */
96 enum
97 {
98     /* Sub states for getting system codes */
99     RW_T3T_GET_SC_SST_POLL_WILDCARD,        /* Waiting for wilcard poll response */
100     RW_T3T_GET_SC_SST_POLL_NDEF,            /* Waiting for NDEF poll response */
101     RW_T3T_GET_SC_SST_REQUEST_SC,           /* Waiting for REQUEST_SYSTEM_CODE response */
102 
103     /* Sub states for formatting Felica-Lite */
104     RW_T3T_FMT_SST_POLL_FELICA_LITE,        /* Waiting for POLL Felica-Lite response (for formatting) */
105     RW_T3T_FMT_SST_CHECK_MC_BLK,            /* Waiting for Felica-Lite MC (MemoryControl) block-read to complete */
106     RW_T3T_FMT_SST_UPDATE_MC_BLK,           /* Waiting for Felica-Lite MC (MemoryControl) block-write to complete */
107     RW_T3T_FMT_SST_UPDATE_NDEF_ATTRIB,      /* Waiting for NDEF attribute block-write to complete */
108 
109     /* Sub states for setting Felica-Lite read only */
110     RW_T3T_SRO_SST_POLL_FELICA_LITE,        /* Waiting for POLL Felica-Lite response (for setting read only) */
111     RW_T3T_SRO_SST_UPDATE_NDEF_ATTRIB,      /* Waiting for NDEF attribute block-write to complete */
112     RW_T3T_SRO_SST_CHECK_MC_BLK,            /* Waiting for Felica-Lite MC (MemoryControl) block-read to complete */
113     RW_T3T_SRO_SST_UPDATE_MC_BLK            /* Waiting for Felica-Lite MC (MemoryControl) block-write to complete */
114 };
115 
116 #if (BT_TRACE_VERBOSE == TRUE)
117 static char *rw_t3t_cmd_str (UINT8 cmd_id);
118 static char *rw_t3t_state_str (UINT8 state_id);
119 #endif
120 
121 
122 /* Local static functions */
123 static tNFC_STATUS rw_t3t_unselect (UINT8 peer_nfcid2[]);
124 static BT_HDR *rw_t3t_get_cmd_buf (void);
125 static tNFC_STATUS rw_t3t_send_to_lower (BT_HDR *p_msg);
126 static void rw_t3t_handle_get_system_codes_cplt (void);
127 static void rw_t3t_handle_get_sc_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf);
128 static void rw_t3t_handle_ndef_detect_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf);
129 static void rw_t3t_handle_fmt_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf);
130 static void rw_t3t_handle_sro_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf);
131 
132 
133 /* Default NDEF attribute information block (used when formatting Felica-Lite tags) */
134 #define RW_T3T_DEFAULT_FELICALITE_NBR       4   /* NBr (max block reads per cmd)*/
135 #define RW_T3T_DEFAULT_FELICALITE_NBW       1   /* NBw (max block write per cmd)*/
136 #define RW_T3T_DEFAULT_FELICALITE_NMAXB     (T3T_FELICALITE_NMAXB)
137 #define RW_T3T_DEFAULT_FELICALITE_ATTRIB_INFO_CHECKSUM  ((T3T_MSG_NDEF_VERSION +                    \
138                                                             RW_T3T_DEFAULT_FELICALITE_NBR +         \
139                                                             RW_T3T_DEFAULT_FELICALITE_NBW +         \
140                                                             (RW_T3T_DEFAULT_FELICALITE_NMAXB>>8) +  \
141                                                             (RW_T3T_DEFAULT_FELICALITE_NMAXB&0xFF) +\
142                                                             T3T_MSG_NDEF_WRITEF_OFF +               \
143                                                             T3T_MSG_NDEF_RWFLAG_RW) & 0xFFFF)
144 
145 const UINT8 rw_t3t_default_attrib_info[T3T_MSG_BLOCKSIZE] =
146 {
147     T3T_MSG_NDEF_VERSION,                   /* Ver                          */
148     RW_T3T_DEFAULT_FELICALITE_NBR,          /* NBr (max block reads per cmd)*/
149     RW_T3T_DEFAULT_FELICALITE_NBW,          /* NBw (max block write per cmd)*/
150     (RW_T3T_DEFAULT_FELICALITE_NMAXB>>8),   /* Nmaxb (max size in blocks)   */
151     (RW_T3T_DEFAULT_FELICALITE_NMAXB&0xFF), /* Nmaxb (max size in blocks)   */
152     0, 0, 0, 0,                             /* Unused                       */
153     T3T_MSG_NDEF_WRITEF_OFF,                /* WriteF                       */
154     T3T_MSG_NDEF_RWFLAG_RW,                 /* RW Flag                      */
155     0, 0, 0,                                /* Ln (current size in bytes)   */
156 
157     (RW_T3T_DEFAULT_FELICALITE_ATTRIB_INFO_CHECKSUM >> 8),     /* checksum (high-byte) */
158     (RW_T3T_DEFAULT_FELICALITE_ATTRIB_INFO_CHECKSUM & 0xFF)    /* checksum (low-byte)  */
159 
160 };
161 
162 /*****************************************************************************
163 **  Type3 TAG COMMANDS
164 *****************************************************************************/
165 
166 /*******************************************************************************
167 **
168 ** Function         rw_t3t_process_error
169 **
170 ** Description      Process error (timeout or CRC error)
171 **
172 ** Returns          none
173 **
174 *******************************************************************************/
rw_t3t_process_error(tNFC_STATUS status)175 void rw_t3t_process_error (tNFC_STATUS status)
176 {
177     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
178     UINT8 evt;
179     tRW_DATA evt_data;
180     BT_HDR *p_cmd_buf;
181 
182     if (p_cb->rw_state == RW_T3T_STATE_COMMAND_PENDING)
183     {
184         if (p_cb->cur_cmd == RW_T3T_CMD_GET_SYSTEM_CODES)
185         {
186             /* For GetSystemCode: either tag did not respond to requested POLL, or REQUEST_SYSTEM_CODE command */
187             rw_t3t_handle_get_system_codes_cplt ();
188             return;
189         }
190         /* Retry sending command if retry-count < max */
191         else if (rw_cb.cur_retry < RW_MAX_RETRIES)
192         {
193             /* retry sending the command */
194             rw_cb.cur_retry++;
195 
196             RW_TRACE_DEBUG2 ("T3T retransmission attempt %i of %i", rw_cb.cur_retry, RW_MAX_RETRIES);
197 
198             /* allocate a new buffer for message */
199             if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
200             {
201                 memcpy (p_cmd_buf, p_cb->p_cur_cmd_buf, sizeof (BT_HDR) + p_cb->p_cur_cmd_buf->offset + p_cb->p_cur_cmd_buf->len);
202 
203                 if (rw_t3t_send_to_lower (p_cmd_buf) == NFC_STATUS_OK)
204                 {
205                     /* Start timer for waiting for response */
206                     nfc_start_quick_timer (&p_cb->timer, NFC_TTYPE_RW_T3T_RESPONSE, p_cb->cur_tout);
207                     return;
208                 }
209                 else
210                 {
211                     /* failure - could not send buffer */
212                     GKI_freebuf (p_cmd_buf);
213                 }
214             }
215         }
216         else
217         {
218             RW_TRACE_DEBUG1 ("T3T maximum retransmission attempts reached (%i)", RW_MAX_RETRIES);
219         }
220 
221 #if (defined (RW_STATS_INCLUDED) && (RW_STATS_INCLUDED == TRUE))
222         /* update failure count */
223         rw_main_update_fail_stats ();
224 #endif  /* RW_STATS_INCLUDED */
225 
226         p_cb->rw_state = RW_T3T_STATE_IDLE;
227 
228         /* Notify app of result (if there was a pending command) */
229         if (p_cb->cur_cmd < RW_T3T_CMD_MAX)
230         {
231             /* If doing presence check, use status=NFC_STATUS_FAILED, otherwise NFC_STATUS_TIMEOUT */
232             evt_data.status = status;
233             evt = rw_t3t_api_res_evt[p_cb->cur_cmd];
234 
235             /* Set additional flags for RW_T3T_NDEF_DETECT_EVT */
236             if (evt == RW_T3T_NDEF_DETECT_EVT)
237             {
238                 evt_data.ndef.flags = RW_NDEF_FL_UNKNOWN;
239             }
240 
241             (*(rw_cb.p_cback)) (evt, &evt_data);
242         }
243     }
244     else
245     {
246         evt_data.status = status;
247         (*(rw_cb.p_cback)) (RW_T3T_INTF_ERROR_EVT, &evt_data);
248     }
249 }
250 
251 /*******************************************************************************
252 **
253 ** Function         rw_t3t_start_poll_timer
254 **
255 ** Description      Start the timer for T3T POLL Command
256 **
257 ** Returns          none
258 **
259 *******************************************************************************/
rw_t3t_start_poll_timer(tRW_T3T_CB * p_cb)260 void rw_t3t_start_poll_timer (tRW_T3T_CB *p_cb)
261 {
262     nfc_start_quick_timer (&p_cb->poll_timer, NFC_TTYPE_RW_T3T_RESPONSE, RW_T3T_POLL_CMD_TIMEOUT_TICKS);
263 }
264 
265 /*******************************************************************************
266 **
267 ** Function         rw_t3t_handle_nci_poll_ntf
268 **
269 ** Description      Handle NCI_T3T_POLLING_NTF
270 **
271 ** Returns          none
272 **
273 *******************************************************************************/
rw_t3t_handle_nci_poll_ntf(UINT8 nci_status,UINT8 num_responses,UINT8 sensf_res_buf_size,UINT8 * p_sensf_res_buf)274 void rw_t3t_handle_nci_poll_ntf (UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf)
275 {
276     tRW_DATA evt_data;
277     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
278 
279     /* stop timer for poll response */
280     nfc_stop_quick_timer (&p_cb->poll_timer);
281 
282     /* Stop t3t timer (if started) */
283     if (p_cb->flags & RW_T3T_FL_W4_PRESENCE_CHECK_POLL_RSP)
284     {
285         p_cb->flags &= ~RW_T3T_FL_W4_PRESENCE_CHECK_POLL_RSP;
286         evt_data.status = nci_status;
287         p_cb->rw_state = RW_T3T_STATE_IDLE;
288         (*(rw_cb.p_cback)) (RW_T3T_PRESENCE_CHECK_EVT, (tRW_DATA *) &evt_data);
289     }
290     else if (p_cb->flags & RW_T3T_FL_W4_GET_SC_POLL_RSP)
291     {
292         /* Handle POLL ntf in response to get system codes */
293         p_cb->flags &= ~RW_T3T_FL_W4_GET_SC_POLL_RSP;
294         rw_t3t_handle_get_sc_poll_rsp (p_cb, nci_status, num_responses, sensf_res_buf_size, p_sensf_res_buf);
295     }
296     else if (p_cb->flags & RW_T3T_FL_W4_FMT_FELICA_LITE_POLL_RSP)
297     {
298         /* Handle POLL ntf in response to get system codes */
299         p_cb->flags &= ~RW_T3T_FL_W4_FMT_FELICA_LITE_POLL_RSP;
300         rw_t3t_handle_fmt_poll_rsp (p_cb, nci_status, num_responses, sensf_res_buf_size, p_sensf_res_buf);
301     }
302     else if (p_cb->flags & RW_T3T_FL_W4_SRO_FELICA_LITE_POLL_RSP)
303     {
304         /* Handle POLL ntf in response to get system codes */
305         p_cb->flags &= ~RW_T3T_FL_W4_SRO_FELICA_LITE_POLL_RSP;
306         rw_t3t_handle_sro_poll_rsp (p_cb, nci_status, num_responses, sensf_res_buf_size, p_sensf_res_buf);
307     }
308     else if (p_cb->flags & RW_T3T_FL_W4_NDEF_DETECT_POLL_RSP)
309     {
310         /* Handle POLL ntf in response to ndef detection */
311         p_cb->flags &= ~RW_T3T_FL_W4_NDEF_DETECT_POLL_RSP;
312         rw_t3t_handle_ndef_detect_poll_rsp (p_cb, nci_status, num_responses, sensf_res_buf_size, p_sensf_res_buf);
313     }
314     else
315     {
316         /* Handle POLL ntf in response to RW_T3tPoll */
317         if ((evt_data.t3t_poll.status = nci_status) == NCI_STATUS_OK)
318         {
319             evt_data.t3t_poll.rc = p_cb->cur_poll_rc;
320             evt_data.t3t_poll.response_num = num_responses;
321             evt_data.t3t_poll.response_bufsize = sensf_res_buf_size;
322             evt_data.t3t_poll.response_buf = p_sensf_res_buf;
323         }
324 
325         p_cb->rw_state = RW_T3T_STATE_IDLE;
326         (*(rw_cb.p_cback)) (RW_T3T_POLL_EVT, &evt_data);
327     }
328 }
329 
330 
331 /*******************************************************************************
332 **
333 ** Function         rw_t3t_handle_get_system_codes_cplt
334 **
335 ** Description      Notify upper layer of system codes
336 **
337 ** Returns          none
338 **
339 *******************************************************************************/
rw_t3t_handle_get_system_codes_cplt(void)340 void rw_t3t_handle_get_system_codes_cplt (void)
341 {
342     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
343     tRW_DATA evt_data;
344     UINT8 i;
345 
346     evt_data.t3t_sc.status = NFC_STATUS_OK;
347     evt_data.t3t_sc.num_system_codes = p_cb->num_system_codes;
348     evt_data.t3t_sc.p_system_codes   = p_cb->system_codes;
349 
350     RW_TRACE_DEBUG1 ("rw_t3t_handle_get_system_codes_cplt, number of systems: %i", evt_data.t3t_sc.num_system_codes);
351     for (i = 0; i < evt_data.t3t_sc.num_system_codes; i++)
352     {
353         RW_TRACE_DEBUG2 ("   system %i: %04X", i, evt_data.t3t_sc.p_system_codes[i]);
354     }
355 
356     p_cb->rw_state = RW_T3T_STATE_IDLE;
357     (*(rw_cb.p_cback)) (RW_T3T_GET_SYSTEM_CODES_EVT, &evt_data);
358 
359 
360 }
361 
362 /*******************************************************************************
363 **
364 ** Function         rw_t3t_format_cplt
365 **
366 ** Description      Notify upper layer of format complete
367 **
368 ** Returns          none
369 **
370 *******************************************************************************/
rw_t3t_format_cplt(tNFC_STATUS status)371 void rw_t3t_format_cplt (tNFC_STATUS status)
372 {
373     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
374     tRW_DATA evt_data;
375 
376     p_cb->rw_state = RW_T3T_STATE_IDLE;
377 
378     /* Update ndef info */
379     p_cb->ndef_attrib.status = status;
380     if (status == NFC_STATUS_OK)
381     {
382         p_cb->ndef_attrib.version = T3T_MSG_NDEF_VERSION;
383         p_cb->ndef_attrib.nbr = RW_T3T_DEFAULT_FELICALITE_NBR;
384         p_cb->ndef_attrib.nbw = RW_T3T_DEFAULT_FELICALITE_NBW;
385         p_cb->ndef_attrib.nmaxb = RW_T3T_DEFAULT_FELICALITE_NMAXB;
386         p_cb->ndef_attrib.writef = T3T_MSG_NDEF_WRITEF_OFF;
387         p_cb->ndef_attrib.rwflag = T3T_MSG_NDEF_RWFLAG_RW;
388         p_cb->ndef_attrib.ln = 0;
389     }
390 
391     /* Notify upper layer of format complete */
392     evt_data.status = status;
393     (*(rw_cb.p_cback)) (RW_T3T_FORMAT_CPLT_EVT, &evt_data);
394 }
395 
396 /*******************************************************************************
397 **
398 ** Function         rw_t3t_set_readonly_cplt
399 **
400 ** Description      Notify upper layer of set read only complete
401 **
402 ** Returns          none
403 **
404 *******************************************************************************/
rw_t3t_set_readonly_cplt(tNFC_STATUS status)405 void rw_t3t_set_readonly_cplt (tNFC_STATUS status)
406 {
407     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
408     tRW_DATA evt_data;
409 
410     p_cb->rw_state = RW_T3T_STATE_IDLE;
411 
412     /* Notify upper layer of format complete */
413     evt_data.status = status;
414     (*(rw_cb.p_cback)) (RW_T3T_SET_READ_ONLY_CPLT_EVT, &evt_data);
415 }
416 
417 /*******************************************************************************
418 **
419 ** Function         rw_t3t_process_timeout
420 **
421 ** Description      Process timeout
422 **
423 ** Returns          none
424 **
425 *******************************************************************************/
rw_t3t_process_timeout(TIMER_LIST_ENT * p_tle)426 void rw_t3t_process_timeout (TIMER_LIST_ENT *p_tle)
427 {
428     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
429     tRW_DATA evt_data;
430 
431     /* Check which timer timed out */
432     if (p_tle == &p_cb->timer)
433     {
434         /* UPDATE/CHECK response timeout */
435 #if (BT_TRACE_VERBOSE == TRUE)
436         RW_TRACE_ERROR3 ("T3T timeout. state=%s cur_cmd=0x%02X (%s)", rw_t3t_state_str (rw_cb.tcb.t3t.rw_state), rw_cb.tcb.t3t.cur_cmd, rw_t3t_cmd_str (rw_cb.tcb.t3t.cur_cmd));
437 #else
438         RW_TRACE_ERROR2 ("T3T timeout. state=0x%02X cur_cmd=0x%02X", rw_cb.tcb.t3t.rw_state, rw_cb.tcb.t3t.cur_cmd);
439 #endif
440 
441         rw_t3t_process_error (NFC_STATUS_TIMEOUT);
442     }
443     else
444     {
445         RW_TRACE_ERROR0 ("T3T POLL timeout.");
446 
447         /* POLL response timeout */
448         if (p_cb->flags & RW_T3T_FL_W4_PRESENCE_CHECK_POLL_RSP)
449         {
450             /* POLL timeout for presence check */
451             p_cb->flags &= ~RW_T3T_FL_W4_PRESENCE_CHECK_POLL_RSP;
452             evt_data.status = NFC_STATUS_FAILED;
453             p_cb->rw_state = RW_T3T_STATE_IDLE;
454             (*(rw_cb.p_cback)) (RW_T3T_PRESENCE_CHECK_EVT, (tRW_DATA *) &evt_data);
455         }
456         else if (p_cb->flags & RW_T3T_FL_W4_GET_SC_POLL_RSP)
457         {
458             /* POLL timeout for getting system codes */
459             p_cb->flags &= ~RW_T3T_FL_W4_GET_SC_POLL_RSP;
460             rw_t3t_handle_get_system_codes_cplt ();
461         }
462         else if (p_cb->flags & RW_T3T_FL_W4_FMT_FELICA_LITE_POLL_RSP)
463         {
464             /* POLL timeout for formatting Felica Lite */
465             p_cb->flags &= ~RW_T3T_FL_W4_FMT_FELICA_LITE_POLL_RSP;
466             RW_TRACE_ERROR0 ("Felica-Lite tag not detected");
467             rw_t3t_format_cplt (NFC_STATUS_FAILED);
468         }
469         else if (p_cb->flags & RW_T3T_FL_W4_SRO_FELICA_LITE_POLL_RSP)
470         {
471             /* POLL timeout for configuring Felica Lite read only */
472             p_cb->flags &= ~RW_T3T_FL_W4_SRO_FELICA_LITE_POLL_RSP;
473             RW_TRACE_ERROR0 ("Felica-Lite tag not detected");
474             rw_t3t_set_readonly_cplt (NFC_STATUS_FAILED);
475         }
476         else if (p_cb->flags & RW_T3T_FL_W4_NDEF_DETECT_POLL_RSP)
477         {
478             /* POLL timeout for ndef detection */
479             p_cb->flags &= ~RW_T3T_FL_W4_NDEF_DETECT_POLL_RSP;
480             rw_t3t_handle_ndef_detect_poll_rsp (p_cb, NFC_STATUS_TIMEOUT, 0, 0, NULL);
481         }
482         else
483         {
484             /* Timeout waiting for response for RW_T3tPoll */
485             evt_data.t3t_poll.status = NFC_STATUS_FAILED;
486             p_cb->rw_state = RW_T3T_STATE_IDLE;
487             (*(rw_cb.p_cback)) (RW_T3T_POLL_EVT, (tRW_DATA *) &evt_data);
488         }
489     }
490 }
491 
492 
493 /*******************************************************************************
494 **
495 ** Function         rw_t3t_process_frame_error
496 **
497 ** Description      Process frame crc error
498 **
499 ** Returns          none
500 **
501 *******************************************************************************/
rw_t3t_process_frame_error(void)502 void rw_t3t_process_frame_error (void)
503 {
504 #if (BT_TRACE_VERBOSE == TRUE)
505     RW_TRACE_ERROR3 ("T3T frame error. state=%s cur_cmd=0x%02X (%s)", rw_t3t_state_str (rw_cb.tcb.t3t.rw_state), rw_cb.tcb.t3t.cur_cmd, rw_t3t_cmd_str (rw_cb.tcb.t3t.cur_cmd));
506 #else
507     RW_TRACE_ERROR2 ("T3T frame error. state=0x%02X cur_cmd=0x%02X", rw_cb.tcb.t3t.rw_state, rw_cb.tcb.t3t.cur_cmd);
508 #endif
509 
510 #if (defined (RW_STATS_INCLUDED) && (RW_STATS_INCLUDED == TRUE))
511     /* Update stats */
512     rw_main_update_crc_error_stats ();
513 #endif  /* RW_STATS_INCLUDED */
514 
515     /* Process the error */
516     rw_t3t_process_error (NFC_STATUS_MSG_CORRUPTED);
517 }
518 
519 /*******************************************************************************
520 **
521 ** Function         rw_t3t_send_to_lower
522 **
523 ** Description      Send command to lower layer
524 **
525 ** Returns          status of the send
526 **
527 *******************************************************************************/
rw_t3t_send_to_lower(BT_HDR * p_msg)528 tNFC_STATUS rw_t3t_send_to_lower (BT_HDR *p_msg)
529 {
530     UINT8 *p;
531 
532 #if (defined (RW_STATS_INCLUDED) && (RW_STATS_INCLUDED == TRUE))
533     BOOLEAN is_retry;
534     /* Update stats */
535     rw_main_update_tx_stats (p_msg->len, ((rw_cb.cur_retry==0) ? FALSE : TRUE));
536 #endif  /* RW_STATS_INCLUDED */
537 
538     /* Set NFC-F SoD field (payload len + 1) */
539     p_msg->offset -= 1;         /* Point to SoD field */
540     p = (UINT8 *) (p_msg+1) + p_msg->offset;
541     UINT8_TO_STREAM (p, (p_msg->len+1));
542     p_msg->len += 1;            /* Increment len to include SoD */
543 
544 #if (BT_TRACE_PROTOCOL == TRUE)
545     DispT3TagMessage (p_msg, FALSE);
546 #endif
547 
548     return (NFC_SendData (NFC_RF_CONN_ID, p_msg));
549 }
550 
551 /*****************************************************************************
552 **
553 ** Function         rw_t3t_get_cmd_buf
554 **
555 ** Description      Get a buffer for sending T3T messages
556 **
557 ** Returns          BT_HDR *
558 **
559 *****************************************************************************/
rw_t3t_get_cmd_buf(void)560 BT_HDR *rw_t3t_get_cmd_buf (void)
561 {
562     BT_HDR *p_cmd_buf;
563 
564     if ((p_cmd_buf = (BT_HDR *) GKI_getpoolbuf (NFC_RW_POOL_ID)) != NULL)
565     {
566         /* Reserve offset for NCI_DATA_HDR and NFC-F Sod (LEN) field */
567         p_cmd_buf->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE + 1;
568         p_cmd_buf->len = 0;
569     }
570 
571     return (p_cmd_buf);
572 }
573 
574 /*****************************************************************************
575 **
576 ** Function         rw_t3t_send_cmd
577 **
578 ** Description      Send command to tag, and start timer for response
579 **
580 ** Returns          tNFC_STATUS
581 **
582 *****************************************************************************/
rw_t3t_send_cmd(tRW_T3T_CB * p_cb,UINT8 rw_t3t_cmd,BT_HDR * p_cmd_buf,UINT32 timeout_ms)583 tNFC_STATUS rw_t3t_send_cmd (tRW_T3T_CB *p_cb, UINT8 rw_t3t_cmd, BT_HDR *p_cmd_buf, UINT32 timeout_ms)
584 {
585     tNFC_STATUS retval;
586 
587     /* Indicate first attempt to send command, back up cmd buffer in case needed for retransmission */
588     rw_cb.cur_retry = 0;
589     memcpy (p_cb->p_cur_cmd_buf, p_cmd_buf, sizeof (BT_HDR) + p_cmd_buf->offset + p_cmd_buf->len);
590 
591     p_cb->cur_cmd = rw_t3t_cmd;
592     p_cb->cur_tout = timeout_ms;
593     p_cb->rw_state = RW_T3T_STATE_COMMAND_PENDING;
594 
595     if ((retval = rw_t3t_send_to_lower (p_cmd_buf)) == NFC_STATUS_OK)
596     {
597         /* Start timer for waiting for response */
598         nfc_start_quick_timer (&p_cb->timer, NFC_TTYPE_RW_T3T_RESPONSE, timeout_ms);
599     }
600     else
601     {
602         /* Error sending */
603         p_cb->rw_state = RW_T3T_STATE_IDLE;
604     }
605 
606     return (retval);
607 }
608 
609 /*****************************************************************************
610 **
611 ** Function         rw_t3t_send_update_ndef_attribute_cmd
612 **
613 ** Description      Send UPDATE command for Attribute Information
614 **
615 ** Returns          tNFC_STATUS
616 **
617 *****************************************************************************/
rw_t3t_send_update_ndef_attribute_cmd(tRW_T3T_CB * p_cb,BOOLEAN write_in_progress)618 tNFC_STATUS rw_t3t_send_update_ndef_attribute_cmd (tRW_T3T_CB *p_cb, BOOLEAN write_in_progress)
619 {
620     tNFC_STATUS retval = NFC_STATUS_OK;
621     BT_HDR *p_cmd_buf;
622     UINT8 *p_cmd_start, *p;
623     UINT16 checksum, i;
624     UINT8 write_f;
625     UINT32 ln;
626     UINT8 *p_ndef_attr_info_start;
627 
628     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
629     {
630         /* Construct T3T message */
631         p = p_cmd_start  = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
632 
633         /* Add UPDATE opcode to message  */
634         UINT8_TO_STREAM (p, T3T_MSG_OPC_UPDATE_CMD);
635 
636         /* Add IDm to message */
637         ARRAY_TO_STREAM (p, p_cb->peer_nfcid2, NCI_NFCID2_LEN);
638 
639         /* Add Service code list */
640         UINT8_TO_STREAM (p, 1);                       /* Number of services (only 1 service: NDEF) */
641         UINT16_TO_STREAM (p, T3T_MSG_NDEF_SC_RW);     /* Service code (little-endian format) */
642 
643         /* Add number of blocks in this UPDATE command */
644         UINT8_TO_STREAM (p, 1);                      /* Number of blocks to write in this command */
645 
646         /* Block List element: the NDEF attribute information block (block 0) */
647         UINT8_TO_STREAM (p, T3T_MSG_MASK_TWO_BYTE_BLOCK_DESC_FORMAT);
648         UINT8_TO_STREAM (p, 0);
649 
650         /* Add payload (Attribute information block) */
651         p_ndef_attr_info_start = p;                              /* Save start of a NDEF attribute info block for checksum */
652         UINT8_TO_STREAM (p, T3T_MSG_NDEF_VERSION);
653         UINT8_TO_STREAM (p, p_cb->ndef_attrib.nbr);
654         UINT8_TO_STREAM (p, p_cb->ndef_attrib.nbw);
655         UINT16_TO_BE_STREAM (p, p_cb->ndef_attrib.nmaxb);
656         UINT32_TO_STREAM (p, 0);
657 
658         /* If starting NDEF write: set WriteF=ON, and ln=current ndef length */
659         if (write_in_progress)
660         {
661             write_f = T3T_MSG_NDEF_WRITEF_ON;
662             ln = p_cb->ndef_attrib.ln;
663         }
664         /* If finishing NDEF write: set WriteF=OFF, and ln=new ndef len */
665         else
666         {
667             write_f = T3T_MSG_NDEF_WRITEF_OFF;
668             ln = p_cb->ndef_msg_len;
669         }
670         UINT8_TO_STREAM (p, write_f);
671         UINT8_TO_STREAM (p, p_cb->ndef_attrib.rwflag);
672         UINT8_TO_STREAM (p, (ln>>16) & 0xFF);    /* High byte (of 3) of Ln */
673         UINT8_TO_STREAM (p, (ln>>8) & 0xFF);     /* Middle byte (of 3) of Ln */
674         UINT8_TO_STREAM (p, (ln) & 0xFF);        /* Low byte (of 3) of Ln */
675 
676         /* Calculate and append Checksum */
677         checksum = 0;
678         for (i = 0; i < T3T_MSG_NDEF_ATTR_INFO_SIZE; i++)
679         {
680             checksum+=p_ndef_attr_info_start[i];
681         }
682         UINT16_TO_BE_STREAM (p, checksum);
683 
684 
685         /* Calculate length of message */
686         p_cmd_buf->len = (UINT16) (p - p_cmd_start);
687 
688         /* Send the T3T message */
689         retval = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_UPDATE_NDEF, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS);
690     }
691     else
692     {
693         retval = NFC_STATUS_NO_BUFFERS;
694     }
695 
696     return (retval);
697 }
698 
699 /*****************************************************************************
700 **
701 ** Function         rw_t3t_send_next_ndef_update_cmd
702 **
703 ** Description      Send next segment of NDEF message to update
704 **
705 ** Returns          tNFC_STATUS
706 **
707 *****************************************************************************/
rw_t3t_send_next_ndef_update_cmd(tRW_T3T_CB * p_cb)708 tNFC_STATUS rw_t3t_send_next_ndef_update_cmd (tRW_T3T_CB *p_cb)
709 {
710     tNFC_STATUS retval = NFC_STATUS_OK;
711     UINT16 block_id;
712     UINT16 first_block_to_write;
713     UINT16 ndef_blocks_to_write, ndef_blocks_remaining;
714     UINT32 ndef_bytes_remaining, ndef_padding = 0;
715     UINT8 flags = 0;
716     UINT8 *p_cur_ndef_src_offset;
717     BT_HDR *p_cmd_buf;
718     UINT8 *p_cmd_start, *p;
719     UINT8 blocks_per_update;
720     UINT32 timeout;
721 
722     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
723     {
724         /* Construct T3T message */
725         p = p_cmd_start  = (UINT8 *) (p_cmd_buf + 1) + p_cmd_buf->offset;
726 
727         /* Calculate number of ndef bytes remaining to write */
728         ndef_bytes_remaining = p_cb->ndef_msg_len - p_cb->ndef_msg_bytes_sent;
729 
730         /* Calculate number of blocks remaining to write */
731         ndef_blocks_remaining = (UINT16) ((ndef_bytes_remaining+15) >> 4);      /* ndef blocks remaining (rounded upward) */
732 
733         /* Calculate first NDEF block ID for this UPDATE command */
734         first_block_to_write = (UINT16) ((p_cb->ndef_msg_bytes_sent >> 4) + 1);
735 
736         /* Calculate max number of blocks per write. */
737         if ((first_block_to_write + RW_T3T_MAX_NDEF_BLOCKS_PER_UPDATE_1_BYTE_FORMAT) < 0x100)
738         {
739             /* All block-numbers are < 0x100 (i.e. can be specified using one-byte format) */
740             blocks_per_update = RW_T3T_MAX_NDEF_BLOCKS_PER_UPDATE_1_BYTE_FORMAT;
741         }
742         else
743         {
744             /* Block-numbers are >= 0x100 (i.e. need to be specified using two-byte format) */
745             blocks_per_update = RW_T3T_MAX_NDEF_BLOCKS_PER_UPDATE_2_BYTE_FORMAT;
746         }
747 
748         /* Check if blocks_per_update is bigger than what peer allows */
749         if (blocks_per_update > p_cb->ndef_attrib.nbw)
750             blocks_per_update = p_cb->ndef_attrib.nbw;
751 
752         /* Check if remaining blocks can fit into one UPDATE command */
753         if (ndef_blocks_remaining <= blocks_per_update)
754         {
755             /* remaining blocks can fit into one UPDATE command */
756             ndef_blocks_to_write = ndef_blocks_remaining;
757         }
758         else
759         {
760             /* Remaining blocks cannot fit into one UPDATE command */
761             ndef_blocks_to_write = blocks_per_update;
762         }
763 
764 
765         /* Write to command header for UPDATE */
766 
767         /* Add UPDATE opcode to message  */
768         UINT8_TO_STREAM (p, T3T_MSG_OPC_UPDATE_CMD);
769 
770         /* Add IDm to message */
771         ARRAY_TO_STREAM (p, p_cb->peer_nfcid2, NCI_NFCID2_LEN);
772 
773         /* Add Service code list */
774         UINT8_TO_STREAM (p, 1);                       /* Number of services (only 1 service: NDEF) */
775         UINT16_TO_STREAM (p, T3T_MSG_NDEF_SC_RW);     /* Service code (little-endian format) */
776 
777 
778         /* Add number of blocks in this UPDATE command */
779         UINT8_TO_STREAM (p, ndef_blocks_to_write);   /* Number of blocks to write in this command */
780         timeout = RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS * (UINT32) ndef_blocks_to_write;
781 
782         for (block_id = first_block_to_write; block_id < (first_block_to_write + ndef_blocks_to_write); block_id++)
783         {
784             if (block_id<256)
785             {
786                 /* Block IDs 0-255 can be specified in '2-byte' format: byte0=0, byte1=blocknumber */
787                 UINT8_TO_STREAM (p, T3T_MSG_MASK_TWO_BYTE_BLOCK_DESC_FORMAT);    /* byte0: len=1; access-mode=0; service code list order=0 */
788                 UINT8_TO_STREAM (p, block_id);                                   /* byte1: block number */
789             }
790             else
791             {
792                 /* Block IDs 256+ must be specified in '3-byte' format: byte0=80h, followed by blocknumber */
793                 UINT8_TO_STREAM (p, 0x00);               /* byte0: len=0; access-mode=0; service code list order=0 */
794                 UINT16_TO_STREAM (p, block_id);          /* byte1-2: block number in little-endian format */
795             }
796 
797         }
798 
799         /* Add NDEF payload */
800 
801         /* If this sending last block of NDEF,  check if padding is needed to make payload a multiple of 16 bytes */
802         if (ndef_blocks_to_write == ndef_blocks_remaining)
803         {
804             ndef_padding = (16 - (ndef_bytes_remaining & 0x0F)) & 0x0F;
805             if (ndef_padding)
806             {
807                 flags |= RW_T3T_FL_PADDING;
808                 ndef_blocks_to_write--;         /* handle the last block separately if it needs padding */
809             }
810         }
811 
812         /* Add NDEF payload to the message */
813         p_cur_ndef_src_offset = &p_cb->ndef_msg[p_cb->ndef_msg_bytes_sent];
814 
815 
816         ARRAY_TO_STREAM (p, p_cur_ndef_src_offset, (ndef_blocks_to_write * 16));
817         p_cb->ndef_msg_bytes_sent += ((UINT32) ndef_blocks_to_write * 16);
818 
819         if (flags & RW_T3T_FL_PADDING)
820         {
821             /* Add last of the NDEF message */
822             p_cur_ndef_src_offset = &p_cb->ndef_msg[p_cb->ndef_msg_bytes_sent];
823             ARRAY_TO_STREAM (p, p_cur_ndef_src_offset, (int) (16-ndef_padding));
824             p_cb->ndef_msg_bytes_sent += (16-ndef_padding);
825 
826             /* Add padding */
827             memset (p, 0, ndef_padding);
828             p+=ndef_padding;
829         }
830 
831         /* Calculate length of message */
832         p_cmd_buf->len = (UINT16) (p - p_cmd_start);
833 
834         /* Send the T3T message */
835         retval = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_UPDATE_NDEF, p_cmd_buf, timeout);
836     }
837     else
838     {
839         retval = NFC_STATUS_NO_BUFFERS;
840     }
841 
842     return (retval);
843 }
844 
845 
846 
847 /*****************************************************************************
848 **
849 ** Function         rw_t3t_send_next_ndef_check_cmd
850 **
851 ** Description      Send command for reading next segment of NDEF message
852 **
853 ** Returns          tNFC_STATUS
854 **
855 *****************************************************************************/
rw_t3t_send_next_ndef_check_cmd(tRW_T3T_CB * p_cb)856 tNFC_STATUS rw_t3t_send_next_ndef_check_cmd (tRW_T3T_CB *p_cb)
857 {
858     tNFC_STATUS retval = NFC_STATUS_OK;
859     UINT16 block_id;
860     UINT16 ndef_blocks_remaining, first_block_to_read, cur_blocks_to_read;
861     UINT32 ndef_bytes_remaining;
862     BT_HDR *p_cmd_buf;
863     UINT8 *p_cmd_start, *p;
864 
865     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
866     {
867         /* Construct T3T message */
868         p = p_cmd_start  = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
869 
870         /* Calculate number of ndef bytes remaining to read */
871         ndef_bytes_remaining = p_cb->ndef_attrib.ln - p_cb->ndef_rx_offset;
872 
873         /* Calculate number of blocks remaining to read */
874         ndef_blocks_remaining = (UINT16) ((ndef_bytes_remaining+15) >> 4);      /* ndef blocks remaining (rounded upward) */
875 
876         /* Calculate first NDEF block ID */
877         first_block_to_read = (UINT16) ((p_cb->ndef_rx_offset >> 4) + 1);
878 
879         /* Check if remaining blocks can fit into one CHECK command */
880         if (ndef_blocks_remaining <= p_cb->ndef_attrib.nbr)
881         {
882             /* remaining blocks can fit into one CHECK command */
883             cur_blocks_to_read = ndef_blocks_remaining;
884             p_cb->ndef_rx_readlen = ndef_bytes_remaining;
885             p_cb->flags |= RW_T3T_FL_IS_FINAL_NDEF_SEGMENT;
886         }
887         else
888         {
889             /* Remaining blocks cannot fit into one CHECK command */
890             cur_blocks_to_read = p_cb->ndef_attrib.nbr;            /* Read maximum number of blocks allowed by the peer */
891             p_cb->ndef_rx_readlen = ((UINT32) p_cb->ndef_attrib.nbr * 16);
892         }
893 
894         RW_TRACE_DEBUG3 ("rw_t3t_send_next_ndef_check_cmd: bytes_remaining: %i, cur_blocks_to_read: %i, is_final: %i",
895             ndef_bytes_remaining, cur_blocks_to_read, (p_cb->flags & RW_T3T_FL_IS_FINAL_NDEF_SEGMENT));
896 
897         /* Write to command header for UPDATE */
898 
899         /* Add UPDATE opcode to message  */
900         UINT8_TO_STREAM (p, T3T_MSG_OPC_CHECK_CMD);
901 
902         /* Add IDm to message */
903         ARRAY_TO_STREAM (p, p_cb->peer_nfcid2, NCI_NFCID2_LEN);
904 
905         /* Add Service code list */
906         UINT8_TO_STREAM (p, 1);                       /* Number of services (only 1 service: NDEF) */
907 
908         /* Service code (little-endian format) . If NDEF is read-only, then use T3T_MSG_NDEF_SC_RO, otherwise use T3T_MSG_NDEF_SC_RW */
909         if (p_cb->ndef_attrib.rwflag == T3T_MSG_NDEF_RWFLAG_RO)
910         {
911             UINT16_TO_STREAM (p, T3T_MSG_NDEF_SC_RO);
912         }
913         else
914         {
915             UINT16_TO_STREAM (p, T3T_MSG_NDEF_SC_RW);
916         }
917 
918         /* Add number of blocks in this UPDATE command */
919         UINT8_TO_STREAM (p, cur_blocks_to_read);     /* Number of blocks to check in this command */
920 
921         for (block_id = first_block_to_read; block_id < (first_block_to_read + cur_blocks_to_read); block_id++)
922         {
923             if (block_id<256)
924             {
925                 /* Block IDs 0-255 can be specified in '2-byte' format: byte0=0, byte1=blocknumber */
926                 UINT8_TO_STREAM (p, T3T_MSG_MASK_TWO_BYTE_BLOCK_DESC_FORMAT);    /* byte1: len=0; access-mode=0; service code list order=0 */
927                 UINT8_TO_STREAM (p, block_id);                                   /* byte1: block number */
928             }
929             else
930             {
931                 /* Block IDs 256+ must be specified in '3-byte' format: byte0=80h, followed by blocknumber */
932                 UINT8_TO_STREAM (p, 0x00);           /* byte0: len=1; access-mode=0; service code list order=0 */
933                 UINT16_TO_STREAM (p, block_id);      /* byte1-2: block number in little-endian format */
934             }
935 
936         }
937 
938         /* Calculate length of message */
939         p_cmd_buf->len = (UINT16) (p - p_cmd_start);
940 
941         /* Send the T3T message */
942         retval = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_CHECK_NDEF, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS * (UINT32) cur_blocks_to_read);
943     }
944     else
945     {
946         retval = NFC_STATUS_NO_BUFFERS;
947     }
948 
949     return(retval);
950 }
951 
952 
953 /*****************************************************************************
954 **
955 ** Function         rw_t3t_message_set_block_list
956 **
957 ** Description      Add block list to T3T message
958 **
959 ** Returns          Number of bytes added to message
960 **
961 *****************************************************************************/
rw_t3t_message_set_block_list(tRW_T3T_CB * p_cb,UINT8 ** p,UINT8 num_blocks,tT3T_BLOCK_DESC * p_t3t_blocks)962 void rw_t3t_message_set_block_list (tRW_T3T_CB *p_cb, UINT8 **p, UINT8 num_blocks, tT3T_BLOCK_DESC *p_t3t_blocks)
963 {
964     UINT16 i, cur_service_code;
965     UINT8 service_code_idx, num_services = 0;
966     UINT8 *p_msg_num_services;
967     UINT16 service_list[T3T_MSG_SERVICE_LIST_MAX];
968 
969     /* Add CHECK or UPDATE opcode to message  */
970     UINT8_TO_STREAM ((*p), ((p_cb->cur_cmd == RW_T3T_CMD_CHECK) ? T3T_MSG_OPC_CHECK_CMD:T3T_MSG_OPC_UPDATE_CMD));
971 
972     /* Add IDm to message */
973     ARRAY_TO_STREAM ((*p), p_cb->peer_nfcid2, NCI_NFCID2_LEN);
974 
975     /* Skip over Number of Services field */
976     p_msg_num_services = (*p);      /* pointer to Number of Services offset */
977     (*p)++;
978 
979     /* Count number of different services are specified in the list, and add services to Service Code list */
980     for (i = 0; i < num_blocks; i++)
981     {
982         cur_service_code = p_t3t_blocks[i].service_code;
983 
984         /* Check if current service_code is already in the service_list */
985         for (service_code_idx=0; service_code_idx<num_services; service_code_idx++)
986         {
987             if (service_list[service_code_idx] == cur_service_code)
988                 break;
989         }
990 
991         if (service_code_idx == num_services)
992         {
993             /* Service not in the list yet. Add it. */
994             service_list[service_code_idx] = cur_service_code;
995             num_services++;
996 
997             /* Add service code to T3T message */
998             UINT16_TO_STREAM ((*p), cur_service_code);
999         }
1000     }
1001 
1002     /* Add 'Number of Sservices' to the message */
1003     *p_msg_num_services = num_services;
1004 
1005     /* Add 'number of blocks' to the message */
1006     UINT8_TO_STREAM ((*p), num_blocks);
1007 
1008     /* Add block descriptors */
1009     for (i = 0; i < num_blocks; i++)
1010     {
1011         cur_service_code = p_t3t_blocks[i].service_code;
1012 
1013         /* Check if current service_code is already in the service_list */
1014         for (service_code_idx=0; service_code_idx<num_services; service_code_idx++)
1015         {
1016             if (service_list[service_code_idx] == cur_service_code)
1017                 break;
1018         }
1019 
1020         /* Add decriptor to T3T message */
1021         if (p_t3t_blocks[i].block_number > 0xFF)
1022         {
1023             UINT8_TO_STREAM ((*p), service_code_idx);
1024             UINT16_TO_STREAM ((*p), p_t3t_blocks[i].block_number);
1025         }
1026         else
1027         {
1028             service_code_idx |= T3T_MSG_MASK_TWO_BYTE_BLOCK_DESC_FORMAT;
1029             UINT8_TO_STREAM ((*p), service_code_idx);
1030             UINT8_TO_STREAM ((*p), p_t3t_blocks[i].block_number);
1031         }
1032     }
1033 }
1034 
1035 /*****************************************************************************
1036 **
1037 ** Function         rw_t3t_send_check_cmd
1038 **
1039 ** Description      Send CHECK command
1040 **
1041 ** Returns          tNFC_STATUS
1042 **
1043 *****************************************************************************/
rw_t3t_send_check_cmd(tRW_T3T_CB * p_cb,UINT8 num_blocks,tT3T_BLOCK_DESC * p_t3t_blocks)1044 tNFC_STATUS rw_t3t_send_check_cmd (tRW_T3T_CB *p_cb, UINT8 num_blocks, tT3T_BLOCK_DESC *p_t3t_blocks)
1045 {
1046     BT_HDR *p_cmd_buf;
1047     UINT8 *p, *p_cmd_start;
1048     tNFC_STATUS retval = NFC_STATUS_OK;
1049 
1050     p_cb->cur_cmd = RW_T3T_CMD_CHECK;
1051     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
1052     {
1053         /* Construct T3T message */
1054         p = p_cmd_start = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
1055         rw_t3t_message_set_block_list (p_cb, &p, num_blocks, p_t3t_blocks);
1056 
1057         /* Calculate length of message */
1058         p_cmd_buf->len = (UINT16) (p - p_cmd_start);
1059 
1060         /* Send the T3T message */
1061         retval = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_CHECK, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS * (UINT32) num_blocks);
1062     }
1063     else
1064     {
1065         retval = NFC_STATUS_NO_BUFFERS;
1066     }
1067 
1068     return(retval);
1069 }
1070 
1071 /*****************************************************************************
1072 **
1073 ** Function         rw_t3t_send_update_cmd
1074 **
1075 ** Description      Send UPDATE command
1076 **
1077 ** Returns          tNFC_STATUS
1078 **
1079 *****************************************************************************/
rw_t3t_send_update_cmd(tRW_T3T_CB * p_cb,UINT8 num_blocks,tT3T_BLOCK_DESC * p_t3t_blocks,UINT8 * p_data)1080 tNFC_STATUS rw_t3t_send_update_cmd (tRW_T3T_CB *p_cb, UINT8 num_blocks, tT3T_BLOCK_DESC *p_t3t_blocks, UINT8 *p_data)
1081 {
1082     BT_HDR *p_cmd_buf;
1083     UINT8 *p, *p_cmd_start;
1084     tNFC_STATUS retval = NFC_STATUS_OK;
1085 
1086     p_cb->cur_cmd = RW_T3T_CMD_UPDATE;
1087     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
1088     {
1089         /* Construct T3T message */
1090         p = p_cmd_start = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
1091         rw_t3t_message_set_block_list (p_cb, &p, num_blocks, p_t3t_blocks);
1092 
1093         /* Add data blocks to the message */
1094         ARRAY_TO_STREAM (p, p_data, num_blocks*16);
1095 
1096         /* Calculate length of message */
1097         p_cmd_buf->len = (UINT16) (p - p_cmd_start);
1098 
1099         /* Send the T3T message */
1100         retval = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_UPDATE, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS * (UINT32) num_blocks);
1101     }
1102     else
1103     {
1104         retval = NFC_STATUS_NO_BUFFERS;
1105     }
1106 
1107     return(retval);
1108 }
1109 
1110 /*****************************************************************************
1111 **
1112 ** Function         rw_t3t_check_mc_block
1113 **
1114 ** Description      Send command to check Memory Configuration Block
1115 **
1116 ** Returns          tNFC_STATUS
1117 **
1118 *****************************************************************************/
rw_t3t_check_mc_block(tRW_T3T_CB * p_cb)1119 tNFC_STATUS rw_t3t_check_mc_block (tRW_T3T_CB *p_cb)
1120 {
1121     BT_HDR *p_cmd_buf;
1122     UINT8 *p, *p_cmd_start;
1123 
1124     /* Read Memory Configuration block */
1125     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
1126     {
1127         /* Construct T3T message */
1128         p = p_cmd_start = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
1129 
1130         /* Add CHECK opcode to message  */
1131         UINT8_TO_STREAM (p, T3T_MSG_OPC_CHECK_CMD);
1132 
1133         /* Add IDm to message */
1134         ARRAY_TO_STREAM (p, p_cb->peer_nfcid2, NCI_NFCID2_LEN);
1135 
1136         /* Add Service code list */
1137         UINT8_TO_STREAM (p, 1);                       /* Number of services (only 1 service: NDEF) */
1138         UINT16_TO_STREAM (p, T3T_MSG_NDEF_SC_RO);     /* Service code (little-endian format) */
1139 
1140         /* Number of blocks */
1141         UINT8_TO_STREAM (p, 1);                       /* Number of blocks (only 1 block: Memory Configuration Information ) */
1142 
1143         /* Block List element: the Memory Configuration block (block 0x88) */
1144         UINT8_TO_STREAM (p, T3T_MSG_MASK_TWO_BYTE_BLOCK_DESC_FORMAT);
1145         UINT8_TO_STREAM (p, T3T_MSG_FELICALITE_BLOCK_ID_MC);
1146 
1147         /* Calculate length of message */
1148         p_cmd_buf->len = (UINT16) (p - p_cmd_start);
1149 
1150         /* Send the T3T message */
1151         return rw_t3t_send_cmd (p_cb, p_cb->cur_cmd, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS);
1152     }
1153     else
1154     {
1155         RW_TRACE_ERROR0 ("Unable to allocate buffer to read MC block");
1156         return (NFC_STATUS_NO_BUFFERS);
1157     }
1158 }
1159 
1160 /*****************************************************************************
1161 **
1162 ** Function         rw_t3t_send_raw_frame
1163 **
1164 ** Description      Send raw frame
1165 **
1166 ** Returns          tNFC_STATUS
1167 **
1168 *****************************************************************************/
rw_t3t_send_raw_frame(tRW_T3T_CB * p_cb,UINT16 len,UINT8 * p_data)1169 tNFC_STATUS rw_t3t_send_raw_frame (tRW_T3T_CB *p_cb, UINT16 len, UINT8 *p_data)
1170 {
1171     BT_HDR *p_cmd_buf;
1172     UINT8 *p;
1173     tNFC_STATUS retval = NFC_STATUS_OK;
1174 
1175     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
1176     {
1177         /* Construct T3T message */
1178         p = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
1179 
1180         /* Add data blocks to the message */
1181         ARRAY_TO_STREAM (p, p_data, len);
1182 
1183         /* Calculate length of message */
1184         p_cmd_buf->len = len;
1185 
1186         /* Send the T3T message */
1187         retval = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_SEND_RAW_FRAME, p_cmd_buf, RW_T3T_RAW_FRAME_CMD_TIMEOUT_TICKS);
1188     }
1189     else
1190     {
1191         retval = NFC_STATUS_NO_BUFFERS;
1192     }
1193 
1194     return (retval);
1195 }
1196 
1197 
1198 /*****************************************************************************
1199 **  TAG RESPONSE HANDLERS
1200 *****************************************************************************/
1201 
1202 
1203 /*****************************************************************************
1204 **
1205 ** Function         rw_t3t_act_handle_ndef_detect_rsp
1206 **
1207 ** Description      Handle response to NDEF detection
1208 **
1209 ** Returns          Nothing
1210 **
1211 *****************************************************************************/
rw_t3t_act_handle_ndef_detect_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1212 void rw_t3t_act_handle_ndef_detect_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1213 {
1214     UINT8 *p;
1215     UINT32 temp;
1216     UINT8 i;
1217     UINT16 checksum_calc, checksum_rx;
1218     tRW_DETECT_NDEF_DATA evt_data;
1219     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
1220 
1221     evt_data.status = NFC_STATUS_FAILED;
1222     evt_data.flags  = RW_NDEF_FL_UNKNOWN;
1223 
1224     /* Check if response code is CHECK resp (for reading NDEF attribute block) */
1225     if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_CHECK_RSP)
1226     {
1227         RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_CHECK_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
1228         evt_data.status = NFC_STATUS_FAILED;
1229     }
1230     /* Validate status code and NFCID2 response from tag */
1231     else if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)                          /* verify response status code */
1232              ||(memcmp (p_cb->peer_nfcid2, &p_t3t_rsp[T3T_MSG_RSP_OFFSET_IDM], NCI_NFCID2_LEN) != 0)  )   /* verify response IDm */
1233     {
1234         evt_data.status = NFC_STATUS_FAILED;
1235     }
1236     else
1237     {
1238         /* Get checksum from received ndef attribute msg */
1239         p = &p_t3t_rsp[T3T_MSG_RSP_OFFSET_CHECK_DATA+T3T_MSG_NDEF_ATTR_INFO_SIZE];
1240         BE_STREAM_TO_UINT16 (checksum_rx, p);
1241 
1242         /* Calculate checksum - move check for checsum to beginning */
1243         checksum_calc = 0;
1244         p = &p_t3t_rsp[T3T_MSG_RSP_OFFSET_CHECK_DATA];
1245         for (i = 0; i < T3T_MSG_NDEF_ATTR_INFO_SIZE; i++)
1246         {
1247             checksum_calc+=p[i];
1248         }
1249 
1250         /* Validate checksum */
1251         if (checksum_calc != checksum_rx)
1252         {
1253             p_cb->ndef_attrib.status = NFC_STATUS_FAILED;       /* only ok or failed passed to the app. can be boolean*/
1254 
1255             RW_TRACE_ERROR0 ("RW_T3tDetectNDEF checksum failed");
1256         }
1257         else
1258         {
1259             p_cb->ndef_attrib.status = NFC_STATUS_OK;
1260 
1261             /* Validate version number */
1262             STREAM_TO_UINT8 (p_cb->ndef_attrib.version, p);
1263 
1264             if (T3T_GET_MAJOR_VERSION (T3T_MSG_NDEF_VERSION) < T3T_GET_MAJOR_VERSION (p_cb->ndef_attrib.version))
1265             {
1266                 /* Remote tag's MajorVer is newer than our's. Reject NDEF as per T3TOP RQ_T3T_NDA_024 */
1267                 RW_TRACE_ERROR2 ("RW_T3tDetectNDEF: incompatible NDEF version. Local=0x%02x, Remote=0x%02x", T3T_MSG_NDEF_VERSION, p_cb->ndef_attrib.version);
1268                 p_cb->ndef_attrib.status = NFC_STATUS_FAILED;
1269                 evt_data.status = NFC_STATUS_BAD_RESP;
1270             }
1271             else
1272             {
1273                 /* Remote tag's MajorVer is equal or older than our's. NDEF is compatible with our version. */
1274 
1275                 /* Update NDEF info */
1276                 STREAM_TO_UINT8 (p_cb->ndef_attrib.nbr, p);              /* NBr: number of blocks that can be read using one Check command */
1277                 STREAM_TO_UINT8 (p_cb->ndef_attrib.nbw, p);              /* Nbw: number of blocks that can be written using one Update command */
1278                 BE_STREAM_TO_UINT16 (p_cb->ndef_attrib.nmaxb, p);        /* Nmaxb: maximum number of blocks available for NDEF data */
1279                 BE_STREAM_TO_UINT32 (temp, p);
1280                 STREAM_TO_UINT8 (p_cb->ndef_attrib.writef, p);           /* WriteFlag: 00h if writing data finished; 0Fh if writing data in progress */
1281                 STREAM_TO_UINT8 (p_cb->ndef_attrib.rwflag, p);           /* RWFlag: 00h NDEF is read-only; 01h if read/write available */
1282 
1283                 /* Get length (3-byte, big-endian) */
1284                 STREAM_TO_UINT8 (temp, p);                               /* Ln: high-byte */
1285                 BE_STREAM_TO_UINT16 (p_cb->ndef_attrib.ln, p);           /* Ln: lo-word */
1286                 p_cb->ndef_attrib.ln += (temp << 16);
1287 
1288 
1289                 RW_TRACE_DEBUG1 ("Detected NDEF Ver: 0x%02x", p_cb->ndef_attrib.version);
1290                 RW_TRACE_DEBUG6 ("Detected NDEF Attributes: Nbr=%i, Nbw=%i, Nmaxb=%i, WriteF=%i, RWFlag=%i, Ln=%i",
1291                     p_cb->ndef_attrib.nbr,
1292                     p_cb->ndef_attrib.nbw,
1293                     p_cb->ndef_attrib.nmaxb,
1294                     p_cb->ndef_attrib.writef,
1295                     p_cb->ndef_attrib.rwflag,
1296                     p_cb->ndef_attrib.ln);
1297 
1298                 /* Set data for RW_T3T_NDEF_DETECT_EVT */
1299                 evt_data.status = p_cb->ndef_attrib.status;
1300                 evt_data.cur_size = p_cb->ndef_attrib.ln;
1301                 evt_data.max_size = (UINT32) p_cb->ndef_attrib.nmaxb * 16;
1302                 evt_data.protocol = NFC_PROTOCOL_T3T;
1303                 evt_data.flags    = (RW_NDEF_FL_SUPPORTED | RW_NDEF_FL_FORMATED);
1304                 if (p_cb->ndef_attrib.rwflag == T3T_MSG_NDEF_RWFLAG_RO)
1305                     evt_data.flags    |= RW_NDEF_FL_READ_ONLY;
1306             }
1307         }
1308     }
1309 
1310     RW_TRACE_DEBUG1 ("RW_T3tDetectNDEF response: %i", evt_data.status);
1311 
1312     p_cb->rw_state = RW_T3T_STATE_IDLE;
1313 
1314     /* Notify app of NDEF detection result */
1315     (*(rw_cb.p_cback)) (RW_T3T_NDEF_DETECT_EVT, (tRW_DATA *) &evt_data);
1316 
1317     GKI_freebuf (p_msg_rsp);
1318 }
1319 
1320 
1321 /*****************************************************************************
1322 **
1323 ** Function         rw_t3t_act_handle_check_rsp
1324 **
1325 ** Description      Handle response to CHECK command
1326 **
1327 ** Returns          Nothing
1328 **
1329 *****************************************************************************/
rw_t3t_act_handle_check_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1330 void rw_t3t_act_handle_check_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1331 {
1332     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
1333     tRW_READ_DATA evt_data;
1334     tNFC_STATUS nfc_status = NFC_STATUS_OK;
1335 
1336     /* Validate response from tag */
1337     if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)                           /* verify response status code */
1338         ||(memcmp (p_cb->peer_nfcid2, &p_t3t_rsp[T3T_MSG_RSP_OFFSET_IDM], NCI_NFCID2_LEN) != 0)  )   /* verify response IDm */
1339     {
1340         nfc_status = NFC_STATUS_FAILED;
1341         GKI_freebuf (p_msg_rsp);
1342     }
1343     else if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_CHECK_RSP)
1344     {
1345         RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_CHECK_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
1346         nfc_status = NFC_STATUS_FAILED;
1347         GKI_freebuf (p_msg_rsp);
1348     }
1349     else
1350     {
1351         /* Copy incoming data into buffer */
1352         p_msg_rsp->offset += T3T_MSG_RSP_OFFSET_CHECK_DATA;     /* Skip over t3t header */
1353         p_msg_rsp->len -= T3T_MSG_RSP_OFFSET_CHECK_DATA;
1354         evt_data.status = NFC_STATUS_OK;
1355         evt_data.p_data = p_msg_rsp;
1356         (*(rw_cb.p_cback)) (RW_T3T_CHECK_EVT, (tRW_DATA *) &evt_data);
1357     }
1358 
1359 
1360     p_cb->rw_state = RW_T3T_STATE_IDLE;
1361 
1362     (*(rw_cb.p_cback)) (RW_T3T_CHECK_CPLT_EVT, (tRW_DATA *) &nfc_status);
1363 }
1364 
1365 /*****************************************************************************
1366 **
1367 ** Function         rw_t3t_act_handle_update_rsp
1368 **
1369 ** Description      Handle response to UPDATE command
1370 **
1371 ** Returns          Nothing
1372 **
1373 *****************************************************************************/
rw_t3t_act_handle_update_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1374 void rw_t3t_act_handle_update_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1375 {
1376     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
1377     tRW_READ_DATA evt_data;
1378 
1379     /* Validate response from tag */
1380     if (   (p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)                          /* verify response status code */
1381         ||(memcmp (p_cb->peer_nfcid2, &p_t3t_rsp[T3T_MSG_RSP_OFFSET_IDM], NCI_NFCID2_LEN) != 0)  )   /* verify response IDm */
1382     {
1383         evt_data.status = NFC_STATUS_FAILED;
1384     }
1385     else if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_UPDATE_RSP)
1386     {
1387         RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_UPDATE_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
1388         evt_data.status = NFC_STATUS_FAILED;
1389     }
1390     else
1391     {
1392         /* Copy incoming data into buffer */
1393         evt_data.status = NFC_STATUS_OK;
1394     }
1395 
1396     p_cb->rw_state = RW_T3T_STATE_IDLE;
1397 
1398     (*(rw_cb.p_cback)) (RW_T3T_UPDATE_CPLT_EVT, (tRW_DATA *)&evt_data);
1399 
1400     GKI_freebuf (p_msg_rsp);
1401 }
1402 
1403 /*****************************************************************************
1404 **
1405 ** Function         rw_t3t_act_handle_raw_senddata_rsp
1406 **
1407 ** Description      Handle response to NDEF detection
1408 **
1409 ** Returns          Nothing
1410 **
1411 *****************************************************************************/
rw_t3t_act_handle_raw_senddata_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1412 void rw_t3t_act_handle_raw_senddata_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1413 {
1414     tRW_READ_DATA evt_data;
1415 
1416     /* Copy incoming data into buffer */
1417     evt_data.status = NFC_STATUS_OK;
1418     evt_data.p_data = p_msg_rsp;
1419 
1420     p_cb->rw_state = RW_T3T_STATE_IDLE;
1421 
1422     (*(rw_cb.p_cback)) (RW_T3T_RAW_FRAME_EVT, (tRW_DATA *) &evt_data);
1423 }
1424 
1425 /*****************************************************************************
1426 **
1427 ** Function         rw_t3t_act_handle_check_ndef_rsp
1428 **
1429 ** Description      Handle response to NDEF read segment
1430 **
1431 ** Returns          Nothing
1432 **
1433 *****************************************************************************/
rw_t3t_act_handle_check_ndef_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1434 void rw_t3t_act_handle_check_ndef_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1435 {
1436     BOOLEAN check_complete = TRUE;
1437     tNFC_STATUS nfc_status = NFC_STATUS_OK;
1438     tRW_READ_DATA read_data;
1439     tRW_DATA evt_data;
1440     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
1441     UINT8 rsp_num_bytes_rx;
1442 
1443     /* Validate response from tag */
1444     if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)                      /* verify response status code */
1445         ||(memcmp (p_cb->peer_nfcid2, &p_t3t_rsp[T3T_MSG_RSP_OFFSET_IDM], NCI_NFCID2_LEN) != 0) /* verify response IDm */
1446         ||(p_t3t_rsp[T3T_MSG_RSP_OFFSET_NUMBLOCKS] != ((p_cb->ndef_rx_readlen+15) >> 4))  )     /* verify length of response */
1447     {
1448         RW_TRACE_ERROR2 ("Response error: bad status, nfcid2, or invalid len: %i %i", p_t3t_rsp[T3T_MSG_RSP_OFFSET_NUMBLOCKS], ((p_cb->ndef_rx_readlen+15)>>4));
1449         nfc_status = NFC_STATUS_FAILED;
1450         GKI_freebuf (p_msg_rsp);
1451     }
1452     else if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_CHECK_RSP)
1453     {
1454         RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_CHECK_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
1455         nfc_status = NFC_STATUS_FAILED;
1456         GKI_freebuf (p_msg_rsp);
1457     }
1458     else
1459     {
1460         /* Notify app of NDEF segment received */
1461         rsp_num_bytes_rx = p_t3t_rsp[T3T_MSG_RSP_OFFSET_NUMBLOCKS] * 16;    /* Number of bytes received, according to header */
1462         p_cb->ndef_rx_offset += p_cb->ndef_rx_readlen;
1463         read_data.status = NFC_STATUS_OK;
1464         p_msg_rsp->offset += T3T_MSG_RSP_OFFSET_CHECK_DATA;     /* Skip over t3t header (point to block data) */
1465         p_msg_rsp->len -= T3T_MSG_RSP_OFFSET_CHECK_DATA;
1466 
1467         /* Verify that the bytes received is really the amount indicated in the check-response header */
1468         if (rsp_num_bytes_rx > p_msg_rsp->len)
1469         {
1470             RW_TRACE_ERROR2 ("Response error: CHECK rsp header indicates %i bytes, but only received %i bytes", rsp_num_bytes_rx, p_msg_rsp->len);
1471             nfc_status = NFC_STATUS_FAILED;
1472             GKI_freebuf (p_msg_rsp);
1473         }
1474         else
1475         {
1476             /* If this is the the final block, then set len to reflect only valid bytes (do not include padding to 16-byte boundary) */
1477             if ((p_cb->flags & RW_T3T_FL_IS_FINAL_NDEF_SEGMENT) && (p_cb->ndef_attrib.ln & 0x000F))
1478             {
1479                 rsp_num_bytes_rx -= (16 - (p_cb->ndef_attrib.ln & 0x000F));
1480             }
1481 
1482             p_msg_rsp->len = rsp_num_bytes_rx;
1483             read_data.p_data = p_msg_rsp;
1484             (*(rw_cb.p_cback)) (RW_T3T_CHECK_EVT, (tRW_DATA *) &read_data);
1485 
1486             /* Send CHECK cmd for next NDEF segment, if needed */
1487             if (!(p_cb->flags & RW_T3T_FL_IS_FINAL_NDEF_SEGMENT))
1488             {
1489                 if ((nfc_status = rw_t3t_send_next_ndef_check_cmd (p_cb)) == NFC_STATUS_OK)
1490                 {
1491                     /* Still getting more segments. Don't send RW_T3T_CHECK_CPLT_EVT yet */
1492                     check_complete = FALSE;
1493                 }
1494             }
1495         }
1496     }
1497 
1498     /* Notify app of RW_T3T_CHECK_CPLT_EVT if entire NDEF has been read, or if failure */
1499     if (check_complete)
1500     {
1501         p_cb->rw_state = RW_T3T_STATE_IDLE;
1502         evt_data.status = nfc_status;
1503         (*(rw_cb.p_cback)) (RW_T3T_CHECK_CPLT_EVT, (tRW_DATA *) &evt_data);
1504     }
1505 }
1506 
1507 
1508 /*****************************************************************************
1509 **
1510 ** Function         rw_t3t_act_handle_update_ndef_rsp
1511 **
1512 ** Description      Handle response to NDEF write segment
1513 **
1514 ** Returns          Nothing
1515 **
1516 *****************************************************************************/
rw_t3t_act_handle_update_ndef_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1517 void rw_t3t_act_handle_update_ndef_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1518 {
1519     BOOLEAN update_complete = TRUE;
1520     tNFC_STATUS nfc_status = NFC_STATUS_OK;
1521     tRW_DATA evt_data;
1522     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
1523 
1524     /* Check nfcid2 and status of response */
1525     if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)                           /* verify response status code */
1526         ||(memcmp (p_cb->peer_nfcid2, &p_t3t_rsp[T3T_MSG_RSP_OFFSET_IDM], NCI_NFCID2_LEN) != 0)  )   /* verify response IDm */
1527     {
1528         nfc_status = NFC_STATUS_FAILED;
1529     }
1530     /* Validate response opcode */
1531     else if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_UPDATE_RSP)
1532     {
1533         RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_UPDATE_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
1534         nfc_status = NFC_STATUS_FAILED;
1535     }
1536     /* If this is response to final UPDATE, then update NDEF local size */
1537     else if (p_cb->flags & RW_T3T_FL_IS_FINAL_NDEF_SEGMENT)
1538     {
1539         /* If successful, update current NDEF size */
1540         p_cb->ndef_attrib.ln = p_cb->ndef_msg_len;
1541     }
1542     /*  If any more NDEF bytes to update, then send next UPDATE command */
1543     else if (p_cb->ndef_msg_bytes_sent < p_cb->ndef_msg_len)
1544     {
1545         /* Send UPDATE command for next segment of NDEF */
1546         if ((nfc_status = rw_t3t_send_next_ndef_update_cmd (p_cb)) == NFC_STATUS_OK)
1547         {
1548             /* Wait for update response */
1549             update_complete = FALSE;
1550         }
1551     }
1552     /*  Otherwise, no more NDEF bytes. Send final UPDATE for Attribute Information block */
1553     else
1554     {
1555         p_cb->flags |= RW_T3T_FL_IS_FINAL_NDEF_SEGMENT;
1556         if ((nfc_status = rw_t3t_send_update_ndef_attribute_cmd (p_cb, FALSE)) == NFC_STATUS_OK)
1557         {
1558             /* Wait for update response */
1559             update_complete = FALSE;
1560         }
1561     }
1562 
1563     /* If update is completed, then notify app */
1564     if (update_complete)
1565     {
1566         p_cb->rw_state = RW_T3T_STATE_IDLE;
1567         evt_data.status = nfc_status;
1568         (*(rw_cb.p_cback)) (RW_T3T_UPDATE_CPLT_EVT, (tRW_DATA *) &evt_data);
1569     }
1570 
1571 
1572     GKI_freebuf (p_msg_rsp);
1573 
1574     return;
1575 }
1576 
1577 
1578 /*****************************************************************************
1579 **
1580 ** Function         rw_t3t_handle_get_sc_poll_rsp
1581 **
1582 ** Description      Handle POLL response for getting system codes
1583 **
1584 ** Returns          Nothing
1585 **
1586 *****************************************************************************/
rw_t3t_handle_get_sc_poll_rsp(tRW_T3T_CB * p_cb,UINT8 nci_status,UINT8 num_responses,UINT8 sensf_res_buf_size,UINT8 * p_sensf_res_buf)1587 static void rw_t3t_handle_get_sc_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf)
1588 {
1589     BT_HDR *p_cmd_buf;
1590     UINT8 *p, *p_cmd_start;
1591     UINT16 sc;
1592     tNFC_STATUS status = NFC_STATUS_FAILED;
1593 
1594     /* If waiting for wildcard POLL */
1595     if (p_cb->rw_substate == RW_T3T_GET_SC_SST_POLL_WILDCARD)
1596     {
1597         /* Get the system code from the response */
1598         if (  (nci_status == NCI_STATUS_OK)
1599             &&(num_responses > 0)
1600             &&(sensf_res_buf_size >= (RW_T3T_SENSF_RES_RD_OFFSET + RW_T3T_SENSF_RES_RD_LEN))  )
1601         {
1602             p = &p_sensf_res_buf[RW_T3T_SENSF_RES_RD_OFFSET];
1603             BE_STREAM_TO_UINT16 (sc, p);
1604 
1605             /* Handle felica lite */
1606             if (sc == T3T_SYSTEM_CODE_FELICA_LITE)
1607             {
1608                 RW_TRACE_DEBUG1 ("FeliCa Lite tag detected (system code %04X)", sc);
1609                 /* Store system code */
1610                 p_cb->system_codes[p_cb->num_system_codes++] = sc;
1611 
1612                 /* Poll for NDEF system code */
1613                 if ((status = (tNFC_STATUS) nci_snd_t3t_polling (T3T_SYSTEM_CODE_NDEF, 0, 0)) == NCI_STATUS_OK)
1614                 {
1615                     p_cb->rw_substate = RW_T3T_GET_SC_SST_POLL_NDEF;
1616                     p_cb->cur_poll_rc = 0;
1617                     p_cb->flags |= RW_T3T_FL_W4_GET_SC_POLL_RSP;
1618 
1619                     /* start timer for waiting for responses */
1620                     rw_t3t_start_poll_timer (p_cb);
1621                 }
1622             }
1623             else
1624             {
1625                 /* All other types, send REQUEST_SYSTEM_CODE command */
1626                 if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
1627                 {
1628                     p_cb->rw_substate = RW_T3T_GET_SC_SST_REQUEST_SC;
1629 
1630                     /* Construct T3T message */
1631                     p_cmd_start = p = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
1632                     UINT8_TO_STREAM (p, T3T_MSG_OPC_REQ_SYSTEMCODE_CMD);
1633                     ARRAY_TO_STREAM (p, p_cb->peer_nfcid2, NCI_NFCID2_LEN);
1634 
1635                     /* Fill in length field */
1636                     p_cmd_buf->len = (UINT16) (p - p_cmd_start);
1637 
1638                     /* Send the T3T message */
1639                     status = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_GET_SYSTEM_CODES, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS);
1640                 }
1641             }
1642         }
1643 
1644         /* Error proceeding. Notify upper layer of system codes found so far */
1645         if (status != NFC_STATUS_OK)
1646         {
1647             rw_t3t_handle_get_system_codes_cplt ();
1648         }
1649     }
1650     /* If waiting for NDEF POLL */
1651     else if (p_cb->rw_substate == RW_T3T_GET_SC_SST_POLL_NDEF)
1652     {
1653         /* Validate response for NDEF poll */
1654         if ((nci_status == NCI_STATUS_OK) && (num_responses > 0))
1655         {
1656             /* Tag responded for NDEF poll */
1657             p_cb->system_codes[p_cb->num_system_codes++] = T3T_SYSTEM_CODE_NDEF;
1658         }
1659         rw_t3t_handle_get_system_codes_cplt ();
1660     }
1661 }
1662 
1663 /*****************************************************************************
1664 **
1665 ** Function         rw_t3t_handle_ndef_detect_poll_rsp
1666 **
1667 ** Description      Handle POLL response for getting system codes
1668 **
1669 ** Returns          Nothing
1670 **
1671 *****************************************************************************/
rw_t3t_handle_ndef_detect_poll_rsp(tRW_T3T_CB * p_cb,UINT8 nci_status,UINT8 num_responses,UINT8 sensf_res_buf_size,UINT8 * p_sensf_res_buf)1672 static void rw_t3t_handle_ndef_detect_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf)
1673 {
1674     BT_HDR *p_cmd_buf;
1675     UINT8 *p, *p_cmd_start;
1676     tRW_DATA evt_data;
1677 
1678     /* Validate response for NDEF poll */
1679     if ((nci_status == NCI_STATUS_OK) && (num_responses > 0))
1680     {
1681         /* Tag responded for NDEF poll */
1682 
1683         /* Read NDEF attribute block */
1684         if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
1685         {
1686             /* Construct T3T message */
1687             p = p_cmd_start = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
1688 
1689             /* Add CHECK opcode to message  */
1690             UINT8_TO_STREAM (p, T3T_MSG_OPC_CHECK_CMD);
1691 
1692             /* Add IDm to message */
1693             ARRAY_TO_STREAM (p, p_cb->peer_nfcid2, NCI_NFCID2_LEN);
1694 
1695             /* Add Service code list */
1696             UINT8_TO_STREAM (p, 1);                       /* Number of services (only 1 service: NDEF) */
1697             UINT16_TO_STREAM (p, T3T_MSG_NDEF_SC_RO);     /* Service code (little-endian format) */
1698 
1699             /* Number of blocks */
1700             UINT8_TO_STREAM (p, 1);                       /* Number of blocks (only 1 block: NDEF Attribute Information ) */
1701 
1702             /* Block List element: the NDEF attribute information block (block 0) */
1703             UINT8_TO_STREAM (p, T3T_MSG_MASK_TWO_BYTE_BLOCK_DESC_FORMAT);
1704             UINT8_TO_STREAM (p, 0);
1705 
1706             /* Calculate length of message */
1707             p_cmd_buf->len = (UINT16) (p - p_cmd_start);
1708 
1709             /* Send the T3T message */
1710             if ((evt_data.status = rw_t3t_send_cmd (p_cb, RW_T3T_CMD_DETECT_NDEF, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS)) == NFC_STATUS_OK)
1711             {
1712                 /* CHECK command sent. Wait for response */
1713                 return;
1714             }
1715         }
1716         nci_status = NFC_STATUS_FAILED;
1717     }
1718 
1719     /* NDEF detection failed */
1720     p_cb->rw_state = RW_T3T_STATE_IDLE;
1721     evt_data.ndef.status = nci_status;
1722     evt_data.ndef.flags  = RW_NDEF_FL_UNKNOWN;
1723     (*(rw_cb.p_cback)) (RW_T3T_NDEF_DETECT_EVT, &evt_data);
1724 }
1725 
1726 
1727 /*****************************************************************************
1728 **
1729 ** Function         rw_t3t_act_handle_get_sc_rsp
1730 **
1731 ** Description      Handle response for getting system codes
1732 **
1733 ** Returns          Nothing
1734 **
1735 *****************************************************************************/
rw_t3t_act_handle_get_sc_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1736 void rw_t3t_act_handle_get_sc_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1737 {
1738     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
1739     UINT8 *p;
1740     UINT16 sc;
1741     UINT8 num_sc, i;
1742 
1743     /* Validate response opcode */
1744     if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_REQ_SYSTEMCODE_RSP)
1745     {
1746         RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_REQ_SYSTEMCODE_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
1747     }
1748     else
1749     {
1750         /* Point to number of systems parameter */
1751         p = &p_t3t_rsp[T3T_MSG_RSP_OFFSET_NUMSYS];
1752         STREAM_TO_UINT8 (num_sc, p);
1753 
1754         /* Validate maximum */
1755         if (num_sc>T3T_MAX_SYSTEM_CODES)
1756         {
1757             RW_TRACE_DEBUG2 ("Tag's number of systems (%i) exceeds NFA max (%i)", num_sc, T3T_MAX_SYSTEM_CODES);
1758             num_sc = T3T_MAX_SYSTEM_CODES;
1759         }
1760 
1761         for (i = 0; i < num_sc; i++)
1762         {
1763             BE_STREAM_TO_UINT16 (sc, p);
1764             p_cb->system_codes[p_cb->num_system_codes++] = sc;
1765         }
1766     }
1767     rw_t3t_handle_get_system_codes_cplt ();
1768 
1769     GKI_freebuf (p_msg_rsp);
1770 }
1771 
1772 /*****************************************************************************
1773 **
1774 ** Function         rw_t3t_update_block
1775 **
1776 ** Description      Send UPDATE command for single block
1777 **                  (for formatting/configuring read only)
1778 **
1779 ** Returns          tNFC_STATUS
1780 **
1781 *****************************************************************************/
rw_t3t_update_block(tRW_T3T_CB * p_cb,UINT8 block_id,UINT8 * p_block_data)1782 tNFC_STATUS rw_t3t_update_block (tRW_T3T_CB *p_cb, UINT8 block_id, UINT8 *p_block_data)
1783 {
1784     UINT8 *p_dst, *p_cmd_start;
1785     BT_HDR *p_cmd_buf;
1786     tNFC_STATUS status;
1787 
1788     if ((p_cmd_buf = rw_t3t_get_cmd_buf ()) != NULL)
1789     {
1790         p_dst = p_cmd_start = (UINT8 *) (p_cmd_buf+1) + p_cmd_buf->offset;
1791 
1792         /* Add UPDATE opcode to message  */
1793         UINT8_TO_STREAM (p_dst, T3T_MSG_OPC_UPDATE_CMD);
1794 
1795         /* Add IDm to message */
1796         ARRAY_TO_STREAM (p_dst, p_cb->peer_nfcid2, NCI_NFCID2_LEN);
1797 
1798         /* Add Service code list */
1799         UINT8_TO_STREAM (p_dst, 1);                      /* Number of services (only 1 service: NDEF) */
1800         UINT16_TO_STREAM (p_dst, T3T_MSG_NDEF_SC_RW);    /* Service code (little-endian format) */
1801 
1802         /* Number of blocks */
1803         UINT8_TO_STREAM (p_dst, 1);
1804 
1805         /* Add Block list element for MC */
1806         UINT8_TO_STREAM (p_dst, T3T_MSG_MASK_TWO_BYTE_BLOCK_DESC_FORMAT);
1807         UINT8_TO_STREAM (p_dst, block_id);
1808 
1809         /* Copy MC data to UPDATE message */
1810         ARRAY_TO_STREAM (p_dst, p_block_data, T3T_MSG_BLOCKSIZE);
1811 
1812         /* Calculate length of message */
1813         p_cmd_buf->len = (UINT16) (p_dst - p_cmd_start);
1814 
1815         /* Send the T3T message */
1816         status = rw_t3t_send_cmd (p_cb, p_cb->cur_cmd, p_cmd_buf, RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS);
1817     }
1818     else
1819     {
1820         /* Unable to send UPDATE command */
1821         status = NFC_STATUS_NO_BUFFERS;
1822     }
1823 
1824     return (status);
1825 }
1826 
1827 /*****************************************************************************
1828 **
1829 ** Function         rw_t3t_handle_fmt_poll_rsp
1830 **
1831 ** Description      Handle POLL response for formatting felica-lite
1832 **
1833 ** Returns          Nothing
1834 **
1835 *****************************************************************************/
rw_t3t_handle_fmt_poll_rsp(tRW_T3T_CB * p_cb,UINT8 nci_status,UINT8 num_responses,UINT8 sensf_res_buf_size,UINT8 * p_sensf_res_buf)1836 static void rw_t3t_handle_fmt_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf)
1837 {
1838     tRW_DATA evt_data;
1839 
1840     evt_data.status = NFC_STATUS_OK;
1841 
1842     /* Validate response for poll response */
1843     if ((nci_status == NCI_STATUS_OK) && (num_responses > 0))
1844     {
1845         /* Tag responded for Felica-Lite poll */
1846         /* Get MemoryControl block */
1847         RW_TRACE_DEBUG0 ("Felica-Lite tag detected...getting Memory Control block.");
1848 
1849         p_cb->rw_substate = RW_T3T_FMT_SST_CHECK_MC_BLK;
1850 
1851         /* Send command to check Memory Configuration block */
1852         evt_data.status = rw_t3t_check_mc_block (p_cb);
1853     }
1854     else
1855     {
1856         RW_TRACE_ERROR0 ("Felica-Lite tag not detected");
1857         evt_data.status = NFC_STATUS_FAILED;
1858     }
1859 
1860     /* If error, notify upper layer */
1861     if (evt_data.status != NFC_STATUS_OK)
1862     {
1863         rw_t3t_format_cplt (evt_data.status);
1864     }
1865 }
1866 
1867 /*****************************************************************************
1868 **
1869 ** Function         rw_t3t_act_handle_fmt_rsp
1870 **
1871 ** Description      Handle response for formatting codes
1872 **
1873 ** Returns          Nothing
1874 **
1875 *****************************************************************************/
rw_t3t_act_handle_fmt_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)1876 void rw_t3t_act_handle_fmt_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
1877 {
1878     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
1879     UINT8 *p_mc;
1880     tRW_DATA evt_data;
1881 
1882     evt_data.status = NFC_STATUS_OK;
1883 
1884     /* Check tags's response for reading MemoryControl block */
1885     if (p_cb->rw_substate == RW_T3T_FMT_SST_CHECK_MC_BLK)
1886     {
1887         /* Validate response opcode */
1888         if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_CHECK_RSP)
1889         {
1890             RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_CHECK_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
1891             evt_data.status = NFC_STATUS_FAILED;
1892         }
1893         /* Validate status code and NFCID2 response from tag */
1894         else if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)                           /* verify response status code */
1895                  ||(memcmp (p_cb->peer_nfcid2, &p_t3t_rsp[T3T_MSG_RSP_OFFSET_IDM], NCI_NFCID2_LEN) != 0)  )   /* verify response IDm */
1896         {
1897             evt_data.status = NFC_STATUS_FAILED;
1898         }
1899         else
1900         {
1901             /* Check if memory configuration (MC) block to see if SYS_OP=1 (NDEF enabled) */
1902             p_mc = &p_t3t_rsp[T3T_MSG_RSP_OFFSET_CHECK_DATA];  /* Point to MC data of CHECK response */
1903 
1904             if (p_mc[T3T_MSG_FELICALITE_MC_OFFSET_SYS_OP] != 0x01)
1905             {
1906                 /* Tag is not currently enabled for NDEF. Indicate that we need to update the MC block */
1907 
1908                 /* Set SYS_OP field to 0x01 (enable NDEF) */
1909                 p_mc[T3T_MSG_FELICALITE_MC_OFFSET_SYS_OP] = 0x01;
1910 
1911                 /* Set RF_PRM field to 0x07 (procedure of issuance) */
1912                 p_mc[T3T_MSG_FELICALITE_MC_OFFSET_RF_PRM] = 0x07;
1913 
1914                 /* Construct and send UPDATE message to write MC block */
1915                 p_cb->rw_substate = RW_T3T_FMT_SST_UPDATE_MC_BLK;
1916                 evt_data.status = rw_t3t_update_block (p_cb, T3T_MSG_FELICALITE_BLOCK_ID_MC, p_mc);
1917             }
1918             else
1919             {
1920                 /* SYS_OP=1: ndef already enabled. Just need to update attribute information block */
1921                 p_cb->rw_substate = RW_T3T_FMT_SST_UPDATE_NDEF_ATTRIB;
1922                 evt_data.status = rw_t3t_update_block (p_cb, 0, (UINT8 *) rw_t3t_default_attrib_info);
1923             }
1924         }
1925 
1926         /* If error, notify upper layer */
1927         if (evt_data.status != NFC_STATUS_OK)
1928         {
1929             rw_t3t_format_cplt (evt_data.status);
1930         }
1931     }
1932     else if (p_cb->rw_substate == RW_T3T_FMT_SST_UPDATE_MC_BLK)
1933     {
1934         /* Validate response opcode */
1935         if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_UPDATE_RSP)
1936             ||(p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)  )
1937 
1938         {
1939             RW_TRACE_ERROR2 ("Response error: rsp_code=%02X, status=%02X", p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE], p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1]);
1940             evt_data.status = NFC_STATUS_FAILED;
1941         }
1942         else
1943         {
1944             /* SYS_OP=1: ndef already enabled. Just need to update attribute information block */
1945             p_cb->rw_substate = RW_T3T_FMT_SST_UPDATE_NDEF_ATTRIB;
1946             evt_data.status = rw_t3t_update_block (p_cb, 0, (UINT8 *) rw_t3t_default_attrib_info);
1947         }
1948 
1949         /* If error, notify upper layer */
1950         if (evt_data.status != NFC_STATUS_OK)
1951         {
1952             rw_t3t_format_cplt (evt_data.status);
1953         }
1954     }
1955     else if (p_cb->rw_substate == RW_T3T_FMT_SST_UPDATE_NDEF_ATTRIB)
1956     {
1957         /* Validate response opcode */
1958         if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_UPDATE_RSP)
1959             ||(p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)  )
1960 
1961         {
1962             RW_TRACE_ERROR2 ("Response error: rsp_code=%02X, status=%02X", p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE], p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1]);
1963             evt_data.status = NFC_STATUS_FAILED;
1964         }
1965 
1966 
1967         rw_t3t_format_cplt (evt_data.status);
1968     }
1969 
1970     GKI_freebuf (p_msg_rsp);
1971 }
1972 
1973 /*****************************************************************************
1974 **
1975 ** Function         rw_t3t_handle_sro_poll_rsp
1976 **
1977 ** Description      Handle POLL response for configuring felica-lite read only
1978 **
1979 ** Returns          Nothing
1980 **
1981 *****************************************************************************/
rw_t3t_handle_sro_poll_rsp(tRW_T3T_CB * p_cb,UINT8 nci_status,UINT8 num_responses,UINT8 sensf_res_buf_size,UINT8 * p_sensf_res_buf)1982 static void rw_t3t_handle_sro_poll_rsp (tRW_T3T_CB *p_cb, UINT8 nci_status, UINT8 num_responses, UINT8 sensf_res_buf_size, UINT8 *p_sensf_res_buf)
1983 {
1984     tRW_DATA evt_data;
1985     UINT8 rw_t3t_ndef_attrib_info[T3T_MSG_BLOCKSIZE];
1986     UINT8 *p;
1987     UINT8 tempU8;
1988     UINT16 checksum, i;
1989     UINT32 tempU32 = 0;
1990 
1991     evt_data.status = NFC_STATUS_OK;
1992 
1993     /* Validate response for poll response */
1994     if ((nci_status == NCI_STATUS_OK) && (num_responses > 0))
1995     {
1996         /* Tag responded for Felica-Lite poll */
1997         if (p_cb->ndef_attrib.rwflag != T3T_MSG_NDEF_RWFLAG_RO)
1998         {
1999             /* First update attribute information block */
2000             RW_TRACE_DEBUG0 ("Felica-Lite tag detected...update NDef attribution block.");
2001 
2002             p_cb->rw_substate = RW_T3T_SRO_SST_UPDATE_NDEF_ATTRIB;
2003 
2004             p = rw_t3t_ndef_attrib_info;
2005 
2006             UINT8_TO_STREAM (p, p_cb->ndef_attrib.version);
2007 
2008             /* Update NDEF info */
2009             UINT8_TO_STREAM (p, p_cb->ndef_attrib.nbr);              /* NBr: number of blocks that can be read using one Check command */
2010             UINT8_TO_STREAM (p, p_cb->ndef_attrib.nbw);              /* Nbw: number of blocks that can be written using one Update command */
2011             UINT16_TO_BE_STREAM (p, p_cb->ndef_attrib.nmaxb);        /* Nmaxb: maximum number of blocks available for NDEF data */
2012             UINT32_TO_BE_STREAM (p, tempU32);
2013             UINT8_TO_STREAM (p, p_cb->ndef_attrib.writef);           /* WriteFlag: 00h if writing data finished; 0Fh if writing data in progress */
2014             UINT8_TO_STREAM (p, 0x00);                               /* RWFlag: 00h NDEF is read-only */
2015 
2016             tempU8 = (UINT8) (p_cb->ndef_attrib.ln >> 16);
2017             /* Get length (3-byte, big-endian) */
2018             UINT8_TO_STREAM (p, tempU8);                               /* Ln: high-byte */
2019             UINT16_TO_BE_STREAM (p, p_cb->ndef_attrib.ln);           /* Ln: lo-word */
2020 
2021             /* Calculate and append Checksum */
2022             checksum = 0;
2023             for (i = 0; i < T3T_MSG_NDEF_ATTR_INFO_SIZE; i++)
2024             {
2025                 checksum+=rw_t3t_ndef_attrib_info[i];
2026             }
2027             UINT16_TO_BE_STREAM (p, checksum);
2028 
2029             evt_data.status = rw_t3t_update_block (p_cb, 0, (UINT8 *) rw_t3t_ndef_attrib_info);
2030         }
2031         else if (p_cb->cur_cmd == RW_T3T_CMD_SET_READ_ONLY_HARD)
2032         {
2033             /* NDEF is already read only, Read and update MemoryControl block */
2034             RW_TRACE_DEBUG0 ("Felica-Lite tag detected...getting Memory Control block.");
2035             p_cb->rw_substate = RW_T3T_SRO_SST_CHECK_MC_BLK;
2036 
2037             /* Send command to check Memory Configuration block */
2038             evt_data.status = rw_t3t_check_mc_block (p_cb);
2039         }
2040     }
2041     else
2042     {
2043         RW_TRACE_ERROR0 ("Felica-Lite tag not detected");
2044         evt_data.status = NFC_STATUS_FAILED;
2045     }
2046 
2047     /* If error, notify upper layer */
2048     if (evt_data.status != NFC_STATUS_OK)
2049     {
2050         rw_t3t_set_readonly_cplt (evt_data.status);
2051     }
2052 }
2053 
2054 /*****************************************************************************
2055 **
2056 ** Function         rw_t3t_act_handle_sro_rsp
2057 **
2058 ** Description      Handle response for setting read only codes
2059 **
2060 ** Returns          Nothing
2061 **
2062 *****************************************************************************/
rw_t3t_act_handle_sro_rsp(tRW_T3T_CB * p_cb,BT_HDR * p_msg_rsp)2063 void rw_t3t_act_handle_sro_rsp (tRW_T3T_CB *p_cb, BT_HDR *p_msg_rsp)
2064 {
2065     UINT8 *p_t3t_rsp = (UINT8 *) (p_msg_rsp+1) + p_msg_rsp->offset;
2066     UINT8 *p_mc;
2067     tRW_DATA evt_data;
2068 
2069     evt_data.status = NFC_STATUS_OK;
2070 
2071     if (p_cb->rw_substate == RW_T3T_SRO_SST_UPDATE_NDEF_ATTRIB)
2072     {
2073         /* Validate response opcode */
2074         if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_UPDATE_RSP)
2075             ||(p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)  )
2076 
2077         {
2078             RW_TRACE_ERROR2 ("Response error: rsp_code=%02X, status=%02X", p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE], p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1]);
2079             evt_data.status = NFC_STATUS_FAILED;
2080         }
2081         else
2082         {
2083             p_cb->ndef_attrib.rwflag = T3T_MSG_NDEF_RWFLAG_RO;
2084             if (p_cb->cur_cmd == RW_T3T_CMD_SET_READ_ONLY_HARD)
2085             {
2086                 p_cb->rw_substate = RW_T3T_SRO_SST_CHECK_MC_BLK;
2087 
2088                 /* Send command to check Memory Configuration block */
2089                 evt_data.status = rw_t3t_check_mc_block (p_cb);
2090             }
2091             else
2092             {
2093                 rw_t3t_set_readonly_cplt (evt_data.status);
2094             }
2095         }
2096     }
2097     else if (p_cb->rw_substate == RW_T3T_SRO_SST_CHECK_MC_BLK)
2098     {
2099         /* Check tags's response for reading MemoryControl block, Validate response opcode */
2100         if (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_CHECK_RSP)
2101         {
2102             RW_TRACE_ERROR2 ("Response error: expecting rsp_code %02X, but got %02X", T3T_MSG_OPC_CHECK_RSP, p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE]);
2103             evt_data.status = NFC_STATUS_FAILED;
2104         }
2105         /* Validate status code and NFCID2 response from tag */
2106         else if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)                           /* verify response status code */
2107                  ||(memcmp (p_cb->peer_nfcid2, &p_t3t_rsp[T3T_MSG_RSP_OFFSET_IDM], NCI_NFCID2_LEN) != 0)  )   /* verify response IDm */
2108         {
2109             evt_data.status = NFC_STATUS_FAILED;
2110         }
2111         else
2112         {
2113             /* Check if memory configuration (MC) block to see if SYS_OP=1 (NDEF enabled) */
2114             p_mc = &p_t3t_rsp[T3T_MSG_RSP_OFFSET_CHECK_DATA];  /* Point to MC data of CHECK response */
2115 
2116             if (p_mc[T3T_MSG_FELICALITE_MC_OFFSET_SYS_OP] != 0x01)
2117             {
2118                 /* Tag is not currently enabled for NDEF */
2119                 evt_data.status = NFC_STATUS_FAILED;
2120             }
2121             else
2122             {
2123                 /* Set MC_SP field with MC[0] = 0x00 & MC[1] = 0xC0 (Hardlock) to change access permission from RW to RO */
2124                 p_mc[T3T_MSG_FELICALITE_MC_OFFSET_MC_SP]     = 0x00;
2125                 /* Not changing the access permission of Subtraction Register and MC[0:1] */
2126                 p_mc[T3T_MSG_FELICALITE_MC_OFFSET_MC_SP + 1] = 0xC0;
2127 
2128                /* Set RF_PRM field to 0x07 (procedure of issuance) */
2129                 p_mc[T3T_MSG_FELICALITE_MC_OFFSET_RF_PRM] = 0x07;
2130 
2131                 /* Construct and send UPDATE message to write MC block */
2132                 p_cb->rw_substate = RW_T3T_SRO_SST_UPDATE_MC_BLK;
2133                 evt_data.status = rw_t3t_update_block (p_cb, T3T_MSG_FELICALITE_BLOCK_ID_MC, p_mc);
2134             }
2135         }
2136     }
2137     else if (p_cb->rw_substate == RW_T3T_SRO_SST_UPDATE_MC_BLK)
2138     {
2139         /* Validate response opcode */
2140         if (  (p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE] != T3T_MSG_OPC_UPDATE_RSP)
2141             ||(p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1] != T3T_MSG_RSP_STATUS_OK)  )
2142 
2143         {
2144             RW_TRACE_ERROR2 ("Response error: rsp_code=%02X, status=%02X", p_t3t_rsp[T3T_MSG_RSP_OFFSET_RSPCODE], p_t3t_rsp[T3T_MSG_RSP_OFFSET_STATUS1]);
2145             evt_data.status = NFC_STATUS_FAILED;
2146         }
2147         else
2148         {
2149             rw_t3t_set_readonly_cplt (evt_data.status);
2150         }
2151     }
2152 
2153     /* If error, notify upper layer */
2154     if (evt_data.status != NFC_STATUS_OK)
2155     {
2156         rw_t3t_set_readonly_cplt (evt_data.status);
2157     }
2158 
2159     GKI_freebuf (p_msg_rsp);
2160 }
2161 
2162 /*******************************************************************************
2163 **
2164 ** Function         rw_t3t_data_cback
2165 **
2166 ** Description      This callback function receives the data from NFCC.
2167 **
2168 ** Returns          none
2169 **
2170 *******************************************************************************/
rw_t3t_data_cback(UINT8 conn_id,BT_HDR * p_msg)2171 void rw_t3t_data_cback (UINT8 conn_id, BT_HDR *p_msg)
2172 {
2173     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2174     BOOLEAN free_msg = FALSE;           /* if TRUE, free msg buffer before returning */
2175     UINT8 *p, sod;
2176 
2177     /* Stop rsponse timer */
2178     nfc_stop_quick_timer (&p_cb->timer);
2179 
2180 #if (defined (RW_STATS_INCLUDED) && (RW_STATS_INCLUDED == TRUE))
2181     /* Update rx stats */
2182     rw_main_update_rx_stats (p_msg->len);
2183 #endif  /* RW_STATS_INCLUDED */
2184 
2185     /* Check if we are expecting a response */
2186     if (p_cb->rw_state != RW_T3T_STATE_COMMAND_PENDING)
2187     {
2188         /*
2189         **  This must be raw frame response
2190         **  send raw frame to app with SoD
2191         */
2192         rw_t3t_act_handle_raw_senddata_rsp (p_cb, p_msg);
2193     }
2194     /* Sanity check: verify msg len is big enough to contain t3t header */
2195     else if (p_msg->len < T3T_MSG_RSP_COMMON_HDR_LEN)
2196     {
2197         RW_TRACE_ERROR1 ("T3T: invalid Type3 Tag Message (invalid len: %i)", p_msg->len);
2198         free_msg = TRUE;
2199 
2200         rw_t3t_process_frame_error ();
2201     }
2202     else
2203     {
2204         /* Check for RF frame error */
2205         p = (UINT8 *) (p_msg+1) + p_msg->offset;
2206         sod = p[0];
2207         if (p[sod] != NCI_STATUS_OK)
2208         {
2209             RW_TRACE_ERROR1 ("T3T: rf frame error (crc status=%i)", p[sod]);
2210             GKI_freebuf (p_msg);
2211 
2212             rw_t3t_process_frame_error ();
2213             return;
2214         }
2215 
2216 #if (BT_TRACE_PROTOCOL == TRUE)
2217         DispT3TagMessage (p_msg, TRUE);
2218 #endif
2219 
2220         /* Skip over SoD */
2221         p_msg->offset++;
2222         p_msg->len--;
2223 
2224         /* Get response code */
2225         switch (p_cb->cur_cmd)
2226         {
2227             case RW_T3T_CMD_DETECT_NDEF:
2228                 rw_t3t_act_handle_ndef_detect_rsp (p_cb, p_msg);
2229                 break;
2230 
2231             case RW_T3T_CMD_CHECK_NDEF:
2232                 rw_t3t_act_handle_check_ndef_rsp (p_cb, p_msg);
2233                 break;
2234 
2235             case RW_T3T_CMD_UPDATE_NDEF:
2236                 rw_t3t_act_handle_update_ndef_rsp (p_cb, p_msg);
2237                 break;
2238 
2239             case RW_T3T_CMD_CHECK:
2240                 rw_t3t_act_handle_check_rsp (p_cb, p_msg);
2241                 break;
2242 
2243             case RW_T3T_CMD_UPDATE:
2244                 rw_t3t_act_handle_update_rsp (p_cb, p_msg);
2245                 break;
2246 
2247             case RW_T3T_CMD_SEND_RAW_FRAME:
2248                 rw_t3t_act_handle_raw_senddata_rsp (p_cb, p_msg);
2249                 break;
2250 
2251             case RW_T3T_CMD_GET_SYSTEM_CODES:
2252                 rw_t3t_act_handle_get_sc_rsp (p_cb, p_msg);
2253                 break;
2254 
2255             case RW_T3T_CMD_FORMAT:
2256                 rw_t3t_act_handle_fmt_rsp (p_cb, p_msg);
2257                 break;
2258 
2259             case RW_T3T_CMD_SET_READ_ONLY_SOFT:
2260             case RW_T3T_CMD_SET_READ_ONLY_HARD:
2261                 rw_t3t_act_handle_sro_rsp (p_cb, p_msg);
2262                 break;
2263 
2264             default:
2265                 GKI_freebuf (p_msg);
2266                 break;
2267         }
2268     }
2269 
2270     if (free_msg)
2271     {
2272         GKI_freebuf (p_msg);
2273     }
2274 }
2275 
2276 
2277 /*******************************************************************************
2278 **
2279 ** Function         rw_t3t_conn_cback
2280 **
2281 ** Description      This callback function receives the events/data from NFCC.
2282 **
2283 ** Returns          none
2284 **
2285 *******************************************************************************/
rw_t3t_conn_cback(UINT8 conn_id,tNFC_CONN_EVT event,tNFC_CONN * p_data)2286 void rw_t3t_conn_cback (UINT8 conn_id, tNFC_CONN_EVT event, tNFC_CONN *p_data)
2287 {
2288     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2289     RW_TRACE_DEBUG2 ("rw_t3t_conn_cback: conn_id=%i, evt=0x%02x", conn_id, event);
2290 
2291     /* Only handle NFC_RF_CONN_ID conn_id */
2292     if (conn_id != NFC_RF_CONN_ID)
2293     {
2294         return;
2295     }
2296 
2297     switch (event)
2298     {
2299     case NFC_DEACTIVATE_CEVT:
2300         rw_t3t_unselect (NULL);
2301         break;
2302 
2303     case NFC_DATA_CEVT:     /* check for status in tNFC_CONN */
2304         if (p_data->data.status == NFC_STATUS_OK)
2305         {
2306             rw_t3t_data_cback (conn_id, p_data->data.p_data);
2307             break;
2308         }
2309         /* Data event with error status...fall through to NFC_ERROR_CEVT case */
2310 
2311 
2312     case NFC_ERROR_CEVT:
2313         nfc_stop_quick_timer (&p_cb->timer);
2314 
2315 #if (defined (RW_STATS_INCLUDED) && (RW_STATS_INCLUDED == TRUE))
2316         rw_main_update_trans_error_stats ();
2317 #endif  /* RW_STATS_INCLUDED */
2318 
2319         if (event == NFC_ERROR_CEVT)
2320             rw_t3t_process_error (NFC_STATUS_TIMEOUT);
2321         else
2322             rw_t3t_process_error (p_data->status);
2323         break;
2324 
2325     default:
2326         break;
2327 
2328     }
2329 }
2330 
2331 
2332 /*******************************************************************************
2333 **
2334 ** Function         rw_t3t_select
2335 **
2336 ** Description      Called by NFC manager when a Type3 tag has been activated
2337 **
2338 ** Returns          NFC_STATUS_OK
2339 **
2340 *******************************************************************************/
rw_t3t_select(UINT8 peer_nfcid2[NCI_RF_F_UID_LEN],UINT8 mrti_check,UINT8 mrti_update)2341 tNFC_STATUS rw_t3t_select (UINT8 peer_nfcid2[NCI_RF_F_UID_LEN], UINT8 mrti_check, UINT8 mrti_update)
2342 {
2343     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2344 
2345     RW_TRACE_API0 ("rw_t3t_select");
2346 
2347     memcpy (p_cb->peer_nfcid2, peer_nfcid2, NCI_NFCID2_LEN); /* Store tag's NFCID2 */
2348     p_cb->ndef_attrib.status = NFC_STATUS_NOT_INITIALIZED;  /* Indicate that NDEF detection has not been performed yet */
2349     p_cb->rw_state = RW_T3T_STATE_IDLE;
2350     p_cb->flags = 0;
2351 
2352     /* Alloc cmd buf for retransmissions */
2353     if (p_cb->p_cur_cmd_buf ==  NULL)
2354     {
2355         if ((p_cb->p_cur_cmd_buf = (BT_HDR *) GKI_getpoolbuf (NFC_RW_POOL_ID)) == NULL)
2356         {
2357             RW_TRACE_ERROR0 ("rw_t3t_select: unable to allocate buffer for retransmission");
2358             p_cb->rw_state = RW_T3T_STATE_NOT_ACTIVATED;
2359             return (NFC_STATUS_FAILED);
2360         }
2361     }
2362 
2363 
2364     NFC_SetStaticRfCback (rw_t3t_conn_cback);
2365 
2366     return NFC_STATUS_OK;
2367 }
2368 
2369 
2370 /*******************************************************************************
2371 **
2372 ** Function         rw_t3t_unselect
2373 **
2374 ** Description      Called by NFC manager when a Type3 tag has been de-activated
2375 **
2376 ** Returns          NFC_STATUS_OK
2377 **
2378 *******************************************************************************/
rw_t3t_unselect(UINT8 peer_nfcid2[])2379 static tNFC_STATUS rw_t3t_unselect (UINT8 peer_nfcid2[])
2380 {
2381     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2382 
2383 #if (defined (RW_STATS_INCLUDED) && (RW_STATS_INCLUDED == TRUE))
2384     /* Display stats */
2385     rw_main_log_stats ();
2386 #endif  /* RW_STATS_INCLUDED */
2387 
2388     /* Stop t3t timer (if started) */
2389     nfc_stop_quick_timer (&p_cb->timer);
2390 
2391     /* Free cmd buf for retransmissions */
2392     if (p_cb->p_cur_cmd_buf)
2393     {
2394         GKI_freebuf (p_cb->p_cur_cmd_buf);
2395         p_cb->p_cur_cmd_buf = NULL;
2396     }
2397 
2398     p_cb->rw_state = RW_T3T_STATE_NOT_ACTIVATED;
2399     NFC_SetStaticRfCback (NULL);
2400 
2401     return NFC_STATUS_OK;
2402 }
2403 
2404 
2405 #if (BT_TRACE_VERBOSE == TRUE)
2406 /*******************************************************************************
2407 **
2408 ** Function         rw_t3t_cmd_str
2409 **
2410 ** Description      Converts cmd_id to command string for logging
2411 **
2412 ** Returns          command string
2413 **
2414 *******************************************************************************/
rw_t3t_cmd_str(UINT8 cmd_id)2415 static char *rw_t3t_cmd_str (UINT8 cmd_id)
2416 {
2417     switch (cmd_id)
2418     {
2419     case RW_T3T_CMD_DETECT_NDEF:
2420         return "RW_T3T_CMD_DETECT_NDEF";
2421 
2422     case RW_T3T_CMD_CHECK_NDEF:
2423         return "RW_T3T_CMD_CHECK_NDEF";
2424 
2425     case RW_T3T_CMD_UPDATE_NDEF:
2426         return "RW_T3T_CMD_UPDATE_NDEF";
2427 
2428     case RW_T3T_CMD_CHECK:
2429         return "RW_T3T_CMD_CHECK";
2430 
2431     case RW_T3T_CMD_UPDATE:
2432         return "RW_T3T_CMD_UPDATE";
2433 
2434     case RW_T3T_CMD_SEND_RAW_FRAME:
2435         return "RW_T3T_CMD_SEND_RAW_FRAME";
2436 
2437     case RW_T3T_CMD_GET_SYSTEM_CODES:
2438         return "RW_T3T_CMD_GET_SYSTEM_CODES";
2439 
2440     default:
2441         return "Unknown";
2442     }
2443 }
2444 
2445 
2446 /*******************************************************************************
2447 **
2448 ** Function         rw_t3t_state_str
2449 **
2450 ** Description      Converts state_id to command string for logging
2451 **
2452 ** Returns          command string
2453 **
2454 *******************************************************************************/
rw_t3t_state_str(UINT8 state_id)2455 static char *rw_t3t_state_str (UINT8 state_id)
2456 {
2457     switch (state_id)
2458     {
2459     case RW_T3T_STATE_NOT_ACTIVATED:
2460         return "RW_T3T_STATE_NOT_ACTIVATED";
2461 
2462     case RW_T3T_STATE_IDLE:
2463         return "RW_T3T_STATE_IDLE";
2464 
2465     case RW_T3T_STATE_COMMAND_PENDING:
2466         return "RW_T3T_STATE_COMMAND_PENDING";
2467 
2468     default:
2469         return "Unknown";
2470     }
2471 }
2472 #endif
2473 
2474 /*****************************************************************************
2475 **  Type3 Tag API Functions
2476 *****************************************************************************/
2477 
2478 
2479 /*****************************************************************************
2480 **
2481 ** Function         RW_T3tDetectNDef
2482 **
2483 ** Description
2484 **      This function is used to perform NDEF detection on a Type 3 tag, and
2485 **      retrieve the tag's NDEF attribute information (block 0).
2486 **
2487 **      Before using this API, the application must call RW_SelectTagType to
2488 **      indicate that a Type 3 tag has been activated, and to provide the
2489 **      tag's Manufacture ID (IDm) .
2490 **
2491 ** Returns
2492 **      NFC_STATUS_OK: ndef detection procedure started
2493 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2494 **      NFC_STATUS_FAILED: other error
2495 **
2496 *****************************************************************************/
RW_T3tDetectNDef(void)2497 tNFC_STATUS RW_T3tDetectNDef (void)
2498 {
2499     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2500     tNFC_STATUS retval = NFC_STATUS_OK;
2501 
2502     RW_TRACE_API0 ("RW_T3tDetectNDef");
2503 
2504     /* Check if we are in valid state to handle this API */
2505     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2506     {
2507         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2508         return (NFC_STATUS_FAILED);
2509     }
2510 
2511     if ((retval = (tNFC_STATUS) nci_snd_t3t_polling (T3T_SYSTEM_CODE_NDEF, 0, 0)) == NCI_STATUS_OK)
2512     {
2513         p_cb->cur_cmd = RW_T3T_CMD_DETECT_NDEF;
2514         p_cb->cur_tout = RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS;
2515         p_cb->cur_poll_rc = 0;
2516         p_cb->rw_state = RW_T3T_STATE_COMMAND_PENDING;
2517         p_cb->flags |= RW_T3T_FL_W4_NDEF_DETECT_POLL_RSP;
2518 
2519         /* start timer for waiting for responses */
2520         rw_t3t_start_poll_timer (p_cb);
2521     }
2522 
2523     return (retval);
2524 }
2525 
2526 
2527 /*****************************************************************************
2528 **
2529 ** Function         RW_T3tCheckNDef
2530 **
2531 ** Description
2532 **      Retrieve NDEF contents from a Type3 tag.
2533 **
2534 **      The RW_T3T_CHECK_EVT event is used to notify the application for each
2535 **      segment of NDEF data received. The RW_T3T_CHECK_CPLT_EVT event is used to
2536 **      notify the application all segments have been received.
2537 **
2538 **      Before using this API, the RW_T3tDetectNDef function must be called to
2539 **      verify that the tag contains NDEF data, and to retrieve the NDEF
2540 **      attributes.
2541 **
2542 **      Internally, this command will be separated into multiple Tag 3 Check
2543 **      commands (if necessary) - depending on the tag's Nbr (max number of
2544 **      blocks per read) attribute.
2545 **
2546 ** Returns
2547 **      NFC_STATUS_OK: check command started
2548 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2549 **      NFC_STATUS_FAILED: other error
2550 **
2551 *****************************************************************************/
RW_T3tCheckNDef(void)2552 tNFC_STATUS RW_T3tCheckNDef (void)
2553 {
2554     tNFC_STATUS retval = NFC_STATUS_OK;
2555     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2556 
2557     RW_TRACE_API0 ("RW_T3tCheckNDef");
2558 
2559     /* Check if we are in valid state to handle this API */
2560     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2561     {
2562         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2563         return (NFC_STATUS_FAILED);
2564     }
2565     else if (p_cb->ndef_attrib.status != NFC_STATUS_OK)       /* NDEF detection not performed yet? */
2566     {
2567         RW_TRACE_ERROR0 ("Error: NDEF detection not performed yet");
2568         return (NFC_STATUS_NOT_INITIALIZED);
2569     }
2570     else if (p_cb->ndef_attrib.ln == 0)
2571     {
2572         RW_TRACE_ERROR0 ("Type 3 tag contains empty NDEF message");
2573         return (NFC_STATUS_FAILED);
2574     }
2575 
2576     /* Check number of blocks needed for this update */
2577     p_cb->flags &= ~RW_T3T_FL_IS_FINAL_NDEF_SEGMENT;
2578     p_cb->ndef_rx_offset = 0;
2579     retval = rw_t3t_send_next_ndef_check_cmd (p_cb);
2580 
2581     return (retval);
2582 }
2583 
2584 /*****************************************************************************
2585 **
2586 ** Function         RW_T3tUpdateNDef
2587 **
2588 ** Description
2589 **      Write NDEF contents to a Type3 tag.
2590 **
2591 **      The RW_T3T_UPDATE_CPLT_EVT callback event will be used to notify the
2592 **      application of the response.
2593 **
2594 **      Before using this API, the RW_T3tDetectNDef function must be called to
2595 **      verify that the tag contains NDEF data, and to retrieve the NDEF
2596 **      attributes.
2597 **
2598 **      Internally, this command will be separated into multiple Tag 3 Update
2599 **      commands (if necessary) - depending on the tag's Nbw (max number of
2600 **      blocks per write) attribute.
2601 **
2602 ** Returns
2603 **      NFC_STATUS_OK: check command started
2604 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2605 **      NFC_STATUS_REFUSED: tag is read-only
2606 **      NFC_STATUS_BUFFER_FULL: len exceeds tag's maximum size
2607 **      NFC_STATUS_FAILED: other error
2608 **
2609 *****************************************************************************/
RW_T3tUpdateNDef(UINT32 len,UINT8 * p_data)2610 tNFC_STATUS RW_T3tUpdateNDef (UINT32 len, UINT8 *p_data)
2611 {
2612     tNFC_STATUS retval = NFC_STATUS_OK;
2613     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2614 
2615     RW_TRACE_API1 ("RW_T3tUpdateNDef (len=%i)", len);
2616 
2617     /* Check if we are in valid state to handle this API */
2618     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2619     {
2620         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2621         return (NFC_STATUS_FAILED);
2622     }
2623     else if (p_cb->ndef_attrib.status != NFC_STATUS_OK)       /* NDEF detection not performed yet? */
2624     {
2625         RW_TRACE_ERROR0 ("Error: NDEF detection not performed yet");
2626         return (NFC_STATUS_NOT_INITIALIZED);
2627     }
2628     else if (len > (((UINT32)p_cb->ndef_attrib.nmaxb) * 16))                /* Len exceed's tag's NDEF memory? */
2629     {
2630         return (NFC_STATUS_BUFFER_FULL);
2631     }
2632     else if (p_cb->ndef_attrib.rwflag == T3T_MSG_NDEF_RWFLAG_RO)/* Tag's NDEF memory is read-only? */
2633     {
2634         return (NFC_STATUS_REFUSED);
2635     }
2636 
2637     /* Check number of blocks needed for this update */
2638     p_cb->flags &= ~RW_T3T_FL_IS_FINAL_NDEF_SEGMENT;
2639     p_cb->ndef_msg_bytes_sent = 0;
2640     p_cb->ndef_msg_len = len;
2641     p_cb->ndef_msg = p_data;
2642 
2643     /* Send initial UPDATE command for NDEF Attribute Info */
2644     retval = rw_t3t_send_update_ndef_attribute_cmd (p_cb, TRUE);
2645 
2646     return (retval);
2647 }
2648 
2649 /*****************************************************************************
2650 **
2651 ** Function         RW_T3tCheck
2652 **
2653 ** Description
2654 **      Read (non-NDEF) contents from a Type3 tag.
2655 **
2656 **      The RW_READ_EVT event is used to notify the application for each
2657 **      segment of NDEF data received. The RW_READ_CPLT_EVT event is used to
2658 **      notify the application all segments have been received.
2659 **
2660 **      Before using this API, the application must call RW_SelectTagType to
2661 **      indicate that a Type 3 tag has been activated, and to provide the
2662 **      tag's Manufacture ID (IDm) .
2663 **
2664 ** Returns
2665 **      NFC_STATUS_OK: check command started
2666 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2667 **      NFC_STATUS_FAILED: other error
2668 **
2669 *****************************************************************************/
RW_T3tCheck(UINT8 num_blocks,tT3T_BLOCK_DESC * t3t_blocks)2670 tNFC_STATUS RW_T3tCheck (UINT8 num_blocks, tT3T_BLOCK_DESC *t3t_blocks)
2671 {
2672     tNFC_STATUS retval = NFC_STATUS_OK;
2673     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2674 
2675     RW_TRACE_API1 ("RW_T3tCheck (num_blocks = %i)", num_blocks);
2676 
2677     /* Check if we are in valid state to handle this API */
2678     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2679     {
2680         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2681         return (NFC_STATUS_FAILED);
2682     }
2683 
2684     /* Send the CHECK command */
2685     retval = rw_t3t_send_check_cmd (p_cb, num_blocks, t3t_blocks);
2686 
2687     return (retval);
2688 }
2689 
2690 /*****************************************************************************
2691 **
2692 ** Function         RW_T3tUpdate
2693 **
2694 ** Description
2695 **      Write (non-NDEF) contents to a Type3 tag.
2696 **
2697 **      The RW_WRITE_CPLT_EVT event is used to notify the application all
2698 **      segments have been received.
2699 **
2700 **      Before using this API, the application must call RW_SelectTagType to
2701 **      indicate that a Type 3 tag has been activated, and to provide the tag's
2702 **      Manufacture ID (IDm) .
2703 **
2704 ** Returns
2705 **      NFC_STATUS_OK: check command started
2706 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2707 **      NFC_STATUS_FAILED: other error
2708 **
2709 *****************************************************************************/
RW_T3tUpdate(UINT8 num_blocks,tT3T_BLOCK_DESC * t3t_blocks,UINT8 * p_data)2710 tNFC_STATUS RW_T3tUpdate (UINT8 num_blocks, tT3T_BLOCK_DESC *t3t_blocks, UINT8 *p_data)
2711 {
2712     tNFC_STATUS retval = NFC_STATUS_OK;
2713     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2714 
2715     RW_TRACE_API1 ("RW_T3tUpdate (num_blocks = %i)", num_blocks);
2716 
2717     /* Check if we are in valid state to handle this API */
2718     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2719     {
2720         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2721         return (NFC_STATUS_FAILED);
2722     }
2723 
2724     /* Send the UPDATE command */
2725     retval = rw_t3t_send_update_cmd (p_cb, num_blocks, t3t_blocks, p_data);
2726 
2727     return (retval);
2728 }
2729 
2730 /*****************************************************************************
2731 **
2732 ** Function         RW_T3tPresenceCheck
2733 **
2734 ** Description
2735 **      Check if the tag is still in the field.
2736 **
2737 **      The RW_T3T_PRESENCE_CHECK_EVT w/ status is used to indicate presence
2738 **      or non-presence.
2739 **
2740 ** Returns
2741 **      NFC_STATUS_OK, if raw data frame sent
2742 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2743 **      NFC_STATUS_FAILED: other error
2744 **
2745 *****************************************************************************/
RW_T3tPresenceCheck(void)2746 tNFC_STATUS RW_T3tPresenceCheck (void)
2747 {
2748     tNFC_STATUS retval = NFC_STATUS_OK;
2749     tRW_DATA evt_data;
2750     tRW_CB *p_rw_cb = &rw_cb;
2751 
2752     RW_TRACE_API0 ("RW_T3tPresenceCheck");
2753 
2754     /* If RW_SelectTagType was not called (no conn_callback) return failure */
2755     if (!(p_rw_cb->p_cback))
2756     {
2757         retval = NFC_STATUS_FAILED;
2758     }
2759     /* If we are not activated, then RW_T3T_PRESENCE_CHECK_EVT status=FAIL */
2760     else if (p_rw_cb->tcb.t3t.rw_state == RW_T3T_STATE_NOT_ACTIVATED)
2761     {
2762         evt_data.status = NFC_STATUS_FAILED;
2763         (*p_rw_cb->p_cback) (RW_T3T_PRESENCE_CHECK_EVT, &evt_data);
2764     }
2765     /* If command is pending */
2766     else if (p_rw_cb->tcb.t3t.rw_state == RW_T3T_STATE_COMMAND_PENDING)
2767     {
2768         /* If already performing presence check, return error */
2769         if (p_rw_cb->tcb.t3t.flags & RW_T3T_FL_W4_PRESENCE_CHECK_POLL_RSP)
2770         {
2771             RW_TRACE_DEBUG0 ("RW_T3tPresenceCheck already in progress");
2772             retval = NFC_STATUS_FAILED;
2773         }
2774         /* If busy with any other command, assume that the tag is present */
2775         else
2776         {
2777             evt_data.status = NFC_STATUS_OK;
2778             (*p_rw_cb->p_cback) (RW_T3T_PRESENCE_CHECK_EVT, &evt_data);
2779         }
2780     }
2781     else
2782     {
2783         /* IDLE state: send POLL command */
2784         if ((retval = (tNFC_STATUS) nci_snd_t3t_polling (0xFFFF, T3T_POLL_RC_SC, 0)) == NCI_STATUS_OK)
2785         {
2786             p_rw_cb->tcb.t3t.flags |= RW_T3T_FL_W4_PRESENCE_CHECK_POLL_RSP;
2787             p_rw_cb->tcb.t3t.rw_state = RW_T3T_STATE_COMMAND_PENDING;
2788             p_rw_cb->tcb.t3t.cur_poll_rc = 0;
2789 
2790             /* start timer for waiting for responses */
2791             rw_t3t_start_poll_timer (&p_rw_cb->tcb.t3t);
2792         }
2793         else
2794         {
2795             RW_TRACE_DEBUG1 ("RW_T3tPresenceCheck error sending NCI_RF_T3T_POLLING cmd (status = 0x%0x)", retval);
2796         }
2797     }
2798 
2799     return (retval);
2800 }
2801 
2802 /*****************************************************************************
2803 **
2804 ** Function         RW_T3tPoll
2805 **
2806 ** Description
2807 **      Send POLL command
2808 **
2809 ** Returns
2810 **      NFC_STATUS_OK, if raw data frame sent
2811 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2812 **      NFC_STATUS_FAILED: other error
2813 **
2814 *****************************************************************************/
RW_T3tPoll(UINT16 system_code,tT3T_POLL_RC rc,UINT8 tsn)2815 tNFC_STATUS RW_T3tPoll (UINT16 system_code, tT3T_POLL_RC rc, UINT8 tsn)
2816 {
2817     tNFC_STATUS retval = NFC_STATUS_OK;
2818     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2819 
2820     RW_TRACE_API0 ("RW_T3tPoll");
2821 
2822     /* Check if we are in valid state to handle this API */
2823     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2824     {
2825         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2826         return (NFC_STATUS_FAILED);
2827     }
2828 
2829     if ((retval = (tNFC_STATUS) nci_snd_t3t_polling (system_code, (UINT8) rc, tsn)) == NCI_STATUS_OK)
2830     {
2831         /* start timer for waiting for responses */
2832         p_cb->cur_poll_rc = rc;
2833         p_cb->rw_state = RW_T3T_STATE_COMMAND_PENDING;
2834         rw_t3t_start_poll_timer (p_cb);
2835     }
2836 
2837 
2838     return (retval);
2839 }
2840 
2841 /*****************************************************************************
2842 **
2843 ** Function         RW_T3tSendRawFrame
2844 **
2845 ** Description
2846 **      This function is called to send a raw data frame to the peer device.
2847 **      When type 3 tag receives response from peer, the callback function
2848 **      will be called with a RW_T3T_RAW_FRAME_EVT [Table 6].
2849 **
2850 **      Before using this API, the application must call RW_SelectTagType to
2851 **      indicate that a Type 3 tag has been activated.
2852 **
2853 **      The raw frame should be a properly formatted Type 3 tag message.
2854 **
2855 ** Returns
2856 **      NFC_STATUS_OK, if raw data frame sent
2857 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2858 **      NFC_STATUS_FAILED: other error
2859 **
2860 *****************************************************************************/
RW_T3tSendRawFrame(UINT16 len,UINT8 * p_data)2861 tNFC_STATUS RW_T3tSendRawFrame (UINT16 len, UINT8 *p_data)
2862 {
2863     tNFC_STATUS retval = NFC_STATUS_OK;
2864     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2865 
2866     RW_TRACE_API1 ("RW_T3tSendRawFrame (len = %i)", len);
2867 
2868     /* Check if we are in valid state to handle this API */
2869     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2870     {
2871         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2872         return (NFC_STATUS_FAILED);
2873     }
2874 
2875     /* Send the UPDATE command */
2876     retval = rw_t3t_send_raw_frame (p_cb, len ,p_data);
2877 
2878     return (retval);
2879 }
2880 
2881 /*****************************************************************************
2882 **
2883 ** Function         RW_T3tGetSystemCodes
2884 **
2885 ** Description
2886 **      Get systems codes supported by the activated tag:
2887 **              Poll for wildcard (FFFF):
2888 **                  - If felica-lite code then poll for ndef (12fc)
2889 **                  - Otherwise send RequestSystmCode command to get
2890 **                    system codes.
2891 **
2892 **      Before using this API, the application must call RW_SelectTagType to
2893 **      indicate that a Type 3 tag has been activated.
2894 **
2895 ** Returns
2896 **      NFC_STATUS_OK, if raw data frame sent
2897 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2898 **      NFC_STATUS_FAILED: other error
2899 **
2900 *****************************************************************************/
RW_T3tGetSystemCodes(void)2901 tNFC_STATUS RW_T3tGetSystemCodes (void)
2902 {
2903     tNFC_STATUS retval = NFC_STATUS_OK;
2904     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2905 
2906     RW_TRACE_API0 ("RW_T3tGetSystemCodes");
2907 
2908     /* Check if we are in valid state to handle this API */
2909     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2910     {
2911         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2912         return (NFC_STATUS_FAILED);
2913     }
2914     else
2915     {
2916         if ((retval = (tNFC_STATUS) nci_snd_t3t_polling (0xFFFF, T3T_POLL_RC_SC, 0)) == NCI_STATUS_OK)
2917         {
2918             p_cb->cur_cmd = RW_T3T_CMD_GET_SYSTEM_CODES;
2919             p_cb->cur_tout = RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS;
2920             p_cb->cur_poll_rc = T3T_POLL_RC_SC;
2921             p_cb->rw_state = RW_T3T_STATE_COMMAND_PENDING;
2922             p_cb->rw_substate = RW_T3T_GET_SC_SST_POLL_WILDCARD;
2923             p_cb->flags |= RW_T3T_FL_W4_GET_SC_POLL_RSP;
2924             p_cb->num_system_codes = 0;
2925 
2926             /* start timer for waiting for responses */
2927             rw_t3t_start_poll_timer (p_cb);
2928         }
2929     }
2930 
2931 
2932 
2933     return (retval);
2934 }
2935 
2936 /*****************************************************************************
2937 **
2938 ** Function         RW_T3tFormatNDef
2939 **
2940 ** Description
2941 **      Format a type-3 tag for NDEF.
2942 **
2943 **      Only Felica-Lite tags are supported by this API. The
2944 **      RW_T3T_FORMAT_CPLT_EVT is used to notify the status of the operation.
2945 **
2946 ** Returns
2947 **      NFC_STATUS_OK: ndef detection procedure started
2948 **      NFC_STATUS_NO_BUFFERS: unable to allocate a buffer for this operation
2949 **      NFC_STATUS_FAILED: other error
2950 **
2951 *****************************************************************************/
RW_T3tFormatNDef(void)2952 tNFC_STATUS RW_T3tFormatNDef (void)
2953 {
2954     tNFC_STATUS retval = NFC_STATUS_OK;
2955     tRW_T3T_CB *p_cb = &rw_cb.tcb.t3t;
2956 
2957     RW_TRACE_API0 ("RW_T3tFormatNDef");
2958 
2959     /* Check if we are in valid state to handle this API */
2960     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
2961     {
2962         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
2963         return (NFC_STATUS_FAILED);
2964     }
2965     else
2966     {
2967         /* Poll tag, to see if Felica-Lite system is supported */
2968         if ((retval = (tNFC_STATUS) nci_snd_t3t_polling (T3T_SYSTEM_CODE_FELICA_LITE, T3T_POLL_RC_SC, 0)) == NCI_STATUS_OK)
2969         {
2970             p_cb->cur_cmd = RW_T3T_CMD_FORMAT;
2971             p_cb->cur_tout = RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS;
2972             p_cb->cur_poll_rc = T3T_POLL_RC_SC;
2973             p_cb->rw_state = RW_T3T_STATE_COMMAND_PENDING;
2974             p_cb->rw_substate = RW_T3T_FMT_SST_POLL_FELICA_LITE;
2975             p_cb->flags |= RW_T3T_FL_W4_FMT_FELICA_LITE_POLL_RSP;
2976 
2977             /* start timer for waiting for responses */
2978             rw_t3t_start_poll_timer (p_cb);
2979         }
2980     }
2981 
2982 
2983 
2984     return (retval);
2985 }
2986 
2987 /*****************************************************************************
2988 **
2989 ** Function         RW_T3tSetReadOnly
2990 **
2991 ** Description      This function performs NDEF read-only procedure
2992 **                  Note: Only Felica-Lite tags are supported by this API.
2993 **                        RW_T3tDetectNDef() must be called before using this
2994 **
2995 **                  The RW_T3T_SET_READ_ONLY_CPLT_EVT event will be returned.
2996 **
2997 ** Returns          NFC_STATUS_OK if success
2998 **                  NFC_STATUS_FAILED if T3T is busy or other error
2999 **
3000 *****************************************************************************/
RW_T3tSetReadOnly(BOOLEAN b_hard_lock)3001 tNFC_STATUS RW_T3tSetReadOnly (BOOLEAN b_hard_lock)
3002 {
3003     tNFC_STATUS retval = NFC_STATUS_OK;
3004     tRW_T3T_CB  *p_cb  = &rw_cb.tcb.t3t;
3005     tRW_DATA    evt_data;
3006 
3007     RW_TRACE_API0 ("RW_T4tSetTagReadOnly ()");
3008 
3009     /* Check if we are in valid state to handle this API */
3010     if (p_cb->rw_state != RW_T3T_STATE_IDLE)
3011     {
3012         RW_TRACE_ERROR1 ("Error: invalid state to handle API (0x%x)", p_cb->rw_state);
3013         return (NFC_STATUS_FAILED);
3014     }
3015 
3016     if (p_cb->ndef_attrib.status != NFC_STATUS_OK)       /* NDEF detection not performed yet? */
3017     {
3018         RW_TRACE_ERROR0 ("Error: NDEF detection not performed yet");
3019         return (NFC_STATUS_NOT_INITIALIZED);
3020     }
3021 
3022     if ((!b_hard_lock) && (p_cb->ndef_attrib.rwflag == T3T_MSG_NDEF_RWFLAG_RO))/* Tag's NDEF memory is read-only already */
3023     {
3024         evt_data.status = NFC_STATUS_OK;
3025         (*(rw_cb.p_cback)) (RW_T4T_SET_TO_RO_EVT, &evt_data);
3026         return (retval);
3027     }
3028     else
3029     {
3030         /* Poll tag, to see if Felica-Lite system is supported */
3031         if ((retval = (tNFC_STATUS) nci_snd_t3t_polling (T3T_SYSTEM_CODE_FELICA_LITE, T3T_POLL_RC_SC, 0)) == NCI_STATUS_OK)
3032         {
3033             if (b_hard_lock)
3034                 p_cb->cur_cmd     = RW_T3T_CMD_SET_READ_ONLY_HARD;
3035             else
3036                 p_cb->cur_cmd     = RW_T3T_CMD_SET_READ_ONLY_SOFT;
3037             p_cb->cur_tout    = RW_T3T_DEFAULT_CMD_TIMEOUT_TICKS;
3038             p_cb->cur_poll_rc = T3T_POLL_RC_SC;
3039             p_cb->rw_state    = RW_T3T_STATE_COMMAND_PENDING;
3040             p_cb->rw_substate = RW_T3T_SRO_SST_POLL_FELICA_LITE;
3041             p_cb->flags |= RW_T3T_FL_W4_SRO_FELICA_LITE_POLL_RSP;
3042 
3043             /* start timer for waiting for responses */
3044             rw_t3t_start_poll_timer (p_cb);
3045         }
3046     }
3047     return (retval);
3048 }
3049