blob: c36d8593e9a6cf2995607e6423318b4bf619f1b5 [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;
116 for (i = 0; i < len; i++)
117 card_emu_process_rx_byte(ch, bytes[i]);
118}
119
120static void dump_rctx(struct req_ctx *rctx)
121{
122 struct cardemu_usb_msg_hdr *mh =
123 (struct cardemu_usb_msg_hdr *) rctx->data;
124 struct cardemu_usb_msg_rx_data *rxd;
125 int i;
126
127 printf("req_ctx(%p): state=%u, size=%u, tot_len=%u, idx=%u, data=%p\n",
128 rctx, rctx->state, rctx->size, rctx->tot_len, rctx->idx, rctx->data);
129 printf(" msg_type=%u, seq_nr=%u, data_len=%u\n",
130 mh->msg_type, mh->seq_nr, mh->data_len);
131
132 switch (mh->msg_type) {
133 case CEMU_USB_MSGT_DO_RX_DATA:
134 rxd = (struct cardemu_usb_msg_rx_data *)mh;
135 printf(" flags=%x, data=", rxd->flags);
136 for (i = 0; i < mh->data_len; i++)
137 printf(" %02x", rxd->data[i]);
138 printf("\n");
139 break;
140 }
141}
142
Harald Welteb4362862015-11-14 19:02:33 +0100143/* emulate a TPDU header being sent by the reader/phone */
Harald Welte9d3e3822015-11-09 00:50:54 +0100144static void send_tpdu_hdr(struct card_handle *ch, const uint8_t *tpdu_hdr)
145{
146 struct req_ctx *rctx;
147
148 /* we don't want a receive context to become available during
149 * the first four bytes */
150 send_bytes(ch, tpdu_hdr, 4);
151 assert(!req_ctx_find_get(1, RCTX_S_USB_TX_PENDING, RCTX_S_USB_TX_BUSY));
152
153 send_bytes(ch, tpdu_hdr+4, 1);
154 /* but then after the final byte of the TPDU header, we want a
155 * receive context to be available for USB transmission */
156 rctx = req_ctx_find_get(1, RCTX_S_USB_TX_PENDING, RCTX_S_USB_TX_BUSY);
157 assert(rctx);
158 dump_rctx(rctx);
Harald Welteb4362862015-11-14 19:02:33 +0100159
160 /* free the req_ctx, indicating it has fully arrived on the host */
161 req_ctx_set_state(rctx, RCTX_S_FREE);
162}
163
164/* emulate a CEMU_USB_MSGT_DT_TX_DATA received from USB */
165static void host_to_device_data(const uint8_t *data, uint16_t len, int continue_rx)
166{
167 struct req_ctx *rctx;
168 struct cardemu_usb_msg_rx_data *rd;
169
170 /* allocate a free req_ctx */
171 rctx = req_ctx_find_get(1, RCTX_S_FREE, RCTX_S_USB_RX_BUSY);
172 assert(rctx);
173
174 /* initialize the header */
175 rd = (struct cardemu_usb_msg_rx_data *) rctx->data;
176 rctx->size = sizeof(*rd);
177 cardemu_hdr_set(&rd->hdr, CEMU_USB_MSGT_DT_TX_DATA);
178 if (continue_rx)
179 rd->flags = CEMU_DATA_F_PB_AND_RX;
180 else
181 rd->flags = CEMU_DATA_F_PB_AND_TX;
182 /* copy data and set length */
183 rd->hdr.data_len = len;
184 memcpy(rd->data, data, len);
185
186 /* hand the req_ctx to the UART transmit code */
187 req_ctx_set_state(rctx, RCTX_S_UART_TX_PENDING);
188}
189
190static int print_tx_chars(struct card_handle *ch)
191{
192 uint8_t byte;
193 int count = 0;
194
195 while (card_emu_get_tx_byte(ch, &byte)) {
196 printf("UART_TX(%02x)\n", byte);
197 count++;
198 }
199 return count;
Harald Welte9d3e3822015-11-09 00:50:54 +0100200}
201
202const uint8_t tpdu_hdr_sel_mf[] = { 0xA0, 0xA4, 0x00, 0x00, 0x02 };
Harald Welteb4362862015-11-14 19:02:33 +0100203const uint8_t tpdu_pb_sw[] = { 0xA4, 0x90, 0x00 };
Harald Welte9d3e3822015-11-09 00:50:54 +0100204
205int main(int argc, char **argv)
206{
207 struct card_handle *ch;
208
209 req_ctx_init();
210
211 ch = card_emu_init(0, 23, 42);
212 assert(ch);
213
Harald Welteb4362862015-11-14 19:02:33 +0100214 /* start up the card (VCC/RST, ATR) */
Harald Welte9d3e3822015-11-09 00:50:54 +0100215 io_start_card(ch);
Harald Welteb4362862015-11-14 19:02:33 +0100216 assert(!print_tx_chars(ch));
Harald Welte9d3e3822015-11-09 00:50:54 +0100217
Harald Welteb4362862015-11-14 19:02:33 +0100218 /* emulate the reader sending a TPDU header */
Harald Welte9d3e3822015-11-09 00:50:54 +0100219 send_tpdu_hdr(ch, tpdu_hdr_sel_mf);
Harald Welteb4362862015-11-14 19:02:33 +0100220 assert(!print_tx_chars(ch));
221 /* card emulator sends a response via USB */
222 host_to_device_data(tpdu_pb_sw, sizeof(tpdu_pb_sw), 0);
223 /* obtain any pending tx chars */
224 assert(print_tx_chars(ch) == sizeof(tpdu_pb_sw));
Harald Welte9d3e3822015-11-09 00:50:54 +0100225
226 exit(0);
227}