netconsole.c

来自「linux 内核源代码」· C语言 代码 · 共 816 行 · 第 1/2 页

C
816
字号
/* *  linux/drivers/net/netconsole.c * *  Copyright (C) 2001  Ingo Molnar <mingo@redhat.com> * *  This file contains the implementation of an IRQ-safe, crash-safe *  kernel console implementation that outputs kernel messages to the *  network. * * Modification history: * * 2001-09-17    started by Ingo Molnar. * 2003-08-11    2.6 port by Matt Mackall *               simplified options *               generic card hooks *               works non-modular * 2003-09-07    rewritten with netpoll api *//**************************************************************** *      This program is free software; you can redistribute it and/or modify *      it under the terms of the GNU General Public License as published by *      the Free Software Foundation; either version 2, or (at your option) *      any later version. * *      This program is distributed in the hope that it will be useful, *      but WITHOUT ANY WARRANTY; without even the implied warranty of *      MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the *      GNU General Public License for more details. * *      You should have received a copy of the GNU General Public License *      along with this program; if not, write to the Free Software *      Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * ****************************************************************/#include <linux/mm.h>#include <linux/init.h>#include <linux/module.h>#include <linux/console.h>#include <linux/moduleparam.h>#include <linux/string.h>#include <linux/netpoll.h>#include <linux/inet.h>#include <linux/configfs.h>MODULE_AUTHOR("Maintainer: Matt Mackall <mpm@selenic.com>");MODULE_DESCRIPTION("Console driver for network interfaces");MODULE_LICENSE("GPL");#define MAX_PARAM_LENGTH	256#define MAX_PRINT_CHUNK		1000static char config[MAX_PARAM_LENGTH];module_param_string(netconsole, config, MAX_PARAM_LENGTH, 0);MODULE_PARM_DESC(netconsole, " netconsole=[src-port]@[src-ip]/[dev],[tgt-port]@<tgt-ip>/[tgt-macaddr]\n");#ifndef	MODULEstatic int __init option_setup(char *opt){	strlcpy(config, opt, MAX_PARAM_LENGTH);	return 1;}__setup("netconsole=", option_setup);#endif	/* MODULE *//* Linked list of all configured targets */static LIST_HEAD(target_list);/* This needs to be a spinlock because write_msg() cannot sleep */static DEFINE_SPINLOCK(target_list_lock);/** * struct netconsole_target - Represents a configured netconsole target. * @list:	Links this target into the target_list. * @item:	Links us into the configfs subsystem hierarchy. * @enabled:	On / off knob to enable / disable target. *		Visible from userspace (read-write). *		We maintain a strict 1:1 correspondence between this and *		whether the corresponding netpoll is active or inactive. *		Also, other parameters of a target may be modified at *		runtime only when it is disabled (enabled == 0). * @np:		The netpoll structure for this target. *		Contains the other userspace visible parameters: *		dev_name	(read-write) *		local_port	(read-write) *		remote_port	(read-write) *		local_ip	(read-write) *		remote_ip	(read-write) *		local_mac	(read-only) *		remote_mac	(read-write) */struct netconsole_target {	struct list_head	list;#ifdef	CONFIG_NETCONSOLE_DYNAMIC	struct config_item	item;#endif	int			enabled;	struct netpoll		np;};#ifdef	CONFIG_NETCONSOLE_DYNAMICstatic struct configfs_subsystem netconsole_subsys;static int __init dynamic_netconsole_init(void){	config_group_init(&netconsole_subsys.su_group);	mutex_init(&netconsole_subsys.su_mutex);	return configfs_register_subsystem(&netconsole_subsys);}static void __exit dynamic_netconsole_exit(void){	configfs_unregister_subsystem(&netconsole_subsys);}/* * Targets that were created by parsing the boot/module option string * do not exist in the configfs hierarchy (and have NULL names) and will * never go away, so make these a no-op for them. */static void netconsole_target_get(struct netconsole_target *nt){	if (config_item_name(&nt->item))		config_item_get(&nt->item);}static void netconsole_target_put(struct netconsole_target *nt){	if (config_item_name(&nt->item))		config_item_put(&nt->item);}#else	/* !CONFIG_NETCONSOLE_DYNAMIC */static int __init dynamic_netconsole_init(void){	return 0;}static void __exit dynamic_netconsole_exit(void){}/* * No danger of targets going away from under us when dynamic * reconfigurability is off. */static void netconsole_target_get(struct netconsole_target *nt){}static void netconsole_target_put(struct netconsole_target *nt){}#endif	/* CONFIG_NETCONSOLE_DYNAMIC *//* Allocate new target (from boot/module param) and setup netpoll for it */static struct netconsole_target *alloc_param_target(char *target_config){	int err = -ENOMEM;	struct netconsole_target *nt;	/*	 * Allocate and initialize with defaults.	 * Note that these targets get their config_item fields zeroed-out.	 */	nt = kzalloc(sizeof(*nt), GFP_KERNEL);	if (!nt) {		printk(KERN_ERR "netconsole: failed to allocate memory\n");		goto fail;	}	nt->np.name = "netconsole";	strlcpy(nt->np.dev_name, "eth0", IFNAMSIZ);	nt->np.local_port = 6665;	nt->np.remote_port = 6666;	memset(nt->np.remote_mac, 0xff, ETH_ALEN);	/* Parse parameters and setup netpoll */	err = netpoll_parse_options(&nt->np, target_config);	if (err)		goto fail;	err = netpoll_setup(&nt->np);	if (err)		goto fail;	nt->enabled = 1;	return nt;fail:	kfree(nt);	return ERR_PTR(err);}/* Cleanup netpoll for given target (from boot/module param) and free it */static void free_param_target(struct netconsole_target *nt){	netpoll_cleanup(&nt->np);	kfree(nt);}#ifdef	CONFIG_NETCONSOLE_DYNAMIC/* * Our subsystem hierarchy is: * * /sys/kernel/config/netconsole/ *				| *				<target>/ *				|	enabled *				|	dev_name *				|	local_port *				|	remote_port *				|	local_ip *				|	remote_ip *				|	local_mac *				|	remote_mac *				| *				<target>/... */struct netconsole_target_attr {	struct configfs_attribute	attr;	ssize_t				(*show)(struct netconsole_target *nt,						char *buf);	ssize_t				(*store)(struct netconsole_target *nt,						 const char *buf,						 size_t count);};static struct netconsole_target *to_target(struct config_item *item){	return item ?		container_of(item, struct netconsole_target, item) :		NULL;}/* * Wrapper over simple_strtol (base 10) with sanity and range checking. * We return (signed) long only because we may want to return errors. * Do not use this to convert numbers that are allowed to be negative. */static long strtol10_check_range(const char *cp, long min, long max){	long ret;	char *p = (char *) cp;	WARN_ON(min < 0);	WARN_ON(max < min);	ret = simple_strtol(p, &p, 10);	if (*p && (*p != '\n')) {		printk(KERN_ERR "netconsole: invalid input\n");		return -EINVAL;	}	if ((ret < min) || (ret > max)) {		printk(KERN_ERR "netconsole: input %ld must be between "				"%ld and %ld\n", ret, min, max);		return -EINVAL;	}	return ret;}/* * Attribute operations for netconsole_target. */static ssize_t show_enabled(struct netconsole_target *nt, char *buf){	return snprintf(buf, PAGE_SIZE, "%d\n", nt->enabled);}static ssize_t show_dev_name(struct netconsole_target *nt, char *buf){	return snprintf(buf, PAGE_SIZE, "%s\n", nt->np.dev_name);}static ssize_t show_local_port(struct netconsole_target *nt, char *buf){	return snprintf(buf, PAGE_SIZE, "%d\n", nt->np.local_port);}static ssize_t show_remote_port(struct netconsole_target *nt, char *buf){	return snprintf(buf, PAGE_SIZE, "%d\n", nt->np.remote_port);}static ssize_t show_local_ip(struct netconsole_target *nt, char *buf){	return snprintf(buf, PAGE_SIZE, "%d.%d.%d.%d\n",			HIPQUAD(nt->np.local_ip));}static ssize_t show_remote_ip(struct netconsole_target *nt, char *buf){	return snprintf(buf, PAGE_SIZE, "%d.%d.%d.%d\n",			HIPQUAD(nt->np.remote_ip));}static ssize_t show_local_mac(struct netconsole_target *nt, char *buf){	DECLARE_MAC_BUF(mac);	return snprintf(buf, PAGE_SIZE, "%s\n",			print_mac(mac, nt->np.local_mac));}static ssize_t show_remote_mac(struct netconsole_target *nt, char *buf){	DECLARE_MAC_BUF(mac);	return snprintf(buf, PAGE_SIZE, "%s\n",			print_mac(mac, nt->np.remote_mac));}/* * This one is special -- targets created through the configfs interface * are not enabled (and the corresponding netpoll activated) by default. * The user is expected to set the desired parameters first (which * would enable him to dynamically add new netpoll targets for new * network interfaces as and when they come up). */static ssize_t store_enabled(struct netconsole_target *nt,			     const char *buf,			     size_t count){	int err;	long enabled;	enabled = strtol10_check_range(buf, 0, 1);	if (enabled < 0)		return enabled;	if (enabled) {	/* 1 */		/*		 * Skip netpoll_parse_options() -- all the attributes are		 * already configured via configfs. Just print them out.		 */		netpoll_print_options(&nt->np);		err = netpoll_setup(&nt->np);		if (err)			return err;		printk(KERN_INFO "netconsole: network logging started\n");	} else {	/* 0 */		netpoll_cleanup(&nt->np);	}	nt->enabled = enabled;	return strnlen(buf, count);}static ssize_t store_dev_name(struct netconsole_target *nt,			      const char *buf,			      size_t count){	size_t len;	if (nt->enabled) {		printk(KERN_ERR "netconsole: target (%s) is enabled, "				"disable to update parameters\n",				config_item_name(&nt->item));		return -EINVAL;	}	strlcpy(nt->np.dev_name, buf, IFNAMSIZ);	/* Get rid of possible trailing newline from echo(1) */	len = strnlen(nt->np.dev_name, IFNAMSIZ);	if (nt->np.dev_name[len - 1] == '\n')		nt->np.dev_name[len - 1] = '\0';	return strnlen(buf, count);}static ssize_t store_local_port(struct netconsole_target *nt,				const char *buf,				size_t count){	long local_port;#define __U16_MAX	((__u16) ~0U)	if (nt->enabled) {		printk(KERN_ERR "netconsole: target (%s) is enabled, "				"disable to update parameters\n",				config_item_name(&nt->item));		return -EINVAL;	}	local_port = strtol10_check_range(buf, 0, __U16_MAX);	if (local_port < 0)		return local_port;	nt->np.local_port = local_port;	return strnlen(buf, count);}static ssize_t store_remote_port(struct netconsole_target *nt,

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?