blob: 13083842cbd90153f1d714da22ea7652a26b8c7b [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
wdenk91d32562002-09-18 21:21:13 +00002/*
Heiko Schochere0e55bc2012-01-16 21:12:24 +00003 * Copyright (C) 2009 Sergey Kubushyn <ksi@koi8.net>
4 *
5 * Changes for multibus/multiadapter I2C support.
6 *
wdenk91d32562002-09-18 21:21:13 +00007 * (C) Copyright 2000
8 * Paolo Scaffardi, AIRVENT SAM s.p.a - RIMINI(ITALY), arsenio@tin.it
wdenk91d32562002-09-18 21:21:13 +00009 */
10
11#include <config.h>
12#include <common.h>
Simon Glassd3d40352015-10-18 21:17:15 -060013#include <dm.h>
Simon Glass5d609ce2014-07-23 06:55:05 -060014#include <errno.h>
Simon Glass0f2af882020-05-10 11:40:05 -060015#include <log.h>
wdenk91d32562002-09-18 21:21:13 +000016#include <stdarg.h>
17#include <malloc.h>
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +020018#include <stdio_dev.h>
wdenk7ac16102004-08-01 22:48:16 +000019#include <serial.h>
Igor Opaniuk1f6bfad2019-05-29 09:01:43 +000020#include <splash.h>
wdenk91d32562002-09-18 21:21:13 +000021#include <i2c.h>
Simon Glass3ba929a2020-10-30 21:38:53 -060022#include <asm/global_data.h>
Simon Glassd3d40352015-10-18 21:17:15 -060023#include <dm/device-internal.h>
24
Wolfgang Denk6405a152006-03-31 18:32:53 +020025DECLARE_GLOBAL_DATA_PTR;
26
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +020027static struct stdio_dev devs;
28struct stdio_dev *stdio_devices[] = { NULL, NULL, NULL };
wdenk91d32562002-09-18 21:21:13 +000029char *stdio_names[MAX_FILES] = { "stdin", "stdout", "stderr" };
30
Andy Shevchenko94331022021-02-11 17:09:36 +020031int stdio_file_to_flags(const int file)
32{
33 switch (file) {
34 case stdin:
35 return DEV_FLAGS_INPUT;
36 case stdout:
37 case stderr:
38 return DEV_FLAGS_OUTPUT;
39 default:
40 return -EINVAL;
41 }
42}
43
Andy Shevchenko2066f392021-02-11 17:09:35 +020044#if CONFIG_IS_ENABLED(SYS_DEVICE_NULLDEV)
Jeroen Hofstee169eda62014-10-08 22:57:44 +020045static void nulldev_putc(struct stdio_dev *dev, const char c)
wdenk91d32562002-09-18 21:21:13 +000046{
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +020047 /* nulldev is empty! */
wdenk91d32562002-09-18 21:21:13 +000048}
49
Jeroen Hofstee169eda62014-10-08 22:57:44 +020050static void nulldev_puts(struct stdio_dev *dev, const char *s)
wdenk91d32562002-09-18 21:21:13 +000051{
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +020052 /* nulldev is empty! */
wdenk91d32562002-09-18 21:21:13 +000053}
54
Jeroen Hofstee169eda62014-10-08 22:57:44 +020055static int nulldev_input(struct stdio_dev *dev)
wdenk91d32562002-09-18 21:21:13 +000056{
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +020057 /* nulldev is empty! */
58 return 0;
wdenk91d32562002-09-18 21:21:13 +000059}
wdenk91d32562002-09-18 21:21:13 +000060
Andy Shevchenko2066f392021-02-11 17:09:35 +020061static void nulldev_register(void)
62{
63 struct stdio_dev dev;
64
65 memset(&dev, '\0', sizeof(dev));
66
67 strcpy(dev.name, "nulldev");
68 dev.flags = DEV_FLAGS_OUTPUT | DEV_FLAGS_INPUT;
69 dev.putc = nulldev_putc;
70 dev.puts = nulldev_puts;
71 dev.getc = nulldev_input;
72 dev.tstc = nulldev_input;
73
74 stdio_register(&dev);
75}
76#else
77static inline void nulldev_register(void) {}
78#endif /* SYS_DEVICE_NULLDEV */
79
Jeroen Hofstee169eda62014-10-08 22:57:44 +020080static void stdio_serial_putc(struct stdio_dev *dev, const char c)
Simon Glass0d1e1f72014-07-23 06:54:59 -060081{
82 serial_putc(c);
83}
84
Jeroen Hofstee169eda62014-10-08 22:57:44 +020085static void stdio_serial_puts(struct stdio_dev *dev, const char *s)
Simon Glass0d1e1f72014-07-23 06:54:59 -060086{
87 serial_puts(s);
88}
89
Pali Rohára48a24f2022-09-05 11:31:19 +020090#ifdef CONFIG_CONSOLE_FLUSH_SUPPORT
91static void stdio_serial_flush(struct stdio_dev *dev)
92{
93 serial_flush();
94}
95#endif
96
Jeroen Hofstee169eda62014-10-08 22:57:44 +020097static int stdio_serial_getc(struct stdio_dev *dev)
Simon Glass0d1e1f72014-07-23 06:54:59 -060098{
99 return serial_getc();
100}
101
Jeroen Hofstee169eda62014-10-08 22:57:44 +0200102static int stdio_serial_tstc(struct stdio_dev *dev)
Simon Glass0d1e1f72014-07-23 06:54:59 -0600103{
104 return serial_tstc();
105}
106
wdenk91d32562002-09-18 21:21:13 +0000107/**************************************************************************
108 * SYSTEM DRIVERS
109 **************************************************************************
110 */
111
112static void drv_system_init (void)
113{
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200114 struct stdio_dev dev;
wdenk91d32562002-09-18 21:21:13 +0000115
116 memset (&dev, 0, sizeof (dev));
117
118 strcpy (dev.name, "serial");
Bin Meng6abe4b62015-11-03 23:23:37 -0800119 dev.flags = DEV_FLAGS_OUTPUT | DEV_FLAGS_INPUT;
Simon Glass0d1e1f72014-07-23 06:54:59 -0600120 dev.putc = stdio_serial_putc;
121 dev.puts = stdio_serial_puts;
Pali Rohára48a24f2022-09-05 11:31:19 +0200122 STDIO_DEV_ASSIGN_FLUSH(&dev, stdio_serial_flush);
Simon Glass0d1e1f72014-07-23 06:54:59 -0600123 dev.getc = stdio_serial_getc;
124 dev.tstc = stdio_serial_tstc;
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200125 stdio_register (&dev);
wdenk91d32562002-09-18 21:21:13 +0000126
Andy Shevchenko2066f392021-02-11 17:09:35 +0200127 nulldev_register();
wdenk91d32562002-09-18 21:21:13 +0000128}
129
130/**************************************************************************
131 * DEVICES
132 **************************************************************************
133 */
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200134struct list_head* stdio_get_list(void)
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200135{
Simon Glass4b1e3092020-08-11 11:23:40 -0600136 return &devs.list;
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200137}
138
Simon Glass4af407f2016-10-05 20:42:16 -0600139/**
140 * stdio_probe_device() - Find a device which provides the given stdio device
141 *
142 * This looks for a device of the given uclass which provides a particular
143 * stdio device. It is currently really only useful for UCLASS_VIDEO.
144 *
145 * Ultimately we want to be able to probe a device by its stdio name. At
146 * present devices register in their probe function (for video devices this
147 * is done in vidconsole_post_probe()) and we don't know what name they will
148 * use until they do so.
149 * TODO(sjg@chromium.org): We should be able to determine the name before
150 * probing, and probe the required device.
151 *
152 * @name: stdio device name (e.g. "vidconsole")
153 * id: Uclass ID of device to look for (e.g. UCLASS_VIDEO)
154 * @sdevp: Returns stdout device, if found, else NULL
Heinrich Schuchardt47b4c022022-01-19 18:05:50 +0100155 * Return: 0 if found, -ENOENT if no device found with that name, other -ve
Simon Glass4af407f2016-10-05 20:42:16 -0600156 * on other error
157 */
158static int stdio_probe_device(const char *name, enum uclass_id id,
159 struct stdio_dev **sdevp)
160{
161 struct stdio_dev *sdev;
162 struct udevice *dev;
163 int seq, ret;
164
165 *sdevp = NULL;
166 seq = trailing_strtoln(name, NULL);
167 if (seq == -1)
Simon Glass927e6312016-11-13 14:22:00 -0700168 seq = 0;
169 ret = uclass_get_device_by_seq(id, seq, &dev);
170 if (ret == -ENODEV)
Simon Glass4af407f2016-10-05 20:42:16 -0600171 ret = uclass_first_device_err(id, &dev);
Simon Glass4af407f2016-10-05 20:42:16 -0600172 if (ret) {
173 debug("No %s device for seq %d (%s)\n", uclass_get_name(id),
174 seq, name);
175 return ret;
176 }
177 /* The device should be be the last one registered */
178 sdev = list_empty(&devs.list) ? NULL :
179 list_last_entry(&devs.list, struct stdio_dev, list);
180 if (!sdev || strcmp(sdev->name, name)) {
181 debug("Device '%s' did not register with stdio as '%s'\n",
182 dev->name, name);
183 return -ENOENT;
184 }
185 *sdevp = sdev;
186
187 return 0;
188}
Simon Glass4af407f2016-10-05 20:42:16 -0600189
Simon Glass913d7612016-11-13 14:21:59 -0700190struct stdio_dev *stdio_get_by_name(const char *name)
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200191{
192 struct list_head *pos;
Simon Glass4af407f2016-10-05 20:42:16 -0600193 struct stdio_dev *sdev;
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200194
Simon Glass913d7612016-11-13 14:21:59 -0700195 if (!name)
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200196 return NULL;
197
Simon Glass4b1e3092020-08-11 11:23:40 -0600198 list_for_each(pos, &devs.list) {
Simon Glass4af407f2016-10-05 20:42:16 -0600199 sdev = list_entry(pos, struct stdio_dev, list);
200 if (strcmp(sdev->name, name) == 0)
201 return sdev;
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200202 }
Simon Glass5a36d822020-08-11 11:23:39 -0600203 if (IS_ENABLED(CONFIG_DM_VIDEO)) {
204 /*
205 * We did not find a suitable stdio device. If there is a video
206 * driver with a name starting with 'vidconsole', we can try
207 * probing that in the hope that it will produce the required
208 * stdio device.
209 *
210 * This function is sometimes called with the entire value of
211 * 'stdout', which may include a list of devices separate by
212 * commas. Obviously this is not going to work, so we ignore
213 * that case. The call path in that case is
Andy Shevchenkocfaed7f2020-12-21 14:30:03 +0200214 * console_init_r() -> console_search_dev() -> stdio_get_by_name()
Simon Glass5a36d822020-08-11 11:23:39 -0600215 */
216 if (!strncmp(name, "vidconsole", 10) && !strchr(name, ',') &&
217 !stdio_probe_device(name, UCLASS_VIDEO, &sdev))
218 return sdev;
219 }
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200220
221 return NULL;
222}
223
Simon Glass5b4c1222020-08-11 11:23:41 -0600224struct stdio_dev *stdio_clone(struct stdio_dev *dev)
Jean-Christophe PLAGNIOL-VILLARD3d85c432008-09-01 17:11:26 +0200225{
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200226 struct stdio_dev *_dev;
Jean-Christophe PLAGNIOL-VILLARD3d85c432008-09-01 17:11:26 +0200227
Simon Glass5b4c1222020-08-11 11:23:41 -0600228 if (!dev)
Jean-Christophe PLAGNIOL-VILLARD3d85c432008-09-01 17:11:26 +0200229 return NULL;
230
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200231 _dev = calloc(1, sizeof(struct stdio_dev));
Simon Glass5b4c1222020-08-11 11:23:41 -0600232 if (!_dev)
Jean-Christophe PLAGNIOL-VILLARD3d85c432008-09-01 17:11:26 +0200233 return NULL;
234
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200235 memcpy(_dev, dev, sizeof(struct stdio_dev));
Jean-Christophe PLAGNIOL-VILLARD3d85c432008-09-01 17:11:26 +0200236
237 return _dev;
238}
wdenk91d32562002-09-18 21:21:13 +0000239
Simon Glass5d609ce2014-07-23 06:55:05 -0600240int stdio_register_dev(struct stdio_dev *dev, struct stdio_dev **devp)
wdenk91d32562002-09-18 21:21:13 +0000241{
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200242 struct stdio_dev *_dev;
Jean-Christophe PLAGNIOL-VILLARD3d85c432008-09-01 17:11:26 +0200243
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200244 _dev = stdio_clone(dev);
Simon Glass5b4c1222020-08-11 11:23:41 -0600245 if (!_dev)
Simon Glass5d609ce2014-07-23 06:55:05 -0600246 return -ENODEV;
Simon Glass4b1e3092020-08-11 11:23:40 -0600247 list_add_tail(&_dev->list, &devs.list);
Simon Glass5d609ce2014-07-23 06:55:05 -0600248 if (devp)
249 *devp = _dev;
250
wdenk91d32562002-09-18 21:21:13 +0000251 return 0;
252}
253
Simon Glass5d609ce2014-07-23 06:55:05 -0600254int stdio_register(struct stdio_dev *dev)
255{
256 return stdio_register_dev(dev, NULL);
257}
258
Hans de Goede1a586382014-09-20 16:54:37 +0200259int stdio_deregister_dev(struct stdio_dev *dev, int force)
wdenk91d32562002-09-18 21:21:13 +0000260{
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200261 struct list_head *pos;
Bradley Bolen466d7392011-08-22 11:48:05 +0000262 char temp_names[3][16];
Simon Glass5b4c1222020-08-11 11:23:41 -0600263 int i;
wdenk91d32562002-09-18 21:21:13 +0000264
wdenk91d32562002-09-18 21:21:13 +0000265 /* get stdio devices (ListRemoveItem changes the dev list) */
Simon Glass5b4c1222020-08-11 11:23:41 -0600266 for (i = 0 ; i < MAX_FILES; i++) {
267 if (stdio_devices[i] == dev) {
Hans de Goede1a586382014-09-20 16:54:37 +0200268 if (force) {
Simon Glass5b4c1222020-08-11 11:23:41 -0600269 strcpy(temp_names[i], "nulldev");
Hans de Goede1a586382014-09-20 16:54:37 +0200270 continue;
271 }
wdenk91d32562002-09-18 21:21:13 +0000272 /* Device is assigned -> report error */
Simon Glass5b4c1222020-08-11 11:23:41 -0600273 return -EBUSY;
wdenk91d32562002-09-18 21:21:13 +0000274 }
Simon Glass5b4c1222020-08-11 11:23:41 -0600275 memcpy(&temp_names[i][0], stdio_devices[i]->name,
276 sizeof(temp_names[i]));
wdenk91d32562002-09-18 21:21:13 +0000277 }
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200278
Simon Glass4b1e3092020-08-11 11:23:40 -0600279 list_del(&dev->list);
Hans de Goede35e116c2014-09-24 14:06:09 +0200280 free(dev);
Jean-Christophe PLAGNIOL-VILLARD8a4a7842008-08-31 04:24:55 +0200281
Simon Glass5b4c1222020-08-11 11:23:41 -0600282 /* reassign device list */
Simon Glass4b1e3092020-08-11 11:23:40 -0600283 list_for_each(pos, &devs.list) {
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +0200284 dev = list_entry(pos, struct stdio_dev, list);
Simon Glass5b4c1222020-08-11 11:23:41 -0600285 for (i = 0 ; i < MAX_FILES; i++) {
286 if (strcmp(dev->name, temp_names[i]) == 0)
287 stdio_devices[i] = dev;
wdenk91d32562002-09-18 21:21:13 +0000288 }
289 }
Simon Glass5b4c1222020-08-11 11:23:41 -0600290
wdenk91d32562002-09-18 21:21:13 +0000291 return 0;
292}
Simon Glass5d609ce2014-07-23 06:55:05 -0600293
Simon Glass712da5e2014-09-03 17:37:01 -0600294int stdio_init_tables(void)
wdenk91d32562002-09-18 21:21:13 +0000295{
Wolfgang Denkd0813e52010-10-28 20:00:11 +0200296#if defined(CONFIG_NEEDS_MANUAL_RELOC)
Peter Tyser9057cbf2009-09-21 11:20:36 -0500297 /* already relocated for current ARM implementation */
wdenk91d32562002-09-18 21:21:13 +0000298 ulong relocation_offset = gd->reloc_off;
wdenk56958612003-06-22 17:18:28 +0000299 int i;
wdenk91d32562002-09-18 21:21:13 +0000300
301 /* relocate device name pointers */
302 for (i = 0; i < (sizeof (stdio_names) / sizeof (char *)); ++i) {
303 stdio_names[i] = (char *) (((ulong) stdio_names[i]) +
304 relocation_offset);
305 }
Wolfgang Denkd0813e52010-10-28 20:00:11 +0200306#endif /* CONFIG_NEEDS_MANUAL_RELOC */
wdenk91d32562002-09-18 21:21:13 +0000307
308 /* Initialize the list */
Simon Glass4b1e3092020-08-11 11:23:40 -0600309 INIT_LIST_HEAD(&devs.list);
wdenk91d32562002-09-18 21:21:13 +0000310
Simon Glass712da5e2014-09-03 17:37:01 -0600311 return 0;
312}
313
314int stdio_add_devices(void)
315{
Simon Glassd3d40352015-10-18 21:17:15 -0600316 struct udevice *dev;
317 struct uclass *uc;
318 int ret;
319
Simon Glass5a36d822020-08-11 11:23:39 -0600320 if (IS_ENABLED(CONFIG_DM_KEYBOARD)) {
321 /*
322 * For now we probe all the devices here. At some point this
323 * should be done only when the devices are required - e.g. we
324 * have a list of input devices to start up in the stdin
325 * environment variable. That work probably makes more sense
326 * when stdio itself is converted to driver model.
327 *
328 * TODO(sjg@chromium.org): Convert changing
329 * uclass_first_device() etc. to return the device even on
330 * error. Then we could use that here.
331 */
332 ret = uclass_get(UCLASS_KEYBOARD, &uc);
Simon Glassd3d40352015-10-18 21:17:15 -0600333 if (ret)
Simon Glass5a36d822020-08-11 11:23:39 -0600334 return ret;
335
336 /*
337 * Don't report errors to the caller - assume that they are
338 * non-fatal
339 */
340 uclass_foreach_dev(dev, uc) {
341 ret = device_probe(dev);
342 if (ret)
343 printf("Failed to probe keyboard '%s'\n",
344 dev->name);
345 }
Simon Glassd3d40352015-10-18 21:17:15 -0600346 }
Tom Rini52b2e262021-08-18 23:12:24 -0400347#if CONFIG_IS_ENABLED(SYS_I2C_LEGACY)
Heiko Schochere0e55bc2012-01-16 21:12:24 +0000348 i2c_init_all();
Heiko Schochere0e55bc2012-01-16 21:12:24 +0000349#endif
Simon Glass5a36d822020-08-11 11:23:39 -0600350 if (IS_ENABLED(CONFIG_DM_VIDEO)) {
351 /*
352 * If the console setting is not in environment variables then
353 * console_init_r() will not be calling iomux_doenv() (which
Andy Shevchenkocfaed7f2020-12-21 14:30:03 +0200354 * calls console_search_dev()). So we will not dynamically add
Simon Glass5a36d822020-08-11 11:23:39 -0600355 * devices by calling stdio_probe_device().
356 *
357 * So just probe all video devices now so that whichever one is
358 * required will be available.
359 */
360 struct udevice *vdev;
361 int ret;
362
363 if (!IS_ENABLED(CONFIG_SYS_CONSOLE_IS_IN_ENV)) {
364 for (ret = uclass_first_device(UCLASS_VIDEO, &vdev);
365 vdev;
366 ret = uclass_next_device(&vdev))
367 ;
368 if (ret)
369 printf("%s: Video device failed (ret=%d)\n",
370 __func__, ret);
371 }
372 if (IS_ENABLED(CONFIG_SPLASH_SCREEN) &&
373 IS_ENABLED(CONFIG_CMD_BMP))
374 splash_display();
375 } else {
376 if (IS_ENABLED(CONFIG_LCD))
377 drv_lcd_init();
Simon Glass03fbd252022-01-23 07:04:08 -0700378 if (IS_ENABLED(CONFIG_VIDEO_VCXK))
Simon Glass5a36d822020-08-11 11:23:39 -0600379 drv_video_init();
380 }
Simon Glass0005cb22016-01-18 19:52:23 -0700381
Simon Glassd3d40352015-10-18 21:17:15 -0600382#if defined(CONFIG_KEYBOARD) && !defined(CONFIG_DM_KEYBOARD)
Simon Glass5a36d822020-08-11 11:23:39 -0600383 drv_keyboard_init();
wdenk91d32562002-09-18 21:21:13 +0000384#endif
Simon Glass5a36d822020-08-11 11:23:39 -0600385 drv_system_init();
386 serial_stdio_init();
wdenk29e7f5a2004-03-12 00:14:09 +0000387#ifdef CONFIG_USB_TTY
Simon Glass5a36d822020-08-11 11:23:39 -0600388 drv_usbtty_init();
wdenk29e7f5a2004-03-12 00:14:09 +0000389#endif
Loic Poulaineb78f832021-11-25 18:16:15 +0100390#ifdef CONFIG_USB_FUNCTION_ACM
391 drv_usbacm_init ();
392#endif
Simon Glass5a36d822020-08-11 11:23:39 -0600393 if (IS_ENABLED(CONFIG_NETCONSOLE))
394 drv_nc_init();
Mike Frysinger7504e972008-10-11 21:51:20 -0400395#ifdef CONFIG_JTAG_CONSOLE
Simon Glass5a36d822020-08-11 11:23:39 -0600396 drv_jtag_console_init();
Vadim Bendebury263a8bd2012-10-12 18:48:47 +0000397#endif
Simon Glass5a36d822020-08-11 11:23:39 -0600398 if (IS_ENABLED(CONFIG_CBMEM_CONSOLE))
399 cbmemc_init();
Simon Glass712da5e2014-09-03 17:37:01 -0600400
401 return 0;
402}
403
404int stdio_init(void)
405{
406 stdio_init_tables();
407 stdio_add_devices();
408
409 return 0;
wdenk91d32562002-09-18 21:21:13 +0000410}