77ed7a2ac9
Use of strcat() against an uninitialized buffer would lead
to buffer overflow. This patch fixes it.
Fixes: 694cd5618c
("IOMUX: Introduce iomux_replace_device()")
Signed-off-by: Yuichiro Goto <goto@k-tech.co.jp>
Cc: Peter Robinson <pbrobinson@gmail.com>
Cc: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Cc: Nicolas Saenz Julienne <nsaenzjulienne@suse.de>
Reviewed-by: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Tested-by: Peter Robinson <pbrobinson@gmail.com>
180 lines
3.7 KiB
C
180 lines
3.7 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* (C) Copyright 2008
|
|
* Gary Jennejohn, DENX Software Engineering GmbH, garyj@denx.de.
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <console.h>
|
|
#include <serial.h>
|
|
#include <malloc.h>
|
|
|
|
#if CONFIG_IS_ENABLED(CONSOLE_MUX)
|
|
void iomux_printdevs(const int console)
|
|
{
|
|
int i;
|
|
struct stdio_dev *dev;
|
|
|
|
for_each_console_dev(i, console, dev)
|
|
printf("%s ", dev->name);
|
|
printf("\n");
|
|
}
|
|
|
|
int iomux_match_device(struct stdio_dev **set, const int n, struct stdio_dev *sdev)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < n; i++)
|
|
if (sdev == set[i])
|
|
return i;
|
|
return -ENOENT;
|
|
}
|
|
|
|
/* This tries to preserve the old list if an error occurs. */
|
|
int iomux_doenv(const int console, const char *arg)
|
|
{
|
|
char *console_args, *temp, **start;
|
|
int i, j, io_flag, cs_idx, repeat;
|
|
struct stdio_dev **cons_set, **old_set;
|
|
struct stdio_dev *dev;
|
|
|
|
console_args = strdup(arg);
|
|
if (console_args == NULL)
|
|
return 1;
|
|
/*
|
|
* Check whether a comma separated list of devices was
|
|
* entered and count how many devices were entered.
|
|
* The array start[] has pointers to the beginning of
|
|
* each device name (up to MAX_CONSARGS devices).
|
|
*
|
|
* Have to do this twice - once to count the number of
|
|
* commas and then again to populate start.
|
|
*/
|
|
i = 0;
|
|
temp = console_args;
|
|
for (;;) {
|
|
/* There's always one entry more than the number of commas. */
|
|
i++;
|
|
|
|
temp = strchr(temp, ',');
|
|
if (temp == NULL)
|
|
break;
|
|
|
|
temp++;
|
|
}
|
|
start = (char **)malloc(i * sizeof(char *));
|
|
if (start == NULL) {
|
|
free(console_args);
|
|
return 1;
|
|
}
|
|
i = 0;
|
|
start[0] = console_args;
|
|
for (;;) {
|
|
temp = strchr(start[i++], ',');
|
|
if (temp == NULL)
|
|
break;
|
|
*temp = '\0';
|
|
start[i] = temp + 1;
|
|
}
|
|
cons_set = (struct stdio_dev **)calloc(i, sizeof(struct stdio_dev *));
|
|
if (cons_set == NULL) {
|
|
free(start);
|
|
free(console_args);
|
|
return 1;
|
|
}
|
|
|
|
io_flag = stdio_file_to_flags(console);
|
|
if (io_flag < 0) {
|
|
free(start);
|
|
free(console_args);
|
|
free(cons_set);
|
|
return 1;
|
|
}
|
|
|
|
cs_idx = 0;
|
|
for (j = 0; j < i; j++) {
|
|
/*
|
|
* Check whether the device exists and is valid.
|
|
* console_assign() also calls console_search_dev(),
|
|
* but I need the pointer to the device.
|
|
*/
|
|
dev = console_search_dev(io_flag, start[j]);
|
|
if (dev == NULL)
|
|
continue;
|
|
/*
|
|
* Prevent multiple entries for a device.
|
|
*/
|
|
repeat = iomux_match_device(cons_set, cs_idx, dev);
|
|
if (repeat >= 0)
|
|
continue;
|
|
/*
|
|
* Try assigning the specified device.
|
|
* This could screw up the console settings for apps.
|
|
*/
|
|
if (console_assign(console, start[j]) < 0)
|
|
continue;
|
|
cons_set[cs_idx++] = dev;
|
|
}
|
|
free(console_args);
|
|
free(start);
|
|
/* failed to set any console */
|
|
if (cs_idx == 0) {
|
|
free(cons_set);
|
|
return 1;
|
|
}
|
|
|
|
old_set = console_devices[console];
|
|
repeat = cd_count[console];
|
|
|
|
console_devices[console] = cons_set;
|
|
cd_count[console] = cs_idx;
|
|
|
|
/* Stop dropped consoles */
|
|
for (i = 0; i < repeat; i++) {
|
|
j = iomux_match_device(cons_set, cs_idx, old_set[i]);
|
|
if (j == cs_idx)
|
|
console_stop(console, old_set[i]);
|
|
}
|
|
|
|
free(old_set);
|
|
return 0;
|
|
}
|
|
|
|
int iomux_replace_device(const int console, const char *old, const char *new)
|
|
{
|
|
struct stdio_dev *dev;
|
|
char *arg = NULL; /* Initial empty list */
|
|
int size = 1; /* For NUL terminator */
|
|
int i, ret;
|
|
|
|
for_each_console_dev(i, console, dev) {
|
|
const char *name = strcmp(dev->name, old) ? dev->name : new;
|
|
char *tmp;
|
|
|
|
/* Append name with a ',' (comma) separator */
|
|
tmp = realloc(arg, size + strlen(name) + 1);
|
|
if (!tmp) {
|
|
free(arg);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
if (arg) {
|
|
strcat(tmp, ",");
|
|
strcat(tmp, name);
|
|
}
|
|
else
|
|
strcpy(tmp, name);
|
|
|
|
arg = tmp;
|
|
size = strlen(tmp) + 1;
|
|
}
|
|
|
|
ret = iomux_doenv(console, arg);
|
|
if (ret)
|
|
ret = -EINVAL;
|
|
|
|
free(arg);
|
|
return ret;
|
|
}
|
|
#endif /* CONSOLE_MUX */
|