blob: 9a681675cd400b99340551eb7d1a24d7c3b16e5e [file] [log] [blame]
Harald Weltea40c8e52019-09-27 19:22:34 +02001#include <errno.h>
2#include <string.h>
3#include <osmocom/core/linuxlist.h>
4#include <osmocom/core/utils.h>
Harald Welte7b64fc02019-10-09 20:49:35 +02005#include <osmocom/core/timer.h>
Harald Weltea40c8e52019-09-27 19:22:34 +02006
7#include "cuart.h"
8
9static LLIST_HEAD(g_cuart_drivers);
10
11const struct value_string card_uart_event_vals[] = {
12 OSMO_VALUE_STRING(CUART_E_RX_SINGLE),
13 OSMO_VALUE_STRING(CUART_E_RX_COMPLETE),
14 OSMO_VALUE_STRING(CUART_E_RX_TIMEOUT),
15 OSMO_VALUE_STRING(CUART_E_TX_COMPLETE),
16 { 0, NULL }
17};
18
19static struct card_uart_driver *cuart_drv_by_name(const char *driver_name)
20{
21 struct card_uart_driver *drv;
22 llist_for_each_entry(drv, &g_cuart_drivers, list) {
23 if (!strcmp(drv->name, driver_name))
24 return drv;
25 }
26 return NULL;
27}
28
Harald Welte7b64fc02019-10-09 20:49:35 +020029/* obtain the current ETU in us */
30static int get_etu_in_us(struct card_uart *cuart)
31{
32 /* FIXME: actually implement this based on the real baud rate */
33 return (1000000/9600);
34}
35
36/* software waiting-time timer has expired */
37static void card_uart_wtime_cb(void *data)
38{
39 struct card_uart *cuart = (struct card_uart *) data;
40 card_uart_notification(cuart, CUART_E_RX_TIMEOUT, NULL);
41 /* should we automatically disable the receiver? */
42}
43
44void card_uart_wtime_restart(struct card_uart *cuart)
45{
46 int secs, usecs;
47
48 usecs = get_etu_in_us(cuart) * cuart->wtime_etu;
49 if (usecs > 1000000) {
50 secs = usecs / 1000000;
51 usecs = usecs % 1000000;
52 } else
53 secs = 0;
54 osmo_timer_schedule(&cuart->wtime_tmr, secs, usecs);
55}
56
Harald Weltea40c8e52019-09-27 19:22:34 +020057int card_uart_open(struct card_uart *cuart, const char *driver_name, const char *device_name)
58{
59 struct card_uart_driver *drv = cuart_drv_by_name(driver_name);
60 int rc;
61
62 if (!drv)
63 return -ENODEV;
64
Harald Welte7b64fc02019-10-09 20:49:35 +020065 cuart->wtime_etu = 9600; /* ISO 7816-3 Section 8.1 */
Harald Weltea40c8e52019-09-27 19:22:34 +020066 cuart->rx_enabled = true;
67 cuart->rx_threshold = 1;
Harald Welte7b64fc02019-10-09 20:49:35 +020068 osmo_timer_setup(&cuart->wtime_tmr, card_uart_wtime_cb, cuart);
Harald Weltea40c8e52019-09-27 19:22:34 +020069
70 rc = drv->ops->open(cuart, device_name);
71 if (rc < 0)
72 return rc;
73
74 cuart->driver = drv;
75 return 0;
76}
77
78int card_uart_close(struct card_uart *cuart)
79{
80 OSMO_ASSERT(cuart);
81 OSMO_ASSERT(cuart->driver);
82 OSMO_ASSERT(cuart->driver->ops);
83 OSMO_ASSERT(cuart->driver->ops->close);
84 return cuart->driver->ops->close(cuart);
85}
86
Harald Welte9700d392019-10-09 20:17:28 +020087int card_uart_ctrl(struct card_uart *cuart, enum card_uart_ctl ctl, int arg)
Harald Weltea40c8e52019-09-27 19:22:34 +020088{
89 int rc;
90 OSMO_ASSERT(cuart);
91 OSMO_ASSERT(cuart->driver);
92 OSMO_ASSERT(cuart->driver->ops);
93 OSMO_ASSERT(cuart->driver->ops->ctrl);
Harald Welte7b64fc02019-10-09 20:49:35 +020094
Harald Welte9700d392019-10-09 20:17:28 +020095 rc = cuart->driver->ops->ctrl(cuart, ctl, arg);
Harald Weltea40c8e52019-09-27 19:22:34 +020096 if (rc < 0)
97 return rc;
98
99 switch (ctl) {
Harald Welte7b64fc02019-10-09 20:49:35 +0200100 case CUART_CTL_WTIME:
101 cuart->wtime_etu = arg;
102 break;
Harald Weltea40c8e52019-09-27 19:22:34 +0200103 case CUART_CTL_RX:
Harald Welte9700d392019-10-09 20:17:28 +0200104 cuart->rx_enabled = arg ? true : false;
Harald Welte7b64fc02019-10-09 20:49:35 +0200105 if (!cuart->rx_enabled)
106 osmo_timer_del(&cuart->wtime_tmr);
Harald Weltea40c8e52019-09-27 19:22:34 +0200107 break;
108 default:
109 break;
110 }
111
112 return rc;
113}
114
Harald Welte6603d952019-10-09 20:50:13 +0200115int card_uart_tx(struct card_uart *cuart, const uint8_t *data, size_t len, bool rx_after_complete)
Harald Weltea40c8e52019-09-27 19:22:34 +0200116{
117 OSMO_ASSERT(cuart);
118 OSMO_ASSERT(cuart->driver);
119 OSMO_ASSERT(cuart->driver->ops);
120 OSMO_ASSERT(cuart->driver->ops->async_tx);
121
122 OSMO_ASSERT(!cuart->tx_busy);
123 cuart->tx_busy = true;
Harald Welte6603d952019-10-09 20:50:13 +0200124 cuart->rx_after_tx_compl = rx_after_complete;
Harald Weltea40c8e52019-09-27 19:22:34 +0200125 /* disable receiver to avoid receiving what we transmit */
126 card_uart_ctrl(cuart, CUART_CTL_RX, false);
127
Harald Welte02dd9112019-10-01 08:55:29 +0200128 return cuart->driver->ops->async_tx(cuart, data, len);
Harald Weltea40c8e52019-09-27 19:22:34 +0200129}
130
131int card_uart_rx(struct card_uart *cuart, uint8_t *data, size_t len)
132{
133 OSMO_ASSERT(cuart);
134 OSMO_ASSERT(cuart->driver);
135 OSMO_ASSERT(cuart->driver->ops);
136 OSMO_ASSERT(cuart->driver->ops->async_rx);
137 return cuart->driver->ops->async_rx(cuart, data, len);
138}
139
140void card_uart_set_rx_threshold(struct card_uart *cuart, size_t rx_threshold)
141{
142 cuart->rx_threshold = rx_threshold;
143}
144
145void card_uart_notification(struct card_uart *cuart, enum card_uart_event evt, void *data)
146{
147 OSMO_ASSERT(cuart);
148 OSMO_ASSERT(cuart->handle_event);
149
150 switch (evt) {
151 case CUART_E_TX_COMPLETE:
152 cuart->tx_busy = false;
153 /* re-enable receiver if we're done with transmit */
Harald Welte6603d952019-10-09 20:50:13 +0200154 if (cuart->rx_after_tx_compl)
155 card_uart_ctrl(cuart, CUART_CTL_RX, true);
Harald Weltea40c8e52019-09-27 19:22:34 +0200156 break;
157 default:
158 break;
159 }
160
161 cuart->handle_event(cuart, evt, data);
162}
163
164int card_uart_driver_register(struct card_uart_driver *drv)
165{
166 OSMO_ASSERT(!cuart_drv_by_name(drv->name));
167 OSMO_ASSERT(drv->name);
168 OSMO_ASSERT(drv->ops);
169 llist_add_tail(&drv->list, &g_cuart_drivers);
170 return 0;
171}