blob: d17594530d06669d8ad5dd59b194814b62a47397 [file] [log] [blame]
Harald Welte9d3e3822015-11-09 00:50:54 +01001#include <stdint.h>
2#include <stdio.h>
3#include <string.h>
4#include <assert.h>
5#include <stdlib.h>
6
7#include "card_emu.h"
8#include "cardemu_prot.h"
9#include "tc_etu.h"
10#include "req_ctx.h"
11
12/* stub functions required by card_emu.c */
13
14int card_emu_uart_update_fidi(uint8_t uart_chan, unsigned int fidi)
15{
16 printf("uart_update_fidi(uart_chan=%u, fidi=%u)\n", uart_chan, fidi);
17 return 0;
18}
19
20int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte)
21{
22 printf("TX: 0x%02x\n", byte);
23 return 1;
24}
25
26void card_emu_uart_enable(uint8_t uart_chan, uint8_t rxtx)
27{
Harald Welteb4362862015-11-14 19:02:33 +010028 char *rts;
29 switch (rxtx) {
30 case 0:
31 rts = "OFF";
32 break;
33 case ENABLE_TX:
34 rts = "TX";
35 break;
36 case ENABLE_RX:
37 rts = "RX";
38 break;
39 default:
40 rts = "unknown";
41 break;
42 };
43
44 printf("uart_enable(uart_chan=%u, %s)\n", uart_chan, rts);
Harald Welte9d3e3822015-11-09 00:50:54 +010045}
46
47void tc_etu_set_wtime(uint8_t tc_chan, uint16_t wtime)
48{
49 printf("tc_etu_set_wtime(tc_chan=%u, wtime=%u)\n", tc_chan, wtime);
50}
51
52void tc_etu_set_etu(uint8_t tc_chan, uint16_t etu)
53{
54 printf("tc_etu_set_etu(tc_chan=%u, etu=%u)\n", tc_chan, etu);
55}
56
57void tc_etu_init(uint8_t chan_nr, void *handle)
58{
59}
60
61
62
63
64#if 0
65/* process a single byte received from the reader */
66void card_emu_process_rx_byte(struct card_handle *ch, uint8_t byte);
67
68/* return a single byte to be transmitted to the reader */
69int card_emu_get_tx_byte(struct card_handle *ch, uint8_t *byte);
70
71/* hardware driver informs us that a card I/O signal has changed */
72void card_emu_io_statechg(struct card_handle *ch, enum card_io io, int active);
73
74/* User sets a new ATR to be returned during next card reset */
75int card_emu_set_atr(struct card_handle *ch, const uint8_t *atr, uint8_t len);
76#endif
77
78
79static int verify_atr(struct card_handle *ch)
80{
81 uint8_t atr[4];
82 uint8_t byte;
83 unsigned int i;
84
85 printf("receiving + verifying ATR:");
86 for (i = 0; i < sizeof(atr); i++) {
87 assert(card_emu_get_tx_byte(ch, &atr[i]) == 1);
88 printf(" %02x", atr[i]);
89 }
90 printf("\n");
91 assert(card_emu_get_tx_byte(ch, &byte) == 0);
92
93 return 1;
94}
95
96static void io_start_card(struct card_handle *ch)
97{
98 uint8_t byte;
99
100 /* bring the card up from the dead */
101 card_emu_io_statechg(ch, CARD_IO_VCC, 1);
102 assert(card_emu_get_tx_byte(ch, &byte) == 0);
103 card_emu_io_statechg(ch, CARD_IO_CLK, 1);
104 assert(card_emu_get_tx_byte(ch, &byte) == 0);
105 card_emu_io_statechg(ch, CARD_IO_RST, 1);
106 assert(card_emu_get_tx_byte(ch, &byte) == 0);
107
108 /* release from reset and verify th ATR */
109 card_emu_io_statechg(ch, CARD_IO_RST, 0);
110 verify_atr(ch);
111}
112
113static void send_bytes(struct card_handle *ch, const uint8_t *bytes, unsigned int len)
114{
115 unsigned int i;
Harald Welte84ec2522015-11-14 23:03:50 +0100116 for (i = 0; i < len; i++) {
117 printf("UART_RX(%02x)\n", bytes[i]);
Harald Welte9d3e3822015-11-09 00:50:54 +0100118 card_emu_process_rx_byte(ch, bytes[i]);
Harald Welte84ec2522015-11-14 23:03:50 +0100119 }
Harald Welte9d3e3822015-11-09 00:50:54 +0100120}
121
122static void dump_rctx(struct req_ctx *rctx)
123{
124 struct cardemu_usb_msg_hdr *mh =
125 (struct cardemu_usb_msg_hdr *) rctx->data;
126 struct cardemu_usb_msg_rx_data *rxd;
127 int i;
128
129 printf("req_ctx(%p): state=%u, size=%u, tot_len=%u, idx=%u, data=%p\n",
130 rctx, rctx->state, rctx->size, rctx->tot_len, rctx->idx, rctx->data);
131 printf(" msg_type=%u, seq_nr=%u, data_len=%u\n",
132 mh->msg_type, mh->seq_nr, mh->data_len);
133
134 switch (mh->msg_type) {
135 case CEMU_USB_MSGT_DO_RX_DATA:
136 rxd = (struct cardemu_usb_msg_rx_data *)mh;
137 printf(" flags=%x, data=", rxd->flags);
138 for (i = 0; i < mh->data_len; i++)
139 printf(" %02x", rxd->data[i]);
140 printf("\n");
141 break;
142 }
143}
144
Harald Welteb4362862015-11-14 19:02:33 +0100145/* emulate a TPDU header being sent by the reader/phone */
Harald Welte9d3e3822015-11-09 00:50:54 +0100146static void send_tpdu_hdr(struct card_handle *ch, const uint8_t *tpdu_hdr)
147{
148 struct req_ctx *rctx;
149
150 /* we don't want a receive context to become available during
151 * the first four bytes */
152 send_bytes(ch, tpdu_hdr, 4);
Harald Welte84ec2522015-11-14 23:03:50 +0100153 assert(!req_ctx_find_get(0, RCTX_S_USB_TX_PENDING, RCTX_S_USB_TX_BUSY));
Harald Welte9d3e3822015-11-09 00:50:54 +0100154
155 send_bytes(ch, tpdu_hdr+4, 1);
156 /* but then after the final byte of the TPDU header, we want a
157 * receive context to be available for USB transmission */
Harald Welte84ec2522015-11-14 23:03:50 +0100158 rctx = req_ctx_find_get(0, RCTX_S_USB_TX_PENDING, RCTX_S_USB_TX_BUSY);
Harald Welte9d3e3822015-11-09 00:50:54 +0100159 assert(rctx);
160 dump_rctx(rctx);
Harald Welteb4362862015-11-14 19:02:33 +0100161
162 /* free the req_ctx, indicating it has fully arrived on the host */
163 req_ctx_set_state(rctx, RCTX_S_FREE);
164}
165
166/* emulate a CEMU_USB_MSGT_DT_TX_DATA received from USB */
167static void host_to_device_data(const uint8_t *data, uint16_t len, int continue_rx)
168{
169 struct req_ctx *rctx;
Harald Welte84ec2522015-11-14 23:03:50 +0100170 struct cardemu_usb_msg_tx_data *rd;
Harald Welteb4362862015-11-14 19:02:33 +0100171
172 /* allocate a free req_ctx */
Harald Welte84ec2522015-11-14 23:03:50 +0100173 rctx = req_ctx_find_get(0, RCTX_S_FREE, RCTX_S_USB_RX_BUSY);
Harald Welteb4362862015-11-14 19:02:33 +0100174 assert(rctx);
175
176 /* initialize the header */
177 rd = (struct cardemu_usb_msg_rx_data *) rctx->data;
Harald Welte84ec2522015-11-14 23:03:50 +0100178 rctx->tot_len = sizeof(*rd);
Harald Welteb4362862015-11-14 19:02:33 +0100179 cardemu_hdr_set(&rd->hdr, CEMU_USB_MSGT_DT_TX_DATA);
Harald Welte84ec2522015-11-14 23:03:50 +0100180 rd->flags = CEMU_DATA_F_FINAL;
Harald Welteb4362862015-11-14 19:02:33 +0100181 if (continue_rx)
Harald Welte84ec2522015-11-14 23:03:50 +0100182 rd->flags |= CEMU_DATA_F_PB_AND_RX;
Harald Welteb4362862015-11-14 19:02:33 +0100183 else
Harald Welte84ec2522015-11-14 23:03:50 +0100184 rd->flags |= CEMU_DATA_F_PB_AND_TX;
Harald Welteb4362862015-11-14 19:02:33 +0100185 /* copy data and set length */
186 rd->hdr.data_len = len;
187 memcpy(rd->data, data, len);
188
189 /* hand the req_ctx to the UART transmit code */
190 req_ctx_set_state(rctx, RCTX_S_UART_TX_PENDING);
191}
192
193static int print_tx_chars(struct card_handle *ch)
194{
195 uint8_t byte;
196 int count = 0;
197
198 while (card_emu_get_tx_byte(ch, &byte)) {
199 printf("UART_TX(%02x)\n", byte);
200 count++;
201 }
202 return count;
Harald Welte9d3e3822015-11-09 00:50:54 +0100203}
204
Harald Welte84ec2522015-11-14 23:03:50 +0100205const uint8_t tpdu_hdr_sel_mf[] = { 0xA0, 0xA4, 0x00, 0x00, 0x00 };
Harald Welteb4362862015-11-14 19:02:33 +0100206const uint8_t tpdu_pb_sw[] = { 0xA4, 0x90, 0x00 };
Harald Welte9d3e3822015-11-09 00:50:54 +0100207
Harald Welte84ec2522015-11-14 23:03:50 +0100208const uint8_t tpdu_hdr_upd_bin[] = { 0xA0, 0xB2, 0x00, 0x00, 0x0A };
209const uint8_t tpdu_body_upd_bin[] = { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09 };
210
Harald Welte9d3e3822015-11-09 00:50:54 +0100211int main(int argc, char **argv)
212{
213 struct card_handle *ch;
Harald Welte84ec2522015-11-14 23:03:50 +0100214 struct req_ctx *rctx;
215 unsigned int i;
Harald Welte9d3e3822015-11-09 00:50:54 +0100216
217 req_ctx_init();
218
219 ch = card_emu_init(0, 23, 42);
220 assert(ch);
221
Harald Welteb4362862015-11-14 19:02:33 +0100222 /* start up the card (VCC/RST, ATR) */
Harald Welte9d3e3822015-11-09 00:50:54 +0100223 io_start_card(ch);
Harald Welteb4362862015-11-14 19:02:33 +0100224 assert(!print_tx_chars(ch));
Harald Welte9d3e3822015-11-09 00:50:54 +0100225
Harald Welte84ec2522015-11-14 23:03:50 +0100226 for (i = 0; i < 2; i++) {
227 printf("\n==> transmitting APDU (PB + SW $%u)\n", i);
228 /* emulate the reader sending a TPDU header */
229 send_tpdu_hdr(ch, tpdu_hdr_sel_mf);
230 assert(!print_tx_chars(ch));
231 /* card emulator sends a response via USB */
232 host_to_device_data(tpdu_pb_sw, sizeof(tpdu_pb_sw), 0);
233 /* obtain any pending tx chars */
234 assert(print_tx_chars(ch) == sizeof(tpdu_pb_sw));
235
236 /* simulate some clock stop */
237 card_emu_io_statechg(ch, CARD_IO_CLK, 0);
238 card_emu_io_statechg(ch, CARD_IO_CLK, 1);
239 }
240
241 for (i = 0; i < 2; i++) {
242 printf("\n==> transmitting APDU (PB + RX #%u)\n", i);
243 /* emulate the reader sending a TPDU header */
244 send_tpdu_hdr(ch, tpdu_hdr_upd_bin);
245 assert(!print_tx_chars(ch));
246 /* card emulator sends a response via USB */
247 host_to_device_data(tpdu_hdr_upd_bin+1, 1, 1);
248 /* obtain any pending tx chars */
249 assert(print_tx_chars(ch) == 1);
250
251 /* emulate more characters from reader to card */
252 send_bytes(ch, tpdu_body_upd_bin, sizeof(tpdu_body_upd_bin));
253
254 /* check if we have received them on the USB side */
255 rctx = req_ctx_find_get(0, RCTX_S_USB_TX_PENDING, RCTX_S_USB_TX_BUSY);
256 assert(rctx);
257 dump_rctx(rctx);
258 req_ctx_set_state(rctx, RCTX_S_FREE);
259
260 /* ensure there is no extra data received on usb */
261 assert(!req_ctx_find_get(0, RCTX_S_USB_TX_PENDING, RCTX_S_USB_TX_BUSY));
262
263 /* simulate some clock stop */
264 card_emu_io_statechg(ch, CARD_IO_CLK, 0);
265 card_emu_io_statechg(ch, CARD_IO_CLK, 1);
266 }
Harald Welte9d3e3822015-11-09 00:50:54 +0100267
268 exit(0);
269}