Merge tag 'xilinx-for-v2021.07-rc2' of https://source.denx.de/u-boot/custodians/u...
[platform/kernel/u-boot.git] / drivers / serial / serial_octeon_pcie_console.c
1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * Copyright (C) 2019 Marvell International Ltd.
4  * Copyright (C) 2021 Stefan Roese <sr@denx.de>
5  */
6
7 #include <dm.h>
8 #include <dm/uclass.h>
9 #include <errno.h>
10 #include <input.h>
11 #include <iomux.h>
12 #include <log.h>
13 #include <serial.h>
14 #include <stdio_dev.h>
15 #include <string.h>
16 #include <watchdog.h>
17 #include <linux/delay.h>
18 #include <asm/io.h>
19 #include <mach/cvmx-regs.h>
20 #include <mach/cvmx-bootmem.h>
21
22 #define DRIVER_NAME                             "pci-console"
23 #define OCTEONTX_PCIE_CONSOLE_NAME_LEN          16
24
25 /* Current versions */
26 #define OCTEON_PCIE_CONSOLE_MAJOR_VERSION       1
27 #define OCTEON_PCIE_CONSOLE_MINOR_VERSION       0
28
29 #define OCTEON_PCIE_CONSOLE_BLOCK_NAME          "__pci_console"
30
31 /*
32  * Structure that defines a single console.
33  * Note: when read_index == write_index, the buffer is empty.
34  * The actual usable size of each console is console_buf_size -1;
35  */
36 struct octeon_pcie_console {
37         u64 input_base_addr;
38         u32 input_read_index;
39         u32 input_write_index;
40         u64 output_base_addr;
41         u32 output_read_index;
42         u32 output_write_index;
43         u32 lock;
44         u32 buf_size;
45 };
46
47 /*
48  * This is the main container structure that contains all the information
49  * about all PCI consoles. The address of this structure is passed to various
50  * routines that operation on PCI consoles.
51  */
52 struct octeon_pcie_console_desc {
53         u32 major_version;
54         u32 minor_version;
55         u32 lock;
56         u32 flags;
57         u32 num_consoles;
58         u32 pad;
59         /* must be 64 bit aligned here... */
60         /* Array of addresses of octeon_pcie_console_t structures */
61         u64 console_addr_array[0];
62         /* Implicit storage for console_addr_array */
63 };
64
65 struct octeon_pcie_console_priv {
66         struct octeon_pcie_console *console;
67         int console_num;
68         bool console_active;
69 };
70
71 /* Flag definitions for read/write functions */
72 enum {
73         /*
74          * If set, read/write functions won't block waiting for space or data.
75          * For reads, 0 bytes may be read, and for writes not all of the
76          * supplied data may be written.
77          */
78         OCT_PCI_CON_FLAG_NONBLOCK = 1 << 0,
79 };
80
81 static int buffer_free_bytes(u32 buffer_size, u32 wr_idx, u32 rd_idx)
82 {
83         if (rd_idx >= buffer_size || wr_idx >= buffer_size)
84                 return -1;
85
86         return ((buffer_size - 1) - (wr_idx - rd_idx)) % buffer_size;
87 }
88
89 static int buffer_avail_bytes(u32 buffer_size, u32 wr_idx, u32 rd_idx)
90 {
91         if (rd_idx >= buffer_size || wr_idx >= buffer_size)
92                 return -1;
93
94         return buffer_size - 1 - buffer_free_bytes(buffer_size, wr_idx, rd_idx);
95 }
96
97 static int buffer_read_avail(struct udevice *dev, unsigned int console_num)
98 {
99         struct octeon_pcie_console_priv *priv = dev_get_priv(dev);
100         struct octeon_pcie_console *cons_ptr = priv->console;
101         int avail;
102
103         avail = buffer_avail_bytes(cons_ptr->buf_size,
104                                    cons_ptr->input_write_index,
105                                    cons_ptr->input_read_index);
106         if (avail >= 0)
107                 return avail;
108
109         return 0;
110 }
111
112 static int octeon_pcie_console_read(struct udevice *dev,
113                                     unsigned int console_num, char *buffer,
114                                     int buffer_size, u32 flags)
115 {
116         struct octeon_pcie_console_priv *priv = dev_get_priv(dev);
117         struct octeon_pcie_console *cons_ptr = priv->console;
118         int avail;
119         char *buf_ptr;
120         int bytes_read;
121         int read_size;
122
123         buf_ptr = (char *)cvmx_phys_to_ptr(cons_ptr->input_base_addr);
124
125         avail = buffer_avail_bytes(cons_ptr->buf_size,
126                                    cons_ptr->input_write_index,
127                                    cons_ptr->input_read_index);
128         if (avail < 0)
129                 return avail;
130
131         if (!(flags & OCT_PCI_CON_FLAG_NONBLOCK)) {
132                 /* Wait for some data to be available */
133                 while (0 == (avail = buffer_avail_bytes(cons_ptr->buf_size,
134                                                         cons_ptr->input_write_index,
135                                                         cons_ptr->input_read_index))) {
136                         mdelay(10);
137                         WATCHDOG_RESET();
138                 }
139         }
140
141         bytes_read = 0;
142
143         /* Don't overflow the buffer passed to us */
144         read_size = min_t(int, avail, buffer_size);
145
146         /* Limit ourselves to what we can input in a contiguous block */
147         if (cons_ptr->input_read_index + read_size >= cons_ptr->buf_size)
148                 read_size = cons_ptr->buf_size - cons_ptr->input_read_index;
149
150         memcpy(buffer, buf_ptr + cons_ptr->input_read_index, read_size);
151         cons_ptr->input_read_index =
152                 (cons_ptr->input_read_index + read_size) % cons_ptr->buf_size;
153         bytes_read += read_size;
154
155         /* Mark the PCIe console to be active from now on */
156         if (bytes_read)
157                 priv->console_active = true;
158
159         return bytes_read;
160 }
161
162 static int octeon_pcie_console_write(struct udevice *dev,
163                                      unsigned int console_num,
164                                      const char *buffer,
165                                      int bytes_to_write, u32 flags)
166 {
167         struct octeon_pcie_console_priv *priv = dev_get_priv(dev);
168         struct octeon_pcie_console *cons_ptr = priv->console;
169         int avail;
170         char *buf_ptr;
171         int bytes_written;
172
173         buf_ptr = (char *)cvmx_phys_to_ptr(cons_ptr->output_base_addr);
174         bytes_written = 0;
175         while (bytes_to_write > 0) {
176                 avail = buffer_free_bytes(cons_ptr->buf_size,
177                                           cons_ptr->output_write_index,
178                                           cons_ptr->output_read_index);
179
180                 if (avail > 0) {
181                         int write_size = min_t(int, avail, bytes_to_write);
182
183                         /*
184                          * Limit ourselves to what we can output in a contiguous
185                          * block
186                          */
187                         if (cons_ptr->output_write_index + write_size >=
188                             cons_ptr->buf_size) {
189                                 write_size = cons_ptr->buf_size -
190                                              cons_ptr->output_write_index;
191                         }
192
193                         memcpy(buf_ptr + cons_ptr->output_write_index,
194                                buffer + bytes_written, write_size);
195                         /*
196                          * Make sure data is visible before changing write
197                          * index
198                          */
199                         CVMX_SYNCW;
200                         cons_ptr->output_write_index =
201                                 (cons_ptr->output_write_index + write_size) %
202                                 cons_ptr->buf_size;
203                         bytes_to_write -= write_size;
204                         bytes_written += write_size;
205                 } else if (avail == 0) {
206                         /*
207                          * Check to see if we should wait for room, or return
208                          * after a partial write
209                          */
210                         if (flags & OCT_PCI_CON_FLAG_NONBLOCK)
211                                 goto done;
212
213                         WATCHDOG_RESET();
214                         mdelay(10);     /* Delay if we are spinning */
215                 } else {
216                         bytes_written = -1;
217                         goto done;
218                 }
219         }
220
221 done:
222         return bytes_written;
223 }
224
225 static struct octeon_pcie_console_desc *octeon_pcie_console_init(int num_consoles,
226                                                                  int buffer_size)
227 {
228         struct octeon_pcie_console_desc *cons_desc_ptr;
229         struct octeon_pcie_console *cons_ptr;
230         s64 addr;
231         u64 avail_addr;
232         int alloc_size;
233         int i;
234
235         /* Compute size required for pci console structure */
236         alloc_size = num_consoles *
237                 (buffer_size * 2 + sizeof(struct octeon_pcie_console) +
238                  sizeof(u64)) + sizeof(struct octeon_pcie_console_desc);
239
240         /*
241          * Allocate memory for the consoles.  This must be in the range
242          * addresssible by the bootloader.
243          * Try to do so in a manner which minimizes fragmentation.  We try to
244          * put it at the top of DDR0 or bottom of DDR2 first, and only do
245          * generic allocation if those fail
246          */
247         addr = cvmx_bootmem_phy_named_block_alloc(alloc_size,
248                                                   OCTEON_DDR0_SIZE - alloc_size - 128,
249                                                   OCTEON_DDR0_SIZE, 128,
250                                                   OCTEON_PCIE_CONSOLE_BLOCK_NAME,
251                                                   CVMX_BOOTMEM_FLAG_END_ALLOC);
252         if (addr < 0) {
253                 addr = cvmx_bootmem_phy_named_block_alloc(alloc_size, 0,
254                                                           0x1fffffff, 128,
255                                                           OCTEON_PCIE_CONSOLE_BLOCK_NAME,
256                                                           CVMX_BOOTMEM_FLAG_END_ALLOC);
257         }
258         if (addr < 0)
259                 return 0;
260
261         cons_desc_ptr = cvmx_phys_to_ptr(addr);
262
263         /* Clear entire alloc'ed memory */
264         memset(cons_desc_ptr, 0, alloc_size);
265
266         /* Initialize as locked until we are done */
267         cons_desc_ptr->lock = 1;
268         CVMX_SYNCW;
269         cons_desc_ptr->num_consoles = num_consoles;
270         cons_desc_ptr->flags = 0;
271         cons_desc_ptr->major_version = OCTEON_PCIE_CONSOLE_MAJOR_VERSION;
272         cons_desc_ptr->minor_version = OCTEON_PCIE_CONSOLE_MINOR_VERSION;
273
274         avail_addr = addr + sizeof(struct octeon_pcie_console_desc) +
275                 num_consoles * sizeof(u64);
276
277         for (i = 0; i < num_consoles; i++) {
278                 cons_desc_ptr->console_addr_array[i] = avail_addr;
279                 cons_ptr = (void *)cons_desc_ptr->console_addr_array[i];
280                 avail_addr += sizeof(struct octeon_pcie_console);
281                 cons_ptr->input_base_addr = avail_addr;
282                 avail_addr += buffer_size;
283                 cons_ptr->output_base_addr = avail_addr;
284                 avail_addr += buffer_size;
285                 cons_ptr->buf_size = buffer_size;
286         }
287         CVMX_SYNCW;
288         cons_desc_ptr->lock = 0;
289
290         return cvmx_phys_to_ptr(addr);
291 }
292
293 static int octeon_pcie_console_getc(struct udevice *dev)
294 {
295         char c;
296
297         octeon_pcie_console_read(dev, 0, &c, 1, 0);
298         return c;
299 }
300
301 static int octeon_pcie_console_putc(struct udevice *dev, const char c)
302 {
303         struct octeon_pcie_console_priv *priv = dev_get_priv(dev);
304
305         if (priv->console_active)
306                 octeon_pcie_console_write(dev, 0, (char *)&c, 1, 0);
307
308         return 0;
309 }
310
311 static int octeon_pcie_console_pending(struct udevice *dev, bool input)
312 {
313         if (input) {
314                 udelay(100);
315                 return buffer_read_avail(dev, 0) > 0;
316         }
317
318         return 0;
319 }
320
321 static const struct dm_serial_ops octeon_pcie_console_ops = {
322         .getc = octeon_pcie_console_getc,
323         .putc = octeon_pcie_console_putc,
324         .pending = octeon_pcie_console_pending,
325 };
326
327 static int octeon_pcie_console_probe(struct udevice *dev)
328 {
329         struct octeon_pcie_console_priv *priv = dev_get_priv(dev);
330         struct octeon_pcie_console_desc *cons_desc;
331         int console_count;
332         int console_size;
333         int console_num;
334
335         /*
336          * Currently only 1 console is supported. Perhaps we need to add
337          * a console nexus if more than one needs to be supported.
338          */
339         console_count = 1;
340         console_size = 1024;
341         console_num = 0;
342
343         cons_desc = octeon_pcie_console_init(console_count, console_size);
344         priv->console =
345                 cvmx_phys_to_ptr(cons_desc->console_addr_array[console_num]);
346
347         debug("PCI console init succeeded, %d consoles, %d bytes each\n",
348               console_count, console_size);
349
350         return 0;
351 }
352
353 static const struct udevice_id octeon_pcie_console_serial_id[] = {
354         { .compatible = "marvell,pci-console", },
355         { },
356 };
357
358 U_BOOT_DRIVER(octeon_pcie_console) = {
359         .name = DRIVER_NAME,
360         .id = UCLASS_SERIAL,
361         .ops = &octeon_pcie_console_ops,
362         .of_match = of_match_ptr(octeon_pcie_console_serial_id),
363         .probe = octeon_pcie_console_probe,
364         .priv_auto = sizeof(struct octeon_pcie_console_priv),
365 };