Re: [PATCH v3 1/2] printk: console: Create console= parser that supports named options

From: Petr Mladek
Date: Wed Aug 31 2022 - 06:13:14 EST


On Wed 2022-07-20 18:48:11, Chris Down wrote:
> This will be used in the next patch, and for future changes like the
> proposed sync/nothread named options. This avoids having to use sigils
> like "/" to get different semantic meaning for different parts of the
> option string, and helps to make things more human readable and
> easily extensible.
>
> There are no functional changes for existing console= users.
>
> Signed-off-by: Chris Down <chris@xxxxxxxxxxxxxx>
> ---
> kernel/printk/printk.c | 26 +++++++++++++++++++++++++-
> 1 file changed, 25 insertions(+), 1 deletion(-)
>
> diff --git a/kernel/printk/printk.c b/kernel/printk/printk.c
> index b49c6ff6dca0..6094f773ad4a 100644
> --- a/kernel/printk/printk.c
> +++ b/kernel/printk/printk.c
> @@ -2368,6 +2368,30 @@ static void set_user_specified(struct console_cmdline *c, bool user_specified)
> console_set_on_cmdline = 1;
> }
>
> +static void parse_console_cmdline_options(struct console_cmdline *c,
> + char *options)
> +{

I think that the function does not work as expected.

> + bool seen_serial_opts = false;
> + char *key;
> +
> + while ((key = strsep(&options, ",")) != NULL) {
> + char *value;

strsep() replaces ',' with '\0'.

> +
> + value = strchr(key, ':');
> + if (value)
> + *(value++) = '\0';

This replaces ':' with '\0'.

> +
> + if (!seen_serial_opts && isdigit(key[0]) && !value) {

This catches the classic options in the format "bbbbpnf".

> + seen_serial_opts = true;
> + c->options = key;
> + continue;
> + }
> +
> + pr_err("ignoring invalid console option: '%s%s%s'\n", key,
> + value ? ":" : "", value ?: "");

IMHO, this would warn even about "io", "mmio", ... that are used by:

console=uart[8250],io,<addr>[,options]
console=uart[8250],mmio,<addr>[,options]

Warning: I am not completely sure about this. It seems that
this format is handled by univ8250_console_match().

IMHO, the "8250" is taken as "idx" in console_setup().
And "idx" parameter is ignored by univ8250_console_match().
This probably explains why "8250" is optional in console=
parameter.

> + }
> +}

Sigh, the parsing of console= parameter is really hacky. Very old code.
The name and idx is handled in console_setup(). The rest
is handled by particular drivers via "options".

This patch makes it even more tricky. It tries to handle some
*options in add_preferred_console(). But it replaces all ','
and ':' by '\0' so that drivers do not see the original *options
any longer.

I thought a lot how to do it a clean way. IMHO, it would be great to
parse everything at a single place but it might require updating
all drivers. I am not sure if it is worth it.

So, I suggest to do it another way. We could implement a generic
function to find in the new key[:value] format. It would check
if the given option (key) exists and read the optional value.

The optional value would allow to define another new options
that would not need any value, e.g. "kthread" or "atomic" that
might be used in the upcoming code that allows to offload
console handling to kthreads.

The function would look like:

/**
* find_and_remove_console_option() - find and remove particular option from
* console= parameter
*
* @options: pointer to the buffer with original "options"
* @key: option name to be found
* @buf: buffer for the found value, might be NULL
*
* Try to find "key[:value]" in the given @options string. Copy the value
* into the given @buf when any.
*
* Warning: The function is designed only to handle new generic console
* options. The found "key[:value]" is removed from the given
* @options string. The aim is to pass only the driver specific
* options to the legacy driver code.
*
* The new options might stay when all console drivers are able
* to handle it.
*
* The given @options buffer is modified to avoid allocations. It
* makes the life easier during early boot.
*/
bool find_and_remove_console_option(char *options, const char *key,
char *buf, int size)
{
bool found = false, copy_failed = false. trailing_comma = false;
char *opt, *val;

/* Find the option: key[:value] */
while (options && *options) {
opt = options;

options = strchr(options, ",");
if (options)
*options++ = '\0';

val = strchr(opt, ":");
if (val)
*val++ = '\0';

if (strcmp(opt, key) == 0) {
found = true;
break;
}

/* Not our key. Keep it in options. */
if (options) {
*(options - 1) = ',';
trailing_comma = 1;
}
if (val)
*(val - 1) = ':';
}

/* Copy the value if found. */
if (found && val) {
if (buf && size > strlen(val)) {
strscpy(buf, val, size);
} else {
pr_warn("Can't copy console= option value. Ignoring %s:%s\n",
opt, val);
copy_failed = true;
}
}

/* Remove the found option. */
if (found) {
if (options) {
strcpy(opt, options);
options = opt;
} else if (trailing_comma) {
/* Removing last option */
*(opt - 1) = '\0';
} else
*(opt) = '\0';
}
}

return found && !copy_failed;
}

then we could do something like:

void handle_per_console_loglevel_option(struct console_cmdline *c, char *options)
{
char buf[10];
int loglevel, ret;

if (!find_and_remove_console_option("options, "loglevel", buf, sizeof(buf)))
return;

if (kstrtoint(buf, 10, &loglevel)) {
pr_warn("Invalid console loglevel:%s\n", buf);
return;
}

if (loglevel < CONSOLE_LOGLEVEL_MIN || loglevel > CONSOLE_MOTORMOUTH) {
pr_warn("Console loglevel out of range: %d\n", loglevel);
return;
}

c->loglevel = loglevel;
c->flags |= CON_LOGLEVEL;
}

Note that the code is not even compile tested.

Any better ideas are highly appreciated.

Best Regards,
Petr