1 /*
2 * Copyright (c) 2021 EPAM Systems
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/arch/arm64/hypercall.h>
8 #include <zephyr/xen/console.h>
9 #include <zephyr/xen/events.h>
10 #include <zephyr/xen/generic.h>
11 #include <zephyr/xen/hvm.h>
12 #include <zephyr/xen/public/io/console.h>
13 #include <zephyr/xen/public/sched.h>
14 #include <zephyr/xen/public/xen.h>
15
16 #include <zephyr/device.h>
17 #include <zephyr/init.h>
18 #include <zephyr/kernel.h>
19 #include <zephyr/sys/device_mmio.h>
20 #include <zephyr/sys/printk-hooks.h>
21 #include <zephyr/sys/libc-hooks.h>
22
23 #include <zephyr/logging/log.h>
24 LOG_MODULE_REGISTER(uart_hvc_xen, CONFIG_UART_LOG_LEVEL);
25
26 static struct hvc_xen_data xen_hvc_data = {0};
27
28 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
29 static void hvc_uart_evtchn_cb(void *priv);
30 #endif /* CONFIG_UART_INTERRUPT_DRIVEN */
31
read_from_ring(const struct device * dev,char * str,int len)32 static int read_from_ring(const struct device *dev, char *str, int len)
33 {
34 int recv = 0;
35 struct hvc_xen_data *hvc_data = dev->data;
36 XENCONS_RING_IDX cons = hvc_data->intf->in_cons;
37 XENCONS_RING_IDX prod = hvc_data->intf->in_prod;
38 XENCONS_RING_IDX in_idx = 0;
39
40 compiler_barrier();
41 __ASSERT((prod - cons) <= sizeof(hvc_data->intf->in),
42 "Invalid input ring buffer");
43
44 while (cons != prod && recv < len) {
45 in_idx = MASK_XENCONS_IDX(cons, hvc_data->intf->in);
46 str[recv] = hvc_data->intf->in[in_idx];
47 recv++;
48 cons++;
49 }
50
51 compiler_barrier();
52 hvc_data->intf->in_cons = cons;
53
54 notify_evtchn(hvc_data->evtchn);
55 return recv;
56 }
57
write_to_ring(const struct device * dev,const char * str,int len)58 static int write_to_ring(const struct device *dev, const char *str, int len)
59 {
60 int sent = 0;
61 struct hvc_xen_data *hvc_data = dev->data;
62 XENCONS_RING_IDX cons = hvc_data->intf->out_cons;
63 XENCONS_RING_IDX prod = hvc_data->intf->out_prod;
64 XENCONS_RING_IDX out_idx = 0;
65
66 compiler_barrier();
67 __ASSERT((prod - cons) <= sizeof(hvc_data->intf->out),
68 "Invalid output ring buffer");
69
70 while ((sent < len) && ((prod - cons) < sizeof(hvc_data->intf->out))) {
71 out_idx = MASK_XENCONS_IDX(prod, hvc_data->intf->out);
72 hvc_data->intf->out[out_idx] = str[sent];
73 prod++;
74 sent++;
75 }
76
77 compiler_barrier();
78 hvc_data->intf->out_prod = prod;
79
80 if (sent) {
81 notify_evtchn(hvc_data->evtchn);
82 }
83
84 return sent;
85 }
86
xen_hvc_poll_in(const struct device * dev,unsigned char * c)87 static int xen_hvc_poll_in(const struct device *dev,
88 unsigned char *c)
89 {
90 int ret = 0;
91 char temp;
92
93 ret = read_from_ring(dev, &temp, sizeof(temp));
94 if (!ret) {
95 /* Char was not received */
96 return -1;
97 }
98
99 *c = temp;
100 return 0;
101 }
102
xen_hvc_poll_out(const struct device * dev,unsigned char c)103 static void xen_hvc_poll_out(const struct device *dev,
104 unsigned char c)
105 {
106 /* Not a good solution (notifying HV every time), but needed for poll_out */
107 (void) write_to_ring(dev, &c, sizeof(c));
108 }
109
110 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
xen_hvc_fifo_fill(const struct device * dev,const uint8_t * tx_data,int len)111 static int xen_hvc_fifo_fill(const struct device *dev, const uint8_t *tx_data,
112 int len)
113 {
114 int ret = 0, sent = 0;
115
116 while (len) {
117 sent = write_to_ring(dev, tx_data, len);
118
119 ret += sent;
120 tx_data += sent;
121 len -= sent;
122
123 if (len) {
124 /* Need to be able to read it from another domain */
125 HYPERVISOR_sched_op(SCHEDOP_yield, NULL);
126 }
127 }
128
129 return ret;
130 }
131
xen_hvc_fifo_read(const struct device * dev,uint8_t * rx_data,const int size)132 static int xen_hvc_fifo_read(const struct device *dev, uint8_t *rx_data,
133 const int size)
134 {
135 return read_from_ring(dev, rx_data, size);
136 }
137
xen_hvc_irq_tx_enable(const struct device * dev)138 static void xen_hvc_irq_tx_enable(const struct device *dev)
139 {
140 /*
141 * Need to explicitly call UART callback on TX enabling to
142 * process available buffered TX actions, because no HV events
143 * will be generated on tx_enable.
144 */
145 hvc_uart_evtchn_cb(dev->data);
146 }
147
xen_hvc_irq_tx_ready(const struct device * dev)148 static int xen_hvc_irq_tx_ready(const struct device *dev)
149 {
150 return 1;
151 }
152
xen_hvc_irq_rx_enable(const struct device * dev)153 static void xen_hvc_irq_rx_enable(const struct device *dev)
154 {
155 /*
156 * Need to explicitly call UART callback on RX enabling to
157 * process available buffered RX actions, because no HV events
158 * will be generated on rx_enable.
159 */
160 hvc_uart_evtchn_cb(dev->data);
161 }
162
xen_hvc_irq_tx_complete(const struct device * dev)163 static int xen_hvc_irq_tx_complete(const struct device *dev)
164 {
165 /*
166 * TX is performed by copying in ring buffer by fifo_fill,
167 * so it will be always completed.
168 */
169 return 1;
170 }
171
xen_hvc_irq_rx_ready(const struct device * dev)172 static int xen_hvc_irq_rx_ready(const struct device *dev)
173 {
174 struct hvc_xen_data *data = dev->data;
175
176 /* RX is ready only if data is available in ring buffer */
177 return (data->intf->in_prod != data->intf->in_cons);
178 }
179
xen_hvc_irq_is_pending(const struct device * dev)180 static int xen_hvc_irq_is_pending(const struct device *dev)
181 {
182 return xen_hvc_irq_rx_ready(dev);
183 }
184
xen_hvc_irq_update(const struct device * dev)185 static int xen_hvc_irq_update(const struct device *dev)
186 {
187 /* Nothing needs to be updated before actual ISR */
188 return 1;
189 }
190
xen_hvc_irq_callback_set(const struct device * dev,uart_irq_callback_user_data_t cb,void * user_data)191 static void xen_hvc_irq_callback_set(const struct device *dev,
192 uart_irq_callback_user_data_t cb, void *user_data)
193 {
194 struct hvc_xen_data *data = dev->data;
195
196 data->irq_cb = cb;
197 data->irq_cb_data = user_data;
198 }
199 #endif /* CONFIG_UART_INTERRUPT_DRIVEN */
200
201 static DEVICE_API(uart, xen_hvc_api) = {
202 .poll_in = xen_hvc_poll_in,
203 .poll_out = xen_hvc_poll_out,
204 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
205 .fifo_fill = xen_hvc_fifo_fill,
206 .fifo_read = xen_hvc_fifo_read,
207 .irq_tx_enable = xen_hvc_irq_tx_enable,
208 .irq_tx_ready = xen_hvc_irq_tx_ready,
209 .irq_rx_enable = xen_hvc_irq_rx_enable,
210 .irq_tx_complete = xen_hvc_irq_tx_complete,
211 .irq_rx_ready = xen_hvc_irq_rx_ready,
212 .irq_is_pending = xen_hvc_irq_is_pending,
213 .irq_update = xen_hvc_irq_update,
214 .irq_callback_set = xen_hvc_irq_callback_set,
215 #endif /* CONFIG_UART_INTERRUPT_DRIVEN */
216 };
217
218 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
hvc_uart_evtchn_cb(void * priv)219 static void hvc_uart_evtchn_cb(void *priv)
220 {
221 struct hvc_xen_data *data = priv;
222
223 if (data->irq_cb) {
224 data->irq_cb(data->dev, data->irq_cb_data);
225 }
226 }
227 #endif /* CONFIG_UART_INTERRUPT_DRIVEN */
228
xen_console_init(const struct device * dev)229 int xen_console_init(const struct device *dev)
230 {
231 int ret = 0;
232 uint64_t console_pfn = 0;
233 uintptr_t console_addr = 0;
234 struct hvc_xen_data *data = dev->data;
235
236 data->dev = dev;
237
238 ret = hvm_get_parameter(HVM_PARAM_CONSOLE_EVTCHN, DOMID_SELF, &data->evtchn);
239 if (ret) {
240 LOG_ERR("%s: failed to get Xen console evtchn, ret = %d\n",
241 __func__, ret);
242 return ret;
243 }
244
245 ret = hvm_get_parameter(HVM_PARAM_CONSOLE_PFN, DOMID_SELF, &console_pfn);
246 if (ret) {
247 LOG_ERR("%s: failed to get Xen console PFN, ret = %d\n",
248 __func__, ret);
249 return ret;
250 }
251
252 console_addr = (uintptr_t) (console_pfn << XEN_PAGE_SHIFT);
253 device_map(DEVICE_MMIO_RAM_PTR(dev), console_addr, XEN_PAGE_SIZE,
254 K_MEM_CACHE_WB);
255
256 data->intf = (struct xencons_interface *) DEVICE_MMIO_GET(dev);
257
258 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
259 bind_event_channel(data->evtchn, hvc_uart_evtchn_cb, data);
260 #endif /* CONFIG_UART_INTERRUPT_DRIVEN */
261
262 LOG_INF("Xen HVC inited successfully\n");
263
264 return 0;
265 }
266
267 DEVICE_DT_DEFINE(DT_NODELABEL(xen_hvc), xen_console_init, NULL, &xen_hvc_data,
268 NULL, PRE_KERNEL_1, CONFIG_XEN_HVC_INIT_PRIORITY,
269 &xen_hvc_api);
270
271 #ifdef CONFIG_XEN_EARLY_CONSOLEIO
xen_consoleio_putc(int c)272 int xen_consoleio_putc(int c)
273 {
274 char symbol = (char) c;
275
276 HYPERVISOR_console_io(CONSOLEIO_write, sizeof(symbol), &symbol);
277 return c;
278 }
279
280
281
consoleio_hooks_set(void)282 int consoleio_hooks_set(void)
283 {
284
285 /* Will be replaced with poll_in/poll_out by uart_console.c later on boot */
286 __stdout_hook_install(xen_consoleio_putc);
287 __printk_hook_install(xen_consoleio_putc);
288
289 return 0;
290 }
291
292 SYS_INIT(consoleio_hooks_set, PRE_KERNEL_1, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
293 #endif /* CONFIG_XEN_EARLY_CONSOLEIO */
294