blob: 104691f7bef9d651fdedc5160cea39b9a1e14f21 [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 Welte9d90d282018-06-29 22:25:42 +02009#include <osmocom/core/linuxlist.h>
10#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"
Kévin Redona1012b12018-07-01 18:11:01 +020013#include "simtrace_usb.h"
Harald Welte25a9a802017-05-08 13:30:09 +020014#include "simtrace_prot.h"
Harald Welte5c583d32017-05-09 06:46:47 +020015#include "sim_switch.h"
Harald Weltebd717682016-02-28 19:30:05 +010016
Harald Weltefd9c0412016-03-20 18:15:57 +010017#define TRACE_ENTRY() TRACE_DEBUG("%s entering\r\n", __func__)
Harald Welte2a6d3af2016-02-28 19:29:14 +010018
Harald Welte6d1128e2017-05-05 20:23:10 +020019#ifdef PINS_CARDSIM
Harald Welte2a6d3af2016-02-28 19:29:14 +010020static const Pin pins_cardsim[] = PINS_CARDSIM;
Harald Welte6d1128e2017-05-05 20:23:10 +020021#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +010022
23/* UART pins */
24static const Pin pins_usim1[] = {PINS_USIM1};
25static const Pin pin_usim1_rst = PIN_USIM1_nRST;
26static const Pin pin_usim1_vcc = PIN_USIM1_VCC;
27
28#ifdef CARDEMU_SECOND_UART
Harald Welte6dcacf32016-03-19 14:01:31 +010029static const Pin pins_usim2[] = {PINS_USIM2};
Harald Welte2a6d3af2016-02-28 19:29:14 +010030static const Pin pin_usim2_rst = PIN_USIM2_nRST;
31static const Pin pin_usim2_vcc = PIN_USIM2_VCC;
32#endif
33
Harald Welte13e82022016-03-02 15:17:53 +010034struct cardem_inst {
Harald Weltedde112e2016-03-20 16:42:11 +010035 uint32_t num;
Harald Welte13e82022016-03-02 15:17:53 +010036 struct card_handle *ch;
37 struct llist_head usb_out_queue;
38 struct ringbuf rb;
39 struct Usart_info usart_info;
40 int usb_pending_old;
41 uint8_t ep_out;
42 uint8_t ep_in;
43 uint8_t ep_int;
Harald Welte0eaa9922016-03-04 03:03:49 +010044 const Pin pin_insert;
Harald Welte07872b62016-03-20 11:45:36 +010045 uint32_t vcc_uv;
46 uint32_t vcc_uv_last;
Harald Welte13e82022016-03-02 15:17:53 +010047};
Harald Welte2a6d3af2016-02-28 19:29:14 +010048
Harald Welte8e7fca32017-05-07 16:14:33 +020049struct cardem_inst cardem_inst[] = {
Harald Welte2a6d3af2016-02-28 19:29:14 +010050 {
Harald Weltedde112e2016-03-20 16:42:11 +010051 .num = 0,
Harald Welte13e82022016-03-02 15:17:53 +010052 .usart_info = {
53 .base = USART1,
54 .id = ID_USART1,
55 .state = USART_RCV
56 },
Kévin Redona1012b12018-07-01 18:11:01 +020057 .ep_out = SIMTRACE_CARDEM_USB_EP_USIM1_DATAOUT,
58 .ep_in = SIMTRACE_CARDEM_USB_EP_USIM1_DATAIN,
59 .ep_int = SIMTRACE_CARDEM_USB_EP_USIM1_INT,
Harald Welte6d1128e2017-05-05 20:23:10 +020060#ifdef PIN_SET_USIM1_PRES
Harald Welte0eaa9922016-03-04 03:03:49 +010061 .pin_insert = PIN_SET_USIM1_PRES,
Harald Welte6d1128e2017-05-05 20:23:10 +020062#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +010063 },
64#ifdef CARDEMU_SECOND_UART
65 {
Harald Weltedde112e2016-03-20 16:42:11 +010066 .num = 1,
Harald Welte13e82022016-03-02 15:17:53 +010067 .usart_info = {
68 .base = USART0,
69 .id = ID_USART0,
70 .state = USART_RCV
71 },
Kévin Redona1012b12018-07-01 18:11:01 +020072 .ep_out = SIMTRACE_CARDEM_USB_EP_USIM2_DATAOUT,
73 .ep_in = SIMTRACE_CARDEM_USB_EP_USIM2_DATAIN,
74 .ep_int = SIMTRACE_CARDEM_USB_EP_USIM2_INT,
Harald Welte6d1128e2017-05-05 20:23:10 +020075#ifdef PIN_SET_USIM2_PRES
Harald Welte0eaa9922016-03-04 03:03:49 +010076 .pin_insert = PIN_SET_USIM2_PRES,
Harald Welte6d1128e2017-05-05 20:23:10 +020077#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +010078 },
79#endif
80};
81
82static Usart *get_usart_by_chan(uint8_t uart_chan)
83{
84 switch (uart_chan) {
85 case 0:
86 return USART1;
87#ifdef CARDEMU_SECOND_UART
88 case 1:
89 return USART0;
90#endif
91 }
92 return NULL;
93}
94
95/***********************************************************************
96 * Call-Backs from card_emu.c
97 ***********************************************************************/
98
Harald Weltec8beefb2016-03-20 14:40:47 +010099static void wait_tx_idle(Usart *usart)
100{
101 int i = 1;
102
103 /* wait until last char has been fully transmitted */
104 while ((usart->US_CSR & (US_CSR_TXEMPTY)) == 0) {
105 if (!(i%1000000)) {
106 TRACE_ERROR("s: %x \r\n", usart->US_CSR);
107 }
108 i++;
109 }
110}
111
Harald Weltec58bba02016-03-20 14:57:53 +0100112void card_emu_uart_wait_tx_idle(uint8_t uart_chan)
113{
114 Usart *usart = get_usart_by_chan(uart_chan);
115 wait_tx_idle(usart);
116}
117
Harald Welte2a6d3af2016-02-28 19:29:14 +0100118/* call-back from card_emu.c to enable/disable transmit and/or receive */
119void card_emu_uart_enable(uint8_t uart_chan, uint8_t rxtx)
120{
121 Usart *usart = get_usart_by_chan(uart_chan);
122 switch (rxtx) {
123 case ENABLE_TX:
Harald Welte9dbc46e2016-02-29 10:05:10 +0100124 USART_DisableIt(usart, ~US_IER_TXRDY);
Harald Weltec8beefb2016-03-20 14:40:47 +0100125 /* as irritating as it is, we actually want to keep the
126 * receiver enabled during transmit */
127 USART_SetReceiverEnabled(usart, 1);
Harald Welte8a416b12016-03-01 00:42:04 +0100128 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
129 USART_EnableIt(usart, US_IER_TXRDY);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100130 USART_SetTransmitterEnabled(usart, 1);
131 break;
132 case ENABLE_RX:
Harald Welte9dbc46e2016-02-29 10:05:10 +0100133 USART_DisableIt(usart, ~US_IER_RXRDY);
Harald Weltec8beefb2016-03-20 14:40:47 +0100134 /* as irritating as it is, we actually want to keep the
135 * transmitter enabled during receive */
136 USART_SetTransmitterEnabled(usart, 1);
137 wait_tx_idle(usart);
Harald Welte8a416b12016-03-01 00:42:04 +0100138 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
139 USART_EnableIt(usart, US_IER_RXRDY);
140 USART_SetReceiverEnabled(usart, 1);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100141 break;
142 case 0:
143 default:
144 USART_SetTransmitterEnabled(usart, 0);
145 USART_SetReceiverEnabled(usart, 0);
Harald Welte9dbc46e2016-02-29 10:05:10 +0100146 USART_DisableIt(usart, 0xFFFFFFFF);
Harald Welte8a416b12016-03-01 00:42:04 +0100147 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
Harald Welte2a6d3af2016-02-28 19:29:14 +0100148 break;
149 }
150}
151
152/* call-back from card_emu.c to transmit a byte */
153int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte)
154{
Harald Welte13e82022016-03-02 15:17:53 +0100155 Usart *usart = get_usart_by_chan(uart_chan);
156#if 0
Harald Welte2a6d3af2016-02-28 19:29:14 +0100157 Usart_info *ui = &usart_info[uart_chan];
158 ISO7816_SendChar(byte, ui);
Harald Welte13e82022016-03-02 15:17:53 +0100159#else
Harald Welte53079bb2016-03-20 14:58:35 +0100160 int i = 1;
Harald Welte13e82022016-03-02 15:17:53 +0100161 while ((usart->US_CSR & (US_CSR_TXRDY)) == 0) {
162 if (!(i%1000000)) {
Harald Weltedde112e2016-03-20 16:42:11 +0100163 TRACE_ERROR("%u: s: %x %02X\r\n",
164 uart_chan, usart->US_CSR,
165 usart->US_RHR & 0xFF);
Harald Welte13e82022016-03-02 15:17:53 +0100166 usart->US_CR = US_CR_RSTTX;
167 usart->US_CR = US_CR_RSTRX;
168 }
Harald Welte53079bb2016-03-20 14:58:35 +0100169 i++;
Harald Welte13e82022016-03-02 15:17:53 +0100170 }
171 usart->US_THR = byte;
Harald Welte53079bb2016-03-20 14:58:35 +0100172 //TRACE_ERROR("Sx%02x\r\n", byte);
Harald Welte13e82022016-03-02 15:17:53 +0100173#endif
Harald Welte2a6d3af2016-02-28 19:29:14 +0100174 return 1;
175}
176
177
178/* FIXME: integrate this with actual irq handler */
Harald Welte3bafe432016-03-20 16:43:12 +0100179static void usart_irq_rx(uint8_t inst_num)
Harald Welte2a6d3af2016-02-28 19:29:14 +0100180{
Harald Welte3bafe432016-03-20 16:43:12 +0100181 Usart *usart = get_usart_by_chan(inst_num);
182 struct cardem_inst *ci = &cardem_inst[inst_num];
Harald Weltec0bd7f02016-02-29 10:13:33 +0100183 uint32_t csr;
184 uint8_t byte = 0;
185
Harald Welteda15ca02016-03-17 21:14:04 +0100186 csr = usart->US_CSR & usart->US_IMR;
Harald Welte2a6d3af2016-02-28 19:29:14 +0100187
Harald Weltec0bd7f02016-02-29 10:13:33 +0100188 if (csr & US_CSR_RXRDY) {
189 byte = (usart->US_RHR) & 0xFF;
Harald Welte05cc7f62018-07-04 04:39:40 +0200190 if (rbuf_write(&ci->rb, byte) < 0)
191 TRACE_ERROR("rbuf overrun\r\n");
Harald Weltec0bd7f02016-02-29 10:13:33 +0100192 }
Harald Welte2a6d3af2016-02-28 19:29:14 +0100193
Harald Welte12d4bdf2016-03-02 10:31:03 +0100194 if (csr & US_CSR_TXRDY) {
Harald Welte13e82022016-03-02 15:17:53 +0100195 if (card_emu_tx_byte(ci->ch) == 0)
Harald Welte12d4bdf2016-03-02 10:31:03 +0100196 USART_DisableIt(usart, US_IER_TXRDY);
197 }
198
Harald Welte2a6d3af2016-02-28 19:29:14 +0100199 if (csr & (US_CSR_OVRE|US_CSR_FRAME|US_CSR_PARE|
200 US_CSR_TIMEOUT|US_CSR_NACK|(1<<10))) {
Harald Weltec0bd7f02016-02-29 10:13:33 +0100201 usart->US_CR = US_CR_RSTSTA | US_CR_RSTIT | US_CR_RSTNACK;
Harald Weltedde112e2016-03-20 16:42:11 +0100202 TRACE_ERROR("%u e 0x%x st: 0x%x\n", ci->num, byte, csr);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100203 }
204}
205
Harald Welte3bafe432016-03-20 16:43:12 +0100206void mode_cardemu_usart0_irq(void)
207{
208 /* USART0 == Instance 1 == USIM 2 */
209 usart_irq_rx(1);
210}
211
212void mode_cardemu_usart1_irq(void)
213{
214 /* USART1 == Instance 0 == USIM 1 */
215 usart_irq_rx(0);
216}
217
Harald Weltebd717682016-02-28 19:30:05 +0100218/* call-back from card_emu.c to change UART baud rate */
219int card_emu_uart_update_fidi(uint8_t uart_chan, unsigned int fidi)
220{
221 int rc;
Harald Weltebd717682016-02-28 19:30:05 +0100222 Usart *usart = get_usart_by_chan(uart_chan);
223
Harald Welte99f62a62016-02-29 10:08:49 +0100224 usart->US_CR |= US_CR_RXDIS | US_CR_RSTRX;
225 usart->US_FIDI = fidi & 0x3ff;
226 usart->US_CR |= US_CR_RXEN | US_CR_STTTO;
227 return 0;
Harald Weltebd717682016-02-28 19:30:05 +0100228}
Harald Welte2a6d3af2016-02-28 19:29:14 +0100229
230/***********************************************************************
Harald Welte419eb8a2016-03-20 10:04:03 +0100231 * ADC for VCC voltage detection
232 ***********************************************************************/
233
Harald Welte07872b62016-03-20 11:45:36 +0100234#ifdef DETECT_VCC_BY_ADC
235
236static int adc_triggered = 0;
Harald Welte28772eb2016-03-20 15:44:52 +0100237static int adc_sam3s_reva_errata = 0;
Harald Welte07872b62016-03-20 11:45:36 +0100238
Harald Welte419eb8a2016-03-20 10:04:03 +0100239static int card_vcc_adc_init(void)
240{
Harald Welte622b6be2016-03-20 16:43:59 +0100241 uint32_t chip_arch = CHIPID->CHIPID_CIDR & CHIPID_CIDR_ARCH_Msk;
242 uint32_t chip_ver = CHIPID->CHIPID_CIDR & CHIPID_CIDR_VERSION_Msk;
Harald Welte28772eb2016-03-20 15:44:52 +0100243
Harald Welte07872b62016-03-20 11:45:36 +0100244 PMC_EnablePeripheral(ID_ADC);
245
Harald Welte419eb8a2016-03-20 10:04:03 +0100246 ADC->ADC_CR |= ADC_CR_SWRST;
Harald Welte28772eb2016-03-20 15:44:52 +0100247 if (chip_ver == 0 &&
248 (chip_arch == CHIPID_CIDR_ARCH_SAM3SxA ||
249 chip_arch == CHIPID_CIDR_ARCH_SAM3SxB ||
250 chip_arch == CHIPID_CIDR_ARCH_SAM3SxC)) {
251 TRACE_INFO("Enabling Rev.A ADC Errata work-around\r\n");
252 adc_sam3s_reva_errata = 1;
253 }
254
255 if (adc_sam3s_reva_errata) {
256 /* Errata Work-Around to clear EOCx flags */
Harald Welte07872b62016-03-20 11:45:36 +0100257 volatile uint32_t foo;
258 int i;
259 for (i = 0; i < 16; i++)
260 foo = ADC->ADC_CDR[i];
261 }
262
263 /* Initialize ADC for AD7 / AD6, fADC=48/24=2MHz */
Harald Welte419eb8a2016-03-20 10:04:03 +0100264 ADC->ADC_MR = ADC_MR_TRGEN_DIS | ADC_MR_LOWRES_BITS_12 |
265 ADC_MR_SLEEP_NORMAL | ADC_MR_FWUP_OFF |
Harald Welte07872b62016-03-20 11:45:36 +0100266 ADC_MR_FREERUN_OFF | ADC_MR_PRESCAL(23) |
267 ADC_MR_STARTUP_SUT8 | ADC_MR_SETTLING(3) |
268 ADC_MR_ANACH_NONE | ADC_MR_TRACKTIM(4) |
Harald Welte419eb8a2016-03-20 10:04:03 +0100269 ADC_MR_TRANSFER(1) | ADC_MR_USEQ_NUM_ORDER;
270 /* enable AD6 + AD7 channels */
Harald Welte07872b62016-03-20 11:45:36 +0100271 ADC->ADC_CHER = ADC_CHER_CH7;
272 ADC->ADC_IER = ADC_IER_EOC7;
273#ifdef CARDEMU_SECOND_UART
274 ADC->ADC_CHER |= ADC_CHER_CH6;
275 ADC->ADC_IER |= ADC_IER_EOC6;
276#endif
277 NVIC_EnableIRQ(ADC_IRQn);
Harald Welte419eb8a2016-03-20 10:04:03 +0100278 ADC->ADC_CR |= ADC_CR_START;
Harald Welte07872b62016-03-20 11:45:36 +0100279
280 return 0;
Harald Welte419eb8a2016-03-20 10:04:03 +0100281}
282
Harald Welte07872b62016-03-20 11:45:36 +0100283#define UV_PER_LSB ((3300 * 1000) / 4096)
284#define VCC_UV_THRESH_1V8 1500000
285#define VCC_UV_THRESH_3V 2800000
286
287static void process_vcc_adc(struct cardem_inst *ci)
Harald Welte419eb8a2016-03-20 10:04:03 +0100288{
Harald Welte07872b62016-03-20 11:45:36 +0100289 if (ci->vcc_uv >= VCC_UV_THRESH_3V &&
290 ci->vcc_uv_last < VCC_UV_THRESH_3V) {
291 card_emu_io_statechg(ci->ch, CARD_IO_VCC, 1);
292 /* FIXME do this for real */
293 card_emu_io_statechg(ci->ch, CARD_IO_CLK, 1);
294 } else if (ci->vcc_uv < VCC_UV_THRESH_3V &&
295 ci->vcc_uv_last >= VCC_UV_THRESH_3V) {
296 /* FIXME do this for real */
297 card_emu_io_statechg(ci->ch, CARD_IO_CLK, 0);
298 card_emu_io_statechg(ci->ch, CARD_IO_VCC, 0);
Harald Welte419eb8a2016-03-20 10:04:03 +0100299 }
Harald Welte07872b62016-03-20 11:45:36 +0100300 ci->vcc_uv_last = ci->vcc_uv;
301}
302
303static uint32_t adc2uv(uint16_t adc)
304{
305 uint32_t uv = (uint32_t) adc * UV_PER_LSB;
Harald Welte622b6be2016-03-20 16:43:59 +0100306 return uv;
Harald Welte07872b62016-03-20 11:45:36 +0100307}
308
309void ADC_IrqHandler(void)
310{
311#ifdef CARDEMU_SECOND_UART
312 if (ADC->ADC_ISR & ADC_ISR_EOC6) {
313 uint16_t val = ADC->ADC_CDR[6] & 0xFFF;
314 cardem_inst[1].vcc_uv = adc2uv(val);
315 process_vcc_adc(&cardem_inst[1]);
Harald Welte28772eb2016-03-20 15:44:52 +0100316 if (adc_sam3s_reva_errata) {
317 /* Errata: START doesn't start a conversion
318 * sequence, but only a single conversion */
319 ADC->ADC_CR |= ADC_CR_START;
320 }
Harald Welte07872b62016-03-20 11:45:36 +0100321 }
322#endif
Harald Welte419eb8a2016-03-20 10:04:03 +0100323
324 if (ADC->ADC_ISR & ADC_ISR_EOC7) {
Harald Welte07872b62016-03-20 11:45:36 +0100325 uint16_t val = ADC->ADC_CDR[7] & 0xFFF;
326 cardem_inst[0].vcc_uv = adc2uv(val);
327 process_vcc_adc(&cardem_inst[0]);
328 ADC->ADC_CR |= ADC_CR_START;
Harald Welte419eb8a2016-03-20 10:04:03 +0100329 }
330}
Harald Welte07872b62016-03-20 11:45:36 +0100331#endif /* DETECT_VCC_BY_ADC */
Harald Welte419eb8a2016-03-20 10:04:03 +0100332
333/***********************************************************************
Harald Welte2a6d3af2016-02-28 19:29:14 +0100334 * Core USB / mainloop integration
335 ***********************************************************************/
336
Harald Weltec0bd7f02016-02-29 10:13:33 +0100337static void usim1_rst_irqhandler(const Pin *pPin)
338{
339 int active = PIO_Get(&pin_usim1_rst) ? 0 : 1;
Harald Welte13e82022016-03-02 15:17:53 +0100340 card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_RST, active);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100341}
342
Harald Welte07872b62016-03-20 11:45:36 +0100343#ifndef DETECT_VCC_BY_ADC
Harald Weltec0bd7f02016-02-29 10:13:33 +0100344static void usim1_vcc_irqhandler(const Pin *pPin)
345{
346 int active = PIO_Get(&pin_usim1_vcc) ? 1 : 0;
Harald Welte13e82022016-03-02 15:17:53 +0100347 card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_VCC, active);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100348 /* FIXME do this for real */
Harald Welte13e82022016-03-02 15:17:53 +0100349 card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_CLK, active);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100350}
Harald Welte07872b62016-03-20 11:45:36 +0100351#endif /* !DETECT_VCC_BY_ADC */
Harald Weltec0bd7f02016-02-29 10:13:33 +0100352
Harald Welte13e82022016-03-02 15:17:53 +0100353#ifdef CARDEMU_SECOND_UART
354static void usim2_rst_irqhandler(const Pin *pPin)
355{
356 int active = PIO_Get(&pin_usim2_rst) ? 0 : 1;
357 card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_RST, active);
358}
359
Harald Welte07872b62016-03-20 11:45:36 +0100360#ifndef DETECT_VCC_BY_ADC
Harald Welte13e82022016-03-02 15:17:53 +0100361static void usim2_vcc_irqhandler(const Pin *pPin)
362{
363 int active = PIO_Get(&pin_usim2_vcc) ? 1 : 0;
364 card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_VCC, active);
365 /* FIXME do this for real */
366 card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_CLK, active);
367}
Harald Welte07872b62016-03-20 11:45:36 +0100368#endif /* !DETECT_VCC_BY_ADC */
369#endif /* CARDEMU_SECOND_UART */
Harald Welte13e82022016-03-02 15:17:53 +0100370
Harald Welte2a6d3af2016-02-28 19:29:14 +0100371/* executed once at system boot for each config */
372void mode_cardemu_configure(void)
373{
Harald Weltebd717682016-02-28 19:30:05 +0100374 TRACE_ENTRY();
Harald Welte2a6d3af2016-02-28 19:29:14 +0100375}
376
377/* called if config is activated */
378void mode_cardemu_init(void)
379{
Harald Welte13e82022016-03-02 15:17:53 +0100380 int i;
Harald Weltebd717682016-02-28 19:30:05 +0100381
Harald Welte13e82022016-03-02 15:17:53 +0100382 TRACE_ENTRY();
Harald Weltec0bd7f02016-02-29 10:13:33 +0100383
Harald Welte6d1128e2017-05-05 20:23:10 +0200384#ifdef PINS_CARDSIM
Harald Welte2a6d3af2016-02-28 19:29:14 +0100385 PIO_Configure(pins_cardsim, PIO_LISTSIZE(pins_cardsim));
Harald Welte6d1128e2017-05-05 20:23:10 +0200386#endif
Harald Welte07872b62016-03-20 11:45:36 +0100387#ifdef DETECT_VCC_BY_ADC
388 card_vcc_adc_init();
389#endif /* DETECT_VCC_BY_ADC */
Harald Welte2a6d3af2016-02-28 19:29:14 +0100390
Harald Welte13e82022016-03-02 15:17:53 +0100391 INIT_LLIST_HEAD(&cardem_inst[0].usb_out_queue);
392 rbuf_reset(&cardem_inst[0].rb);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100393 PIO_Configure(pins_usim1, PIO_LISTSIZE(pins_usim1));
Harald Welte13e82022016-03-02 15:17:53 +0100394 ISO7816_Init(&cardem_inst[0].usart_info, CLK_SLAVE);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100395 NVIC_EnableIRQ(USART1_IRQn);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100396 PIO_ConfigureIt(&pin_usim1_rst, usim1_rst_irqhandler);
397 PIO_EnableIt(&pin_usim1_rst);
Harald Welte07872b62016-03-20 11:45:36 +0100398#ifndef DETECT_VCC_BY_ADC
Harald Weltec0bd7f02016-02-29 10:13:33 +0100399 PIO_ConfigureIt(&pin_usim1_vcc, usim1_vcc_irqhandler);
400 PIO_EnableIt(&pin_usim1_vcc);
Harald Welte07872b62016-03-20 11:45:36 +0100401#endif /* DETECT_VCC_BY_ADC */
Kévin Redona1012b12018-07-01 18:11:01 +0200402 cardem_inst[0].ch = card_emu_init(0, 2, 0, SIMTRACE_CARDEM_USB_EP_USIM1_DATAIN, SIMTRACE_CARDEM_USB_EP_USIM1_INT);
Harald Weltec47fc5f2017-05-11 16:51:57 +0200403 sim_switch_use_physical(0, 1);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100404
405#ifdef CARDEMU_SECOND_UART
Harald Welte13e82022016-03-02 15:17:53 +0100406 INIT_LLIST_HEAD(&cardem_inst[1].usb_out_queue);
407 rbuf_reset(&cardem_inst[1].rb);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100408 PIO_Configure(pins_usim2, PIO_LISTSIZE(pins_usim2));
Harald Welte13e82022016-03-02 15:17:53 +0100409 ISO7816_Init(&cardem_inst[1].usart_info, CLK_SLAVE);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100410 NVIC_EnableIRQ(USART0_IRQn);
Harald Welte13e82022016-03-02 15:17:53 +0100411 PIO_ConfigureIt(&pin_usim2_rst, usim2_rst_irqhandler);
412 PIO_EnableIt(&pin_usim2_rst);
Harald Welte07872b62016-03-20 11:45:36 +0100413#ifndef DETECT_VCC_BY_ADC
Harald Welte13e82022016-03-02 15:17:53 +0100414 PIO_ConfigureIt(&pin_usim2_vcc, usim2_vcc_irqhandler);
415 PIO_EnableIt(&pin_usim2_vcc);
Harald Welte07872b62016-03-20 11:45:36 +0100416#endif /* DETECT_VCC_BY_ADC */
Kévin Redona1012b12018-07-01 18:11:01 +0200417 cardem_inst[1].ch = card_emu_init(1, 0, 1, SIMTRACE_CARDEM_USB_EP_USIM2_DATAIN, SIMTRACE_CARDEM_USB_EP_USIM2_INT);
Harald Weltec47fc5f2017-05-11 16:51:57 +0200418 sim_switch_use_physical(1, 1);
Harald Welte07872b62016-03-20 11:45:36 +0100419#endif /* CARDEMU_SECOND_UART */
Harald Weltec47fc5f2017-05-11 16:51:57 +0200420
Harald Welte2a6d3af2016-02-28 19:29:14 +0100421}
422
423/* called if config is deactivated */
424void mode_cardemu_exit(void)
425{
Harald Weltebd717682016-02-28 19:30:05 +0100426 TRACE_ENTRY();
427
Harald Welte7abdb512016-03-03 17:48:32 +0100428 /* FIXME: stop tc_fdt */
Harald Welte8e7fca32017-05-07 16:14:33 +0200429 /* FIXME: release all msg, unlink them from any queue */
Harald Welte7abdb512016-03-03 17:48:32 +0100430
Harald Weltec0bd7f02016-02-29 10:13:33 +0100431 PIO_DisableIt(&pin_usim1_rst);
432 PIO_DisableIt(&pin_usim1_vcc);
433
Harald Welte2a6d3af2016-02-28 19:29:14 +0100434 NVIC_DisableIRQ(USART1_IRQn);
435 USART_SetTransmitterEnabled(USART1, 0);
436 USART_SetReceiverEnabled(USART1, 0);
437
438#ifdef CARDEMU_SECOND_UART
Harald Welte13e82022016-03-02 15:17:53 +0100439 PIO_DisableIt(&pin_usim2_rst);
440 PIO_DisableIt(&pin_usim2_vcc);
441
Harald Welte2a6d3af2016-02-28 19:29:14 +0100442 NVIC_DisableIRQ(USART0_IRQn);
443 USART_SetTransmitterEnabled(USART0, 0);
444 USART_SetReceiverEnabled(USART0, 0);
445#endif
446}
447
Harald Welteebb80ed2016-03-02 13:56:59 +0100448/* handle a single USB command as received from the USB host */
Harald Welte25a9a802017-05-08 13:30:09 +0200449static void dispatch_usb_command_generic(struct msgb *msg, struct cardem_inst *ci)
Harald Welteebb80ed2016-03-02 13:56:59 +0100450{
Harald Welte25a9a802017-05-08 13:30:09 +0200451 struct simtrace_msg_hdr *hdr;
452
453 hdr = (struct simtrace_msg_hdr *) msg->l1h;
454 switch (hdr->msg_type) {
455 case SIMTRACE_CMD_BD_BOARD_INFO:
456 break;
457 default:
458 break;
459 }
460 usb_buf_free(msg);
461}
462
463/* handle a single USB command as received from the USB host */
464static void dispatch_usb_command_cardem(struct msgb *msg, struct cardem_inst *ci)
465{
466 struct simtrace_msg_hdr *hdr;
Harald Welte06b27f62016-03-02 14:26:38 +0100467 struct cardemu_usb_msg_set_atr *atr;
Harald Welte0eaa9922016-03-04 03:03:49 +0100468 struct cardemu_usb_msg_cardinsert *cardins;
Harald Welteebb80ed2016-03-02 13:56:59 +0100469 struct llist_head *queue;
470
Harald Welte25a9a802017-05-08 13:30:09 +0200471 hdr = (struct simtrace_msg_hdr *) msg->l1h;
Harald Welteebb80ed2016-03-02 13:56:59 +0100472 switch (hdr->msg_type) {
Harald Welte25a9a802017-05-08 13:30:09 +0200473 case SIMTRACE_MSGT_DT_CEMU_TX_DATA:
Harald Welte0eaa9922016-03-04 03:03:49 +0100474 queue = card_emu_get_uart_tx_queue(ci->ch);
Harald Welte8e7fca32017-05-07 16:14:33 +0200475 llist_add_tail(&msg->list, queue);
Harald Welte0eaa9922016-03-04 03:03:49 +0100476 card_emu_have_new_uart_tx(ci->ch);
Harald Welteebb80ed2016-03-02 13:56:59 +0100477 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200478 case SIMTRACE_MSGT_DT_CEMU_SET_ATR:
479 atr = (struct cardemu_usb_msg_set_atr *) msg->l2h;
Harald Welted295b922016-03-18 21:01:36 +0100480 card_emu_set_atr(ci->ch, atr->atr, atr->atr_len);
Harald Welte8e7fca32017-05-07 16:14:33 +0200481 usb_buf_free(msg);
Harald Welte06b27f62016-03-02 14:26:38 +0100482 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200483 case SIMTRACE_MSGT_DT_CEMU_CARDINSERT:
484 cardins = (struct cardemu_usb_msg_cardinsert *) msg->l2h;
Harald Welte965d5c92017-11-03 20:47:12 +0100485 if (!ci->pin_insert.pio) {
486 TRACE_INFO("%u: skipping unsupported card_insert to %s\r\n",
487 ci->num, cardins->card_insert ? "INSERTED" : "REMOVED");
488 break;
489 }
Harald Weltedde112e2016-03-20 16:42:11 +0100490 TRACE_INFO("%u: set card_insert to %s\r\n", ci->num,
491 cardins->card_insert ? "INSERTED" : "REMOVED");
Harald Welte0eaa9922016-03-04 03:03:49 +0100492 if (cardins->card_insert)
493 PIO_Set(&ci->pin_insert);
494 else
495 PIO_Clear(&ci->pin_insert);
Harald Welte8e7fca32017-05-07 16:14:33 +0200496 usb_buf_free(msg);
Harald Welte0eaa9922016-03-04 03:03:49 +0100497 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200498 case SIMTRACE_MSGT_BD_CEMU_STATUS:
Harald Welteff160652016-03-19 21:59:06 +0100499 card_emu_report_status(ci->ch);
Harald Welte25a9a802017-05-08 13:30:09 +0200500 usb_buf_free(msg);
Harald Welteff160652016-03-19 21:59:06 +0100501 break;
Harald Welte25a9a802017-05-08 13:30:09 +0200502 case SIMTRACE_MSGT_BD_CEMU_STATS:
Harald Welteebb80ed2016-03-02 13:56:59 +0100503 default:
Harald Welte25a9a802017-05-08 13:30:09 +0200504 /* FIXME: Send Error */
505 usb_buf_free(msg);
506 break;
507 }
508}
509
Harald Weltefc87c242017-11-28 19:17:27 +0100510#ifdef PINS_PERST
511#include "wwan_perst.h"
512#endif
513
Harald Welte2e9254a2017-05-09 06:30:04 +0200514static int usb_command_modem_reset(struct msgb *msg, struct cardem_inst *ci)
515{
Harald Welteb91f6ad2017-05-10 22:51:30 +0200516 struct st_modem_reset *mr = (struct st_modem_reset *) msg->l2h;
Harald Welte2e9254a2017-05-09 06:30:04 +0200517
518 if (msgb_l2len(msg) < sizeof(*mr))
519 return -1;
520
521 switch (mr->asserted) {
Harald Weltefc87c242017-11-28 19:17:27 +0100522#ifdef PINS_PERST
Harald Welte2e9254a2017-05-09 06:30:04 +0200523 case 0:
524 wwan_perst_set(ci->num, 0);
525 break;
526 case 1:
527 wwan_perst_set(ci->num, 1);
528 break;
529 case 2:
530 wwan_perst_do_reset_pulse(ci->num, mr->pulse_duration_msec);
531 break;
Harald Weltefc87c242017-11-28 19:17:27 +0100532#endif
Harald Welte2e9254a2017-05-09 06:30:04 +0200533 default:
534 return -1;
535 }
536
537 return 0;
538}
539
Harald Welte5c583d32017-05-09 06:46:47 +0200540static int usb_command_sim_select(struct msgb *msg, struct cardem_inst *ci)
541{
Harald Welteb91f6ad2017-05-10 22:51:30 +0200542 struct st_modem_sim_select *mss = (struct st_modem_sim_select *) msg->l2h;
Harald Welte5c583d32017-05-09 06:46:47 +0200543
544 if (msgb_l2len(msg) < sizeof(*mss))
545 return -1;
546
547 if (mss->remote_sim)
548 sim_switch_use_physical(ci->num, 0);
549 else
550 sim_switch_use_physical(ci->num, 1);
551
552 return 0;
553}
554
Harald Welte25a9a802017-05-08 13:30:09 +0200555/* handle a single USB command as received from the USB host */
556static void dispatch_usb_command_modem(struct msgb *msg, struct cardem_inst *ci)
557{
558 struct simtrace_msg_hdr *hdr;
559
560 hdr = (struct simtrace_msg_hdr *) msg->l1h;
561 switch (hdr->msg_type) {
562 case SIMTRACE_MSGT_DT_MODEM_RESET:
Harald Welte2e9254a2017-05-09 06:30:04 +0200563 usb_command_modem_reset(msg, ci);
Harald Welte25a9a802017-05-08 13:30:09 +0200564 break;
565 case SIMTRACE_MSGT_DT_MODEM_SIM_SELECT:
Harald Welte5c583d32017-05-09 06:46:47 +0200566 usb_command_sim_select(msg, ci);
Harald Welte25a9a802017-05-08 13:30:09 +0200567 break;
568 case SIMTRACE_MSGT_BD_MODEM_STATUS:
569 break;
570 default:
571 break;
572 }
573 usb_buf_free(msg);
574}
575
576/* handle a single USB command as received from the USB host */
577static void dispatch_usb_command(struct msgb *msg, struct cardem_inst *ci)
578{
Harald Welteb91f6ad2017-05-10 22:51:30 +0200579 struct simtrace_msg_hdr *sh = (struct simtrace_msg_hdr *) msg->l1h;
Harald Welte25a9a802017-05-08 13:30:09 +0200580
581 if (msgb_length(msg) < sizeof(*sh)) {
582 /* FIXME: Error */
583 usb_buf_free(msg);
584 return;
585 }
586
587 switch (sh->msg_class) {
588 case SIMTRACE_MSGC_GENERIC:
589 dispatch_usb_command_generic(msg, ci);
590 break;
591 case SIMTRACE_MSGC_CARDEM:
592 dispatch_usb_command_cardem(msg, ci);
593 break;
594 case SIMTRACE_MSGC_MODEM:
Harald Welte23c00b62017-05-11 01:11:43 +0200595 /* FIXME: Find out why this fails if used for !=
596 * MSGC_MODEM ?!? */
597 msg->l2h = msg->l1h + sizeof(*sh);
Harald Welte25a9a802017-05-08 13:30:09 +0200598 dispatch_usb_command_modem(msg, ci);
599 break;
600 default:
601 /* FIXME: Send Error */
Harald Welte8e7fca32017-05-07 16:14:33 +0200602 usb_buf_free(msg);
Harald Welteebb80ed2016-03-02 13:56:59 +0100603 break;
604 }
605}
606
Harald Welte8e7fca32017-05-07 16:14:33 +0200607static void dispatch_received_msg(struct msgb *msg, struct cardem_inst *ci)
Harald Welteb26d0032016-03-19 13:33:02 +0100608{
Harald Welte8e7fca32017-05-07 16:14:33 +0200609 struct msgb *segm;
Harald Welte25a9a802017-05-08 13:30:09 +0200610 struct simtrace_msg_hdr *mh;
Harald Welteb26d0032016-03-19 13:33:02 +0100611
612 /* check if we have multiple concatenated commands in
613 * one message. USB endpoints are streams that don't
614 * preserve the message boundaries */
Harald Welte25a9a802017-05-08 13:30:09 +0200615 mh = (struct simtrace_msg_hdr *) msg->data;
Harald Welte8e7fca32017-05-07 16:14:33 +0200616 if (mh->msg_len == msgb_length(msg)) {
Harald Welteb26d0032016-03-19 13:33:02 +0100617 /* fast path: only one message in buffer */
Harald Welte8e7fca32017-05-07 16:14:33 +0200618 dispatch_usb_command(msg, ci);
Harald Welteb26d0032016-03-19 13:33:02 +0100619 return;
620 }
621
622 /* slow path: iterate over list of messages, allocating one new
623 * reqe_ctx per segment */
Harald Welte8e7fca32017-05-07 16:14:33 +0200624 while (1) {
Harald Welte25a9a802017-05-08 13:30:09 +0200625 mh = (struct simtrace_msg_hdr *) msg->data;
Harald Welte8e7fca32017-05-07 16:14:33 +0200626
627 segm = usb_buf_alloc(ci->ep_out);
Harald Welteb26d0032016-03-19 13:33:02 +0100628 if (!segm) {
Harald Welte8e7fca32017-05-07 16:14:33 +0200629 TRACE_ERROR("%u: ENOMEM during msg segmentation\r\n",
Harald Weltedde112e2016-03-20 16:42:11 +0100630 ci->num);
Harald Welteb26d0032016-03-19 13:33:02 +0100631 break;
632 }
Harald Welte8e7fca32017-05-07 16:14:33 +0200633
634 if (mh->msg_len > msgb_length(msg)) {
635 TRACE_ERROR("%u: Unexpected large message (%u bytes)\n",
636 ci->num, mh->msg_len);
637 usb_buf_free(segm);
638 } else {
639 uint8_t *cur = msgb_put(segm, mh->msg_len);
640 segm->l1h = segm->head;
641 memcpy(cur, mh, mh->msg_len);
642 dispatch_usb_command(segm, ci);
643 }
644 /* pull this message */
645 msgb_pull(msg, mh->msg_len);
646 /* abort if we're done */
647 if (msgb_length(msg) <= 0)
648 break;
Harald Welteb26d0032016-03-19 13:33:02 +0100649 }
650
Harald Welte8e7fca32017-05-07 16:14:33 +0200651 usb_buf_free(msg);
Harald Welteb26d0032016-03-19 13:33:02 +0100652}
653
Harald Welteebb80ed2016-03-02 13:56:59 +0100654/* iterate over the queue of incoming USB commands and dispatch/execute
655 * them */
Harald Welte9f240b62016-03-18 10:32:56 +0100656static void process_any_usb_commands(struct llist_head *main_q,
657 struct cardem_inst *ci)
Harald Welteebb80ed2016-03-02 13:56:59 +0100658{
Harald Welte9f240b62016-03-18 10:32:56 +0100659 struct llist_head *lh;
Harald Welte8e7fca32017-05-07 16:14:33 +0200660 struct msgb *msg;
Harald Welte9f240b62016-03-18 10:32:56 +0100661 int i;
Harald Welteebb80ed2016-03-02 13:56:59 +0100662
Harald Welte9f240b62016-03-18 10:32:56 +0100663 /* limit the number of iterations to 10, to ensure we don't get
664 * stuck here without returning to main loop processing */
665 for (i = 0; i < 10; i++) {
666 /* de-queue the list head in an irq-safe way */
667 lh = llist_head_dequeue_irqsafe(main_q);
668 if (!lh)
669 break;
Harald Welte8e7fca32017-05-07 16:14:33 +0200670 msg = llist_entry(lh, struct msgb, list);
671 dispatch_received_msg(msg, ci);
Harald Welte5820ea92016-03-16 22:12:00 +0100672 }
Harald Welteebb80ed2016-03-02 13:56:59 +0100673}
674
Harald Welte2a6d3af2016-02-28 19:29:14 +0100675/* main loop function, called repeatedly */
676void mode_cardemu_run(void)
677{
Harald Welteacae4122016-03-02 10:27:58 +0100678 struct llist_head *queue;
Harald Welte13e82022016-03-02 15:17:53 +0100679 unsigned int i;
Harald Welteacae4122016-03-02 10:27:58 +0100680
Harald Welte13e82022016-03-02 15:17:53 +0100681 for (i = 0; i < ARRAY_SIZE(cardem_inst); i++) {
682 struct cardem_inst *ci = &cardem_inst[i];
683
Harald Welte54cb3d02016-02-29 14:12:40 +0100684 /* drain the ring buffer from UART into card_emu */
Harald Weltec0bd7f02016-02-29 10:13:33 +0100685 while (1) {
686 __disable_irq();
Harald Welte13e82022016-03-02 15:17:53 +0100687 if (rbuf_is_empty(&ci->rb)) {
Harald Weltec0bd7f02016-02-29 10:13:33 +0100688 __enable_irq();
689 break;
690 }
Harald Welte13e82022016-03-02 15:17:53 +0100691 uint8_t byte = rbuf_read(&ci->rb);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100692 __enable_irq();
Harald Welte13e82022016-03-02 15:17:53 +0100693 card_emu_process_rx_byte(ci->ch, byte);
Harald Weltedde112e2016-03-20 16:42:11 +0100694 //TRACE_ERROR("%uRx%02x\r\n", i, byte);
Harald Weltec0bd7f02016-02-29 10:13:33 +0100695 }
Harald Welteacae4122016-03-02 10:27:58 +0100696
Harald Welte8e7fca32017-05-07 16:14:33 +0200697 /* first try to send any pending messages on IRQ */
698 usb_refill_to_host(ci->ep_int);
699
700 /* then try to send any pending messages on IN */
701 usb_refill_to_host(ci->ep_in);
Harald Welteacae4122016-03-02 10:27:58 +0100702
Harald Welteebb80ed2016-03-02 13:56:59 +0100703 /* ensure we can handle incoming USB messages from the
704 * host */
Harald Welte8e7fca32017-05-07 16:14:33 +0200705 usb_refill_from_host(ci->ep_out);
706 queue = usb_get_queue(ci->ep_out);
Harald Welte0eaa9922016-03-04 03:03:49 +0100707 process_any_usb_commands(queue, ci);
Harald Welte2a6d3af2016-02-28 19:29:14 +0100708 }
Harald Welte2a6d3af2016-02-28 19:29:14 +0100709}