blob: 1d777f9007507495a1b8d85206b537c0bba6ace3 [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{
Eric Wilddbff4212019-11-27 19:02:31 +010032 OSMO_ASSERT(cuart);
33 OSMO_ASSERT(cuart->driver);
34 OSMO_ASSERT(cuart->driver->ops);
35 OSMO_ASSERT(cuart->driver->ops->ctrl);
36
37 return 1e6 / cuart->driver->ops->ctrl(cuart, CUART_CTL_GET_BAUDRATE, 0);
Harald Welte7b64fc02019-10-09 20:49:35 +020038}
39
40/* software waiting-time timer has expired */
41static void card_uart_wtime_cb(void *data)
42{
43 struct card_uart *cuart = (struct card_uart *) data;
44 card_uart_notification(cuart, CUART_E_RX_TIMEOUT, NULL);
45 /* should we automatically disable the receiver? */
46}
47
48void card_uart_wtime_restart(struct card_uart *cuart)
49{
50 int secs, usecs;
51
Eric Wild2f5cdd12019-11-27 18:40:44 +010052 if(!cuart->current_wtime_byte)
53 return;
54
55 /* timemout is wtime * ETU + expected number of bytes * (12ETU+1 slack)ETU */
56 usecs = get_etu_in_us(cuart) * cuart->wtime_etu +
57 get_etu_in_us(cuart) * cuart->current_wtime_byte * (12+1);
Harald Welte7b64fc02019-10-09 20:49:35 +020058 if (usecs > 1000000) {
59 secs = usecs / 1000000;
60 usecs = usecs % 1000000;
61 } else
62 secs = 0;
63 osmo_timer_schedule(&cuart->wtime_tmr, secs, usecs);
64}
65
Harald Weltea40c8e52019-09-27 19:22:34 +020066int card_uart_open(struct card_uart *cuart, const char *driver_name, const char *device_name)
67{
68 struct card_uart_driver *drv = cuart_drv_by_name(driver_name);
69 int rc;
70
71 if (!drv)
72 return -ENODEV;
73
Harald Welte7b64fc02019-10-09 20:49:35 +020074 cuart->wtime_etu = 9600; /* ISO 7816-3 Section 8.1 */
Harald Weltea40c8e52019-09-27 19:22:34 +020075 cuart->rx_enabled = true;
76 cuart->rx_threshold = 1;
Harald Welte7b64fc02019-10-09 20:49:35 +020077 osmo_timer_setup(&cuart->wtime_tmr, card_uart_wtime_cb, cuart);
Harald Weltea40c8e52019-09-27 19:22:34 +020078
79 rc = drv->ops->open(cuart, device_name);
80 if (rc < 0)
81 return rc;
82
83 cuart->driver = drv;
84 return 0;
85}
86
87int card_uart_close(struct card_uart *cuart)
88{
89 OSMO_ASSERT(cuart);
90 OSMO_ASSERT(cuart->driver);
91 OSMO_ASSERT(cuart->driver->ops);
92 OSMO_ASSERT(cuart->driver->ops->close);
93 return cuart->driver->ops->close(cuart);
94}
95
Harald Welte9700d392019-10-09 20:17:28 +020096int card_uart_ctrl(struct card_uart *cuart, enum card_uart_ctl ctl, int arg)
Harald Weltea40c8e52019-09-27 19:22:34 +020097{
98 int rc;
99 OSMO_ASSERT(cuart);
100 OSMO_ASSERT(cuart->driver);
101 OSMO_ASSERT(cuart->driver->ops);
102 OSMO_ASSERT(cuart->driver->ops->ctrl);
Harald Welte7b64fc02019-10-09 20:49:35 +0200103
Harald Welte9700d392019-10-09 20:17:28 +0200104 rc = cuart->driver->ops->ctrl(cuart, ctl, arg);
Harald Weltea40c8e52019-09-27 19:22:34 +0200105 if (rc < 0)
106 return rc;
107
108 switch (ctl) {
Harald Welte7b64fc02019-10-09 20:49:35 +0200109 case CUART_CTL_WTIME:
110 cuart->wtime_etu = arg;
111 break;
Harald Weltea40c8e52019-09-27 19:22:34 +0200112 case CUART_CTL_RX:
Harald Welte9700d392019-10-09 20:17:28 +0200113 cuart->rx_enabled = arg ? true : false;
Harald Welte7b64fc02019-10-09 20:49:35 +0200114 if (!cuart->rx_enabled)
115 osmo_timer_del(&cuart->wtime_tmr);
Eric Wild2f5cdd12019-11-27 18:40:44 +0100116// else
117// card_uart_wtime_restart(cuart);
118 break;
119 case CUART_CTL_RX_TIMER_HINT:
120 cuart->current_wtime_byte = arg;
121 if(arg)
122 card_uart_wtime_restart(cuart);
123 else
124 osmo_timer_del(&cuart->wtime_tmr);
Harald Weltea40c8e52019-09-27 19:22:34 +0200125 break;
126 default:
127 break;
128 }
129
130 return rc;
131}
132
Harald Welte6603d952019-10-09 20:50:13 +0200133int 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 +0200134{
135 OSMO_ASSERT(cuart);
136 OSMO_ASSERT(cuart->driver);
137 OSMO_ASSERT(cuart->driver->ops);
138 OSMO_ASSERT(cuart->driver->ops->async_tx);
139
140 OSMO_ASSERT(!cuart->tx_busy);
141 cuart->tx_busy = true;
Harald Welte6603d952019-10-09 20:50:13 +0200142 cuart->rx_after_tx_compl = rx_after_complete;
Harald Weltea40c8e52019-09-27 19:22:34 +0200143 /* disable receiver to avoid receiving what we transmit */
144 card_uart_ctrl(cuart, CUART_CTL_RX, false);
145
Harald Welte02dd9112019-10-01 08:55:29 +0200146 return cuart->driver->ops->async_tx(cuart, data, len);
Harald Weltea40c8e52019-09-27 19:22:34 +0200147}
148
149int card_uart_rx(struct card_uart *cuart, uint8_t *data, size_t len)
150{
151 OSMO_ASSERT(cuart);
152 OSMO_ASSERT(cuart->driver);
153 OSMO_ASSERT(cuart->driver->ops);
154 OSMO_ASSERT(cuart->driver->ops->async_rx);
155 return cuart->driver->ops->async_rx(cuart, data, len);
156}
157
158void card_uart_set_rx_threshold(struct card_uart *cuart, size_t rx_threshold)
159{
160 cuart->rx_threshold = rx_threshold;
161}
162
163void card_uart_notification(struct card_uart *cuart, enum card_uart_event evt, void *data)
164{
165 OSMO_ASSERT(cuart);
166 OSMO_ASSERT(cuart->handle_event);
167
168 switch (evt) {
169 case CUART_E_TX_COMPLETE:
170 cuart->tx_busy = false;
171 /* re-enable receiver if we're done with transmit */
Harald Welte6603d952019-10-09 20:50:13 +0200172 if (cuart->rx_after_tx_compl)
173 card_uart_ctrl(cuart, CUART_CTL_RX, true);
Harald Weltea40c8e52019-09-27 19:22:34 +0200174 break;
Eric Wild2f5cdd12019-11-27 18:40:44 +0100175// case CUART_E_RX_COMPLETE:
176// osmo_timer_del(&cuart->wtime_tmr);
177// break;
178// case CUART_E_RX_SINGLE:
179// card_uart_wtime_restart(cuart);
180// break;
Harald Weltea40c8e52019-09-27 19:22:34 +0200181 default:
182 break;
183 }
184
185 cuart->handle_event(cuart, evt, data);
186}
187
188int card_uart_driver_register(struct card_uart_driver *drv)
189{
190 OSMO_ASSERT(!cuart_drv_by_name(drv->name));
191 OSMO_ASSERT(drv->name);
192 OSMO_ASSERT(drv->ops);
193 llist_add_tail(&drv->list, &g_cuart_drivers);
194 return 0;
195}