blob: 52fb53b660f50b743e02c0a62b3967478c83537e [file] [log] [blame]
Sylvain Munautbc9f5c42020-09-14 10:22:29 +02001/*
2 * usb_e1.c
3 *
4 * Copyright (C) 2019-2020 Sylvain Munaut <tnt@246tNt.com>
5 * SPDX-License-Identifier: GPL-3.0-or-later
6 */
7
8#include <stdint.h>
9#include <stdbool.h>
10#include <string.h>
11
12#include <no2usb/usb_hw.h>
13#include <no2usb/usb_priv.h>
14
15#include "console.h"
16#include "misc.h"
17
18struct {
19 bool running;
20 int out_bdi;
21 int in_bdi;
22} g_usb_e1;
23
24
25/* Hack */
26unsigned int e1_rx_need_data(unsigned int usb_addr, unsigned int max_len);
27unsigned int e1_tx_feed_data(unsigned int usb_addr, unsigned int len);
28unsigned int e1_tx_level(void);
29unsigned int e1_rx_level(void);
30/* ---- */
31
32bool
33usb_ep_boot(const struct usb_intf_desc *intf, uint8_t ep_addr, bool dual_bd);
34
35static void
36_usb_fill_feedback_ep(void)
37{
38 static uint16_t ticks_prev = 0;
39 uint16_t ticks;
40 uint32_t val = 8192;
41 unsigned int level;
42
43 /* Compute real E1 tick count (with safety agains bad values) */
44 ticks = e1_tick_read();
45 val = (ticks - ticks_prev) & 0xffff;
46 ticks_prev = ticks;
47 if ((val < 7168) | (val > 9216))
48 val = 8192;
49
50 /* Bias depending on TX fifo level */
51 level = e1_tx_level();
52 if (level < (3 * 16))
53 val += 256;
54 else if (level > (8 * 16))
55 val -= 256;
56
57 /* Prepare buffer */
58 usb_data_write(64, &val, 4);
59 usb_ep_regs[1].in.bd[0].ptr = 64;
60 usb_ep_regs[1].in.bd[0].csr = USB_BD_STATE_RDY_DATA | USB_BD_LEN(3);
61}
62
63
64void
65usb_e1_run(void)
66{
67 int bdi;
68
69 if (!g_usb_e1.running)
70 return;
71
72 /* EP2 IN */
73 bdi = g_usb_e1.in_bdi;
74
75 while ((usb_ep_regs[2].in.bd[bdi].csr & USB_BD_STATE_MSK) != USB_BD_STATE_RDY_DATA)
76 {
77 uint32_t ptr = usb_ep_regs[2].in.bd[bdi].ptr;
78 uint32_t hdr;
79
80 /* Error check */
81 if ((usb_ep_regs[2].in.bd[bdi].csr & USB_BD_STATE_MSK) == USB_BD_STATE_DONE_ERR)
82 puts("Err EP2 IN\n");
83
84 /* Get some data from E1 */
85 int n = e1_rx_level();
86
87 if (n > 64)
88 n = 12;
89 else if (n > 32)
90 n = 10;
91 else if (n > 8)
92 n = 8;
93 else if (!n)
94 break;
95
96 n = e1_rx_need_data((ptr >> 2) + 1, n);
97
98 /* Write header */
99 hdr = 0x616b00b5;
100 usb_data_write(ptr, &hdr, 4);
101
102 /* Resubmit */
103 usb_ep_regs[2].in.bd[bdi].csr = USB_BD_STATE_RDY_DATA | USB_BD_LEN((n * 32) + 4);
104
105 /* Next BDI */
106 bdi ^= 1;
107 g_usb_e1.in_bdi = bdi;
108 }
109
110 /* EP1 OUT */
111 bdi = g_usb_e1.out_bdi;
112
113 while ((usb_ep_regs[1].out.bd[bdi].csr & USB_BD_STATE_MSK) != USB_BD_STATE_RDY_DATA)
114 {
115 uint32_t ptr = usb_ep_regs[1].out.bd[bdi].ptr;
116 uint32_t csr = usb_ep_regs[1].out.bd[bdi].csr;
117 uint32_t hdr;
118
119 /* Error check */
120 if ((csr & USB_BD_STATE_MSK) == USB_BD_STATE_DONE_ERR) {
121 puts("Err EP1 OUT\n");
122 goto refill;
123 }
124
125 /* Grab header */
126 usb_data_read(&hdr, ptr, 4);
127
128 /* Empty data into the FIFO */
129 int n = ((csr & USB_BD_LEN_MSK) - 4) / 32;
130 n = e1_tx_feed_data((ptr >> 2) + 1, n);
131
132refill:
133 /* Refill it */
134 usb_ep_regs[1].out.bd[bdi].csr = USB_BD_STATE_RDY_DATA | USB_BD_LEN(388);
135
136 /* Next BDI */
137 bdi ^= 1;
138 g_usb_e1.out_bdi = bdi;
139
140 static int x = 0;
141 if ((x++ & 0xff) == 0xff)
142 puts(".");
143 }
144
145 /* EP1 IN */
146 if ((usb_ep_regs[1].in.bd[0].csr & USB_BD_STATE_MSK) != USB_BD_STATE_RDY_DATA)
147 {
148 _usb_fill_feedback_ep();
149 }
150}
151
152static const struct usb_intf_desc *
153_find_intf(const struct usb_conf_desc *conf, uint8_t idx)
154{
155 const struct usb_intf_desc *intf = NULL;
156 const void *sod, *eod;
157
158 if (!conf)
159 return NULL;
160
161 sod = conf;
162 eod = sod + conf->wTotalLength;
163
164 while (1) {
165 sod = usb_desc_find(sod, eod, USB_DT_INTF);
166 if (!sod)
167 break;
168
169 intf = (void*)sod;
170 if (intf->bInterfaceNumber == idx)
171 return intf;
172
173 sod = usb_desc_next(sod);
174 }
175
176 return NULL;
177}
178enum usb_fnd_resp
179_e1_set_conf(const struct usb_conf_desc *conf)
180{
181 const struct usb_intf_desc *intf;
182
183 printf("e1 set_conf %08x\n", conf);
184 if (!conf)
185 return USB_FND_SUCCESS;
186
187 intf = _find_intf(conf, 0);
188 if (!intf)
189 return USB_FND_ERROR;
190
191 printf("e1 set_conf %08x\n", intf);
192
193 usb_ep_boot(intf, 0x01, true);
194 usb_ep_boot(intf, 0x81, true);
195 usb_ep_boot(intf, 0x82, true);
196
197 return USB_FND_SUCCESS;
198}
199
200enum usb_fnd_resp
201_e1_set_intf(const struct usb_intf_desc *base, const struct usb_intf_desc *sel)
202{
203 if (base->bInterfaceNumber != 0)
204 return USB_FND_CONTINUE;
205
206 if (sel->bAlternateSetting != 1)
207 return USB_FND_SUCCESS;
208
209 /* Hack to avoid re-setting while running ... avoid BD desync */
210 if (g_usb_e1.running)
211 return USB_FND_SUCCESS;
212
213 g_usb_e1.running = true;
214
215 /* Configure EP1 OUT / EP2 IN */
216 usb_ep_regs[1].out.status = USB_EP_TYPE_ISOC | USB_EP_BD_DUAL; /* Type=Isochronous, dual buffered */
217 usb_ep_regs[2].in.status = USB_EP_TYPE_ISOC | USB_EP_BD_DUAL; /* Type=Isochronous, dual buffered */
218
219 /* Configure EP1 IN (feedback) */
220 usb_ep_regs[1].in.status = USB_EP_TYPE_ISOC; /* Type=Isochronous, single buffered */
221
222 /* EP2 IN: Prepare two buffers */
223 usb_ep_regs[2].in.bd[0].ptr = 1024;
224 usb_ep_regs[2].in.bd[0].csr = 0;
225
226 usb_ep_regs[2].in.bd[1].ptr = 1536;
227 usb_ep_regs[2].in.bd[1].csr = 0;
228
229 /* EP1 OUT: Queue two buffers */
230 usb_ep_regs[1].out.bd[0].ptr = 1024;
231 usb_ep_regs[1].out.bd[0].csr = USB_BD_STATE_RDY_DATA | USB_BD_LEN(388);
232
233 usb_ep_regs[1].out.bd[1].ptr = 1536;
234 usb_ep_regs[1].out.bd[1].csr = USB_BD_STATE_RDY_DATA | USB_BD_LEN(388);
235
236 /* EP1 IN: Queue buffer */
237 _usb_fill_feedback_ep();
238
239 return USB_FND_SUCCESS;
240}
241
242enum usb_fnd_resp
243_e1_get_intf(const struct usb_intf_desc *base, uint8_t *alt)
244{
245 if (base->bInterfaceNumber != 0)
246 return USB_FND_CONTINUE;
247
248 *alt = g_usb_e1.running ? 1 : 0;
249
250 return USB_FND_SUCCESS;
251}
252
253static struct usb_fn_drv _e1_drv = {
254 .set_conf = _e1_set_conf,
255 .set_intf = _e1_set_intf,
256 .get_intf = _e1_get_intf,
257};
258
259void
260usb_e1_init(void)
261{
262 memset(&g_usb_e1, 0x00, sizeof(g_usb_e1));
263 usb_register_function_driver(&_e1_drv);
264}