Module_Payment.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618
  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 CMD_C8[30] = {0xc8,0x01, // Activate the reader
  9. 0x9f,0x02,0x06,0x00,0x00,0x00,0x01,0x00,0x00, // Amount, Authorized, sample: 100.00 dollar
  10. 0x5f,0x2a,0x02,0x08,0x40, // Transaction Currency Code follow ISO-4217, sample: 0840(USD)
  11. 0x9c,0x01,0x00, // Transaction Type, sample: 00
  12. 0x9a,0x03,0x21,0x03,0x24, // Transaction Date, sample: 2021/03/24
  13. 0x9f,0x21,0x03,0x13,0x36,0x10 }; // Transaction Time, sample: 13:36:10
  14. unsigned char CMD_CARD_DETECT[11] = {0x09, 0, 0x07, 'M','F','1','4','1','2','1', 0x32}; // Enable payment, MIFARE, 15693 card; Detect Payment Card First; Enable ApplePay VAS
  15. unsigned char CMD_USI2[7] = {0x09, 0, 0x03, 'P', 'C', '0', 0x29}; // Configure protocol to USI2
  16. unsigned char CMD_SET_BAUD[7] = {0x09, 0, 0x03, 'B', 'R', '7', 0x2d}; // Configure module baud rate to 115200
  17. unsigned char CMD_RESTORE_DEFAULT[7] = {0x09, 0, 0x03, 'D', 'F', 0, 0x08 }; // Restore module configuration to default setting
  18. int system_command(int uart, unsigned char* cmd, int cmd_len, unsigned char* rx);
  19. int USI2_Parse(unsigned char* rx, unsigned char* rx_data);
  20. struct C9_RESULT
  21. {
  22. unsigned char result_data[512];
  23. unsigned char status;
  24. unsigned char pos_entry;
  25. unsigned char u_id[20];
  26. unsigned char tkData[4][128];
  27. } C9_Result;
  28. //==========================================
  29. // Common routine
  30. //==========================================
  31. int StoreLogMsg(const char *fmt, ...)
  32. {
  33. char Buf[65536+256];
  34. char buffer[65536];
  35. //char Buf[4096+256];
  36. //char buffer[4096];
  37. time_t CurrentTime;
  38. struct tm *tm;
  39. struct timeval tv;
  40. va_list args;
  41. va_start(args, fmt);
  42. int rc = vsnprintf(buffer, sizeof(buffer), fmt, args);
  43. va_end(args);
  44. memset(Buf,0,sizeof(Buf));
  45. CurrentTime = time(NULL);
  46. tm=localtime(&CurrentTime);
  47. gettimeofday(&tv, NULL); // get microseconds, 10^-6
  48. sprintf(Buf,"echo -n \"[%04d.%02d.%02d %02d:%02d:%02d.%06ld]%s\" >> /Storage/System/[%04d.%02d]PaymentLog",
  49. tm->tm_year+1900,tm->tm_mon+1,tm->tm_mday,tm->tm_hour,tm->tm_min,tm->tm_sec,tv.tv_usec,
  50. buffer,
  51. tm->tm_year+1900,tm->tm_mon+1);
  52. system((const char*)Buf);
  53. #ifdef ConsloePrintLog
  54. 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);
  55. #endif
  56. return rc;
  57. }
  58. /**
  59. * Execute shell command
  60. * @param cmd: shell command string
  61. * @return shell command execution result
  62. */
  63. int runShellCmd(const char*cmd)
  64. {
  65. int result = FAIL;
  66. char buf[256];
  67. FILE *fp;
  68. fp = popen(cmd, "r");
  69. if(fp != NULL)
  70. {
  71. while(fgets(buf, sizeof(buf), fp) != NULL)
  72. {
  73. DEBUG_INFO("%s\n", buf);
  74. }
  75. result = PASS;
  76. }
  77. pclose(fp);
  78. return result;
  79. }
  80. /**
  81. * Calculate time differential
  82. * @param ST: start time
  83. * @param ET: end time
  84. * @return time differential in million seconds
  85. */
  86. int DiffTimeb(struct timeb ST, struct timeb ET)
  87. {
  88. //return milli-second
  89. unsigned int StartTime,StopTime;
  90. StartTime=(unsigned int)ST.time;
  91. StopTime=(unsigned int)ET.time;
  92. return (StopTime-StartTime)*1000+ET.millitm-ST.millitm;
  93. }
  94. /**
  95. * Show communication raw data to debug info
  96. * @param data: raw data
  97. * @param len: data length
  98. * @param isRX: is receive data
  99. */
  100. void show_raw(uint8_t *data, uint16_t len, uint8_t isRX)
  101. {
  102. uint8_t output[8192];
  103. memset(output, 0x00, ARRAY_SIZE(output));
  104. sprintf((char*)output, "%s", (isRX?"RX: ":"TX: "));
  105. for(uint16_t idx = 0;idx<len;idx++)
  106. {
  107. sprintf((char*)output, "%s%02x ", output, data[idx]);
  108. }
  109. DEBUG_INFO("%s\n", output);
  110. }
  111. /**
  112. * Show data to debug info
  113. * @param dat: data content
  114. * @param len: data length
  115. */
  116. void show_data(unsigned char *dat, unsigned int len)
  117. {
  118. uint8_t output[8192];
  119. memset(output, 0x00, ARRAY_SIZE(output));
  120. sprintf((char*)output, "Data: ");
  121. for(uint16_t idx = 0;idx<len;idx++)
  122. {
  123. if(dat[idx] > 31 && dat[idx] < 128 )
  124. sprintf((char*)output, "%s%c", output, dat[idx]);
  125. else
  126. sprintf((char*)output, "%s<%x>", output, dat[idx]);
  127. }
  128. DEBUG_INFO("%s\n", output);
  129. }
  130. /**
  131. * Get sentinel quantity in data array
  132. * @param data: message array address
  133. * @param dataLen: array seek size
  134. * @return how many sentinel flag found
  135. */
  136. int getSentinelQuantity(unsigned char *data, unsigned int dataLen)
  137. {
  138. int result = 0;
  139. for(uint16_t idx=0;idx<dataLen;idx++)
  140. {
  141. if(data[idx] == '?')
  142. result++;
  143. }
  144. return result;
  145. }
  146. /**
  147. * Get sentinel position in array
  148. * @param data: message array address
  149. * @param dataLen: array seek size
  150. * @param idxSentinel: which sentinel idx want to find, since 0 start
  151. * @return sentinel position in array
  152. */
  153. int getSentinelPosition(unsigned char *data, unsigned int dataLen, unsigned char idxSentinel)
  154. {
  155. int result = -1;
  156. int foundCnt = -1;
  157. for(uint16_t idx=0;idx<dataLen;idx++)
  158. {
  159. if(data[idx] == '?')
  160. foundCnt++;
  161. if(foundCnt == idxSentinel)
  162. {
  163. result = idx;
  164. break;
  165. }
  166. }
  167. return result;
  168. }
  169. //==========================================
  170. // Init share memory
  171. //==========================================
  172. /**
  173. * Share memory initialization
  174. * @return function result
  175. */
  176. int InitShareMemory()
  177. {
  178. int result = PASS;
  179. int MeterSMId;
  180. //init ShmSysConfigAndInfo
  181. if ((MeterSMId = shmget(ShmSysConfigAndInfoKey, sizeof(struct SysConfigAndInfo), 0777)) < 0)
  182. {
  183. DEBUG_ERROR("shmget ShmSysConfigAndInfo NG\n");
  184. result = FAIL;
  185. }
  186. else if ((ShmSysConfigAndInfo = shmat(MeterSMId, NULL, 0)) == (void *) -1)
  187. {
  188. DEBUG_ERROR("shmat ShmSysConfigAndInfo NG\n");
  189. result = FAIL;
  190. }
  191. else
  192. {}
  193. //init ShmStatusCodeData
  194. if ((MeterSMId = shmget(ShmStatusCodeKey, sizeof(struct StatusCodeData), 0777)) < 0)
  195. {
  196. DEBUG_ERROR("shmget ShmStatusCodeData NG\n");
  197. result = FAIL;
  198. }
  199. else if ((ShmStatusCodeData = shmat(MeterSMId, NULL, 0)) == (void *) -1)
  200. {
  201. DEBUG_ERROR("shmat ShmStatusCodeData NG\n");
  202. result = FAIL;
  203. }
  204. else
  205. {}
  206. return result;
  207. }
  208. //==========================================
  209. // Init com port
  210. //==========================================
  211. /**
  212. * TTY port initialization
  213. * @return port initial result
  214. */
  215. int InitComPort()
  216. {
  217. int fd;
  218. struct termios tios;
  219. fd = open("/dev/ttyS3", O_RDWR);
  220. if(fd<=0)
  221. {
  222. return FAIL;
  223. }
  224. ioctl (fd, TCGETS, &tios);
  225. tios.c_cflag = B9600| CS8 | CLOCAL | CREAD;
  226. tios.c_lflag = 0;
  227. tios.c_iflag = 0;
  228. tios.c_oflag = 0;
  229. tios.c_cc[VMIN]=0;
  230. tios.c_cc[VTIME]=(unsigned char)5; // timeout 0.5 seconds
  231. tios.c_lflag=0;
  232. tcflush(fd, TCIFLUSH);
  233. ioctl (fd, TCSETS, &tios);
  234. return fd;
  235. }
  236. /**
  237. * Send command to UIC680fg module.
  238. * @param uart: port handle
  239. * @param cmd: command buffer
  240. * @param cmd_len: command length
  241. * @param rx: receive buffer
  242. * @return receive data length
  243. */
  244. int system_command(int uart, unsigned char* cmd, int cmd_len, unsigned char* rx)
  245. {
  246. int rx_len = 0;
  247. tcflush(uart,TCIOFLUSH);
  248. show_raw(cmd, cmd_len, NO);
  249. if(write(uart, cmd, cmd_len) > 0)
  250. {
  251. /*
  252. * TODO: Improve sleep time.
  253. */
  254. usleep(1000000);
  255. rx_len = read(uart, rx, 512);
  256. show_raw(rx, rx_len, YES);
  257. }
  258. else
  259. {
  260. DEBUG_ERROR("system command write fail.\n");
  261. }
  262. return rx_len;
  263. }
  264. /**
  265. * Parsing raw data to USI data
  266. * @param rx: raw data
  267. * @param rx_data: parsing result data
  268. * @return parsing result data length
  269. */
  270. int USI2_Parse(unsigned char* rx, unsigned char* rx_data)
  271. {
  272. int result = -1;
  273. unsigned int data_len =0;
  274. unsigned int chksum = 0;
  275. if(rx[0] == SOH) // SOH = 0x01
  276. {
  277. data_len = (unsigned int)rx[2] <<8;
  278. data_len |= rx[3];
  279. for(int idx=0;idx<(data_len+4);idx++)
  280. {
  281. chksum ^= rx[idx];
  282. }
  283. if((chksum&0xff) == rx[(data_len+4)])
  284. {
  285. memcpy(rx_data, &rx[4], data_len);
  286. result = data_len;
  287. }
  288. }
  289. else
  290. {
  291. DEBUG_WARN("USI2 message header is not <01>.\n");
  292. }
  293. return result;
  294. }
  295. //==========================================
  296. // Main loop
  297. //==========================================
  298. int main(void)
  299. {
  300. int UartFd;
  301. uint16_t failCount = 0;
  302. unsigned char rx_Array[512]={0}, rx_Data[512]={0};
  303. char C8_Polling = true;
  304. char Wait_C9 = false;
  305. int rx_len = 0;
  306. int data_len = 0;
  307. //===============================================
  308. // Initialization
  309. //===============================================
  310. if(InitShareMemory() == FAIL)
  311. {
  312. DEBUG_ERROR("InitShareMemory NG\n");
  313. if(ShmStatusCodeData!=NULL)
  314. {
  315. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.FailToCreateShareMemory=1;
  316. }
  317. sleep(5);
  318. return FAIL;
  319. }
  320. UartFd=InitComPort();
  321. if(UartFd<0)
  322. {
  323. DEBUG_ERROR("InitComPort NG\n");
  324. if(ShmStatusCodeData!=NULL)
  325. {
  326. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.CsuInitFailed=1;
  327. }
  328. sleep(5);
  329. return FAIL;
  330. }
  331. else
  332. {
  333. DEBUG_INFO("ttyS3 port open success.\n");
  334. }
  335. //===============================================
  336. // Payment module configuration set to default.
  337. //===============================================
  338. do
  339. {
  340. rx_len = system_command(UartFd, CMD_RESTORE_DEFAULT, ARRAY_SIZE(CMD_RESTORE_DEFAULT), rx_Array);
  341. if((rx_Array[0] == ACK) && (rx_len ==1))
  342. {
  343. DEBUG_INFO("Set to the default success.\n");
  344. failCount = 0;
  345. }
  346. else
  347. {
  348. DEBUG_WARN("Set to the default fail (<%02x>).\n", rx_Array[0]);
  349. failCount++;
  350. }
  351. if(failCount > RETRY_LIMIT)
  352. {
  353. DEBUG_ERROR("Set to the default fail over retry limit.\n");
  354. return FAIL;
  355. }
  356. }while((rx_Array[0] != ACK) || (rx_len !=1));
  357. //===============================================
  358. // set payment card detect type.
  359. //===============================================
  360. do
  361. {
  362. rx_len = system_command(UartFd, CMD_CARD_DETECT, ARRAY_SIZE(CMD_CARD_DETECT), rx_Array);
  363. if((rx_Array[0] == ACK) && (rx_len ==1))
  364. {
  365. DEBUG_INFO("Set payment card type success.\n");
  366. failCount = 0;
  367. }
  368. else
  369. {
  370. DEBUG_WARN("Set payment card type fail (<%02x>).\n", rx_Array[0]);
  371. failCount++;
  372. }
  373. if(failCount > RETRY_LIMIT)
  374. {
  375. DEBUG_ERROR("Set payment card type fail over retry limit.\n");
  376. return FAIL;
  377. }
  378. }while((rx_Array[0] != ACK) || (rx_len !=1));
  379. //===============================================
  380. // set to protocol_2
  381. //===============================================
  382. do
  383. {
  384. rx_len =system_command(UartFd, CMD_USI2, ARRAY_SIZE(CMD_USI2), rx_Array);
  385. if((rx_Array[0] == ACK) && (rx_len ==1))
  386. {
  387. DEBUG_INFO("Set protocol to USI2 success.\n");
  388. failCount = 0;
  389. }
  390. else
  391. {
  392. DEBUG_WARN("Set protocol to USI2 fail (<%02x>).\n", rx_Array[0]);
  393. failCount++;
  394. }
  395. if(failCount > RETRY_LIMIT)
  396. {
  397. DEBUG_ERROR("Set protocol to USI2 fail over retry limit.\n");
  398. return FAIL;
  399. }
  400. }while((rx_Array[0] != ACK) || (rx_len !=1));
  401. //===============================================
  402. // Main loop
  403. //===============================================
  404. for(;;)
  405. {
  406. if(C8_Polling == true)
  407. {
  408. Wait_C9 = false;
  409. /*
  410. * TODO:
  411. * 1. C8 parameter configure
  412. */
  413. rx_len =system_command(UartFd, CMD_C8, sizeof(CMD_C8), rx_Array);
  414. if((rx_Array[0] == ACK) && (rx_len ==1))
  415. {
  416. DEBUG_INFO("Polling C8 command get response.\n");
  417. Wait_C9 = true;
  418. failCount = 0;
  419. }
  420. if(Wait_C9 == true)
  421. {
  422. //=============================================
  423. // wait card to attach the reader and wait C9
  424. //=============================================
  425. tcflush(UartFd,TCIOFLUSH);
  426. memset(rx_Array, 0x00, ARRAY_SIZE(rx_Array));
  427. rx_len = 0;
  428. do
  429. {
  430. sleep(1);
  431. rx_len = read(UartFd, rx_Array, ARRAY_SIZE(rx_Array)); // read response if data count match 512 or timeout.
  432. failCount++;
  433. } while ((rx_len == 0) && (failCount < RETRY_LIMIT));
  434. //=============================================
  435. // Parse rx_Array to rx_Data
  436. //=============================================
  437. if((rx_len > 3) && (failCount < RETRY_LIMIT))
  438. {
  439. // print this raw data before parse it.
  440. show_data(rx_Array, rx_len);
  441. rx_len = USI2_Parse( rx_Array, rx_Data);
  442. if(rx_len > 0)
  443. {
  444. // debug the input data message
  445. show_data(rx_Data, rx_len);
  446. // Copy RAW data to structure
  447. memcpy(&C9_Result.result_data, rx_Data, rx_len);
  448. C9_Result.status = C9_Result.result_data[1];
  449. C9_Result.pos_entry = C9_Result.result_data[2];
  450. switch(C9_Result.pos_entry)
  451. {
  452. case VISA_qVSDC:
  453. case VISA_MSD:
  454. case MASTER_MChip:
  455. case Master_MagStripe:
  456. case AMEX_EMV:
  457. case AMEX_MSD:
  458. for (int i=5; i<(5 + 16); i++)
  459. {
  460. C9_Result.u_id[i] = C9_Result.result_data[i];
  461. }
  462. DEBUG_INFO("Payment card\n");
  463. for(uint8_t idx=0;idx<getSentinelQuantity(C9_Result.result_data, rx_len);idx++)
  464. {
  465. memcpy(C9_Result.tkData[idx],
  466. &C9_Result.result_data[((idx==0)?3:getSentinelPosition(C9_Result.result_data, rx_len, idx-1)+2)],
  467. (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));
  468. DEBUG_INFO("TK[%d]: \n", idx);
  469. show_data(C9_Result.tkData[idx], ARRAY_SIZE(C9_Result.tkData[idx]));
  470. }
  471. break;
  472. case Mifare:
  473. data_len = C9_Result.result_data[6];
  474. memcpy(C9_Result.u_id, &C9_Result.result_data[7], data_len);
  475. switch(C9_Result.result_data[3])
  476. {
  477. case MIFARE_ULTRALIGHT:
  478. 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]);
  479. break;
  480. case MIFARE_CLASSIC_1K:
  481. 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]);
  482. break;
  483. case MIFARE_CLASSIC_4K:
  484. 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]);
  485. break;
  486. case MIFARE_DESFIRE:
  487. 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]);
  488. break;
  489. case MIFARE_PLUS_2K:
  490. 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]);
  491. break;
  492. case MIFARE_MINI:
  493. 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]);
  494. break;
  495. case MIFARE_RESERVE:
  496. DEBUG_INFO("MIFARE Reserve, UID: \n");
  497. break;
  498. case MIFARE_JEWEL:
  499. DEBUG_INFO("MIFARE Jewel, UID: \n");
  500. break;
  501. case MIFARE_JCOP31:
  502. DEBUG_INFO("MIFARE JCOP31, UID: \n");
  503. break;
  504. }
  505. break;
  506. case ISO_15693:
  507. data_len =(C9_Result.result_data[5]<<8) | C9_Result.result_data[6];
  508. memcpy(C9_Result.u_id, &C9_Result.result_data[4+data_len-8], 8);
  509. 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]);
  510. break;
  511. case Apple_Pay:
  512. DEBUG_INFO("Apple_Pay VAS only.\n");
  513. for(uint8_t idx=0;idx<getSentinelQuantity(C9_Result.result_data, rx_len);idx++)
  514. {
  515. memcpy(C9_Result.tkData[idx],
  516. &C9_Result.result_data[((idx==0)?3:getSentinelPosition(C9_Result.result_data, rx_len, idx-1)+2)],
  517. (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));
  518. DEBUG_INFO("TK[%d]: \n", idx);
  519. show_data(C9_Result.tkData[idx], ARRAY_SIZE(C9_Result.tkData[idx]));
  520. }
  521. break;
  522. case No_Data:
  523. DEBUG_INFO("--> No any data.\n");
  524. break;
  525. default:
  526. DEBUG_INFO("--> Unknown pos entry.\n");
  527. break;
  528. }
  529. failCount = 0;
  530. }
  531. else
  532. {
  533. DEBUG_INFO("C9 Parsing result fail.\n");
  534. }
  535. }
  536. else
  537. {
  538. DEBUG_WARN("C9 Response timeout: %d \n", failCount);
  539. }
  540. }
  541. }
  542. usleep(500000);
  543. }
  544. return FAIL;
  545. }