Commit c62c0a17 authored by Breno Leitao's avatar Breno Leitao Committed by Paolo Abeni

netconsole: Append kernel version to message

Create a new netconsole runtime option that prepends the kernel version in
the netconsole message. This is useful to map kernel messages to kernel
version in a simple way, i.e., without checking somewhere which kernel
version the host that sent the message is using.

If this option is selected, then the "<release>," is prepended before the
netconsole message. This is an example of a netconsole output, with
release feature enabled:

	6.4.0-01762-ga1ba2ffe946e;12,426,112883998,-;this is a test

Cc: Dave Jones <davej@codemonkey.org.uk>
Signed-off-by: default avatarBreno Leitao <leitao@debian.org>
Link: https://lore.kernel.org/r/20230714111330.3069605-1-leitao@debian.orgSigned-off-by: default avatarPaolo Abeni <pabeni@redhat.com>
parent a7f6eb19
...@@ -13,6 +13,8 @@ IPv6 support by Cong Wang <xiyou.wangcong@gmail.com>, Jan 1 2013 ...@@ -13,6 +13,8 @@ IPv6 support by Cong Wang <xiyou.wangcong@gmail.com>, Jan 1 2013
Extended console support by Tejun Heo <tj@kernel.org>, May 1 2015 Extended console support by Tejun Heo <tj@kernel.org>, May 1 2015
Release prepend support by Breno Leitao <leitao@debian.org>, Jul 7 2023
Please send bug reports to Matt Mackall <mpm@selenic.com> Please send bug reports to Matt Mackall <mpm@selenic.com>
Satyam Sharma <satyam.sharma@gmail.com>, and Cong Wang <xiyou.wangcong@gmail.com> Satyam Sharma <satyam.sharma@gmail.com>, and Cong Wang <xiyou.wangcong@gmail.com>
...@@ -34,10 +36,11 @@ Sender and receiver configuration: ...@@ -34,10 +36,11 @@ Sender and receiver configuration:
It takes a string configuration parameter "netconsole" in the It takes a string configuration parameter "netconsole" in the
following format:: following format::
netconsole=[+][src-port]@[src-ip]/[<dev>],[tgt-port]@<tgt-ip>/[tgt-macaddr] netconsole=[+][r][src-port]@[src-ip]/[<dev>],[tgt-port]@<tgt-ip>/[tgt-macaddr]
where where
+ if present, enable extended console support + if present, enable extended console support
r if present, prepend kernel version (release) to the message
src-port source for UDP packets (defaults to 6665) src-port source for UDP packets (defaults to 6665)
src-ip source IP to use (interface address) src-ip source IP to use (interface address)
dev network interface (eth0) dev network interface (eth0)
...@@ -125,6 +128,7 @@ The interface exposes these parameters of a netconsole target to userspace: ...@@ -125,6 +128,7 @@ The interface exposes these parameters of a netconsole target to userspace:
============== ================================= ============ ============== ================================= ============
enabled Is this target currently enabled? (read-write) enabled Is this target currently enabled? (read-write)
extended Extended mode enabled (read-write) extended Extended mode enabled (read-write)
release Prepend kernel release to message (read-write)
dev_name Local network interface name (read-write) dev_name Local network interface name (read-write)
local_port Source UDP port to use (read-write) local_port Source UDP port to use (read-write)
remote_port Remote agent's UDP port (read-write) remote_port Remote agent's UDP port (read-write)
...@@ -165,6 +169,11 @@ following format which is the same as /dev/kmsg:: ...@@ -165,6 +169,11 @@ following format which is the same as /dev/kmsg::
<level>,<sequnum>,<timestamp>,<contflag>;<message text> <level>,<sequnum>,<timestamp>,<contflag>;<message text>
If 'r' (release) feature is enabled, the kernel release version is
prepended to the start of the message. Example::
6.4.0,6,444,501151268,-;netconsole: network logging started
Non printable characters in <message text> are escaped using "\xff" Non printable characters in <message text> are escaped using "\xff"
notation. If the message contains optional dictionary, verbatim notation. If the message contains optional dictionary, verbatim
newline is used as the delimiter. newline is used as the delimiter.
......
...@@ -36,6 +36,7 @@ ...@@ -36,6 +36,7 @@
#include <linux/inet.h> #include <linux/inet.h>
#include <linux/configfs.h> #include <linux/configfs.h>
#include <linux/etherdevice.h> #include <linux/etherdevice.h>
#include <linux/utsname.h>
MODULE_AUTHOR("Maintainer: Matt Mackall <mpm@selenic.com>"); MODULE_AUTHOR("Maintainer: Matt Mackall <mpm@selenic.com>");
MODULE_DESCRIPTION("Console driver for network interfaces"); MODULE_DESCRIPTION("Console driver for network interfaces");
...@@ -84,6 +85,8 @@ static struct console netconsole_ext; ...@@ -84,6 +85,8 @@ static struct console netconsole_ext;
* Also, other parameters of a target may be modified at * Also, other parameters of a target may be modified at
* runtime only when it is disabled (enabled == 0). * runtime only when it is disabled (enabled == 0).
* @extended: Denotes whether console is extended or not. * @extended: Denotes whether console is extended or not.
* @release: Denotes whether kernel release version should be prepended
* to the message. Depends on extended console.
* @np: The netpoll structure for this target. * @np: The netpoll structure for this target.
* Contains the other userspace visible parameters: * Contains the other userspace visible parameters:
* dev_name (read-write) * dev_name (read-write)
...@@ -101,6 +104,7 @@ struct netconsole_target { ...@@ -101,6 +104,7 @@ struct netconsole_target {
#endif #endif
bool enabled; bool enabled;
bool extended; bool extended;
bool release;
struct netpoll np; struct netpoll np;
}; };
...@@ -188,6 +192,15 @@ static struct netconsole_target *alloc_param_target(char *target_config) ...@@ -188,6 +192,15 @@ static struct netconsole_target *alloc_param_target(char *target_config)
target_config++; target_config++;
} }
if (*target_config == 'r') {
if (!nt->extended) {
pr_err("Netconsole configuration error. Release feature requires extended log message");
goto fail;
}
nt->release = true;
target_config++;
}
/* Parse parameters and setup netpoll */ /* Parse parameters and setup netpoll */
err = netpoll_parse_options(&nt->np, target_config); err = netpoll_parse_options(&nt->np, target_config);
if (err) if (err)
...@@ -222,6 +235,7 @@ static void free_param_target(struct netconsole_target *nt) ...@@ -222,6 +235,7 @@ static void free_param_target(struct netconsole_target *nt)
* | * |
* <target>/ * <target>/
* | enabled * | enabled
* | release
* | dev_name * | dev_name
* | local_port * | local_port
* | remote_port * | remote_port
...@@ -254,6 +268,11 @@ static ssize_t extended_show(struct config_item *item, char *buf) ...@@ -254,6 +268,11 @@ static ssize_t extended_show(struct config_item *item, char *buf)
return snprintf(buf, PAGE_SIZE, "%d\n", to_target(item)->extended); return snprintf(buf, PAGE_SIZE, "%d\n", to_target(item)->extended);
} }
static ssize_t release_show(struct config_item *item, char *buf)
{
return snprintf(buf, PAGE_SIZE, "%d\n", to_target(item)->release);
}
static ssize_t dev_name_show(struct config_item *item, char *buf) static ssize_t dev_name_show(struct config_item *item, char *buf)
{ {
return snprintf(buf, PAGE_SIZE, "%s\n", to_target(item)->np.dev_name); return snprintf(buf, PAGE_SIZE, "%s\n", to_target(item)->np.dev_name);
...@@ -332,6 +351,11 @@ static ssize_t enabled_store(struct config_item *item, ...@@ -332,6 +351,11 @@ static ssize_t enabled_store(struct config_item *item,
} }
if (enabled) { /* true */ if (enabled) { /* true */
if (nt->release && !nt->extended) {
pr_err("Not enabling netconsole. Release feature requires extended log message");
goto out_unlock;
}
if (nt->extended && !console_is_registered(&netconsole_ext)) if (nt->extended && !console_is_registered(&netconsole_ext))
register_console(&netconsole_ext); register_console(&netconsole_ext);
...@@ -366,6 +390,38 @@ static ssize_t enabled_store(struct config_item *item, ...@@ -366,6 +390,38 @@ static ssize_t enabled_store(struct config_item *item,
return err; return err;
} }
static ssize_t release_store(struct config_item *item, const char *buf,
size_t count)
{
struct netconsole_target *nt = to_target(item);
int release;
int err;
mutex_lock(&dynamic_netconsole_mutex);
if (nt->enabled) {
pr_err("target (%s) is enabled, disable to update parameters\n",
config_item_name(&nt->item));
err = -EINVAL;
goto out_unlock;
}
err = kstrtoint(buf, 10, &release);
if (err < 0)
goto out_unlock;
if (release < 0 || release > 1) {
err = -EINVAL;
goto out_unlock;
}
nt->release = release;
mutex_unlock(&dynamic_netconsole_mutex);
return strnlen(buf, count);
out_unlock:
mutex_unlock(&dynamic_netconsole_mutex);
return err;
}
static ssize_t extended_store(struct config_item *item, const char *buf, static ssize_t extended_store(struct config_item *item, const char *buf,
size_t count) size_t count)
{ {
...@@ -576,10 +632,12 @@ CONFIGFS_ATTR(, local_ip); ...@@ -576,10 +632,12 @@ CONFIGFS_ATTR(, local_ip);
CONFIGFS_ATTR(, remote_ip); CONFIGFS_ATTR(, remote_ip);
CONFIGFS_ATTR_RO(, local_mac); CONFIGFS_ATTR_RO(, local_mac);
CONFIGFS_ATTR(, remote_mac); CONFIGFS_ATTR(, remote_mac);
CONFIGFS_ATTR(, release);
static struct configfs_attribute *netconsole_target_attrs[] = { static struct configfs_attribute *netconsole_target_attrs[] = {
&attr_enabled, &attr_enabled,
&attr_extended, &attr_extended,
&attr_release,
&attr_dev_name, &attr_dev_name,
&attr_local_port, &attr_local_port,
&attr_remote_port, &attr_remote_port,
...@@ -772,9 +830,23 @@ static void send_ext_msg_udp(struct netconsole_target *nt, const char *msg, ...@@ -772,9 +830,23 @@ static void send_ext_msg_udp(struct netconsole_target *nt, const char *msg,
const char *header, *body; const char *header, *body;
int offset = 0; int offset = 0;
int header_len, body_len; int header_len, body_len;
const char *msg_ready = msg;
const char *release;
int release_len = 0;
if (msg_len <= MAX_PRINT_CHUNK) { if (nt->release) {
netpoll_send_udp(&nt->np, msg, msg_len); release = init_utsname()->release;
release_len = strlen(release) + 1;
}
if (msg_len + release_len <= MAX_PRINT_CHUNK) {
/* No fragmentation needed */
if (nt->release) {
scnprintf(buf, MAX_PRINT_CHUNK, "%s,%s", release, msg);
msg_len += release_len;
msg_ready = buf;
}
netpoll_send_udp(&nt->np, msg_ready, msg_len);
return; return;
} }
...@@ -792,7 +864,10 @@ static void send_ext_msg_udp(struct netconsole_target *nt, const char *msg, ...@@ -792,7 +864,10 @@ static void send_ext_msg_udp(struct netconsole_target *nt, const char *msg,
* Transfer multiple chunks with the following extra header. * Transfer multiple chunks with the following extra header.
* "ncfrag=<byte-offset>/<total-bytes>" * "ncfrag=<byte-offset>/<total-bytes>"
*/ */
memcpy(buf, header, header_len); if (nt->release)
scnprintf(buf, MAX_PRINT_CHUNK, "%s,", release);
memcpy(buf + release_len, header, header_len);
header_len += release_len;
while (offset < body_len) { while (offset < body_len) {
int this_header = header_len; int this_header = header_len;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment