blob: c7d5390b435712aff77de19dc7924e35876c5ced [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>
Marek Vasutcef46b72012-09-14 22:33:21 +020019#include <linux/compiler.h>
Simon Glassffb87902014-02-27 13:26:22 -070020#include <asm/state.h>
Simon Glass7accb6e2011-10-03 19:26:46 +000021
Simon Glass890fcef2014-09-04 16:27:27 -060022DECLARE_GLOBAL_DATA_PTR;
23
Simon Glass8a8d24b2020-12-03 16:55:23 -070024struct sandbox_serial_plat {
Simon Glass72e98222014-09-04 16:27:28 -060025 int colour; /* Text colour to use for output, -1 for none */
26};
27
Simon Glass21f90752020-11-08 20:36:50 -070028/**
29 * struct sandbox_serial_priv - Private data for this driver
30 *
31 * @buf: holds input characters available to be read by this driver
32 */
Simon Glass72e98222014-09-04 16:27:28 -060033struct sandbox_serial_priv {
Simon Glass21f90752020-11-08 20:36:50 -070034 struct membuff buf;
35 char serial_buf[16];
Simon Glass72e98222014-09-04 16:27:28 -060036 bool start_of_line;
37};
38
39/**
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 Glass8a8d24b2020-12-03 16:55:23 -070075 struct sandbox_serial_plat *plat = dev->plat;
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
Simon Glass890fcef2014-09-04 16:27:27 -060083static int sandbox_serial_putc(struct udevice *dev, const char ch)
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 Glass8a8d24b2020-12-03 16:55:23 -070086 struct sandbox_serial_plat *plat = dev->plat;
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 }
94
Simon Glass7accb6e2011-10-03 19:26:46 +000095 os_write(1, &ch, 1);
Simon Glass72e98222014-09-04 16:27:28 -060096 if (ch == '\n')
97 priv->start_of_line = true;
Simon Glass7accb6e2011-10-03 19:26:46 +000098
Simon Glass890fcef2014-09-04 16:27:27 -060099 return 0;
Simon Glass7accb6e2011-10-03 19:26:46 +0000100}
101
Simon Glass890fcef2014-09-04 16:27:27 -0600102static int sandbox_serial_pending(struct udevice *dev, bool input)
Simon Glass7accb6e2011-10-03 19:26:46 +0000103{
Simon Glass21f90752020-11-08 20:36:50 -0700104 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Taylor Hutte1015502013-02-24 17:33:13 +0000105 ssize_t count;
Simon Glass21f90752020-11-08 20:36:50 -0700106 char *data;
107 int avail;
Taylor Hutte1015502013-02-24 17:33:13 +0000108
Simon Glass890fcef2014-09-04 16:27:27 -0600109 if (!input)
110 return 0;
111
Taylor Hutte1015502013-02-24 17:33:13 +0000112 os_usleep(100);
Simon Glassc5ea0162020-11-08 20:36:48 -0700113 if (!IS_ENABLED(CONFIG_SPL_BUILD))
114 video_sync_all();
Simon Glass21f90752020-11-08 20:36:50 -0700115 avail = membuff_putraw(&priv->buf, 100, false, &data);
116 if (!avail)
Taylor Hutte1015502013-02-24 17:33:13 +0000117 return 1; /* buffer full */
118
Simon Glass21f90752020-11-08 20:36:50 -0700119 count = os_read(0, data, avail);
120 if (count > 0)
121 membuff_putraw(&priv->buf, count, true, &data);
Simon Glass890fcef2014-09-04 16:27:27 -0600122
Simon Glass21f90752020-11-08 20:36:50 -0700123 return membuff_avail(&priv->buf);
Taylor Hutte1015502013-02-24 17:33:13 +0000124}
125
Simon Glass890fcef2014-09-04 16:27:27 -0600126static int sandbox_serial_getc(struct udevice *dev)
Taylor Hutte1015502013-02-24 17:33:13 +0000127{
Simon Glass21f90752020-11-08 20:36:50 -0700128 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Taylor Hutte1015502013-02-24 17:33:13 +0000129
Simon Glass890fcef2014-09-04 16:27:27 -0600130 if (!sandbox_serial_pending(dev, true))
131 return -EAGAIN; /* buffer empty */
Taylor Hutte1015502013-02-24 17:33:13 +0000132
Simon Glass21f90752020-11-08 20:36:50 -0700133 return membuff_getbyte(&priv->buf);
Simon Glass7accb6e2011-10-03 19:26:46 +0000134}
Marek Vasutcef46b72012-09-14 22:33:21 +0200135
Simon Glassee441762018-10-01 11:55:15 -0600136#ifdef CONFIG_DEBUG_UART_SANDBOX
137
138#include <debug_uart.h>
139
140static inline void _debug_uart_init(void)
141{
142}
143
144static inline void _debug_uart_putc(int ch)
145{
146 os_putc(ch);
147}
148
149DEBUG_UART_FUNCS
150
151#endif /* CONFIG_DEBUG_UART_SANDBOX */
152
Andy Shevchenkoac7f5db2018-11-20 23:52:32 +0200153static int sandbox_serial_getconfig(struct udevice *dev, uint *serial_config)
154{
155 uint config = SERIAL_DEFAULT_CONFIG;
156
157 if (!serial_config)
158 return -EINVAL;
159
160 *serial_config = config;
161
162 return 0;
163}
164
Patrice Chotardd7c09682018-08-03 15:07:41 +0200165static int sandbox_serial_setconfig(struct udevice *dev, uint serial_config)
166{
167 u8 parity = SERIAL_GET_PARITY(serial_config);
168 u8 bits = SERIAL_GET_BITS(serial_config);
169 u8 stop = SERIAL_GET_STOP(serial_config);
170
171 if (bits != SERIAL_8_BITS || stop != SERIAL_ONE_STOP ||
172 parity != SERIAL_PAR_NONE)
173 return -ENOTSUPP; /* not supported in driver*/
174
175 return 0;
176}
177
Andy Shevchenkod5bb4f82018-11-20 23:52:33 +0200178static int sandbox_serial_getinfo(struct udevice *dev,
179 struct serial_device_info *serial_info)
180{
181 struct serial_device_info info = {
182 .type = SERIAL_CHIP_UNKNOWN,
183 .addr_space = SERIAL_ADDRESS_SPACE_IO,
184 .addr = SERIAL_DEFAULT_ADDRESS,
185 .reg_width = 1,
186 .reg_offset = 0,
187 .reg_shift = 0,
Andy Shevchenkobf4661bc2020-02-27 17:21:54 +0200188 .clock = SERIAL_DEFAULT_CLOCK,
Andy Shevchenkod5bb4f82018-11-20 23:52:33 +0200189 };
190
191 if (!serial_info)
192 return -EINVAL;
193
194 *serial_info = info;
195
196 return 0;
197}
198
Simon Glass72e98222014-09-04 16:27:28 -0600199static const char * const ansi_colour[] = {
200 "black", "red", "green", "yellow", "blue", "megenta", "cyan",
201 "white",
202};
203
Simon Glassd1998a92020-12-03 16:55:21 -0700204static int sandbox_serial_of_to_plat(struct udevice *dev)
Simon Glass72e98222014-09-04 16:27:28 -0600205{
Simon Glass8a8d24b2020-12-03 16:55:23 -0700206 struct sandbox_serial_plat *plat = dev->plat;
Simon Glass72e98222014-09-04 16:27:28 -0600207 const char *colour;
208 int i;
209
Simon Glassea147782019-09-25 08:55:53 -0600210 if (CONFIG_IS_ENABLED(OF_PLATDATA))
211 return 0;
Simon Glass72e98222014-09-04 16:27:28 -0600212 plat->colour = -1;
Simon Glass1bf0a402020-11-08 20:36:49 -0700213 colour = dev_read_string(dev, "sandbox,text-colour");
Simon Glass72e98222014-09-04 16:27:28 -0600214 if (colour) {
215 for (i = 0; i < ARRAY_SIZE(ansi_colour); i++) {
216 if (!strcmp(colour, ansi_colour[i])) {
217 plat->colour = i;
218 break;
219 }
220 }
221 }
222
223 return 0;
224}
225
Simon Glass890fcef2014-09-04 16:27:27 -0600226static const struct dm_serial_ops sandbox_serial_ops = {
227 .putc = sandbox_serial_putc,
228 .pending = sandbox_serial_pending,
229 .getc = sandbox_serial_getc,
Andy Shevchenkoac7f5db2018-11-20 23:52:32 +0200230 .getconfig = sandbox_serial_getconfig,
Patrice Chotardd7c09682018-08-03 15:07:41 +0200231 .setconfig = sandbox_serial_setconfig,
Andy Shevchenkod5bb4f82018-11-20 23:52:33 +0200232 .getinfo = sandbox_serial_getinfo,
Marek Vasutcef46b72012-09-14 22:33:21 +0200233};
234
Simon Glass890fcef2014-09-04 16:27:27 -0600235static const struct udevice_id sandbox_serial_ids[] = {
236 { .compatible = "sandbox,serial" },
237 { }
238};
Marek Vasutcef46b72012-09-14 22:33:21 +0200239
Walter Lozanoe3e24702020-06-25 01:10:04 -0300240U_BOOT_DRIVER(sandbox_serial) = {
241 .name = "sandbox_serial",
Simon Glass890fcef2014-09-04 16:27:27 -0600242 .id = UCLASS_SERIAL,
243 .of_match = sandbox_serial_ids,
Simon Glassd1998a92020-12-03 16:55:21 -0700244 .of_to_plat = sandbox_serial_of_to_plat,
Simon Glass8a8d24b2020-12-03 16:55:23 -0700245 .plat_auto = sizeof(struct sandbox_serial_plat),
Simon Glass41575d82020-12-03 16:55:17 -0700246 .priv_auto = sizeof(struct sandbox_serial_priv),
Simon Glass890fcef2014-09-04 16:27:27 -0600247 .probe = sandbox_serial_probe,
Simon Glass72e98222014-09-04 16:27:28 -0600248 .remove = sandbox_serial_remove,
Simon Glass890fcef2014-09-04 16:27:27 -0600249 .ops = &sandbox_serial_ops,
250 .flags = DM_FLAG_PRE_RELOC,
251};
252
Simon Glassc1473292020-10-03 11:31:23 -0600253#if !CONFIG_IS_ENABLED(OF_PLATDATA)
Simon Glass8a8d24b2020-12-03 16:55:23 -0700254static const struct sandbox_serial_plat platdata_non_fdt = {
Simon Glass72e98222014-09-04 16:27:28 -0600255 .colour = -1,
256};
257
Simon Glass890fcef2014-09-04 16:27:27 -0600258U_BOOT_DEVICE(serial_sandbox_non_fdt) = {
Walter Lozanoe3e24702020-06-25 01:10:04 -0300259 .name = "sandbox_serial",
Simon Glasscaa4daa2020-12-03 16:55:18 -0700260 .plat = &platdata_non_fdt,
Simon Glass890fcef2014-09-04 16:27:27 -0600261};
Simon Glassc1473292020-10-03 11:31:23 -0600262#endif