-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathpanicmon.c
147 lines (111 loc) · 3.88 KB
/
panicmon.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
/* Copyright (c) Steve Maresca/Zentific LLC */
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/vmalloc.h>
#include <linux/mm.h>
#include <asm/pgtable.h>
#include <linux/notifier.h> /* For notifier support */
#include <linux/proc_fs.h>
#include <linux/inet.h>
#include <linux/stat.h>
MODULE_AUTHOR("[email protected]");
MODULE_DESCRIPTION("A last-gasp panic notifier");
MODULE_LICENSE("GPL");
#include <linux/netpoll.h>
#define MESSAGE_SIZE 1024
#define MODULE_NAME "panicmon"
/* -------------------------------------------------------------------------*/
static struct netpoll* np = NULL;
static struct netpoll np_t;
static char *iface = NULL;
static char *src_ip = NULL;
static char *dst_ip = NULL;
char message[MESSAGE_SIZE];
/* wish we could use u16 here, but module_param macros
* only handle u32. do range check
*/
static unsigned int src_port = 20000;
static unsigned int dst_port = 20000;
MODULE_PARM_DESC(iface, "Panicmon src iface");
MODULE_PARM_DESC(src_ip, "Panicmon src_ip");
MODULE_PARM_DESC(src_port, "Panicmon dst_ip");
MODULE_PARM_DESC(src_port, "Panicmon src_port");
MODULE_PARM_DESC(dst_port, "Panicmon dst_port");
module_param(iface, charp, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
module_param(src_ip, charp, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
module_param(dst_ip, charp, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
module_param(src_port, uint, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
module_param(dst_port, uint, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
static void update_panicmon_net(void) {
np_t.name = "LRNG";
// to auto-configure ... which is probably not desireable
// http://stackoverflow.com/questions/9941030/get-ip-address-of-local-nic-from-kernel-module
strlcpy(np_t.dev_name, iface, IFNAMSIZ);
/* for systems where this is not a union
*/
//np_t.local_ip = in_aton(src_ip);
//np_t.remote_ip = in_aton(dst_ip);
/* for systems where this is a union
*/
np_t.local_ip.ip = in_aton(src_ip);
np_t.remote_ip.ip = in_aton(dst_ip);
np_t.local_port = src_port;
np_t.remote_port = dst_port;
memset(np_t.remote_mac, 0xff, ETH_ALEN);
}
/*
* Panicing.
*
* Don't. But if you did, this is what happens.
*/
static int panicmon(struct notifier_block *nb, unsigned long l, void *p) {
int len = 0;
int buflen = 0;
char *buf = p;
if(buf){
buflen = strlen(buf);
}
//OH NO...\n == 25 c
//1024 - \0 - 25 = 998
sprintf(message,"OH NO WE ARE PANICING - %.998s\n", buf);
len = strlen(message);
// in case params have been updated
update_panicmon_net();
netpoll_send_udp(np,message,len);
return NOTIFY_DONE;
}
static struct notifier_block onpanic = {
.notifier_call = panicmon
};
int init_module(void) {
printk(KERN_INFO MODULE_NAME ": Loaded.\n");
//validation and init
if ( !src_ip ){
printk(KERN_INFO MODULE_NAME ": Could not initialize (src_ip parameter not set).\n");
return -1;
}
if ( !dst_ip ){
printk(KERN_INFO MODULE_NAME ": Could not initialize (dst_ip parameter not set).\n");
return -1;
}
if ( !iface ){
printk(KERN_INFO MODULE_NAME ": Could not initialize (iface parameter not set).\n");
return -1;
}
if ( src_port > 65535 || dst_port > 65535 || src_port == 0 || dst_port == 0 ){
printk(KERN_INFO MODULE_NAME ": Invalid src/dst port value. Valid range is [1,65535].\n");
return -1;
}
update_panicmon_net();
netpoll_print_options(&np_t);
netpoll_setup(&np_t);
np = &np_t;
atomic_notifier_chain_register(&panic_notifier_list, &onpanic);
printk(KERN_INFO MODULE_NAME ": Registered panic notifier.\n");
return 0;
}
void cleanup_module(void) {
atomic_notifier_chain_unregister(&panic_notifier_list, &onpanic);
printk(KERN_INFO MODULE_NAME ": Unregistered panic notifier.\n");
printk(KERN_INFO MODULE_NAME ": Uninitialized.\n");
}