blob: e726e19c46f7bd288968da8d58d91a77117786a6 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Simon Glass7accb6e2011-10-03 19:26:46 +00002/*
3 * Copyright (c) 2011 The Chromium OS Authors.
Simon Glass7accb6e2011-10-03 19:26:46 +00004 */
5
6/*
7 * This provide a test serial port. It provides an emulated serial port where
8 * a test program and read out the serial output and inject serial input for
9 * U-Boot.
10 */
11
12#include <common.h>
Joe Hershberger82a115f2018-07-02 20:06:49 -050013#include <console.h>
Simon Glass890fcef2014-09-04 16:27:27 -060014#include <dm.h>
Simon Glass7d95f2a2014-02-27 13:26:19 -070015#include <lcd.h>
Simon Glass7accb6e2011-10-03 19:26:46 +000016#include <os.h>
Marek Vasutcef46b72012-09-14 22:33:21 +020017#include <serial.h>
Simon Glass3ade5bc2016-01-18 19:52:25 -070018#include <video.h>
Simon Glass401d1c42020-10-30 21:38:53 -060019#include <asm/global_data.h>
Marek Vasutcef46b72012-09-14 22:33:21 +020020#include <linux/compiler.h>
Simon Glassbfae6cc2020-12-19 10:39:53 -070021#include <asm/serial.h>
Simon Glassffb87902014-02-27 13:26:22 -070022#include <asm/state.h>
Simon Glass7accb6e2011-10-03 19:26:46 +000023
Simon Glass890fcef2014-09-04 16:27:27 -060024DECLARE_GLOBAL_DATA_PTR;
25
Sean Anderson679190c2022-04-04 14:17:59 -040026static size_t _sandbox_serial_written = 1;
27static bool sandbox_serial_enabled = true;
28
29size_t sandbox_serial_written(void)
30{
31 return _sandbox_serial_written;
32}
33
34void sandbox_serial_endisable(bool enabled)
35{
36 sandbox_serial_enabled = enabled;
37}
38
Simon Glass72e98222014-09-04 16:27:28 -060039/**
40 * output_ansi_colour() - Output an ANSI colour code
41 *
42 * @colour: Colour to output (0-7)
43 */
44static void output_ansi_colour(int colour)
45{
46 char ansi_code[] = "\x1b[1;3Xm";
47
48 ansi_code[5] = '0' + colour;
49 os_write(1, ansi_code, sizeof(ansi_code) - 1);
50}
51
52static void output_ansi_reset(void)
53{
54 os_write(1, "\x1b[0m", 4);
55}
56
Simon Glass890fcef2014-09-04 16:27:27 -060057static int sandbox_serial_probe(struct udevice *dev)
Simon Glass7accb6e2011-10-03 19:26:46 +000058{
Simon Glassffb87902014-02-27 13:26:22 -070059 struct sandbox_state *state = state_get_current();
Simon Glass72e98222014-09-04 16:27:28 -060060 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Simon Glassffb87902014-02-27 13:26:22 -070061
62 if (state->term_raw != STATE_TERM_COOKED)
63 os_tty_raw(0, state->term_raw == STATE_TERM_RAW_WITH_SIGS);
Simon Glass72e98222014-09-04 16:27:28 -060064 priv->start_of_line = 0;
65
Joe Hershberger82a115f2018-07-02 20:06:49 -050066 if (state->term_raw != STATE_TERM_RAW)
67 disable_ctrlc(1);
Simon Glass21f90752020-11-08 20:36:50 -070068 membuff_init(&priv->buf, priv->serial_buf, sizeof(priv->serial_buf));
Joe Hershberger82a115f2018-07-02 20:06:49 -050069
Simon Glass72e98222014-09-04 16:27:28 -060070 return 0;
71}
72
73static int sandbox_serial_remove(struct udevice *dev)
74{
Simon Glass0fd3d912020-12-22 19:30:28 -070075 struct sandbox_serial_plat *plat = dev_get_plat(dev);
Simon Glass72e98222014-09-04 16:27:28 -060076
77 if (plat->colour != -1)
78 output_ansi_reset();
Simon Glass890fcef2014-09-04 16:27:27 -060079
Simon Glass7accb6e2011-10-03 19:26:46 +000080 return 0;
81}
82
Sean Andersonefa51f22022-04-04 14:17:58 -040083static void sandbox_print_color(struct udevice *dev)
Simon Glass7accb6e2011-10-03 19:26:46 +000084{
Simon Glass72e98222014-09-04 16:27:28 -060085 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Simon Glass0fd3d912020-12-22 19:30:28 -070086 struct sandbox_serial_plat *plat = dev_get_plat(dev);
Simon Glass72e98222014-09-04 16:27:28 -060087
Walter Lozanoe3e24702020-06-25 01:10:04 -030088 /* With of-platdata we don't real the colour correctly, so disable it */
89 if (!CONFIG_IS_ENABLED(OF_PLATDATA) && priv->start_of_line &&
90 plat->colour != -1) {
Simon Glass72e98222014-09-04 16:27:28 -060091 priv->start_of_line = false;
92 output_ansi_colour(plat->colour);
93 }
Sean Andersonefa51f22022-04-04 14:17:58 -040094}
Simon Glass72e98222014-09-04 16:27:28 -060095
Sean Andersonefa51f22022-04-04 14:17:58 -040096static int sandbox_serial_putc(struct udevice *dev, const char ch)
97{
98 struct sandbox_serial_priv *priv = dev_get_priv(dev);
99
Simon Glass72e98222014-09-04 16:27:28 -0600100 if (ch == '\n')
101 priv->start_of_line = true;
Simon Glass7accb6e2011-10-03 19:26:46 +0000102
Sean Anderson679190c2022-04-04 14:17:59 -0400103 if (sandbox_serial_enabled) {
104 sandbox_print_color(dev);
105 os_write(1, &ch, 1);
106 }
107 _sandbox_serial_written += 1;
Simon Glass890fcef2014-09-04 16:27:27 -0600108 return 0;
Simon Glass7accb6e2011-10-03 19:26:46 +0000109}
110
Sean Andersonefa51f22022-04-04 14:17:58 -0400111static ssize_t sandbox_serial_puts(struct udevice *dev, const char *s,
112 size_t len)
113{
114 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Sean Anderson679190c2022-04-04 14:17:59 -0400115 ssize_t ret;
Sean Andersonefa51f22022-04-04 14:17:58 -0400116
Sean Andersonefa51f22022-04-04 14:17:58 -0400117 if (s[len - 1] == '\n')
118 priv->start_of_line = true;
119
Sean Anderson679190c2022-04-04 14:17:59 -0400120 if (sandbox_serial_enabled) {
121 sandbox_print_color(dev);
122 ret = os_write(1, s, len);
123 if (ret < 0)
124 return ret;
125 } else {
126 ret = len;
127 }
128 _sandbox_serial_written += ret;
129 return ret;
Sean Andersonefa51f22022-04-04 14:17:58 -0400130}
131
Simon Glass890fcef2014-09-04 16:27:27 -0600132static int sandbox_serial_pending(struct udevice *dev, bool input)
Simon Glass7accb6e2011-10-03 19:26:46 +0000133{
Simon Glass21f90752020-11-08 20:36:50 -0700134 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Taylor Hutte1015502013-02-24 17:33:13 +0000135 ssize_t count;
Simon Glass21f90752020-11-08 20:36:50 -0700136 char *data;
137 int avail;
Taylor Hutte1015502013-02-24 17:33:13 +0000138
Simon Glass890fcef2014-09-04 16:27:27 -0600139 if (!input)
140 return 0;
141
Taylor Hutte1015502013-02-24 17:33:13 +0000142 os_usleep(100);
Heinrich Schuchardtce518842022-01-21 16:07:30 +0100143 if (IS_ENABLED(CONFIG_DM_VIDEO) && !IS_ENABLED(CONFIG_SPL_BUILD))
Simon Glassc5ea0162020-11-08 20:36:48 -0700144 video_sync_all();
Simon Glass21f90752020-11-08 20:36:50 -0700145 avail = membuff_putraw(&priv->buf, 100, false, &data);
146 if (!avail)
Taylor Hutte1015502013-02-24 17:33:13 +0000147 return 1; /* buffer full */
148
Simon Glass21f90752020-11-08 20:36:50 -0700149 count = os_read(0, data, avail);
150 if (count > 0)
151 membuff_putraw(&priv->buf, count, true, &data);
Simon Glass890fcef2014-09-04 16:27:27 -0600152
Simon Glass21f90752020-11-08 20:36:50 -0700153 return membuff_avail(&priv->buf);
Taylor Hutte1015502013-02-24 17:33:13 +0000154}
155
Simon Glass890fcef2014-09-04 16:27:27 -0600156static int sandbox_serial_getc(struct udevice *dev)
Taylor Hutte1015502013-02-24 17:33:13 +0000157{
Simon Glass21f90752020-11-08 20:36:50 -0700158 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Taylor Hutte1015502013-02-24 17:33:13 +0000159
Simon Glass890fcef2014-09-04 16:27:27 -0600160 if (!sandbox_serial_pending(dev, true))
161 return -EAGAIN; /* buffer empty */
Taylor Hutte1015502013-02-24 17:33:13 +0000162
Simon Glass21f90752020-11-08 20:36:50 -0700163 return membuff_getbyte(&priv->buf);
Simon Glass7accb6e2011-10-03 19:26:46 +0000164}
Marek Vasutcef46b72012-09-14 22:33:21 +0200165
Simon Glassee441762018-10-01 11:55:15 -0600166#ifdef CONFIG_DEBUG_UART_SANDBOX
167
168#include <debug_uart.h>
169
170static inline void _debug_uart_init(void)
171{
172}
173
174static inline void _debug_uart_putc(int ch)
175{
176 os_putc(ch);
177}
178
179DEBUG_UART_FUNCS
180
181#endif /* CONFIG_DEBUG_UART_SANDBOX */
182
Andy Shevchenkoac7f5db2018-11-20 23:52:32 +0200183static int sandbox_serial_getconfig(struct udevice *dev, uint *serial_config)
184{
185 uint config = SERIAL_DEFAULT_CONFIG;
186
187 if (!serial_config)
188 return -EINVAL;
189
190 *serial_config = config;
191
192 return 0;
193}
194
Patrice Chotardd7c09682018-08-03 15:07:41 +0200195static int sandbox_serial_setconfig(struct udevice *dev, uint serial_config)
196{
197 u8 parity = SERIAL_GET_PARITY(serial_config);
198 u8 bits = SERIAL_GET_BITS(serial_config);
199 u8 stop = SERIAL_GET_STOP(serial_config);
200
201 if (bits != SERIAL_8_BITS || stop != SERIAL_ONE_STOP ||
202 parity != SERIAL_PAR_NONE)
203 return -ENOTSUPP; /* not supported in driver*/
204
205 return 0;
206}
207
Andy Shevchenkod5bb4f82018-11-20 23:52:33 +0200208static int sandbox_serial_getinfo(struct udevice *dev,
209 struct serial_device_info *serial_info)
210{
211 struct serial_device_info info = {
212 .type = SERIAL_CHIP_UNKNOWN,
213 .addr_space = SERIAL_ADDRESS_SPACE_IO,
214 .addr = SERIAL_DEFAULT_ADDRESS,
215 .reg_width = 1,
216 .reg_offset = 0,
217 .reg_shift = 0,
Andy Shevchenkobf4661bc2020-02-27 17:21:54 +0200218 .clock = SERIAL_DEFAULT_CLOCK,
Andy Shevchenkod5bb4f82018-11-20 23:52:33 +0200219 };
220
221 if (!serial_info)
222 return -EINVAL;
223
224 *serial_info = info;
225
226 return 0;
227}
228
Simon Glass72e98222014-09-04 16:27:28 -0600229static const char * const ansi_colour[] = {
230 "black", "red", "green", "yellow", "blue", "megenta", "cyan",
231 "white",
232};
233
Simon Glassd1998a92020-12-03 16:55:21 -0700234static int sandbox_serial_of_to_plat(struct udevice *dev)
Simon Glass72e98222014-09-04 16:27:28 -0600235{
Simon Glass0fd3d912020-12-22 19:30:28 -0700236 struct sandbox_serial_plat *plat = dev_get_plat(dev);
Simon Glass72e98222014-09-04 16:27:28 -0600237 const char *colour;
238 int i;
239
Simon Glassea147782019-09-25 08:55:53 -0600240 if (CONFIG_IS_ENABLED(OF_PLATDATA))
241 return 0;
Simon Glass72e98222014-09-04 16:27:28 -0600242 plat->colour = -1;
Simon Glass1bf0a402020-11-08 20:36:49 -0700243 colour = dev_read_string(dev, "sandbox,text-colour");
Simon Glass72e98222014-09-04 16:27:28 -0600244 if (colour) {
245 for (i = 0; i < ARRAY_SIZE(ansi_colour); i++) {
246 if (!strcmp(colour, ansi_colour[i])) {
247 plat->colour = i;
248 break;
249 }
250 }
251 }
252
253 return 0;
254}
255
Simon Glass890fcef2014-09-04 16:27:27 -0600256static const struct dm_serial_ops sandbox_serial_ops = {
257 .putc = sandbox_serial_putc,
Sean Andersonefa51f22022-04-04 14:17:58 -0400258 .puts = sandbox_serial_puts,
Simon Glass890fcef2014-09-04 16:27:27 -0600259 .pending = sandbox_serial_pending,
260 .getc = sandbox_serial_getc,
Andy Shevchenkoac7f5db2018-11-20 23:52:32 +0200261 .getconfig = sandbox_serial_getconfig,
Patrice Chotardd7c09682018-08-03 15:07:41 +0200262 .setconfig = sandbox_serial_setconfig,
Andy Shevchenkod5bb4f82018-11-20 23:52:33 +0200263 .getinfo = sandbox_serial_getinfo,
Marek Vasutcef46b72012-09-14 22:33:21 +0200264};
265
Simon Glass890fcef2014-09-04 16:27:27 -0600266static const struct udevice_id sandbox_serial_ids[] = {
267 { .compatible = "sandbox,serial" },
268 { }
269};
Marek Vasutcef46b72012-09-14 22:33:21 +0200270
Walter Lozanoe3e24702020-06-25 01:10:04 -0300271U_BOOT_DRIVER(sandbox_serial) = {
272 .name = "sandbox_serial",
Simon Glass890fcef2014-09-04 16:27:27 -0600273 .id = UCLASS_SERIAL,
274 .of_match = sandbox_serial_ids,
Simon Glassd1998a92020-12-03 16:55:21 -0700275 .of_to_plat = sandbox_serial_of_to_plat,
Simon Glass8a8d24b2020-12-03 16:55:23 -0700276 .plat_auto = sizeof(struct sandbox_serial_plat),
Simon Glass41575d82020-12-03 16:55:17 -0700277 .priv_auto = sizeof(struct sandbox_serial_priv),
Simon Glass890fcef2014-09-04 16:27:27 -0600278 .probe = sandbox_serial_probe,
Simon Glass72e98222014-09-04 16:27:28 -0600279 .remove = sandbox_serial_remove,
Simon Glass890fcef2014-09-04 16:27:27 -0600280 .ops = &sandbox_serial_ops,
281 .flags = DM_FLAG_PRE_RELOC,
282};
283
Simon Glass95397382021-08-07 07:24:04 -0600284#if CONFIG_IS_ENABLED(OF_REAL)
Simon Glass8a8d24b2020-12-03 16:55:23 -0700285static const struct sandbox_serial_plat platdata_non_fdt = {
Simon Glass72e98222014-09-04 16:27:28 -0600286 .colour = -1,
287};
288
Simon Glass20e442a2020-12-28 20:34:54 -0700289U_BOOT_DRVINFO(serial_sandbox_non_fdt) = {
Walter Lozanoe3e24702020-06-25 01:10:04 -0300290 .name = "sandbox_serial",
Simon Glasscaa4daa2020-12-03 16:55:18 -0700291 .plat = &platdata_non_fdt,
Simon Glass890fcef2014-09-04 16:27:27 -0600292};
Simon Glassc1473292020-10-03 11:31:23 -0600293#endif