blob: 19368ba256030d995b7c1b72950b785989084dcb [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Simon Glass479349a2011-10-03 19:26:46 +00002/*
3 * Copyright (c) 2011 The Chromium OS Authors.
Simon Glass479349a2011-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 Hershbergerb3eff9b2018-07-02 20:06:49 -050013#include <console.h>
Simon Glasse5870692014-09-04 16:27:27 -060014#include <dm.h>
Simon Glassb9ddbf42014-02-27 13:26:19 -070015#include <lcd.h>
Simon Glass479349a2011-10-03 19:26:46 +000016#include <os.h>
Marek Vasut289c3002012-09-14 22:33:21 +020017#include <serial.h>
Simon Glass38a2ae22016-01-18 19:52:25 -070018#include <video.h>
Marek Vasut289c3002012-09-14 22:33:21 +020019#include <linux/compiler.h>
Simon Glass2bfb7ef2020-12-19 10:39:53 -070020#include <asm/serial.h>
Simon Glass678ef472014-02-27 13:26:22 -070021#include <asm/state.h>
Simon Glass479349a2011-10-03 19:26:46 +000022
Simon Glasse5870692014-09-04 16:27:27 -060023DECLARE_GLOBAL_DATA_PTR;
24
Simon Glass1a823a92014-09-04 16:27:28 -060025/**
26 * output_ansi_colour() - Output an ANSI colour code
27 *
28 * @colour: Colour to output (0-7)
29 */
30static void output_ansi_colour(int colour)
31{
32 char ansi_code[] = "\x1b[1;3Xm";
33
34 ansi_code[5] = '0' + colour;
35 os_write(1, ansi_code, sizeof(ansi_code) - 1);
36}
37
38static void output_ansi_reset(void)
39{
40 os_write(1, "\x1b[0m", 4);
41}
42
Simon Glasse5870692014-09-04 16:27:27 -060043static int sandbox_serial_probe(struct udevice *dev)
Simon Glass479349a2011-10-03 19:26:46 +000044{
Simon Glass678ef472014-02-27 13:26:22 -070045 struct sandbox_state *state = state_get_current();
Simon Glass1a823a92014-09-04 16:27:28 -060046 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Simon Glass678ef472014-02-27 13:26:22 -070047
48 if (state->term_raw != STATE_TERM_COOKED)
49 os_tty_raw(0, state->term_raw == STATE_TERM_RAW_WITH_SIGS);
Simon Glass1a823a92014-09-04 16:27:28 -060050 priv->start_of_line = 0;
Simon Glass479349a2011-10-03 19:26:46 +000051
Joe Hershbergerb3eff9b2018-07-02 20:06:49 -050052 if (state->term_raw != STATE_TERM_RAW)
53 disable_ctrlc(1);
Simon Glass54209572020-11-08 20:36:50 -070054 membuff_init(&priv->buf, priv->serial_buf, sizeof(priv->serial_buf));
Joe Hershbergerb3eff9b2018-07-02 20:06:49 -050055
Simon Glasse5870692014-09-04 16:27:27 -060056 return 0;
Simon Glass479349a2011-10-03 19:26:46 +000057}
58
Simon Glass1a823a92014-09-04 16:27:28 -060059static int sandbox_serial_remove(struct udevice *dev)
60{
Simon Glass95588622020-12-22 19:30:28 -070061 struct sandbox_serial_plat *plat = dev_get_plat(dev);
Simon Glass1a823a92014-09-04 16:27:28 -060062
63 if (plat->colour != -1)
64 output_ansi_reset();
65
66 return 0;
67}
68
Simon Glasse5870692014-09-04 16:27:27 -060069static int sandbox_serial_putc(struct udevice *dev, const char ch)
Simon Glass479349a2011-10-03 19:26:46 +000070{
Simon Glass1a823a92014-09-04 16:27:28 -060071 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Simon Glass95588622020-12-22 19:30:28 -070072 struct sandbox_serial_plat *plat = dev_get_plat(dev);
Simon Glass1a823a92014-09-04 16:27:28 -060073
Walter Lozano2901ac62020-06-25 01:10:04 -030074 /* With of-platdata we don't real the colour correctly, so disable it */
75 if (!CONFIG_IS_ENABLED(OF_PLATDATA) && priv->start_of_line &&
76 plat->colour != -1) {
Simon Glass1a823a92014-09-04 16:27:28 -060077 priv->start_of_line = false;
78 output_ansi_colour(plat->colour);
79 }
80
Simon Glass479349a2011-10-03 19:26:46 +000081 os_write(1, &ch, 1);
Simon Glass1a823a92014-09-04 16:27:28 -060082 if (ch == '\n')
83 priv->start_of_line = true;
Simon Glass479349a2011-10-03 19:26:46 +000084
Simon Glasse5870692014-09-04 16:27:27 -060085 return 0;
Simon Glass479349a2011-10-03 19:26:46 +000086}
87
Simon Glasse5870692014-09-04 16:27:27 -060088static int sandbox_serial_pending(struct udevice *dev, bool input)
Simon Glass479349a2011-10-03 19:26:46 +000089{
Simon Glass54209572020-11-08 20:36:50 -070090 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Mike Frysingerd174fd92011-10-26 00:21:01 +000091 ssize_t count;
Simon Glass54209572020-11-08 20:36:50 -070092 char *data;
93 int avail;
Simon Glass479349a2011-10-03 19:26:46 +000094
Simon Glasse5870692014-09-04 16:27:27 -060095 if (!input)
96 return 0;
97
Taylor Hutt868aa3a2013-02-24 17:33:13 +000098 os_usleep(100);
Simon Glass0b8ffa72020-11-08 20:36:48 -070099 if (!IS_ENABLED(CONFIG_SPL_BUILD))
100 video_sync_all();
Simon Glass54209572020-11-08 20:36:50 -0700101 avail = membuff_putraw(&priv->buf, 100, false, &data);
102 if (!avail)
Taylor Hutt868aa3a2013-02-24 17:33:13 +0000103 return 1; /* buffer full */
104
Simon Glass54209572020-11-08 20:36:50 -0700105 count = os_read(0, data, avail);
106 if (count > 0)
107 membuff_putraw(&priv->buf, count, true, &data);
Simon Glasse5870692014-09-04 16:27:27 -0600108
Simon Glass54209572020-11-08 20:36:50 -0700109 return membuff_avail(&priv->buf);
Simon Glass479349a2011-10-03 19:26:46 +0000110}
111
Simon Glasse5870692014-09-04 16:27:27 -0600112static int sandbox_serial_getc(struct udevice *dev)
Simon Glass479349a2011-10-03 19:26:46 +0000113{
Simon Glass54209572020-11-08 20:36:50 -0700114 struct sandbox_serial_priv *priv = dev_get_priv(dev);
Taylor Hutt868aa3a2013-02-24 17:33:13 +0000115
Simon Glasse5870692014-09-04 16:27:27 -0600116 if (!sandbox_serial_pending(dev, true))
117 return -EAGAIN; /* buffer empty */
Taylor Hutt868aa3a2013-02-24 17:33:13 +0000118
Simon Glass54209572020-11-08 20:36:50 -0700119 return membuff_getbyte(&priv->buf);
Simon Glass479349a2011-10-03 19:26:46 +0000120}
Marek Vasut289c3002012-09-14 22:33:21 +0200121
Simon Glasse3057f32018-10-01 11:55:15 -0600122#ifdef CONFIG_DEBUG_UART_SANDBOX
123
124#include <debug_uart.h>
125
126static inline void _debug_uart_init(void)
127{
128}
129
130static inline void _debug_uart_putc(int ch)
131{
132 os_putc(ch);
133}
134
135DEBUG_UART_FUNCS
136
137#endif /* CONFIG_DEBUG_UART_SANDBOX */
138
Andy Shevchenko08e98792018-11-20 23:52:32 +0200139static int sandbox_serial_getconfig(struct udevice *dev, uint *serial_config)
140{
141 uint config = SERIAL_DEFAULT_CONFIG;
142
143 if (!serial_config)
144 return -EINVAL;
145
146 *serial_config = config;
147
148 return 0;
149}
150
Patrice Chotard70ed0ea2018-08-03 15:07:41 +0200151static int sandbox_serial_setconfig(struct udevice *dev, uint serial_config)
152{
153 u8 parity = SERIAL_GET_PARITY(serial_config);
154 u8 bits = SERIAL_GET_BITS(serial_config);
155 u8 stop = SERIAL_GET_STOP(serial_config);
156
157 if (bits != SERIAL_8_BITS || stop != SERIAL_ONE_STOP ||
158 parity != SERIAL_PAR_NONE)
159 return -ENOTSUPP; /* not supported in driver*/
160
161 return 0;
162}
163
Andy Shevchenko44f21da2018-11-20 23:52:33 +0200164static int sandbox_serial_getinfo(struct udevice *dev,
165 struct serial_device_info *serial_info)
166{
167 struct serial_device_info info = {
168 .type = SERIAL_CHIP_UNKNOWN,
169 .addr_space = SERIAL_ADDRESS_SPACE_IO,
170 .addr = SERIAL_DEFAULT_ADDRESS,
171 .reg_width = 1,
172 .reg_offset = 0,
173 .reg_shift = 0,
Andy Shevchenko106930e2020-02-27 17:21:54 +0200174 .clock = SERIAL_DEFAULT_CLOCK,
Andy Shevchenko44f21da2018-11-20 23:52:33 +0200175 };
176
177 if (!serial_info)
178 return -EINVAL;
179
180 *serial_info = info;
181
182 return 0;
183}
184
Simon Glass1a823a92014-09-04 16:27:28 -0600185static const char * const ansi_colour[] = {
186 "black", "red", "green", "yellow", "blue", "megenta", "cyan",
187 "white",
188};
189
Simon Glassaad29ae2020-12-03 16:55:21 -0700190static int sandbox_serial_of_to_plat(struct udevice *dev)
Simon Glass1a823a92014-09-04 16:27:28 -0600191{
Simon Glass95588622020-12-22 19:30:28 -0700192 struct sandbox_serial_plat *plat = dev_get_plat(dev);
Simon Glass1a823a92014-09-04 16:27:28 -0600193 const char *colour;
194 int i;
195
Simon Glassa91e48b2019-09-25 08:55:53 -0600196 if (CONFIG_IS_ENABLED(OF_PLATDATA))
197 return 0;
Simon Glass1a823a92014-09-04 16:27:28 -0600198 plat->colour = -1;
Simon Glassfb6268d2020-11-08 20:36:49 -0700199 colour = dev_read_string(dev, "sandbox,text-colour");
Simon Glass1a823a92014-09-04 16:27:28 -0600200 if (colour) {
201 for (i = 0; i < ARRAY_SIZE(ansi_colour); i++) {
202 if (!strcmp(colour, ansi_colour[i])) {
203 plat->colour = i;
204 break;
205 }
206 }
207 }
208
209 return 0;
210}
211
Simon Glasse5870692014-09-04 16:27:27 -0600212static const struct dm_serial_ops sandbox_serial_ops = {
213 .putc = sandbox_serial_putc,
214 .pending = sandbox_serial_pending,
215 .getc = sandbox_serial_getc,
Andy Shevchenko08e98792018-11-20 23:52:32 +0200216 .getconfig = sandbox_serial_getconfig,
Patrice Chotard70ed0ea2018-08-03 15:07:41 +0200217 .setconfig = sandbox_serial_setconfig,
Andy Shevchenko44f21da2018-11-20 23:52:33 +0200218 .getinfo = sandbox_serial_getinfo,
Marek Vasut289c3002012-09-14 22:33:21 +0200219};
220
Simon Glasse5870692014-09-04 16:27:27 -0600221static const struct udevice_id sandbox_serial_ids[] = {
222 { .compatible = "sandbox,serial" },
223 { }
224};
Marek Vasut289c3002012-09-14 22:33:21 +0200225
Walter Lozano2901ac62020-06-25 01:10:04 -0300226U_BOOT_DRIVER(sandbox_serial) = {
227 .name = "sandbox_serial",
Simon Glasse5870692014-09-04 16:27:27 -0600228 .id = UCLASS_SERIAL,
229 .of_match = sandbox_serial_ids,
Simon Glassaad29ae2020-12-03 16:55:21 -0700230 .of_to_plat = sandbox_serial_of_to_plat,
Simon Glassb75b15b2020-12-03 16:55:23 -0700231 .plat_auto = sizeof(struct sandbox_serial_plat),
Simon Glass8a2b47f2020-12-03 16:55:17 -0700232 .priv_auto = sizeof(struct sandbox_serial_priv),
Simon Glasse5870692014-09-04 16:27:27 -0600233 .probe = sandbox_serial_probe,
Simon Glass1a823a92014-09-04 16:27:28 -0600234 .remove = sandbox_serial_remove,
Simon Glasse5870692014-09-04 16:27:27 -0600235 .ops = &sandbox_serial_ops,
236 .flags = DM_FLAG_PRE_RELOC,
237};
238
Simon Glasscb90bd32020-10-03 11:31:23 -0600239#if !CONFIG_IS_ENABLED(OF_PLATDATA)
Simon Glassb75b15b2020-12-03 16:55:23 -0700240static const struct sandbox_serial_plat platdata_non_fdt = {
Simon Glass1a823a92014-09-04 16:27:28 -0600241 .colour = -1,
242};
243
Simon Glasse5870692014-09-04 16:27:27 -0600244U_BOOT_DEVICE(serial_sandbox_non_fdt) = {
Walter Lozano2901ac62020-06-25 01:10:04 -0300245 .name = "sandbox_serial",
Simon Glass71fa5b42020-12-03 16:55:18 -0700246 .plat = &platdata_non_fdt,
Simon Glasse5870692014-09-04 16:27:27 -0600247};
Simon Glasscb90bd32020-10-03 11:31:23 -0600248#endif