blob: ed18c11116f38590209515afe729c0d3e9092ccc [file] [log] [blame]
Harald Welte53079bb2016-03-20 14:58:35 +01001//#define TRACE_LEVEL 6
Harald Welte7abdb512016-03-03 17:48:32 +01002
Harald Welte2a6d3af2016-02-28 19:29:14 +01003#include "board.h"
Harald Welte16055642016-03-03 11:02:45 +01004#include "simtrace.h"
5#include "ringbuffer.h"
Harald Welte2a6d3af2016-02-28 19:29:14 +01006#include "card_emu.h"
Harald Weltebd717682016-02-28 19:30:05 +01007#include "iso7816_fidi.h"
Harald Welte54cb3d02016-02-29 14:12:40 +01008#include "utils.h"
Harald Welteb41598b2017-02-04 12:15:58 +01009#include "osmocom/core/linuxlist.h"
Harald Welte8e7fca32017-05-07 16:14:33 +020010#include "osmocom/core/msgb.h"
Harald Welte9f240b62016-03-18 10:32:56 +010011#include "llist_irqsafe.h"
Harald Welte8e7fca32017-05-07 16:14:33 +020012#include "usb_buf.h"
Harald Welte25a9a802017-05-08 13:30:09 +020013#include "simtrace_prot.h"
Harald Weltebd717682016-02-28 19:30:05 +010014
Harald Weltefd9c0412016-03-20 18:15:57 +010015#define TRACE_ENTRY() TRACE_DEBUG("%s entering\r\n", __func__)
Harald Welte2a6d3af2016-02-28 19:29:14 +010016
Harald Welte6d1128e2017-05-05 20:23:10 +020017#ifdef PINS_CARDSIM
Harald Welte2a6d3af2016-02-28 19:29:14 +010018static const Pin pins_cardsim[] = PINS_CARDSIM;
Harald Welte6d1128e2017-05-05 20:23:10 +020019#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +010020
21/* UART pins */
22static const Pin pins_usim1[] = {PINS_USIM1};
23static const Pin pin_usim1_rst = PIN_USIM1_nRST;
24static const Pin pin_usim1_vcc = PIN_USIM1_VCC;
25
26#ifdef CARDEMU_SECOND_UART
Harald Welte6dcacf32016-03-19 14:01:31 +010027static const Pin pins_usim2[] = {PINS_USIM2};
Harald Welte2a6d3af2016-02-28 19:29:14 +010028static const Pin pin_usim2_rst = PIN_USIM2_nRST;
29static const Pin pin_usim2_vcc = PIN_USIM2_VCC;
30#endif
31
Harald Welte13e82022016-03-02 15:17:53 +010032struct cardem_inst {
Harald Weltedde112e2016-03-20 16:42:11 +010033 uint32_t num;
Harald Welte13e82022016-03-02 15:17:53 +010034 struct card_handle *ch;
35 struct llist_head usb_out_queue;
36 struct ringbuf rb;
37 struct Usart_info usart_info;
38 int usb_pending_old;
39 uint8_t ep_out;
40 uint8_t ep_in;
41 uint8_t ep_int;
Harald Welte0eaa9922016-03-04 03:03:49 +010042 const Pin pin_insert;
Harald Welte07872b62016-03-20 11:45:36 +010043 uint32_t vcc_uv;
44 uint32_t vcc_uv_last;
Harald Welte13e82022016-03-02 15:17:53 +010045};
Harald Welte2a6d3af2016-02-28 19:29:14 +010046
Harald Welte8e7fca32017-05-07 16:14:33 +020047struct cardem_inst cardem_inst[] = {
Harald Welte2a6d3af2016-02-28 19:29:14 +010048 {
Harald Weltedde112e2016-03-20 16:42:11 +010049 .num = 0,
Harald Welte13e82022016-03-02 15:17:53 +010050 .usart_info = {
51 .base = USART1,
52 .id = ID_USART1,
53 .state = USART_RCV
54 },
55 .ep_out = PHONE_DATAOUT,
56 .ep_in = PHONE_DATAIN,
57 .ep_int = PHONE_INT,
Harald Welte6d1128e2017-05-05 20:23:10 +020058#ifdef PIN_SET_USIM1_PRES
Harald Welte0eaa9922016-03-04 03:03:49 +010059 .pin_insert = PIN_SET_USIM1_PRES,
Harald Welte6d1128e2017-05-05 20:23:10 +020060#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +010061 },
62#ifdef CARDEMU_SECOND_UART
63 {
Harald Weltedde112e2016-03-20 16:42:11 +010064 .num = 1,
Harald Welte13e82022016-03-02 15:17:53 +010065 .usart_info = {
66 .base = USART0,
67 .id = ID_USART0,
68 .state = USART_RCV
69 },
70 .ep_out = CARDEM_USIM2_DATAOUT,
71 .ep_in = CARDEM_USIM2_DATAIN,
72 .ep_int = CARDEM_USIM2_INT,
Harald Welte6d1128e2017-05-05 20:23:10 +020073#ifdef PIN_SET_USIM2_PRES
Harald Welte0eaa9922016-03-04 03:03:49 +010074 .pin_insert = PIN_SET_USIM2_PRES,
Harald Welte6d1128e2017-05-05 20:23:10 +020075#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +010076 },
77#endif
78};
79
80static Usart *get_usart_by_chan(uint8_t uart_chan)
81{
82 switch (uart_chan) {
83 case 0:
84 return USART1;
85#ifdef CARDEMU_SECOND_UART
86 case 1:
87 return USART0;
88#endif
89 }
90 return NULL;
91}
92
93/***********************************************************************
94 * Call-Backs from card_emu.c
95 ***********************************************************************/
96
Harald Weltec8beefb2016-03-20 14:40:47 +010097static void wait_tx_idle(Usart *usart)
98{
99 int i = 1;
100
101 /* wait until last char has been fully transmitted */
102 while ((usart->US_CSR & (US_CSR_TXEMPTY)) == 0) {
103 if (!(i%1000000)) {
104 TRACE_ERROR("s: %x \r\n", usart->US_CSR);
105 }
106 i++;
107 }
108}
109
Harald Weltec58bba02016-03-20 14:57:53 +0100110void card_emu_uart_wait_tx_idle(uint8_t uart_chan)
111{
112 Usart *usart = get_usart_by_chan(uart_chan);
113 wait_tx_idle(usart);
114}
115
Harald Welte2a6d3af2016-02-28 19:29:14 +0100116/* call-back from card_emu.c to enable/disable transmit and/or receive */
117void card_emu_uart_enable(uint8_t uart_chan, uint8_t rxtx)
118{
119 Usart *usart = get_usart_by_chan(uart_chan);
120 switch (rxtx) {
121 case ENABLE_TX:
Harald Welte9dbc46e2016-02-29 10:05:10 +0100122 USART_DisableIt(usart, ~US_IER_TXRDY);
Harald Weltec8beefb2016-03-20 14:40:47 +0100123 /* as irritating as it is, we actually want to keep the
124 * receiver enabled during transmit */
125 USART_SetReceiverEnabled(usart, 1);
Harald Welte8a416b12016-03-01 00:42:04 +0100126 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
127 USART_EnableIt(usart, US_IER_TXRDY);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100128 USART_SetTransmitterEnabled(usart, 1);
129 break;
130 case ENABLE_RX:
Harald Welte9dbc46e2016-02-29 10:05:10 +0100131 USART_DisableIt(usart, ~US_IER_RXRDY);
Harald Weltec8beefb2016-03-20 14:40:47 +0100132 /* as irritating as it is, we actually want to keep the
133 * transmitter enabled during receive */
134 USART_SetTransmitterEnabled(usart, 1);
135 wait_tx_idle(usart);
Harald Welte8a416b12016-03-01 00:42:04 +0100136 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
137 USART_EnableIt(usart, US_IER_RXRDY);
138 USART_SetReceiverEnabled(usart, 1);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100139 break;
140 case 0:
141 default:
142 USART_SetTransmitterEnabled(usart, 0);
143 USART_SetReceiverEnabled(usart, 0);
Harald Welte9dbc46e2016-02-29 10:05:10 +0100144 USART_DisableIt(usart, 0xFFFFFFFF);
Harald Welte8a416b12016-03-01 00:42:04 +0100145 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
Harald Welte2a6d3af2016-02-28 19:29:14 +0100146 break;
147 }
148}
149
150/* call-back from card_emu.c to transmit a byte */
151int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte)
152{
Harald Welte13e82022016-03-02 15:17:53 +0100153 Usart *usart = get_usart_by_chan(uart_chan);
154#if 0
Harald Welte2a6d3af2016-02-28 19:29:14 +0100155 Usart_info *ui = &usart_info[uart_chan];
156 ISO7816_SendChar(byte, ui);
Harald Welte13e82022016-03-02 15:17:53 +0100157#else
Harald Welte53079bb2016-03-20 14:58:35 +0100158 int i = 1;
Harald Welte13e82022016-03-02 15:17:53 +0100159 while ((usart->US_CSR & (US_CSR_TXRDY)) == 0) {
160 if (!(i%1000000)) {
Harald Weltedde112e2016-03-20 16:42:11 +0100161 TRACE_ERROR("%u: s: %x %02X\r\n",
162 uart_chan, usart->US_CSR,
163 usart->US_RHR & 0xFF);
Harald Welte13e82022016-03-02 15:17:53 +0100164 usart->US_CR = US_CR_RSTTX;
165 usart->US_CR = US_CR_RSTRX;
166 }
Harald Welte53079bb2016-03-20 14:58:35 +0100167 i++;
Harald Welte13e82022016-03-02 15:17:53 +0100168 }
169 usart->US_THR = byte;
Harald Welte53079bb2016-03-20 14:58:35 +0100170 //TRACE_ERROR("Sx%02x\r\n", byte);
Harald Welte13e82022016-03-02 15:17:53 +0100171#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +0100172 return 1;
173}
174
175
176/* FIXME: integrate this with actual irq handler */
Harald Welte3bafe432016-03-20 16:43:12 +0100177static void usart_irq_rx(uint8_t inst_num)
Harald Welte2a6d3af2016-02-28 19:29:14 +0100178{
Harald Welte3bafe432016-03-20 16:43:12 +0100179 Usart *usart = get_usart_by_chan(inst_num);
180 struct cardem_inst *ci = &cardem_inst[inst_num];
Harald Weltec0bd7f02016-02-29 10:13:33 +0100181 uint32_t csr;
182 uint8_t byte = 0;
183
Harald Welteda15ca02016-03-17 21:14:04 +0100184 csr = usart->US_CSR & usart->US_IMR;
Harald Welte2a6d3af2016-02-28 19:29:14 +0100185
Harald Weltec0bd7f02016-02-29 10:13:33 +0100186 if (csr & US_CSR_RXRDY) {
187 byte = (usart->US_RHR) & 0xFF;
Harald Welte13e82022016-03-02 15:17:53 +0100188 rbuf_write(&ci->rb, byte);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100189 }
Harald Welte2a6d3af2016-02-28 19:29:14 +0100190
Harald Welte12d4bdf2016-03-02 10:31:03 +0100191 if (csr & US_CSR_TXRDY) {
Harald Welte13e82022016-03-02 15:17:53 +0100192 if (card_emu_tx_byte(ci->ch) == 0)
Harald Welte12d4bdf2016-03-02 10:31:03 +0100193 USART_DisableIt(usart, US_IER_TXRDY);
194 }
195
Harald Welte2a6d3af2016-02-28 19:29:14 +0100196 if (csr & (US_CSR_OVRE|US_CSR_FRAME|US_CSR_PARE|
197 US_CSR_TIMEOUT|US_CSR_NACK|(1<<10))) {
Harald Weltec0bd7f02016-02-29 10:13:33 +0100198 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
Harald Weltedde112e2016-03-20 16:42:11 +0100199 TRACE_ERROR("%u e 0x%x st: 0x%x\n", ci->num, byte, csr);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100200 }
201}
202
Harald Welte3bafe432016-03-20 16:43:12 +0100203void mode_cardemu_usart0_irq(void)
204{
205 /* USART0 == Instance 1 == USIM 2 */
206 usart_irq_rx(1);
207}
208
209void mode_cardemu_usart1_irq(void)
210{
211 /* USART1 == Instance 0 == USIM 1 */
212 usart_irq_rx(0);
213}
214
Harald Weltebd717682016-02-28 19:30:05 +0100215/* call-back from card_emu.c to change UART baud rate */
216int card_emu_uart_update_fidi(uint8_t uart_chan, unsigned int fidi)
217{
218 int rc;
Harald Weltebd717682016-02-28 19:30:05 +0100219 Usart *usart = get_usart_by_chan(uart_chan);
220
Harald Welte99f62a62016-02-29 10:08:49 +0100221 usart->US_CR |= US_CR_RXDIS | US_CR_RSTRX;
222 usart->US_FIDI = fidi & 0x3ff;
223 usart->US_CR |= US_CR_RXEN | US_CR_STTTO;
224 return 0;
Harald Weltebd717682016-02-28 19:30:05 +0100225}
Harald Welte2a6d3af2016-02-28 19:29:14 +0100226
227/***********************************************************************
Harald Welte419eb8a2016-03-20 10:04:03 +0100228 * ADC for VCC voltage detection
229 ***********************************************************************/
230
Harald Welte07872b62016-03-20 11:45:36 +0100231#ifdef DETECT_VCC_BY_ADC
232
233static int adc_triggered = 0;
Harald Welte28772eb2016-03-20 15:44:52 +0100234static int adc_sam3s_reva_errata = 0;
Harald Welte07872b62016-03-20 11:45:36 +0100235
Harald Welte419eb8a2016-03-20 10:04:03 +0100236static int card_vcc_adc_init(void)
237{
Harald Welte622b6be2016-03-20 16:43:59 +0100238 uint32_t chip_arch = CHIPID->CHIPID_CIDR & CHIPID_CIDR_ARCH_Msk;
239 uint32_t chip_ver = CHIPID->CHIPID_CIDR & CHIPID_CIDR_VERSION_Msk;
Harald Welte28772eb2016-03-20 15:44:52 +0100240
Harald Welte07872b62016-03-20 11:45:36 +0100241 PMC_EnablePeripheral(ID_ADC);
242
Harald Welte419eb8a2016-03-20 10:04:03 +0100243 ADC->ADC_CR |= ADC_CR_SWRST;
Harald Welte28772eb2016-03-20 15:44:52 +0100244 if (chip_ver == 0 &&
245 (chip_arch == CHIPID_CIDR_ARCH_SAM3SxA ||
246 chip_arch == CHIPID_CIDR_ARCH_SAM3SxB ||
247 chip_arch == CHIPID_CIDR_ARCH_SAM3SxC)) {
248 TRACE_INFO("Enabling Rev.A ADC Errata work-around\r\n");
249 adc_sam3s_reva_errata = 1;
250 }
251
252 if (adc_sam3s_reva_errata) {
253 /* Errata Work-Around to clear EOCx flags */
Harald Welte07872b62016-03-20 11:45:36 +0100254 volatile uint32_t foo;
255 int i;
256 for (i = 0; i < 16; i++)
257 foo = ADC->ADC_CDR[i];
258 }
259
260 /* Initialize ADC for AD7 / AD6, fADC=48/24=2MHz */
Harald Welte419eb8a2016-03-20 10:04:03 +0100261 ADC->ADC_MR = ADC_MR_TRGEN_DIS | ADC_MR_LOWRES_BITS_12 |
262 ADC_MR_SLEEP_NORMAL | ADC_MR_FWUP_OFF |
Harald Welte07872b62016-03-20 11:45:36 +0100263 ADC_MR_FREERUN_OFF | ADC_MR_PRESCAL(23) |
264 ADC_MR_STARTUP_SUT8 | ADC_MR_SETTLING(3) |
265 ADC_MR_ANACH_NONE | ADC_MR_TRACKTIM(4) |
Harald Welte419eb8a2016-03-20 10:04:03 +0100266 ADC_MR_TRANSFER(1) | ADC_MR_USEQ_NUM_ORDER;
267 /* enable AD6 + AD7 channels */
Harald Welte07872b62016-03-20 11:45:36 +0100268 ADC->ADC_CHER = ADC_CHER_CH7;
269 ADC->ADC_IER = ADC_IER_EOC7;
270#ifdef CARDEMU_SECOND_UART
271 ADC->ADC_CHER |= ADC_CHER_CH6;
272 ADC->ADC_IER |= ADC_IER_EOC6;
273#endif
274 NVIC_EnableIRQ(ADC_IRQn);
Harald Welte419eb8a2016-03-20 10:04:03 +0100275 ADC->ADC_CR |= ADC_CR_START;
Harald Welte07872b62016-03-20 11:45:36 +0100276
277 return 0;
Harald Welte419eb8a2016-03-20 10:04:03 +0100278}
279
Harald Welte07872b62016-03-20 11:45:36 +0100280#define UV_PER_LSB ((3300 * 1000) / 4096)
281#define VCC_UV_THRESH_1V8 1500000
282#define VCC_UV_THRESH_3V 2800000
283
284static void process_vcc_adc(struct cardem_inst *ci)
Harald Welte419eb8a2016-03-20 10:04:03 +0100285{
Harald Welte07872b62016-03-20 11:45:36 +0100286 if (ci->vcc_uv >= VCC_UV_THRESH_3V &&
287 ci->vcc_uv_last < VCC_UV_THRESH_3V) {
288 card_emu_io_statechg(ci->ch, CARD_IO_VCC, 1);
289 /* FIXME do this for real */
290 card_emu_io_statechg(ci->ch, CARD_IO_CLK, 1);
291 } else if (ci->vcc_uv < VCC_UV_THRESH_3V &&
292 ci->vcc_uv_last >= VCC_UV_THRESH_3V) {
293 /* FIXME do this for real */
294 card_emu_io_statechg(ci->ch, CARD_IO_CLK, 0);
295 card_emu_io_statechg(ci->ch, CARD_IO_VCC, 0);
Harald Welte419eb8a2016-03-20 10:04:03 +0100296 }
Harald Welte07872b62016-03-20 11:45:36 +0100297 ci->vcc_uv_last = ci->vcc_uv;
298}
299
300static uint32_t adc2uv(uint16_t adc)
301{
302 uint32_t uv = (uint32_t) adc * UV_PER_LSB;
Harald Welte622b6be2016-03-20 16:43:59 +0100303 return uv;
Harald Welte07872b62016-03-20 11:45:36 +0100304}
305
306void ADC_IrqHandler(void)
307{
308#ifdef CARDEMU_SECOND_UART
309 if (ADC->ADC_ISR & ADC_ISR_EOC6) {
310 uint16_t val = ADC->ADC_CDR[6] & 0xFFF;
311 cardem_inst[1].vcc_uv = adc2uv(val);
312 process_vcc_adc(&cardem_inst[1]);
Harald Welte28772eb2016-03-20 15:44:52 +0100313 if (adc_sam3s_reva_errata) {
314 /* Errata: START doesn't start a conversion
315 * sequence, but only a single conversion */
316 ADC->ADC_CR |= ADC_CR_START;
317 }
Harald Welte07872b62016-03-20 11:45:36 +0100318 }
319#endif
Harald Welte419eb8a2016-03-20 10:04:03 +0100320
321 if (ADC->ADC_ISR & ADC_ISR_EOC7) {
Harald Welte07872b62016-03-20 11:45:36 +0100322 uint16_t val = ADC->ADC_CDR[7] & 0xFFF;
323 cardem_inst[0].vcc_uv = adc2uv(val);
324 process_vcc_adc(&cardem_inst[0]);
325 ADC->ADC_CR |= ADC_CR_START;
Harald Welte419eb8a2016-03-20 10:04:03 +0100326 }
327}
Harald Welte07872b62016-03-20 11:45:36 +0100328#endif /* DETECT_VCC_BY_ADC */
Harald Welte419eb8a2016-03-20 10:04:03 +0100329
330/***********************************************************************
Harald Welte2a6d3af2016-02-28 19:29:14 +0100331 * Core USB / mainloop integration
332 ***********************************************************************/
333
Harald Weltec0bd7f02016-02-29 10:13:33 +0100334static void usim1_rst_irqhandler(const Pin *pPin)
335{
336 int active = PIO_Get(&pin_usim1_rst) ? 0 : 1;
Harald Welte13e82022016-03-02 15:17:53 +0100337 card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_RST, active);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100338}
339
Harald Welte07872b62016-03-20 11:45:36 +0100340#ifndef DETECT_VCC_BY_ADC
Harald Weltec0bd7f02016-02-29 10:13:33 +0100341static void usim1_vcc_irqhandler(const Pin *pPin)
342{
343 int active = PIO_Get(&pin_usim1_vcc) ? 1 : 0;
Harald Welte13e82022016-03-02 15:17:53 +0100344 card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_VCC, active);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100345 /* FIXME do this for real */
Harald Welte13e82022016-03-02 15:17:53 +0100346 card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_CLK, active);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100347}
Harald Welte07872b62016-03-20 11:45:36 +0100348#endif /* !DETECT_VCC_BY_ADC */
Harald Weltec0bd7f02016-02-29 10:13:33 +0100349
Harald Welte13e82022016-03-02 15:17:53 +0100350#ifdef CARDEMU_SECOND_UART
351static void usim2_rst_irqhandler(const Pin *pPin)
352{
353 int active = PIO_Get(&pin_usim2_rst) ? 0 : 1;
354 card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_RST, active);
355}
356
Harald Welte07872b62016-03-20 11:45:36 +0100357#ifndef DETECT_VCC_BY_ADC
Harald Welte13e82022016-03-02 15:17:53 +0100358static void usim2_vcc_irqhandler(const Pin *pPin)
359{
360 int active = PIO_Get(&pin_usim2_vcc) ? 1 : 0;
361 card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_VCC, active);
362 /* FIXME do this for real */
363 card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_CLK, active);
364}
Harald Welte07872b62016-03-20 11:45:36 +0100365#endif /* !DETECT_VCC_BY_ADC */
366#endif /* CARDEMU_SECOND_UART */
Harald Welte13e82022016-03-02 15:17:53 +0100367
Harald Welte2a6d3af2016-02-28 19:29:14 +0100368/* executed once at system boot for each config */
369void mode_cardemu_configure(void)
370{
Harald Weltebd717682016-02-28 19:30:05 +0100371 TRACE_ENTRY();
Harald Welte2a6d3af2016-02-28 19:29:14 +0100372}
373
374/* called if config is activated */
375void mode_cardemu_init(void)
376{
Harald Welte13e82022016-03-02 15:17:53 +0100377 int i;
Harald Weltebd717682016-02-28 19:30:05 +0100378
Harald Welte13e82022016-03-02 15:17:53 +0100379 TRACE_ENTRY();
Harald Weltec0bd7f02016-02-29 10:13:33 +0100380
Harald Welte6d1128e2017-05-05 20:23:10 +0200381#ifdef PINS_CARDSIM
Harald Welte2a6d3af2016-02-28 19:29:14 +0100382 PIO_Configure(pins_cardsim, PIO_LISTSIZE(pins_cardsim));
Harald Welte6d1128e2017-05-05 20:23:10 +0200383#endif
Harald Welte07872b62016-03-20 11:45:36 +0100384#ifdef DETECT_VCC_BY_ADC
385 card_vcc_adc_init();
386#endif /* DETECT_VCC_BY_ADC */
Harald Welte2a6d3af2016-02-28 19:29:14 +0100387
Harald Welte13e82022016-03-02 15:17:53 +0100388 INIT_LLIST_HEAD(&cardem_inst[0].usb_out_queue);
389 rbuf_reset(&cardem_inst[0].rb);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100390 PIO_Configure(pins_usim1, PIO_LISTSIZE(pins_usim1));
Harald Welte13e82022016-03-02 15:17:53 +0100391 ISO7816_Init(&cardem_inst[0].usart_info, CLK_SLAVE);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100392 NVIC_EnableIRQ(USART1_IRQn);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100393 PIO_ConfigureIt(&pin_usim1_rst, usim1_rst_irqhandler);
394 PIO_EnableIt(&pin_usim1_rst);
Harald Welte07872b62016-03-20 11:45:36 +0100395#ifndef DETECT_VCC_BY_ADC
Harald Weltec0bd7f02016-02-29 10:13:33 +0100396 PIO_ConfigureIt(&pin_usim1_vcc, usim1_vcc_irqhandler);
397 PIO_EnableIt(&pin_usim1_vcc);
Harald Welte07872b62016-03-20 11:45:36 +0100398#endif /* DETECT_VCC_BY_ADC */
Harald Welte8e7fca32017-05-07 16:14:33 +0200399 cardem_inst[0].ch = card_emu_init(0, 2, 0, PHONE_DATAIN, PHONE_INT);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100400
401#ifdef CARDEMU_SECOND_UART
Harald Welte13e82022016-03-02 15:17:53 +0100402 INIT_LLIST_HEAD(&cardem_inst[1].usb_out_queue);
403 rbuf_reset(&cardem_inst[1].rb);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100404 PIO_Configure(pins_usim2, PIO_LISTSIZE(pins_usim2));
Harald Welte13e82022016-03-02 15:17:53 +0100405 ISO7816_Init(&cardem_inst[1].usart_info, CLK_SLAVE);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100406 NVIC_EnableIRQ(USART0_IRQn);
Harald Welte13e82022016-03-02 15:17:53 +0100407 PIO_ConfigureIt(&pin_usim2_rst, usim2_rst_irqhandler);
408 PIO_EnableIt(&pin_usim2_rst);
Harald Welte07872b62016-03-20 11:45:36 +0100409#ifndef DETECT_VCC_BY_ADC
Harald Welte13e82022016-03-02 15:17:53 +0100410 PIO_ConfigureIt(&pin_usim2_vcc, usim2_vcc_irqhandler);
411 PIO_EnableIt(&pin_usim2_vcc);
Harald Welte07872b62016-03-20 11:45:36 +0100412#endif /* DETECT_VCC_BY_ADC */
Harald Welte8e7fca32017-05-07 16:14:33 +0200413 cardem_inst[1].ch = card_emu_init(1, 0, 1, CARDEM_USIM2_DATAIN, CARDEM_USIM2_INT);
Harald Welte07872b62016-03-20 11:45:36 +0100414#endif /* CARDEMU_SECOND_UART */
Harald Welte2a6d3af2016-02-28 19:29:14 +0100415}
416
417/* called if config is deactivated */
418void mode_cardemu_exit(void)
419{
Harald Weltebd717682016-02-28 19:30:05 +0100420 TRACE_ENTRY();
421
Harald Welte7abdb512016-03-03 17:48:32 +0100422 /* FIXME: stop tc_fdt */
Harald Welte8e7fca32017-05-07 16:14:33 +0200423 /* FIXME: release all msg, unlink them from any queue */
Harald Welte7abdb512016-03-03 17:48:32 +0100424
Harald Weltec0bd7f02016-02-29 10:13:33 +0100425 PIO_DisableIt(&pin_usim1_rst);
426 PIO_DisableIt(&pin_usim1_vcc);
427
Harald Welte2a6d3af2016-02-28 19:29:14 +0100428 NVIC_DisableIRQ(USART1_IRQn);
429 USART_SetTransmitterEnabled(USART1, 0);
430 USART_SetReceiverEnabled(USART1, 0);
431
432#ifdef CARDEMU_SECOND_UART
Harald Welte13e82022016-03-02 15:17:53 +0100433 PIO_DisableIt(&pin_usim2_rst);
434 PIO_DisableIt(&pin_usim2_vcc);
435
Harald Welte2a6d3af2016-02-28 19:29:14 +0100436 NVIC_DisableIRQ(USART0_IRQn);
437 USART_SetTransmitterEnabled(USART0, 0);
438 USART_SetReceiverEnabled(USART0, 0);
439#endif
440}
441
Harald Welteebb80ed2016-03-02 13:56:59 +0100442/* handle a single USB command as received from the USB host */
Harald Welte25a9a802017-05-08 13:30:09 +0200443static void dispatch_usb_command_generic(struct msgb *msg, struct cardem_inst *ci)
Harald Welteebb80ed2016-03-02 13:56:59 +0100444{
Harald Welte25a9a802017-05-08 13:30:09 +0200445 struct simtrace_msg_hdr *hdr;
446
447 hdr = (struct simtrace_msg_hdr *) msg->l1h;
448 switch (hdr->msg_type) {
449 case SIMTRACE_CMD_BD_BOARD_INFO:
450 break;
451 default:
452 break;
453 }
454 usb_buf_free(msg);
455}
456
457/* handle a single USB command as received from the USB host */
458static void dispatch_usb_command_cardem(struct msgb *msg, struct cardem_inst *ci)
459{
460 struct simtrace_msg_hdr *hdr;
Harald Welte06b27f62016-03-02 14:26:38 +0100461 struct cardemu_usb_msg_set_atr *atr;
Harald Welte0eaa9922016-03-04 03:03:49 +0100462 struct cardemu_usb_msg_cardinsert *cardins;
Harald Welteebb80ed2016-03-02 13:56:59 +0100463 struct llist_head *queue;
464
Harald Welte25a9a802017-05-08 13:30:09 +0200465 hdr = (struct simtrace_msg_hdr *) msg->l1h;
Harald Welteebb80ed2016-03-02 13:56:59 +0100466 switch (hdr->msg_type) {
Harald Welte25a9a802017-05-08 13:30:09 +0200467 case SIMTRACE_MSGT_DT_CEMU_TX_DATA:
Harald Welte0eaa9922016-03-04 03:03:49 +0100468 queue = card_emu_get_uart_tx_queue(ci->ch);
Harald Welte8e7fca32017-05-07 16:14:33 +0200469 llist_add_tail(&msg->list, queue);
Harald Welte0eaa9922016-03-04 03:03:49 +0100470 card_emu_have_new_uart_tx(ci->ch);
Harald Welteebb80ed2016-03-02 13:56:59 +0100471 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200472 case SIMTRACE_MSGT_DT_CEMU_SET_ATR:
473 atr = (struct cardemu_usb_msg_set_atr *) msg->l2h;
Harald Welted295b922016-03-18 21:01:36 +0100474 card_emu_set_atr(ci->ch, atr->atr, atr->atr_len);
Harald Welte8e7fca32017-05-07 16:14:33 +0200475 usb_buf_free(msg);
Harald Welte06b27f62016-03-02 14:26:38 +0100476 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200477 case SIMTRACE_MSGT_DT_CEMU_CARDINSERT:
478 cardins = (struct cardemu_usb_msg_cardinsert *) msg->l2h;
Harald Weltedde112e2016-03-20 16:42:11 +0100479 TRACE_INFO("%u: set card_insert to %s\r\n", ci->num,
480 cardins->card_insert ? "INSERTED" : "REMOVED");
Harald Welte0eaa9922016-03-04 03:03:49 +0100481 if (cardins->card_insert)
482 PIO_Set(&ci->pin_insert);
483 else
484 PIO_Clear(&ci->pin_insert);
Harald Welte8e7fca32017-05-07 16:14:33 +0200485 usb_buf_free(msg);
Harald Welte0eaa9922016-03-04 03:03:49 +0100486 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200487 case SIMTRACE_MSGT_BD_CEMU_STATUS:
Harald Welteff160652016-03-19 21:59:06 +0100488 card_emu_report_status(ci->ch);
Harald Welte25a9a802017-05-08 13:30:09 +0200489 usb_buf_free(msg);
Harald Welteff160652016-03-19 21:59:06 +0100490 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200491 case SIMTRACE_MSGT_BD_CEMU_STATS:
Harald Welteebb80ed2016-03-02 13:56:59 +0100492 default:
Harald Welte25a9a802017-05-08 13:30:09 +0200493 /* FIXME: Send Error */
494 usb_buf_free(msg);
495 break;
496 }
497}
498
499/* handle a single USB command as received from the USB host */
500static void dispatch_usb_command_modem(struct msgb *msg, struct cardem_inst *ci)
501{
502 struct simtrace_msg_hdr *hdr;
503
504 hdr = (struct simtrace_msg_hdr *) msg->l1h;
505 switch (hdr->msg_type) {
506 case SIMTRACE_MSGT_DT_MODEM_RESET:
507 break;
508 case SIMTRACE_MSGT_DT_MODEM_SIM_SELECT:
509 break;
510 case SIMTRACE_MSGT_BD_MODEM_STATUS:
511 break;
512 default:
513 break;
514 }
515 usb_buf_free(msg);
516}
517
518/* handle a single USB command as received from the USB host */
519static void dispatch_usb_command(struct msgb *msg, struct cardem_inst *ci)
520{
521 struct simtrace_msg_hdr *sh = msg->l1h;
522
523 if (msgb_length(msg) < sizeof(*sh)) {
524 /* FIXME: Error */
525 usb_buf_free(msg);
526 return;
527 }
528
529 switch (sh->msg_class) {
530 case SIMTRACE_MSGC_GENERIC:
531 dispatch_usb_command_generic(msg, ci);
532 break;
533 case SIMTRACE_MSGC_CARDEM:
534 dispatch_usb_command_cardem(msg, ci);
535 break;
536 case SIMTRACE_MSGC_MODEM:
537 dispatch_usb_command_modem(msg, ci);
538 break;
539 default:
540 /* FIXME: Send Error */
Harald Welte8e7fca32017-05-07 16:14:33 +0200541 usb_buf_free(msg);
Harald Welteebb80ed2016-03-02 13:56:59 +0100542 break;
543 }
544}
545
Harald Welte8e7fca32017-05-07 16:14:33 +0200546static void dispatch_received_msg(struct msgb *msg, struct cardem_inst *ci)
Harald Welteb26d0032016-03-19 13:33:02 +0100547{
Harald Welte8e7fca32017-05-07 16:14:33 +0200548 struct msgb *segm;
Harald Welte25a9a802017-05-08 13:30:09 +0200549 struct simtrace_msg_hdr *mh;
Harald Welteb26d0032016-03-19 13:33:02 +0100550
551 /* check if we have multiple concatenated commands in
552 * one message. USB endpoints are streams that don't
553 * preserve the message boundaries */
Harald Welte25a9a802017-05-08 13:30:09 +0200554 mh = (struct simtrace_msg_hdr *) msg->data;
Harald Welte8e7fca32017-05-07 16:14:33 +0200555 if (mh->msg_len == msgb_length(msg)) {
Harald Welteb26d0032016-03-19 13:33:02 +0100556 /* fast path: only one message in buffer */
Harald Welte8e7fca32017-05-07 16:14:33 +0200557 dispatch_usb_command(msg, ci);
Harald Welteb26d0032016-03-19 13:33:02 +0100558 return;
559 }
560
561 /* slow path: iterate over list of messages, allocating one new
562 * reqe_ctx per segment */
Harald Welte8e7fca32017-05-07 16:14:33 +0200563 while (1) {
Harald Welte25a9a802017-05-08 13:30:09 +0200564 mh = (struct simtrace_msg_hdr *) msg->data;
Harald Welte8e7fca32017-05-07 16:14:33 +0200565
566 segm = usb_buf_alloc(ci->ep_out);
Harald Welteb26d0032016-03-19 13:33:02 +0100567 if (!segm) {
Harald Welte8e7fca32017-05-07 16:14:33 +0200568 TRACE_ERROR("%u: ENOMEM during msg segmentation\r\n",
Harald Weltedde112e2016-03-20 16:42:11 +0100569 ci->num);
Harald Welteb26d0032016-03-19 13:33:02 +0100570 break;
571 }
Harald Welte8e7fca32017-05-07 16:14:33 +0200572
573 if (mh->msg_len > msgb_length(msg)) {
574 TRACE_ERROR("%u: Unexpected large message (%u bytes)\n",
575 ci->num, mh->msg_len);
576 usb_buf_free(segm);
577 } else {
578 uint8_t *cur = msgb_put(segm, mh->msg_len);
579 segm->l1h = segm->head;
580 memcpy(cur, mh, mh->msg_len);
581 dispatch_usb_command(segm, ci);
582 }
583 /* pull this message */
584 msgb_pull(msg, mh->msg_len);
585 /* abort if we're done */
586 if (msgb_length(msg) <= 0)
587 break;
Harald Welteb26d0032016-03-19 13:33:02 +0100588 }
589
Harald Welte8e7fca32017-05-07 16:14:33 +0200590 usb_buf_free(msg);
Harald Welteb26d0032016-03-19 13:33:02 +0100591}
592
Harald Welteebb80ed2016-03-02 13:56:59 +0100593/* iterate over the queue of incoming USB commands and dispatch/execute
594 * them */
Harald Welte9f240b62016-03-18 10:32:56 +0100595static void process_any_usb_commands(struct llist_head *main_q,
596 struct cardem_inst *ci)
Harald Welteebb80ed2016-03-02 13:56:59 +0100597{
Harald Welte9f240b62016-03-18 10:32:56 +0100598 struct llist_head *lh;
Harald Welte8e7fca32017-05-07 16:14:33 +0200599 struct msgb *msg;
Harald Welte9f240b62016-03-18 10:32:56 +0100600 int i;
Harald Welteebb80ed2016-03-02 13:56:59 +0100601
Harald Welte9f240b62016-03-18 10:32:56 +0100602 /* limit the number of iterations to 10, to ensure we don't get
603 * stuck here without returning to main loop processing */
604 for (i = 0; i < 10; i++) {
605 /* de-queue the list head in an irq-safe way */
606 lh = llist_head_dequeue_irqsafe(main_q);
607 if (!lh)
608 break;
Harald Welte8e7fca32017-05-07 16:14:33 +0200609 msg = llist_entry(lh, struct msgb, list);
610 dispatch_received_msg(msg, ci);
Harald Welte5820ea92016-03-16 22:12:00 +0100611 }
Harald Welteebb80ed2016-03-02 13:56:59 +0100612}
613
Harald Welte2a6d3af2016-02-28 19:29:14 +0100614/* main loop function, called repeatedly */
615void mode_cardemu_run(void)
616{
Harald Welteacae4122016-03-02 10:27:58 +0100617 struct llist_head *queue;
Harald Welte13e82022016-03-02 15:17:53 +0100618 unsigned int i;
Harald Welteacae4122016-03-02 10:27:58 +0100619
Harald Welte13e82022016-03-02 15:17:53 +0100620 for (i = 0; i < ARRAY_SIZE(cardem_inst); i++) {
621 struct cardem_inst *ci = &cardem_inst[i];
622
Harald Welte54cb3d02016-02-29 14:12:40 +0100623 /* drain the ring buffer from UART into card_emu */
Harald Weltec0bd7f02016-02-29 10:13:33 +0100624 while (1) {
625 __disable_irq();
Harald Welte13e82022016-03-02 15:17:53 +0100626 if (rbuf_is_empty(&ci->rb)) {
Harald Weltec0bd7f02016-02-29 10:13:33 +0100627 __enable_irq();
628 break;
629 }
Harald Welte13e82022016-03-02 15:17:53 +0100630 uint8_t byte = rbuf_read(&ci->rb);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100631 __enable_irq();
Harald Welte13e82022016-03-02 15:17:53 +0100632 card_emu_process_rx_byte(ci->ch, byte);
Harald Weltedde112e2016-03-20 16:42:11 +0100633 //TRACE_ERROR("%uRx%02x\r\n", i, byte);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100634 }
Harald Welteacae4122016-03-02 10:27:58 +0100635
Harald Welte8e7fca32017-05-07 16:14:33 +0200636 /* first try to send any pending messages on IRQ */
637 usb_refill_to_host(ci->ep_int);
638
639 /* then try to send any pending messages on IN */
640 usb_refill_to_host(ci->ep_in);
Harald Welteacae4122016-03-02 10:27:58 +0100641
Harald Welteebb80ed2016-03-02 13:56:59 +0100642 /* ensure we can handle incoming USB messages from the
643 * host */
Harald Welte8e7fca32017-05-07 16:14:33 +0200644 usb_refill_from_host(ci->ep_out);
645 queue = usb_get_queue(ci->ep_out);
Harald Welte0eaa9922016-03-04 03:03:49 +0100646 process_any_usb_commands(queue, ci);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100647 }
Harald Welte2a6d3af2016-02-28 19:29:14 +0100648}