blob: b9088aa3b58b2dea12d77ee66d3ad77193fc6bd5 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Gary Jennejohnc6dc7552008-11-06 15:04:23 +01002/*
3 * (C) Copyright 2008
4 * Gary Jennejohn, DENX Software Engineering GmbH, garyj@denx.de.
Gary Jennejohnc6dc7552008-11-06 15:04:23 +01005 */
6
7#include <common.h>
Simon Glassa73bda42015-11-08 23:47:45 -07008#include <console.h>
Gary Jennejohnc6dc7552008-11-06 15:04:23 +01009#include <serial.h>
10#include <malloc.h>
11
Simon Glassdfc25352017-01-16 07:03:26 -070012#if CONFIG_IS_ENABLED(CONSOLE_MUX)
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010013void iomux_printdevs(const int console)
14{
15 int i;
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +020016 struct stdio_dev *dev;
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010017
Andy Shevchenkoe29d0192021-02-11 17:09:42 +020018 for_each_console_dev(i, console, dev)
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010019 printf("%s ", dev->name);
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010020 printf("\n");
21}
22
Andy Shevchenko75039f52021-02-11 17:09:41 +020023int iomux_match_device(struct stdio_dev **set, const int n, struct stdio_dev *sdev)
24{
25 int i;
26
27 for (i = 0; i < n; i++)
28 if (sdev == set[i])
29 return i;
30 return -ENOENT;
31}
32
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010033/* This tries to preserve the old list if an error occurs. */
34int iomux_doenv(const int console, const char *arg)
35{
36 char *console_args, *temp, **start;
Andy Shevchenko75039f52021-02-11 17:09:41 +020037 int i, j, io_flag, cs_idx, repeat;
Andy Shevchenkocb96e222020-12-21 14:30:08 +020038 struct stdio_dev **cons_set, **old_set;
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +020039 struct stdio_dev *dev;
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010040
41 console_args = strdup(arg);
42 if (console_args == NULL)
43 return 1;
44 /*
45 * Check whether a comma separated list of devices was
46 * entered and count how many devices were entered.
47 * The array start[] has pointers to the beginning of
48 * each device name (up to MAX_CONSARGS devices).
49 *
50 * Have to do this twice - once to count the number of
51 * commas and then again to populate start.
52 */
53 i = 0;
54 temp = console_args;
55 for (;;) {
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010056 /* There's always one entry more than the number of commas. */
57 i++;
Andy Shevchenkoc9d642b2020-12-21 14:30:06 +020058
59 temp = strchr(temp, ',');
60 if (temp == NULL)
61 break;
62
63 temp++;
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010064 }
65 start = (char **)malloc(i * sizeof(char *));
66 if (start == NULL) {
67 free(console_args);
68 return 1;
69 }
70 i = 0;
71 start[0] = console_args;
72 for (;;) {
73 temp = strchr(start[i++], ',');
74 if (temp == NULL)
75 break;
76 *temp = '\0';
77 start[i] = temp + 1;
78 }
Jean-Christophe PLAGNIOL-VILLARD2a7a0312009-05-16 12:14:54 +020079 cons_set = (struct stdio_dev **)calloc(i, sizeof(struct stdio_dev *));
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010080 if (cons_set == NULL) {
81 free(start);
82 free(console_args);
83 return 1;
84 }
85
Andy Shevchenko20a5c5e2021-02-11 17:09:40 +020086 io_flag = stdio_file_to_flags(console);
87 if (io_flag < 0) {
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010088 free(start);
89 free(console_args);
90 free(cons_set);
91 return 1;
92 }
93
94 cs_idx = 0;
95 for (j = 0; j < i; j++) {
96 /*
97 * Check whether the device exists and is valid.
Andy Shevchenkocfaed7f2020-12-21 14:30:03 +020098 * console_assign() also calls console_search_dev(),
Gary Jennejohnc6dc7552008-11-06 15:04:23 +010099 * but I need the pointer to the device.
100 */
Andy Shevchenkocfaed7f2020-12-21 14:30:03 +0200101 dev = console_search_dev(io_flag, start[j]);
Gary Jennejohnc6dc7552008-11-06 15:04:23 +0100102 if (dev == NULL)
103 continue;
104 /*
105 * Prevent multiple entries for a device.
106 */
Andy Shevchenko75039f52021-02-11 17:09:41 +0200107 repeat = iomux_match_device(cons_set, cs_idx, dev);
108 if (repeat >= 0)
Gary Jennejohnc6dc7552008-11-06 15:04:23 +0100109 continue;
110 /*
111 * Try assigning the specified device.
112 * This could screw up the console settings for apps.
113 */
114 if (console_assign(console, start[j]) < 0)
115 continue;
Gary Jennejohnc6dc7552008-11-06 15:04:23 +0100116 cons_set[cs_idx++] = dev;
117 }
118 free(console_args);
119 free(start);
120 /* failed to set any console */
121 if (cs_idx == 0) {
122 free(cons_set);
123 return 1;
Gary Jennejohnc6dc7552008-11-06 15:04:23 +0100124 }
Andy Shevchenko32e85982020-12-21 14:30:07 +0200125
Andy Shevchenkocb96e222020-12-21 14:30:08 +0200126 old_set = console_devices[console];
127 repeat = cd_count[console];
128
Andy Shevchenko32e85982020-12-21 14:30:07 +0200129 console_devices[console] = cons_set;
130 cd_count[console] = cs_idx;
Andy Shevchenkocb96e222020-12-21 14:30:08 +0200131
132 /* Stop dropped consoles */
133 for (i = 0; i < repeat; i++) {
Andy Shevchenko75039f52021-02-11 17:09:41 +0200134 j = iomux_match_device(cons_set, cs_idx, old_set[i]);
Andy Shevchenkocb96e222020-12-21 14:30:08 +0200135 if (j == cs_idx)
136 console_stop(console, old_set[i]);
137 }
138
139 free(old_set);
Gary Jennejohnc6dc7552008-11-06 15:04:23 +0100140 return 0;
141}
Andy Shevchenko8ee40d32021-02-11 17:09:43 +0200142
143int iomux_replace_device(const int console, const char *old, const char *new)
144{
145 struct stdio_dev *dev;
146 char *arg = NULL; /* Initial empty list */
147 int size = 1; /* For NUL terminator */
148 int i, ret;
149
150 for_each_console_dev(i, console, dev) {
151 const char *name = strcmp(dev->name, old) ? dev->name : new;
152 char *tmp;
153
154 /* Append name with a ',' (comma) separator */
155 tmp = realloc(arg, size + strlen(name) + 1);
156 if (!tmp) {
157 free(arg);
158 return -ENOMEM;
159 }
160
161 strcat(tmp, ",");
162 strcat(tmp, name);
163
164 arg = tmp;
165 size = strlen(tmp) + 1;
166 }
167
168 ret = iomux_doenv(console, arg);
169 if (ret)
170 ret = -EINVAL;
171
172 free(arg);
173 return ret;
174}
Simon Glassdfc25352017-01-16 07:03:26 -0700175#endif /* CONSOLE_MUX */