Module_Payment.c 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180
  1. /*
  2. * Module_Payment.c
  3. *
  4. * Created on: 2021/03/24
  5. * Author: Henry Yeh
  6. */
  7. #include "Module_Payment.h"
  8. unsigned char version[] = {'D', '0', '.', '0', '3'};
  9. unsigned char BLP_CMD_CARD_DETECT[] = {0x09, 0, 0x07, 'M','F','1','4','1','2','1', 0x32}; // Enable payment, MIFARE, 15693 card; Detect Payment Card First; Enable ApplePay VAS
  10. unsigned char BLP_CMD_USI2[] = {0x09, 0, 0x03, 'P', 'C', '0', 0x29}; // Configure protocol to USI2
  11. unsigned char BLP_CMD_SET_BAUD[] = {0x09, 0, 0x03, 'B', 'R', '7', 0x2d}; // Configure module baud rate to 115200
  12. unsigned char BLP_CMD_RESTORE_DEFAULT[] = {0x09, 0, 0x03, 'D', 'F', 0, 0x08 }; // Restore module configuration to default setting
  13. unsigned char BLP_CMD_TLV_ENABLE[] = {0x09, 0, 0x03, 'A', 'A', 'E', 'O'}; // Enable TLV command to support EMV transaction
  14. unsigned char BLP_CMD_SET_AID[] = {0x09, 0, 0x04, 'A', 'D', '0', 0, 0x38}; // Set AID type, 0x00: user defined AID 0x01: Default AID
  15. unsigned char BLP_CMD_LED_IND[] = {0x09, 0, 0x05, 'R', 'I', 'E', 0x01, 0x01, 'R'}; // Reader LED indicator
  16. unsigned char BLP_CMD_CA_PKEY[] = {0x09, 0, 0x03, 'C', 'K', 0x01, 0x03}; // Enable user CA Key
  17. unsigned char BLP_CMD_SOFTCARD_DISABLE[]= {0x09, 0, 0x03, 'I', 'S', 'D', 'T'}; // Disable soft card wallet application
  18. unsigned char BLP_CMD_BEEP_DISABLE[] = {0x09, 0, 0x03, 'B', 'A', 'D', 'M'}; // Beep disable
  19. unsigned char BLP_CMD_ERR_BEEP_STYLE[] = {0x09, 0, 0x03, 'B', 'D', 0, 0x0c}; // Error beep style configure 2 beeps
  20. unsigned char BLP_CMD_ERR_BEEP[] = {0x09, 0, 0x03, 'B', 'T', 'D', 'X'}; // Error beep give sound
  21. unsigned char BLP_CMD_LRC_ENABLE[] = {0x09, 0, 0x03, 'L', 'C', 'E', '@'}; // Enable LRC character of track data
  22. unsigned char BLP_CMD_TX_TRACK[] = {0x09, 0, 0x03, 'T', 'K', '3', '&'}; // Configure output track data 1&2
  23. unsigned char USI_CMD_WARM_RESET[] = {0x01, 0x00, 0x00, 0x01, 0x7f, 0x7f}; // Payment module warm reset
  24. unsigned char USI_CMD_MODULE_STATUS[] = {0x01, 0x00, 0x00, 0x01, 0x24, 0x24}; // Deactivate reader
  25. unsigned char USI_CMD_C8_DEACTIVAVE[] = {0x01, 0x00, 0x00, 0x02, 0xc8, 0x00, 0x00}; // Deactivate reader
  26. unsigned char USI_CMD_C8_POLL[] = {0x01, 0x00, 0x00, 0x2b, // STX, address, command length high byte, command length low byte
  27. 0xc8, 0x01, // Activate the reader
  28. 0x9f, 0x02, 0x06, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, // Amount, Authorized, sample: 100.00 dollar
  29. 0x5f, 0x2a, 0x02, 0x08, 0x40, // Transaction Currency Code follow ISO-4217, sample: 0840(USD)
  30. 0x5f, 0x36, 0x01, 0x02, // Transaction Currency Exponent, Identifies the decimal point position from the right of the transaction amount according to ISO 4217
  31. 0x9c, 0x01, 0x00, // Transaction Type, sample: 00
  32. 0x9a, 0x03, 0x21, 0x03, 0x24, // Transaction Date, sample: 2021/03/24
  33. 0x9f, 0x21, 0x03, 0x13, 0x36, 0x10, // Transaction Time, sample: 13:36:10
  34. 0xff, 0xff, 0x82, 0x05, 0x04, 0x00, 0x00, 0x01, 0x2c, // Command timeout unit is million second, sample: 300ms
  35. 0x00}; // Checksum
  36. unsigned char USI_CMD_HALT_CARD[] = {0x01, 0x00, 0x00, 0x01, 'x', 0x78}; // Halt card
  37. unsigned char USI_CMD_SET_DATE[] = {0x01, 0, 0, 0x07, '5', '4', 0x14, 0x15, 0x05, 0x02, 0x01, 0x18};
  38. unsigned char USI_CMD_SET_TIME[] = {0x01, 0, 0, 0x07, '5', '5', 0x0f, 0x1e, 0x20, 0, 0, 0x37};
  39. unsigned char USI_CMD_ARM_DISABLE[] = {0x01, 0, 0, 0x02, 'H', '0', '{'};
  40. unsigned char USI_CMD_NFC_ACTIVE[] = {0x01, 0, 0, 0x02, 'N', 0x00, 'M'}; // Enable NFC function
  41. unsigned char USI_CMD_NFC_POLL[] = {0x01, 0, 0, 0x05, 'N', ' ', '3', 0x02, 0x01, 'Z'}; // Polling FELICA(NFC Type 3, 414kbps)
  42. int system_command(int uart, unsigned char* cmd, int cmd_len, unsigned char* rx);
  43. int USI2_Parse(unsigned char* rx, unsigned char* rx_data);
  44. struct C9_RESULT
  45. {
  46. unsigned char result_data[512]; // C9 response data without header & checksum
  47. unsigned char status; // C9 response status
  48. unsigned char pos_entry; // C9 response pos entry
  49. unsigned char u_id[20]; // Card SN
  50. unsigned char tkData[4][128]; // Card TK data for credit card
  51. unsigned char isCardPreset:1; // Card is presented on module, 0: removed 1: presented
  52. unsigned char isCommandError:1; // Polling command result, 0: normal 1: error
  53. } C9_Result;
  54. //==========================================
  55. // Common routine
  56. //==========================================
  57. int StoreLogMsg(const char *fmt, ...)
  58. {
  59. char Buf[65536+256];
  60. char buffer[65536];
  61. //char Buf[4096+256];
  62. //char buffer[4096];
  63. time_t CurrentTime;
  64. struct tm *tm;
  65. struct timeval tv;
  66. va_list args;
  67. va_start(args, fmt);
  68. int rc = vsnprintf(buffer, sizeof(buffer), fmt, args);
  69. va_end(args);
  70. memset(Buf,0,sizeof(Buf));
  71. CurrentTime = time(NULL);
  72. tm=localtime(&CurrentTime);
  73. gettimeofday(&tv, NULL); // get microseconds, 10^-6
  74. sprintf(Buf,"echo -n \"[%04d.%02d.%02d %02d:%02d:%02d.%06ld]%s\" >> /Storage/SystemLog/[%04d.%02d]PaymentLog",
  75. tm->tm_year+1900,tm->tm_mon+1,tm->tm_mday,tm->tm_hour,tm->tm_min,tm->tm_sec,tv.tv_usec,
  76. buffer,
  77. tm->tm_year+1900,tm->tm_mon+1);
  78. system((const char*)Buf);
  79. #ifdef ConsloePrintLog
  80. printf("[%04d.%02d.%02d %02d:%02d:%02d.%06ld]%s", tm->tm_year+1900,tm->tm_mon+1,tm->tm_mday,tm->tm_hour,tm->tm_min,tm->tm_sec,tv.tv_usec, buffer);
  81. #endif
  82. return rc;
  83. }
  84. /**
  85. * Execute shell command
  86. * @param cmd: shell command string
  87. * @return shell command execution result
  88. */
  89. int runShellCmd(const char*cmd)
  90. {
  91. int result = FAIL;
  92. char buf[256];
  93. FILE *fp;
  94. fp = popen(cmd, "r");
  95. if(fp != NULL)
  96. {
  97. while(fgets(buf, sizeof(buf), fp) != NULL)
  98. {
  99. DEBUG_INFO("%s\n", buf);
  100. }
  101. result = PASS;
  102. }
  103. pclose(fp);
  104. return result;
  105. }
  106. /**
  107. * Calculate time differential
  108. * @param ST: start time
  109. * @param ET: end time
  110. * @return time differential in million seconds
  111. */
  112. int DiffTimeb(struct timeb ST, struct timeb ET)
  113. {
  114. //return milli-second
  115. unsigned int StartTime,StopTime;
  116. StartTime=(unsigned int)ST.time;
  117. StopTime=(unsigned int)ET.time;
  118. return (StopTime-StartTime)*1000+ET.millitm-ST.millitm;
  119. }
  120. /**
  121. * Show communication raw data to debug info
  122. * @param data: raw data
  123. * @param len: data length
  124. * @param isRX: is receive data
  125. */
  126. void show_raw(uint8_t *data, uint16_t len, uint8_t isRX)
  127. {
  128. uint8_t output[8192];
  129. memset(output, 0x00, ARRAY_SIZE(output));
  130. sprintf((char*)output, "%s", (isRX?"RX: ":"TX: "));
  131. for(uint16_t idx = 0;idx<len;idx++)
  132. {
  133. sprintf((char*)output, "%s%02x ", output, data[idx]);
  134. }
  135. DEBUG_INFO("%s\n", output);
  136. }
  137. /**
  138. * Show data to debug info
  139. * @param dat: data content
  140. * @param len: data length
  141. */
  142. void show_data(unsigned char *dat, unsigned int len)
  143. {
  144. uint8_t output[8192];
  145. memset(output, 0x00, ARRAY_SIZE(output));
  146. sprintf((char*)output, "Data: ");
  147. for(uint16_t idx = 0;idx<len;idx++)
  148. {
  149. if(dat[idx] > 31 && dat[idx] < 128 )
  150. sprintf((char*)output, "%s%c", output, dat[idx]);
  151. else
  152. sprintf((char*)output, "%s<%x>", output, dat[idx]);
  153. }
  154. DEBUG_INFO("%s\n", output);
  155. }
  156. /**
  157. * Get sentinel quantity in data array
  158. * @param data: message array address
  159. * @param dataLen: array seek size
  160. * @return how many sentinel flag found
  161. */
  162. int getSentinelQuantity(unsigned char *data, unsigned int dataLen)
  163. {
  164. int result = 0;
  165. for(uint16_t idx=0;idx<dataLen;idx++)
  166. {
  167. if(data[idx] == '?')
  168. result++;
  169. }
  170. return result;
  171. }
  172. /**
  173. * Get sentinel position in array
  174. * @param data: message array address
  175. * @param dataLen: array seek size
  176. * @param idxSentinel: which sentinel idx want to find, since 0 start
  177. * @return sentinel position in array
  178. */
  179. int getSentinelPosition(unsigned char *data, unsigned int dataLen, unsigned char idxSentinel)
  180. {
  181. int result = -1;
  182. int foundCnt = -1;
  183. for(uint16_t idx=0;idx<dataLen;idx++)
  184. {
  185. if(data[idx] == '?')
  186. foundCnt++;
  187. if(foundCnt == idxSentinel)
  188. {
  189. result = idx;
  190. break;
  191. }
  192. }
  193. return result;
  194. }
  195. /**
  196. *
  197. * @param dec: number in dec
  198. * @return number in bcd
  199. */
  200. int decTobcd(int dec)
  201. {
  202. return (dec/10 * 16)+ (dec%10);
  203. }
  204. /**
  205. *
  206. * @param data: message array
  207. * @param dataLen: command & data field length in array
  208. * @return check sum result
  209. */
  210. int calChksum(unsigned char *data, unsigned int dataLen)
  211. {
  212. int result = 0;
  213. for(uint16_t idx=0;idx<dataLen-1;idx++)
  214. {
  215. result ^= data[idx];
  216. }
  217. return (result&0xff);
  218. }
  219. //==========================================
  220. // Init share memory
  221. //==========================================
  222. /**
  223. * Share memory initialization
  224. * @return function result
  225. */
  226. int InitShareMemory()
  227. {
  228. int result = PASS;
  229. int MeterSMId;
  230. #ifndef X86
  231. //init ShmSysConfigAndInfo
  232. if ((MeterSMId = shmget(ShmSysConfigAndInfoKey, sizeof(struct SysConfigAndInfo), 0777)) < 0)
  233. {
  234. DEBUG_ERROR("shmget ShmSysConfigAndInfo NG\n");
  235. result = FAIL;
  236. }
  237. else if ((ShmSysConfigAndInfo = shmat(MeterSMId, NULL, 0)) == (void *) -1)
  238. {
  239. DEBUG_ERROR("shmat ShmSysConfigAndInfo NG\n");
  240. result = FAIL;
  241. }
  242. else
  243. {}
  244. //init ShmStatusCodeData
  245. if ((MeterSMId = shmget(ShmStatusCodeKey, sizeof(struct StatusCodeData), 0777)) < 0)
  246. {
  247. DEBUG_ERROR("shmget ShmStatusCodeData NG\n");
  248. result = FAIL;
  249. }
  250. else if ((ShmStatusCodeData = shmat(MeterSMId, NULL, 0)) == (void *) -1)
  251. {
  252. DEBUG_ERROR("shmat ShmStatusCodeData NG\n");
  253. result = FAIL;
  254. }
  255. else
  256. {}
  257. #endif
  258. return result;
  259. }
  260. //==========================================
  261. // Init com port
  262. //==========================================
  263. /**
  264. * TTY port initialization
  265. * @return port initial result
  266. */
  267. int InitComPort()
  268. {
  269. int fd;
  270. struct termios tios;
  271. fd = open(TTY_PORT, O_RDWR);
  272. if(fd<=0)
  273. {
  274. return FAIL;
  275. }
  276. ioctl (fd, TCGETS, &tios);
  277. tios.c_cflag = B9600| CS8 | CLOCAL | CREAD;
  278. tios.c_lflag = 0;
  279. tios.c_iflag = 0;
  280. tios.c_oflag = 0;
  281. tios.c_cc[VMIN]=0; // data length threshold, 0 bytes
  282. tios.c_cc[VTIME]=(unsigned char)5; // timeout threshold, 0.5 seconds
  283. tios.c_lflag=0;
  284. tcflush(fd, TCIFLUSH);
  285. ioctl (fd, TCSETS, &tios);
  286. return fd;
  287. }
  288. /**
  289. * Send command to UIC680fg module.
  290. * @param uart: port handle
  291. * @param cmd: command buffer
  292. * @param cmd_len: command length
  293. * @param rx: receive buffer
  294. * @return receive data length
  295. */
  296. int system_command(int uart, unsigned char* cmd, int cmd_len, unsigned char* rx)
  297. {
  298. int rx_len = 0;
  299. tcflush(uart,TCIOFLUSH);
  300. //show_raw(cmd, cmd_len, NO);
  301. if(write(uart, cmd, cmd_len) > 0)
  302. {
  303. usleep(100000);
  304. rx_len = read(uart, rx, 512);
  305. //show_raw(rx, rx_len, YES);
  306. }
  307. else
  308. {
  309. DEBUG_ERROR("system command write fail.\n");
  310. }
  311. return rx_len;
  312. }
  313. /**
  314. * Parsing raw data to USI data
  315. * @param rx: raw data
  316. * @param rx_data: parsing result data
  317. * @return parsing result data length
  318. */
  319. int USI2_Parse(unsigned char* rx, unsigned char* rx_data)
  320. {
  321. int result = -1;
  322. unsigned int data_len =0;
  323. unsigned int chksum = 0;
  324. if(rx[0] == SOH) // SOH = 0x01
  325. {
  326. data_len = (unsigned int)rx[2] <<8;
  327. data_len |= rx[3];
  328. for(int idx=0;idx<(data_len+4);idx++)
  329. {
  330. chksum ^= rx[idx];
  331. }
  332. if((chksum&0xff) == rx[(data_len+4)])
  333. {
  334. memcpy(rx_data, &rx[4], data_len);
  335. result = data_len;
  336. }
  337. else
  338. DEBUG_WARN("USI2 message checksum error.\n");
  339. }
  340. else
  341. {
  342. DEBUG_WARN("USI2 message header is not <01>.\n");
  343. }
  344. return result;
  345. }
  346. //==========================================
  347. // Main loop
  348. //==========================================
  349. int main(void)
  350. {
  351. int UartFd;
  352. uint16_t failCount = 0;
  353. unsigned char rx_Array[512]={0}, rx_Data[512]={0};
  354. char C8_Polling = true;
  355. char Wait_C9 = false;
  356. int rx_len = 0;
  357. int data_len = 0;
  358. time_t CurrentTime;
  359. struct tm *tm;
  360. DEBUG_INFO("Task version: %s\n", version);
  361. //===============================================
  362. // Initialization
  363. //===============================================
  364. #ifndef X86
  365. if(InitShareMemory() == FAIL)
  366. {
  367. DEBUG_ERROR("InitShareMemory NG\n");
  368. if(ShmStatusCodeData!=NULL)
  369. {
  370. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.FailToCreateShareMemory=1;
  371. }
  372. sleep(5);
  373. return FAIL;
  374. }
  375. #endif
  376. UartFd=InitComPort();
  377. if(UartFd<0)
  378. {
  379. DEBUG_ERROR("InitComPort NG\n");
  380. if(ShmStatusCodeData!=NULL)
  381. {
  382. #ifndef X86
  383. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.CsuInitFailed=1;
  384. #endif
  385. }
  386. sleep(5);
  387. return FAIL;
  388. }
  389. else
  390. {
  391. DEBUG_INFO("ttyS3 port open success.\n");
  392. }
  393. //===============================================
  394. // Payment module configuration set to default (BLP Protocol)
  395. //===============================================
  396. do
  397. {
  398. rx_len = system_command(UartFd, BLP_CMD_RESTORE_DEFAULT, ARRAY_SIZE(BLP_CMD_RESTORE_DEFAULT), rx_Array);
  399. if((rx_Array[0] == ACK) && (rx_len ==1))
  400. {
  401. DEBUG_INFO("Set to the default success.\n");
  402. failCount = 0;
  403. }
  404. else
  405. {
  406. DEBUG_WARN("Set to the default fail (<%02x>).\n", rx_Array[0]);
  407. failCount++;
  408. }
  409. if(failCount > RETRY_LIMIT)
  410. {
  411. DEBUG_ERROR("Set to the default fail over retry limit.\n");
  412. return FAIL;
  413. }
  414. }while((rx_Array[0] != ACK) || (rx_len != 1));
  415. //===============================================
  416. // set to protocol_2 (BLP Protocol)
  417. //===============================================
  418. do
  419. {
  420. rx_len =system_command(UartFd, BLP_CMD_USI2, ARRAY_SIZE(BLP_CMD_USI2), rx_Array);
  421. if((rx_Array[0] == ACK) && (rx_len ==1))
  422. {
  423. DEBUG_INFO("Set protocol to USI2 success.\n");
  424. DEBUG_INFO("Wait for payment module warm-reset.\n");
  425. sleep(10);
  426. failCount = 0;
  427. }
  428. else
  429. {
  430. DEBUG_WARN("Set protocol to USI2 fail (<%02x>).\n", rx_Array[0]);
  431. failCount++;
  432. }
  433. if(failCount > RETRY_LIMIT)
  434. {
  435. DEBUG_ERROR("Set protocol to USI2 fail over retry limit.\n");
  436. return FAIL;
  437. }
  438. }while((rx_Array[0] != ACK) || (rx_len !=1));
  439. //===============================================
  440. // Disable self-arm mode (USI2 Protocol)
  441. //===============================================
  442. do
  443. {
  444. USI_CMD_ARM_DISABLE[ARRAY_SIZE(USI_CMD_ARM_DISABLE)-1] = calChksum(USI_CMD_ARM_DISABLE, ARRAY_SIZE(USI_CMD_ARM_DISABLE));
  445. system_command(UartFd, USI_CMD_ARM_DISABLE, ARRAY_SIZE(USI_CMD_ARM_DISABLE), rx_Array);
  446. rx_len = USI2_Parse(rx_Array, rx_Data);
  447. if((rx_Data[0] == ACK) && (rx_len >= 0))
  448. {
  449. DEBUG_INFO("Set disable self-arm mode success.\n");
  450. failCount = 0;
  451. }
  452. else
  453. {
  454. DEBUG_WARN("Set disable self-arm mode fail (<%02x>).\n", rx_Data[0]);
  455. failCount++;
  456. }
  457. if(failCount > RETRY_LIMIT)
  458. {
  459. DEBUG_ERROR("Set disable self-arm mode fail over retry limit.\n");
  460. return FAIL;
  461. }
  462. }while((rx_Data[0] != ACK) || (rx_len < 0));
  463. //===============================================
  464. // Enable TLV command (BLP Protocol)
  465. //===============================================
  466. do
  467. {
  468. rx_len = system_command(UartFd, BLP_CMD_TLV_ENABLE, ARRAY_SIZE(BLP_CMD_TLV_ENABLE), rx_Array);
  469. if((rx_Array[0] == ACK) && (rx_len ==1))
  470. {
  471. DEBUG_INFO("Set enable TLV command success.\n");
  472. failCount = 0;
  473. }
  474. else
  475. {
  476. DEBUG_WARN("Set enable TLV command fail (<%02x>).\n", rx_Array[0]);
  477. failCount++;
  478. }
  479. if(failCount > RETRY_LIMIT)
  480. {
  481. DEBUG_ERROR("Set enable TLV command fail over retry limit.\n");
  482. return FAIL;
  483. }
  484. }while((rx_Array[0] != ACK) || (rx_len != 1));
  485. //===============================================
  486. // Set AID type (BLP Protocol)
  487. //===============================================
  488. do
  489. {
  490. rx_len = system_command(UartFd, BLP_CMD_SET_AID, ARRAY_SIZE(BLP_CMD_SET_AID), rx_Array);
  491. if((rx_Array[0] == ACK) && (rx_len ==1))
  492. {
  493. DEBUG_INFO("Set AID type success.\n");
  494. failCount = 0;
  495. }
  496. else
  497. {
  498. DEBUG_WARN("Set AID type fail (<%02x>).\n", rx_Array[0]);
  499. failCount++;
  500. }
  501. if(failCount > RETRY_LIMIT)
  502. {
  503. DEBUG_ERROR("Set AID type fail over retry limit.\n");
  504. return FAIL;
  505. }
  506. }while((rx_Array[0] != ACK) || (rx_len != 1));
  507. //===============================================
  508. // Set LED indicator (BLP Protocol)
  509. //===============================================
  510. do
  511. {
  512. rx_len = system_command(UartFd, BLP_CMD_LED_IND, ARRAY_SIZE(BLP_CMD_LED_IND), rx_Array);
  513. if((rx_Array[0] == ACK) && (rx_len ==1))
  514. {
  515. DEBUG_INFO("Set LED indicator success.\n");
  516. failCount = 0;
  517. }
  518. else
  519. {
  520. DEBUG_WARN("Set LED indicator fail (<%02x>).\n", rx_Array[0]);
  521. failCount++;
  522. }
  523. if(failCount > RETRY_LIMIT)
  524. {
  525. DEBUG_ERROR("Set LED indicator over retry limit.\n");
  526. return FAIL;
  527. }
  528. }while((rx_Array[0] != ACK) || (rx_len != 1));
  529. //===============================================
  530. // Set CA public key type (BLP Protocol)
  531. //===============================================
  532. do
  533. {
  534. rx_len = system_command(UartFd, BLP_CMD_CA_PKEY, ARRAY_SIZE(BLP_CMD_CA_PKEY), rx_Array);
  535. if((rx_Array[0] == ACK) && (rx_len ==1))
  536. {
  537. DEBUG_INFO("Set CA public key type success.\n");
  538. failCount = 0;
  539. }
  540. else
  541. {
  542. DEBUG_WARN("Set CA public key type fail (<%02x>).\n", rx_Array[0]);
  543. failCount++;
  544. }
  545. if(failCount > RETRY_LIMIT)
  546. {
  547. DEBUG_ERROR("Set CA public key type over retry limit.\n");
  548. return FAIL;
  549. }
  550. }while((rx_Array[0] != ACK) || (rx_len != 1));
  551. //===============================================
  552. // Set payment card detect type (BLP Protocol)
  553. //===============================================
  554. do
  555. {
  556. rx_len = system_command(UartFd, BLP_CMD_CARD_DETECT, ARRAY_SIZE(BLP_CMD_CARD_DETECT), rx_Array);
  557. if((rx_Array[0] == ACK) && (rx_len ==1))
  558. {
  559. DEBUG_INFO("Set payment card type success.\n");
  560. failCount = 0;
  561. }
  562. else
  563. {
  564. DEBUG_WARN("Set payment card type fail (<%02x>).\n", rx_Array[0]);
  565. failCount++;
  566. }
  567. if(failCount > RETRY_LIMIT)
  568. {
  569. DEBUG_ERROR("Set payment card type fail over retry limit.\n");
  570. return FAIL;
  571. }
  572. }while((rx_Array[0] != ACK) || (rx_len !=1));
  573. //===============================================
  574. // Disable soft card wallet application (BLP Protocol)
  575. //===============================================
  576. do
  577. {
  578. rx_len = system_command(UartFd, BLP_CMD_SOFTCARD_DISABLE, ARRAY_SIZE(BLP_CMD_SOFTCARD_DISABLE), rx_Array);
  579. if((rx_Array[0] == ACK) && (rx_len ==1))
  580. {
  581. DEBUG_INFO("Set disable soft card success.\n");
  582. failCount = 0;
  583. }
  584. else
  585. {
  586. DEBUG_WARN("Set disable soft card fail (<%02x>).\n", rx_Array[0]);
  587. failCount++;
  588. }
  589. if(failCount > RETRY_LIMIT)
  590. {
  591. DEBUG_ERROR("Set disable soft card fail over retry limit.\n");
  592. return FAIL;
  593. }
  594. }while((rx_Array[0] != ACK) || (rx_len !=1));
  595. //===============================================
  596. // Disable beep (BLP Protocol)
  597. //===============================================
  598. do
  599. {
  600. rx_len = system_command(UartFd, BLP_CMD_BEEP_DISABLE, ARRAY_SIZE(BLP_CMD_BEEP_DISABLE), rx_Array);
  601. if((rx_Array[0] == ACK) && (rx_len ==1))
  602. {
  603. DEBUG_INFO("Set beep disable success.\n");
  604. failCount = 0;
  605. }
  606. else
  607. {
  608. DEBUG_WARN("Set beep disable fail (<%02x>).\n", rx_Array[0]);
  609. failCount++;
  610. }
  611. if(failCount > RETRY_LIMIT)
  612. {
  613. DEBUG_ERROR("Set beep disable fail over retry limit.\n");
  614. return FAIL;
  615. }
  616. }while((rx_Array[0] != ACK) || (rx_len !=1));
  617. //===============================================
  618. // Set error beep style (BLP Protocol)
  619. //===============================================
  620. do
  621. {
  622. rx_len = system_command(UartFd, BLP_CMD_ERR_BEEP_STYLE, ARRAY_SIZE(BLP_CMD_ERR_BEEP_STYLE), rx_Array);
  623. if((rx_Array[0] == ACK) && (rx_len ==1))
  624. {
  625. DEBUG_INFO("Set error beep style success.\n");
  626. failCount = 0;
  627. }
  628. else
  629. {
  630. DEBUG_WARN("Set error beep style fail (<%02x>).\n", rx_Array[0]);
  631. failCount++;
  632. }
  633. if(failCount > RETRY_LIMIT)
  634. {
  635. DEBUG_ERROR("Set error beep style fail over retry limit.\n");
  636. return FAIL;
  637. }
  638. }while((rx_Array[0] != ACK) || (rx_len !=1));
  639. //===============================================
  640. // Set error beep (BLP Protocol)
  641. //===============================================
  642. do
  643. {
  644. rx_len = system_command(UartFd, BLP_CMD_ERR_BEEP, ARRAY_SIZE(BLP_CMD_ERR_BEEP), rx_Array);
  645. if((rx_Array[0] == ACK) && (rx_len ==1))
  646. {
  647. DEBUG_INFO("Set error beep success.\n");
  648. failCount = 0;
  649. }
  650. else
  651. {
  652. DEBUG_WARN("Set error beep fail (<%02x>).\n", rx_Array[0]);
  653. failCount++;
  654. }
  655. if(failCount > RETRY_LIMIT)
  656. {
  657. DEBUG_ERROR("Set error beep fail over retry limit.\n");
  658. return FAIL;
  659. }
  660. }while((rx_Array[0] != ACK) || (rx_len !=1));
  661. //===============================================
  662. // Set enable LRC (BLP Protocol)
  663. //===============================================
  664. do
  665. {
  666. rx_len = system_command(UartFd, BLP_CMD_LRC_ENABLE, ARRAY_SIZE(BLP_CMD_LRC_ENABLE), rx_Array);
  667. if((rx_Array[0] == ACK) && (rx_len ==1))
  668. {
  669. DEBUG_INFO("Set enable LRC success.\n");
  670. failCount = 0;
  671. }
  672. else
  673. {
  674. DEBUG_WARN("Set enable LRC fail (<%02x>).\n", rx_Array[0]);
  675. failCount++;
  676. }
  677. if(failCount > RETRY_LIMIT)
  678. {
  679. DEBUG_ERROR("Set enable LRC fail over retry limit.\n");
  680. return FAIL;
  681. }
  682. }while((rx_Array[0] != ACK) || (rx_len !=1));
  683. //===============================================
  684. // Set TX data tracks (BLP Protocol)
  685. //===============================================
  686. do
  687. {
  688. rx_len = system_command(UartFd, BLP_CMD_TX_TRACK, ARRAY_SIZE(BLP_CMD_TX_TRACK), rx_Array);
  689. if((rx_Array[0] == ACK) && (rx_len ==1))
  690. {
  691. DEBUG_INFO("Set TX data tracks success.\n");
  692. failCount = 0;
  693. }
  694. else
  695. {
  696. DEBUG_WARN("Set TX data tracks fail (<%02x>).\n", rx_Array[0]);
  697. failCount++;
  698. }
  699. if(failCount > RETRY_LIMIT)
  700. {
  701. DEBUG_ERROR("Set TX data tracks over retry limit.\n");
  702. return FAIL;
  703. }
  704. }while((rx_Array[0] != ACK) || (rx_len !=1));
  705. //===============================================
  706. // Set module RTC date (USI2 Protocol)
  707. //===============================================
  708. do
  709. {
  710. CurrentTime = time(NULL);
  711. tm=localtime(&CurrentTime);
  712. USI_CMD_SET_DATE[6] = ((tm->tm_year+1900)/100);
  713. USI_CMD_SET_DATE[7] = ((tm->tm_year+1900)%100);
  714. USI_CMD_SET_DATE[8] = (tm->tm_mon+1);
  715. USI_CMD_SET_DATE[9] = tm->tm_mday;
  716. USI_CMD_SET_DATE[10] = tm->tm_wday==0?0x07:tm->tm_wday;
  717. USI_CMD_SET_DATE[ARRAY_SIZE(USI_CMD_SET_DATE)-1] = calChksum(USI_CMD_SET_DATE, ARRAY_SIZE(USI_CMD_SET_DATE));
  718. system_command(UartFd, USI_CMD_SET_DATE, ARRAY_SIZE(USI_CMD_SET_DATE), rx_Array);
  719. rx_len = USI2_Parse(rx_Array, rx_Data);
  720. if((rx_Data[0] == ACK) && (rx_len >= 0))
  721. {
  722. DEBUG_INFO("Set module RTC date success.\n");
  723. failCount = 0;
  724. }
  725. else
  726. {
  727. DEBUG_WARN("Set module RTC date fail (<%02x>).\n", rx_Data[0]);
  728. failCount++;
  729. }
  730. if(failCount > RETRY_LIMIT)
  731. {
  732. DEBUG_ERROR("Set module RTC date fail over retry limit.\n");
  733. return FAIL;
  734. }
  735. }while((rx_Data[0] != ACK) || (rx_len < 0));
  736. //===============================================
  737. // Set module RTC time (USI2 Protocol)
  738. //===============================================
  739. do
  740. {
  741. CurrentTime = time(NULL);
  742. tm=localtime(&CurrentTime);
  743. USI_CMD_SET_TIME[6] = tm->tm_hour;
  744. USI_CMD_SET_TIME[7] = tm->tm_min;
  745. USI_CMD_SET_TIME[8] = tm->tm_sec;
  746. USI_CMD_SET_TIME[ARRAY_SIZE(USI_CMD_SET_TIME)-1] = calChksum(USI_CMD_SET_TIME, ARRAY_SIZE(USI_CMD_SET_TIME));
  747. system_command(UartFd, USI_CMD_SET_TIME, ARRAY_SIZE(USI_CMD_SET_TIME), rx_Array);
  748. rx_len = USI2_Parse(rx_Array, rx_Data);
  749. if((rx_Data[0] == ACK) && (rx_len >= 0))
  750. {
  751. DEBUG_INFO("Set module RTC time success.\n");
  752. failCount = 0;
  753. }
  754. else
  755. {
  756. DEBUG_WARN("Set module RTC time fail (<%02x>).\n", rx_Data[0]);
  757. failCount++;
  758. }
  759. if(failCount > RETRY_LIMIT)
  760. {
  761. DEBUG_ERROR("Set module RTC time fail over retry limit.\n");
  762. return FAIL;
  763. }
  764. }while((rx_Data[0] != ACK) || (rx_len < 0));
  765. DEBUG_INFO("Payment module initialize completed.\n");
  766. //===============================================
  767. // Main loop
  768. //===============================================
  769. for(;;)
  770. {
  771. USI_CMD_NFC_ACTIVE[ARRAY_SIZE(USI_CMD_NFC_ACTIVE)-1] = calChksum(USI_CMD_NFC_ACTIVE, ARRAY_SIZE(USI_CMD_NFC_ACTIVE));
  772. if(system_command(UartFd, USI_CMD_NFC_ACTIVE, ARRAY_SIZE(USI_CMD_NFC_ACTIVE), rx_Array) > 0)
  773. {
  774. rx_len = USI2_Parse(rx_Array, rx_Data);
  775. if((rx_Data[2] == 0x00) && (rx_len >= 0))
  776. {
  777. USI_CMD_NFC_POLL[ARRAY_SIZE(USI_CMD_NFC_POLL)-1] = calChksum(USI_CMD_NFC_POLL, ARRAY_SIZE(USI_CMD_NFC_POLL));
  778. if(system_command(UartFd, USI_CMD_NFC_POLL, ARRAY_SIZE(USI_CMD_NFC_POLL), rx_Array) > 0)
  779. {
  780. rx_len = USI2_Parse(rx_Array, rx_Data);
  781. if((rx_Data[2] == 0x00) && (rx_len >= 0))
  782. {
  783. // FELICA
  784. memcpy(&C9_Result.result_data, rx_Data, rx_len);
  785. C9_Result.status = C9_Result.result_data[2];
  786. C9_Result.pos_entry = Felica;
  787. memset(C9_Result.u_id, 0x00, ARRAY_SIZE(C9_Result.u_id));
  788. memcpy(C9_Result.u_id, &C9_Result.result_data[6], 6);
  789. DEBUG_INFO("FELICA, UID: %02X-%02X-%02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3], C9_Result.u_id[4], C9_Result.u_id[5]);
  790. // Wait card removed
  791. do
  792. {
  793. USI_CMD_NFC_POLL[ARRAY_SIZE(USI_CMD_NFC_POLL)-1] = calChksum(USI_CMD_NFC_POLL, ARRAY_SIZE(USI_CMD_NFC_POLL));
  794. system_command(UartFd, USI_CMD_NFC_POLL, ARRAY_SIZE(USI_CMD_NFC_POLL), rx_Array);
  795. rx_len = USI2_Parse(rx_Array, rx_Data);
  796. if((rx_Data[2] != 0x00) && (rx_len >= 0))
  797. {
  798. DEBUG_INFO("Card removed.\n");
  799. C9_Result.isCardPreset = OFF;
  800. failCount = 0;
  801. }
  802. else if((rx_Data[0] == 0x00) && (rx_len >= 0))
  803. {
  804. //DEBUG_INFO("Card seated.\n");
  805. C9_Result.isCardPreset = ON;
  806. failCount = 0;
  807. }
  808. else
  809. {
  810. DEBUG_WARN("Module status read fail (<%02x>).\n", rx_Data[0]);
  811. failCount++;
  812. }
  813. if(failCount > RETRY_LIMIT)
  814. {
  815. C9_Result.isCardPreset = OFF;
  816. DEBUG_ERROR("Module status read fail over retry limit.\n");
  817. }
  818. }while(((rx_Data[2] == 0x00) || (rx_len < 0)) && (failCount < RETRY_LIMIT));
  819. }
  820. else
  821. {
  822. // MIFARE, IDO-15693, Credit Card, Apple Pay in query
  823. if(C8_Polling == true)
  824. {
  825. Wait_C9 = false;
  826. CurrentTime = time(NULL);
  827. tm=localtime(&CurrentTime);
  828. USI_CMD_C8_DEACTIVAVE[ARRAY_SIZE(USI_CMD_C8_DEACTIVAVE)-1] = calChksum(USI_CMD_C8_DEACTIVAVE, ARRAY_SIZE(USI_CMD_C8_DEACTIVAVE));
  829. if(system_command(UartFd, USI_CMD_C8_DEACTIVAVE, ARRAY_SIZE(USI_CMD_C8_DEACTIVAVE), rx_Array) > 0)
  830. {
  831. rx_len = USI2_Parse(rx_Array, rx_Data);
  832. if((rx_Data[0] == ACK) && (rx_len >= 0))
  833. {
  834. //DEBUG_INFO("C8 deactivate command success.\n");
  835. failCount = 0;
  836. USI_CMD_C8_POLL[9] = decTobcd((1000/10000000000)%100);
  837. USI_CMD_C8_POLL[10] = decTobcd((1000/100000000)%100);
  838. USI_CMD_C8_POLL[11] = decTobcd((1000/1000000)%100);
  839. USI_CMD_C8_POLL[12] = decTobcd((1000/10000)%100);
  840. USI_CMD_C8_POLL[13] = decTobcd((1000/100)%100);
  841. USI_CMD_C8_POLL[14] = decTobcd((1000/1)%100);
  842. USI_CMD_C8_POLL[18] = (CURRENCY_USD>>0x08)&0xff;
  843. USI_CMD_C8_POLL[19] = (CURRENCY_USD>>0x00)&0xff;
  844. USI_CMD_C8_POLL[29] = decTobcd((tm->tm_year+1900)-2000);
  845. USI_CMD_C8_POLL[30] = decTobcd(tm->tm_mon+1);
  846. USI_CMD_C8_POLL[31] = decTobcd(tm->tm_mday);
  847. USI_CMD_C8_POLL[35] = decTobcd(tm->tm_hour);
  848. USI_CMD_C8_POLL[36] = decTobcd(tm->tm_min);
  849. USI_CMD_C8_POLL[37] = decTobcd(tm->tm_sec);
  850. USI_CMD_C8_POLL[ARRAY_SIZE(USI_CMD_C8_POLL)-1] = calChksum(USI_CMD_C8_POLL, ARRAY_SIZE(USI_CMD_C8_POLL));
  851. if(system_command(UartFd, USI_CMD_C8_POLL, ARRAY_SIZE(USI_CMD_C8_POLL), rx_Array) > 0)
  852. {
  853. rx_len = USI2_Parse(rx_Array, rx_Data);
  854. if((rx_Data[0] == ACK) && (rx_len >= 0))
  855. {
  856. //DEBUG_INFO("C8 polling command success.\n");
  857. Wait_C9 = true;
  858. failCount = 0;
  859. C9_Result.isCommandError = OFF;
  860. }
  861. else
  862. {
  863. DEBUG_INFO("C8 polling command fail (<%02x>).\n", rx_Data[0]);
  864. C9_Result.isCommandError = ON;
  865. }
  866. }
  867. }
  868. else
  869. {
  870. DEBUG_INFO("C8 deactivate command fail.\n");
  871. C9_Result.isCommandError = ON;
  872. }
  873. }
  874. if(Wait_C9 == true)
  875. {
  876. //=============================================
  877. // wait card to attach the reader and wait C9
  878. //=============================================
  879. tcflush(UartFd,TCIOFLUSH);
  880. memset(rx_Array, 0x00, ARRAY_SIZE(rx_Array));
  881. rx_len = 0;
  882. do
  883. {
  884. //DEBUG_INFO("C9 response reading...\n");
  885. usleep(2000000);
  886. rx_len = read(UartFd, rx_Array, ARRAY_SIZE(rx_Array)); // read response if data count match 512 or timeout.
  887. failCount++;
  888. } while ((rx_len == 0) && (failCount < RETRY_LIMIT));
  889. //=============================================
  890. // Parse rx_Array to rx_Data
  891. //=============================================
  892. if((rx_len > 3) && (failCount < RETRY_LIMIT))
  893. {
  894. // print this raw data before parse it.
  895. //show_data(rx_Array, rx_len);
  896. if((rx_len = USI2_Parse( rx_Array, rx_Data)) > 0)
  897. {
  898. // debug the input data message
  899. //show_data(rx_Data, rx_len);
  900. // Copy RAW data to structure
  901. memcpy(&C9_Result.result_data, rx_Data, rx_len);
  902. C9_Result.status = C9_Result.result_data[1];
  903. C9_Result.pos_entry = C9_Result.result_data[2];
  904. switch(C9_Result.pos_entry)
  905. {
  906. case VISA_qVSDC:
  907. case VISA_MSD:
  908. case MASTER_MChip:
  909. case Master_MagStripe:
  910. case AMEX_EMV:
  911. case AMEX_MSD:
  912. memset(C9_Result.u_id, 0x00, ARRAY_SIZE(C9_Result.u_id));
  913. memcpy(C9_Result.u_id, &C9_Result.result_data[5], 16);
  914. DEBUG_INFO("Credit card SN:\n");
  915. show_data(C9_Result.u_id, 16);
  916. for(uint8_t idx=0;idx<getSentinelQuantity(C9_Result.result_data, rx_len);idx++)
  917. {
  918. memset(C9_Result.tkData[idx], 0x00, ARRAY_SIZE(C9_Result.tkData[idx]));
  919. memcpy(C9_Result.tkData[idx],
  920. &C9_Result.result_data[((idx==0)?3:getSentinelPosition(C9_Result.result_data, rx_len, idx-1)+2)],
  921. (idx==0?getSentinelPosition(C9_Result.result_data, rx_len, idx)+1-3+1:getSentinelPosition(C9_Result.result_data, rx_len, idx)+1-getSentinelPosition(C9_Result.result_data, rx_len, idx-1)+2+1));
  922. DEBUG_INFO("TK[%d]: \n", idx);
  923. show_data(C9_Result.tkData[idx], getSentinelPosition(C9_Result.tkData[idx], ARRAY_SIZE(C9_Result.tkData[idx]), 0)+2);
  924. }
  925. break;
  926. case Mifare:
  927. data_len = C9_Result.result_data[6];
  928. memset(C9_Result.u_id, 0x00, ARRAY_SIZE(C9_Result.u_id));
  929. memcpy(C9_Result.u_id, &C9_Result.result_data[7], data_len);
  930. switch(C9_Result.result_data[3])
  931. {
  932. case MIFARE_ULTRALIGHT:
  933. DEBUG_INFO("MIFARE Ultralight, UID: %02X-%02X-%02X-%02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3], C9_Result.u_id[4], C9_Result.u_id[5], C9_Result.u_id[6]);
  934. break;
  935. case MIFARE_CLASSIC_1K:
  936. DEBUG_INFO("MIFARE Classic 1K, UID: %02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3]);
  937. break;
  938. case MIFARE_CLASSIC_4K:
  939. DEBUG_INFO("MIFARE Classic 4K, UID: %02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3]);
  940. break;
  941. case MIFARE_DESFIRE:
  942. DEBUG_INFO("MIFARE DESFire, UID: %02X-%02X-%02X-%02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3], C9_Result.u_id[4], C9_Result.u_id[5], C9_Result.u_id[6]);
  943. break;
  944. case MIFARE_PLUS_2K:
  945. DEBUG_INFO("MIFARE Plus 2k, UID: %02X-%02X-%02X-%02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3], C9_Result.u_id[4], C9_Result.u_id[5], C9_Result.u_id[6]);
  946. break;
  947. case MIFARE_MINI:
  948. DEBUG_INFO("MIFARE Mini, UID: %02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3]);
  949. break;
  950. case MIFARE_RESERVE:
  951. DEBUG_INFO("MIFARE Reserve, UID: \n");
  952. break;
  953. case MIFARE_JEWEL:
  954. DEBUG_INFO("MIFARE Jewel, UID: \n");
  955. break;
  956. case MIFARE_JCOP31:
  957. DEBUG_INFO("MIFARE JCOP31, UID: \n");
  958. break;
  959. }
  960. break;
  961. case ISO_15693:
  962. data_len =(C9_Result.result_data[5]<<8) | C9_Result.result_data[6];
  963. memset(C9_Result.u_id, 0x00, ARRAY_SIZE(C9_Result.u_id));
  964. memcpy(C9_Result.u_id, &C9_Result.result_data[4+data_len-8], 8);
  965. DEBUG_INFO("ISO_15693, UID: %02X-%02X-%02X-%02X-%02X-%02X-%02X-%02X\n", C9_Result.u_id[0], C9_Result.u_id[1], C9_Result.u_id[2], C9_Result.u_id[3], C9_Result.u_id[4], C9_Result.u_id[5], C9_Result.u_id[6], C9_Result.u_id[7]);
  966. break;
  967. case Apple_Pay:
  968. DEBUG_INFO("Apple_Pay VAS only.\n");
  969. for(uint8_t idx=0;idx<getSentinelQuantity(C9_Result.result_data, rx_len);idx++)
  970. {
  971. memcpy(C9_Result.tkData[idx],
  972. &C9_Result.result_data[((idx==0)?3:getSentinelPosition(C9_Result.result_data, rx_len, idx-1)+2)],
  973. (idx==0?getSentinelPosition(C9_Result.result_data, rx_len, idx)+1-3+1:getSentinelPosition(C9_Result.result_data, rx_len, idx)+1-getSentinelPosition(C9_Result.result_data, rx_len, idx-1)+2+1));
  974. DEBUG_INFO("TK[%d]: \n", idx);
  975. show_data(C9_Result.tkData[idx], getSentinelPosition(C9_Result.tkData[idx], ARRAY_SIZE(C9_Result.tkData[idx]), 0)+2);
  976. }
  977. break;
  978. case No_Data:
  979. //DEBUG_INFO("No any data.\n");
  980. break;
  981. default:
  982. DEBUG_WARN("Unknown pos entry.\n");
  983. if(C9_Result.status == C9_RES_C8_CMD_TIMEOUT)
  984. DEBUG_WARN("C8 command timeout.\n");
  985. else if(C9_Result.status == C9_RES_CMD_EXECUTING)
  986. DEBUG_WARN("Command executing or Wait Card to be Removed.\n");
  987. break;
  988. }
  989. failCount = 0;
  990. // Wait card removed
  991. if(C9_Result.pos_entry != No_Data)
  992. {
  993. do
  994. {
  995. system_command(UartFd, USI_CMD_MODULE_STATUS, ARRAY_SIZE(USI_CMD_MODULE_STATUS), rx_Array);
  996. rx_len = USI2_Parse(rx_Array, rx_Data);
  997. if(!(rx_Data[0]&0x02) && (rx_len >= 0))
  998. {
  999. DEBUG_INFO("Card removed.\n");
  1000. C9_Result.isCardPreset = OFF;
  1001. failCount = 0;
  1002. }
  1003. else if((rx_Data[0]&0x02) && (rx_len >= 0))
  1004. {
  1005. //DEBUG_INFO("Card seated.\n");
  1006. C9_Result.isCardPreset = ON;
  1007. failCount = 0;
  1008. }
  1009. else
  1010. {
  1011. DEBUG_WARN("Module status read fail (<%02x>).\n", rx_Data[0]);
  1012. failCount++;
  1013. }
  1014. if(failCount > RETRY_LIMIT)
  1015. {
  1016. C9_Result.isCardPreset = OFF;
  1017. DEBUG_ERROR("Module status read fail over retry limit.\n");
  1018. }
  1019. }while(((rx_Data[0]&0x02) || (rx_len < 0)) && (failCount < RETRY_LIMIT));
  1020. }
  1021. }
  1022. else
  1023. {
  1024. DEBUG_INFO("C9 Parsing result fail.\n");
  1025. C9_Result.isCommandError = ON;
  1026. // Wait card removed
  1027. do
  1028. {
  1029. usleep(500000);
  1030. system_command(UartFd, USI_CMD_MODULE_STATUS, ARRAY_SIZE(USI_CMD_MODULE_STATUS), rx_Array);
  1031. rx_len = USI2_Parse(rx_Array, rx_Data);
  1032. if(!(rx_Data[0]&0x02) && (rx_len >= 0))
  1033. {
  1034. DEBUG_INFO("Card removed.\n");
  1035. C9_Result.isCardPreset = OFF;
  1036. C9_Result.isCommandError = OFF;
  1037. failCount = 0;
  1038. }
  1039. else if((rx_Data[0]&0x02) && (rx_len >= 0))
  1040. {
  1041. //DEBUG_INFO("Card seated.\n");
  1042. C9_Result.isCardPreset = ON;
  1043. failCount = 0;
  1044. }
  1045. else
  1046. {
  1047. DEBUG_WARN("Module status read fail (<%02x>).\n", rx_Data[0]);
  1048. failCount++;
  1049. }
  1050. if(failCount > RETRY_LIMIT)
  1051. {
  1052. C9_Result.isCardPreset = OFF;
  1053. DEBUG_ERROR("Module status read fail over retry limit.\n");
  1054. }
  1055. }while(((rx_Data[0]&0x02) || (rx_len < 0)) && (failCount < RETRY_LIMIT));
  1056. }
  1057. }
  1058. else
  1059. {
  1060. DEBUG_WARN("C9 Response timeout: %d \n", failCount);
  1061. }
  1062. }
  1063. }
  1064. }
  1065. }
  1066. }
  1067. }
  1068. usleep(500000);
  1069. }
  1070. return FAIL;
  1071. }