blob: 49e3b49e552f7f81dea63e708bbb0abf1e32a3f4 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Copyright (C) 2001 Lennert Buytenhek (buytenh@gnu.org)
Jeff Dike827b3f62008-02-04 22:31:29 -08003 * Copyright (C) 2001 - 2008 Jeff Dike (jdike@{addtoit,linux.intel}.com)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004 * Licensed under the GPL
5 */
6
Jeff Dike827b3f62008-02-04 22:31:29 -08007#include <linux/console.h>
8#include <linux/ctype.h>
André Goddard Rosae7d28602009-12-14 18:01:06 -08009#include <linux/string.h>
Jeff Dike827b3f62008-02-04 22:31:29 -080010#include <linux/interrupt.h>
11#include <linux/list.h>
12#include <linux/mm.h>
13#include <linux/module.h>
14#include <linux/notifier.h>
15#include <linux/reboot.h>
16#include <linux/proc_fs.h>
17#include <linux/slab.h>
18#include <linux/syscalls.h>
19#include <linux/utsname.h>
Balbir Singh36137122008-12-09 13:14:07 -080020#include <linux/socket.h>
21#include <linux/un.h>
Jeff Dike827b3f62008-02-04 22:31:29 -080022#include <linux/workqueue.h>
23#include <linux/mutex.h>
Al Viro723a1d72012-08-19 13:00:49 -040024#include <linux/fs.h>
25#include <linux/mount.h>
26#include <linux/file.h>
Jeff Dike827b3f62008-02-04 22:31:29 -080027#include <asm/uaccess.h>
Richard Weinbergera3a85a72012-03-29 18:47:46 +020028#include <asm/switch_to.h>
Jeff Dike827b3f62008-02-04 22:31:29 -080029
Al Viro37185b32012-10-08 03:27:32 +010030#include <init.h>
31#include <irq_kern.h>
32#include <irq_user.h>
33#include <kern_util.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070034#include "mconsole.h"
35#include "mconsole_kern.h"
Al Viro37185b32012-10-08 03:27:32 +010036#include <os.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070037
Jeff Diked50084a2006-01-06 00:18:50 -080038static int do_unlink_socket(struct notifier_block *notifier,
Linus Torvalds1da177e2005-04-16 15:20:36 -070039 unsigned long what, void *data)
40{
Jeff Dikeae2587e2007-10-16 01:26:57 -070041 return mconsole_unlink_socket();
Linus Torvalds1da177e2005-04-16 15:20:36 -070042}
43
44
45static struct notifier_block reboot_notifier = {
46 .notifier_call = do_unlink_socket,
47 .priority = 0,
48};
49
Jeff Diked50084a2006-01-06 00:18:50 -080050/* Safe without explicit locking for now. Tasklets provide their own
Linus Torvalds1da177e2005-04-16 15:20:36 -070051 * locking, and the interrupt handler is safe because it can't interrupt
52 * itself and it can only happen on CPU 0.
53 */
54
Jeff Dike90107722006-01-06 00:18:54 -080055static LIST_HEAD(mc_requests);
Linus Torvalds1da177e2005-04-16 15:20:36 -070056
David Howells6d5aefb2006-12-05 19:36:26 +000057static void mc_work_proc(struct work_struct *unused)
Linus Torvalds1da177e2005-04-16 15:20:36 -070058{
59 struct mconsole_entry *req;
60 unsigned long flags;
61
Jeff Dikeae2587e2007-10-16 01:26:57 -070062 while (!list_empty(&mc_requests)) {
Paolo 'Blaisorblade' Giarrussodbdb4c02006-04-10 22:53:39 -070063 local_irq_save(flags);
Jeff Dikeae2587e2007-10-16 01:26:57 -070064 req = list_entry(mc_requests.next, struct mconsole_entry, list);
Linus Torvalds1da177e2005-04-16 15:20:36 -070065 list_del(&req->list);
66 local_irq_restore(flags);
67 req->request.cmd->handler(&req->request);
68 kfree(req);
69 }
70}
71
David Howells6d5aefb2006-12-05 19:36:26 +000072static DECLARE_WORK(mconsole_work, mc_work_proc);
Linus Torvalds1da177e2005-04-16 15:20:36 -070073
Al Viro7bea96f2006-10-08 22:49:34 +010074static irqreturn_t mconsole_interrupt(int irq, void *dev_id)
Linus Torvalds1da177e2005-04-16 15:20:36 -070075{
76 /* long to avoid size mismatch warnings from gcc */
77 long fd;
78 struct mconsole_entry *new;
Al Viro3a512372006-10-24 11:15:29 +010079 static struct mc_request req; /* that's OK */
Linus Torvalds1da177e2005-04-16 15:20:36 -070080
81 fd = (long) dev_id;
Jeff Dikeae2587e2007-10-16 01:26:57 -070082 while (mconsole_get_request(fd, &req)) {
83 if (req.cmd->context == MCONSOLE_INTR)
Linus Torvalds1da177e2005-04-16 15:20:36 -070084 (*req.cmd->handler)(&req);
85 else {
Jeff Dike60baa152006-04-10 22:53:28 -070086 new = kmalloc(sizeof(*new), GFP_NOWAIT);
Jeff Dikeae2587e2007-10-16 01:26:57 -070087 if (new == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -070088 mconsole_reply(&req, "Out of memory", 1, 0);
89 else {
90 new->request = req;
Al Viro3a512372006-10-24 11:15:29 +010091 new->request.regs = get_irq_regs()->regs;
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 list_add(&new->list, &mc_requests);
93 }
94 }
95 }
Jeff Dikeae2587e2007-10-16 01:26:57 -070096 if (!list_empty(&mc_requests))
Linus Torvalds1da177e2005-04-16 15:20:36 -070097 schedule_work(&mconsole_work);
98 reactivate_fd(fd, MCONSOLE_IRQ);
Jeff Dikeae2587e2007-10-16 01:26:57 -070099 return IRQ_HANDLED;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100}
101
102void mconsole_version(struct mc_request *req)
103{
104 char version[256];
105
Serge E. Hallyne9ff3992006-10-02 02:18:11 -0700106 sprintf(version, "%s %s %s %s %s", utsname()->sysname,
Jeff Dikeae2587e2007-10-16 01:26:57 -0700107 utsname()->nodename, utsname()->release, utsname()->version,
108 utsname()->machine);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700109 mconsole_reply(req, version, 0, 0);
110}
111
112void mconsole_log(struct mc_request *req)
113{
114 int len;
115 char *ptr = req->request.data;
116
117 ptr += strlen("log ");
118
119 len = req->len - (ptr - req->request.data);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700120 printk(KERN_WARNING "%.*s", len, ptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700121 mconsole_reply(req, "", 0, 0);
122}
123
Linus Torvalds1da177e2005-04-16 15:20:36 -0700124void mconsole_proc(struct mc_request *req)
125{
Al Viro4ecf09f2009-12-24 00:44:44 -0500126 struct vfsmount *mnt = current->nsproxy->pid_ns->proc_mnt;
Al Viro723a1d72012-08-19 13:00:49 -0400127 char *buf;
128 int len;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700129 struct file *file;
Al Viro723a1d72012-08-19 13:00:49 -0400130 int first_chunk = 1;
131 char *ptr = req->request.data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700132
133 ptr += strlen("proc");
André Goddard Rosae7d28602009-12-14 18:01:06 -0800134 ptr = skip_spaces(ptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700135
Al Viro73d049a2011-03-11 12:08:24 -0500136 file = file_open_root(mnt->mnt_root, mnt, ptr, O_RDONLY);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700137 if (IS_ERR(file)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700138 mconsole_reply(req, "Failed to open file", 1, 0);
Al Viro723a1d72012-08-19 13:00:49 -0400139 printk(KERN_ERR "open /proc/%s: %ld\n", ptr, PTR_ERR(file));
Al Viro4ecf09f2009-12-24 00:44:44 -0500140 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700141 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700142
143 buf = kmalloc(PAGE_SIZE, GFP_KERNEL);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700144 if (buf == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700145 mconsole_reply(req, "Failed to allocate buffer", 1, 0);
146 goto out_fput;
147 }
148
Al Viro723a1d72012-08-19 13:00:49 -0400149 do {
150 loff_t pos;
151 mm_segment_t old_fs = get_fs();
152 set_fs(KERNEL_DS);
153 len = vfs_read(file, buf, PAGE_SIZE - 1, &pos);
154 set_fs(old_fs);
155 file->f_pos = pos;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700156 if (len < 0) {
157 mconsole_reply(req, "Read of file failed", 1, 0);
158 goto out_free;
159 }
Jeff Dikeae2587e2007-10-16 01:26:57 -0700160 /* Begin the file content on his own line. */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700161 if (first_chunk) {
162 mconsole_reply(req, "\n", 0, 1);
163 first_chunk = 0;
164 }
Al Viro723a1d72012-08-19 13:00:49 -0400165 buf[len] = '\0';
166 mconsole_reply(req, buf, 0, (len > 0));
167 } while (len > 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700168 out_free:
169 kfree(buf);
Al Viro723a1d72012-08-19 13:00:49 -0400170 out_fput:
171 fput(file);
172 out: ;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700173}
174
175#define UML_MCONSOLE_HELPTEXT \
176"Commands: \n\
177 version - Get kernel version \n\
178 help - Print this message \n\
179 halt - Halt UML \n\
180 reboot - Reboot UML \n\
181 config <dev>=<config> - Add a new device to UML; \n\
182 same syntax as command line \n\
183 config <dev> - Query the configuration of a device \n\
184 remove <dev> - Remove a device from UML \n\
185 sysrq <letter> - Performs the SysRq action controlled by the letter \n\
Jeff Dikedb805812006-02-01 03:06:23 -0800186 cad - invoke the Ctrl-Alt-Del handler \n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700187 stop - pause the UML; it will do nothing until it receives a 'go' \n\
188 go - continue the UML after a 'stop' \n\
189 log <string> - make UML enter <string> into the kernel log\n\
190 proc <file> - returns the contents of the UML's /proc/<file>\n\
Jeff Dike3eddddc2005-09-16 19:27:46 -0700191 stack <pid> - returns the stack of the specified pid\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700192"
193
194void mconsole_help(struct mc_request *req)
195{
196 mconsole_reply(req, UML_MCONSOLE_HELPTEXT, 0, 0);
197}
198
199void mconsole_halt(struct mc_request *req)
200{
201 mconsole_reply(req, "", 0, 0);
202 machine_halt();
203}
204
205void mconsole_reboot(struct mc_request *req)
206{
207 mconsole_reply(req, "", 0, 0);
208 machine_restart(NULL);
209}
210
Linus Torvalds1da177e2005-04-16 15:20:36 -0700211void mconsole_cad(struct mc_request *req)
212{
213 mconsole_reply(req, "", 0, 0);
214 ctrl_alt_del();
215}
216
217void mconsole_go(struct mc_request *req)
218{
219 mconsole_reply(req, "Not stopped", 1, 0);
220}
221
222void mconsole_stop(struct mc_request *req)
223{
224 deactivate_fd(req->originating_fd, MCONSOLE_IRQ);
225 os_set_fd_block(req->originating_fd, 1);
Al Viro3a512372006-10-24 11:15:29 +0100226 mconsole_reply(req, "stopped", 0, 0);
Karol Swietlickicc0be0f2008-02-04 22:31:25 -0800227 for (;;) {
228 if (!mconsole_get_request(req->originating_fd, req))
229 continue;
Al Viro3a512372006-10-24 11:15:29 +0100230 if (req->cmd->handler == mconsole_go)
231 break;
232 if (req->cmd->handler == mconsole_stop) {
233 mconsole_reply(req, "Already stopped", 1, 0);
234 continue;
235 }
236 if (req->cmd->handler == mconsole_sysrq) {
237 struct pt_regs *old_regs;
238 old_regs = set_irq_regs((struct pt_regs *)&req->regs);
239 mconsole_sysrq(req);
240 set_irq_regs(old_regs);
241 continue;
242 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243 (*req->cmd->handler)(req);
244 }
245 os_set_fd_block(req->originating_fd, 0);
246 reactivate_fd(req->originating_fd, MCONSOLE_IRQ);
247 mconsole_reply(req, "", 0, 0);
248}
249
Jeff Dike84f48d42007-02-10 01:44:01 -0800250static DEFINE_SPINLOCK(mc_devices_lock);
Paolo 'Blaisorblade' Giarrusso42947cb2006-02-01 03:06:29 -0800251static LIST_HEAD(mconsole_devices);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700252
253void mconsole_register_dev(struct mc_device *new)
254{
Jeff Dike84f48d42007-02-10 01:44:01 -0800255 spin_lock(&mc_devices_lock);
256 BUG_ON(!list_empty(&new->list));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257 list_add(&new->list, &mconsole_devices);
Jeff Dike84f48d42007-02-10 01:44:01 -0800258 spin_unlock(&mc_devices_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700259}
260
261static struct mc_device *mconsole_find_dev(char *name)
262{
263 struct list_head *ele;
264 struct mc_device *dev;
265
Jeff Dikeae2587e2007-10-16 01:26:57 -0700266 list_for_each(ele, &mconsole_devices) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700267 dev = list_entry(ele, struct mc_device, list);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700268 if (!strncmp(name, dev->name, strlen(dev->name)))
269 return dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700270 }
Jeff Dikeae2587e2007-10-16 01:26:57 -0700271 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700272}
273
Jeff Dike02dea082006-03-31 02:30:08 -0800274#define UNPLUGGED_PER_PAGE \
275 ((PAGE_SIZE - sizeof(struct list_head)) / sizeof(unsigned long))
276
277struct unplugged_pages {
278 struct list_head list;
279 void *pages[UNPLUGGED_PER_PAGE];
280};
281
Daniel Walkere98fa282008-02-04 22:31:27 -0800282static DEFINE_MUTEX(plug_mem_mutex);
Jeff Dike02dea082006-03-31 02:30:08 -0800283static unsigned long long unplugged_pages_count = 0;
Jeff Dikec59bce62007-02-10 01:44:04 -0800284static LIST_HEAD(unplugged_pages);
Jeff Dike02dea082006-03-31 02:30:08 -0800285static int unplug_index = UNPLUGGED_PER_PAGE;
286
Jeff Dikef28169d2007-02-10 01:43:53 -0800287static int mem_config(char *str, char **error_out)
Jeff Dike02dea082006-03-31 02:30:08 -0800288{
289 unsigned long long diff;
290 int err = -EINVAL, i, add;
291 char *ret;
292
Jeff Dikeae2587e2007-10-16 01:26:57 -0700293 if (str[0] != '=') {
Jeff Dikef28169d2007-02-10 01:43:53 -0800294 *error_out = "Expected '=' after 'mem'";
Jeff Dike02dea082006-03-31 02:30:08 -0800295 goto out;
Jeff Dikef28169d2007-02-10 01:43:53 -0800296 }
Jeff Dike02dea082006-03-31 02:30:08 -0800297
298 str++;
Jeff Dikeae2587e2007-10-16 01:26:57 -0700299 if (str[0] == '-')
Jeff Dike02dea082006-03-31 02:30:08 -0800300 add = 0;
Jeff Dikeae2587e2007-10-16 01:26:57 -0700301 else if (str[0] == '+') {
Jeff Dike02dea082006-03-31 02:30:08 -0800302 add = 1;
303 }
Jeff Dikef28169d2007-02-10 01:43:53 -0800304 else {
305 *error_out = "Expected increment to start with '-' or '+'";
306 goto out;
307 }
Jeff Dike02dea082006-03-31 02:30:08 -0800308
309 str++;
310 diff = memparse(str, &ret);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700311 if (*ret != '\0') {
Jeff Dikef28169d2007-02-10 01:43:53 -0800312 *error_out = "Failed to parse memory increment";
Jeff Dike02dea082006-03-31 02:30:08 -0800313 goto out;
Jeff Dikef28169d2007-02-10 01:43:53 -0800314 }
Jeff Dike02dea082006-03-31 02:30:08 -0800315
316 diff /= PAGE_SIZE;
317
Daniel Walkere98fa282008-02-04 22:31:27 -0800318 mutex_lock(&plug_mem_mutex);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700319 for (i = 0; i < diff; i++) {
Jeff Dike02dea082006-03-31 02:30:08 -0800320 struct unplugged_pages *unplugged;
321 void *addr;
322
Jeff Dikeae2587e2007-10-16 01:26:57 -0700323 if (add) {
324 if (list_empty(&unplugged_pages))
Jeff Dike02dea082006-03-31 02:30:08 -0800325 break;
326
327 unplugged = list_entry(unplugged_pages.next,
328 struct unplugged_pages, list);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700329 if (unplug_index > 0)
Jeff Dike02dea082006-03-31 02:30:08 -0800330 addr = unplugged->pages[--unplug_index];
331 else {
332 list_del(&unplugged->list);
333 addr = unplugged;
334 unplug_index = UNPLUGGED_PER_PAGE;
335 }
336
337 free_page((unsigned long) addr);
338 unplugged_pages_count--;
339 }
340 else {
341 struct page *page;
342
343 page = alloc_page(GFP_ATOMIC);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700344 if (page == NULL)
Jeff Dike02dea082006-03-31 02:30:08 -0800345 break;
346
347 unplugged = page_address(page);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700348 if (unplug_index == UNPLUGGED_PER_PAGE) {
Jeff Dike02dea082006-03-31 02:30:08 -0800349 list_add(&unplugged->list, &unplugged_pages);
350 unplug_index = 0;
351 }
352 else {
353 struct list_head *entry = unplugged_pages.next;
354 addr = unplugged;
355
356 unplugged = list_entry(entry,
357 struct unplugged_pages,
358 list);
Jeff Dike02dea082006-03-31 02:30:08 -0800359 err = os_drop_memory(addr, PAGE_SIZE);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700360 if (err) {
361 printk(KERN_ERR "Failed to release "
362 "memory - errno = %d\n", err);
Jeff Dikef28169d2007-02-10 01:43:53 -0800363 *error_out = "Failed to release memory";
Jeff Dike84f48d42007-02-10 01:44:01 -0800364 goto out_unlock;
Jeff Dikef28169d2007-02-10 01:43:53 -0800365 }
366 unplugged->pages[unplug_index++] = addr;
Jeff Dike02dea082006-03-31 02:30:08 -0800367 }
368
369 unplugged_pages_count++;
370 }
371 }
372
373 err = 0;
Jeff Dike84f48d42007-02-10 01:44:01 -0800374out_unlock:
Daniel Walkere98fa282008-02-04 22:31:27 -0800375 mutex_unlock(&plug_mem_mutex);
Jeff Dike02dea082006-03-31 02:30:08 -0800376out:
377 return err;
378}
379
380static int mem_get_config(char *name, char *str, int size, char **error_out)
381{
382 char buf[sizeof("18446744073709551615")];
383 int len = 0;
384
385 sprintf(buf, "%ld", uml_physmem);
386 CONFIG_CHUNK(str, size, len, buf, 1);
387
388 return len;
389}
390
391static int mem_id(char **str, int *start_out, int *end_out)
392{
393 *start_out = 0;
394 *end_out = 0;
395
396 return 0;
397}
398
Jeff Dikef28169d2007-02-10 01:43:53 -0800399static int mem_remove(int n, char **error_out)
Jeff Dike02dea082006-03-31 02:30:08 -0800400{
Jeff Dikef28169d2007-02-10 01:43:53 -0800401 *error_out = "Memory doesn't support the remove operation";
Jeff Dike02dea082006-03-31 02:30:08 -0800402 return -EBUSY;
403}
404
405static struct mc_device mem_mc = {
Jeff Dike84f48d42007-02-10 01:44:01 -0800406 .list = LIST_HEAD_INIT(mem_mc.list),
Jeff Dike02dea082006-03-31 02:30:08 -0800407 .name = "mem",
408 .config = mem_config,
409 .get_config = mem_get_config,
410 .id = mem_id,
411 .remove = mem_remove,
412};
413
Jeff Dike97a1fcb2007-07-23 18:43:48 -0700414static int __init mem_mc_init(void)
Jeff Dike02dea082006-03-31 02:30:08 -0800415{
Jeff Dikeae2587e2007-10-16 01:26:57 -0700416 if (can_drop_memory())
Jeff Dike02dea082006-03-31 02:30:08 -0800417 mconsole_register_dev(&mem_mc);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700418 else printk(KERN_ERR "Can't release memory to the host - memory "
419 "hotplug won't be supported\n");
Jeff Dike02dea082006-03-31 02:30:08 -0800420 return 0;
421}
422
423__initcall(mem_mc_init);
424
Linus Torvalds1da177e2005-04-16 15:20:36 -0700425#define CONFIG_BUF_SIZE 64
426
Jeff Diked50084a2006-01-06 00:18:50 -0800427static void mconsole_get_config(int (*get_config)(char *, char *, int,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700428 char **),
429 struct mc_request *req, char *name)
430{
431 char default_buf[CONFIG_BUF_SIZE], *error, *buf;
432 int n, size;
433
Jeff Dikeae2587e2007-10-16 01:26:57 -0700434 if (get_config == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700435 mconsole_reply(req, "No get_config routine defined", 1, 0);
436 return;
437 }
438
439 error = NULL;
Jeff Dike91b165c2006-09-25 23:33:00 -0700440 size = ARRAY_SIZE(default_buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700441 buf = default_buf;
442
Jeff Dikeae2587e2007-10-16 01:26:57 -0700443 while (1) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700444 n = (*get_config)(name, buf, size, &error);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700445 if (error != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700446 mconsole_reply(req, error, 1, 0);
447 goto out;
448 }
449
Jeff Dikeae2587e2007-10-16 01:26:57 -0700450 if (n <= size) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700451 mconsole_reply(req, buf, 0, 0);
452 goto out;
453 }
454
Jeff Dikeae2587e2007-10-16 01:26:57 -0700455 if (buf != default_buf)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700456 kfree(buf);
457
458 size = n;
459 buf = kmalloc(size, GFP_KERNEL);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700460 if (buf == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700461 mconsole_reply(req, "Failed to allocate buffer", 1, 0);
462 return;
463 }
464 }
465 out:
Jeff Dikeae2587e2007-10-16 01:26:57 -0700466 if (buf != default_buf)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700467 kfree(buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700468}
469
470void mconsole_config(struct mc_request *req)
471{
472 struct mc_device *dev;
Jeff Dikef28169d2007-02-10 01:43:53 -0800473 char *ptr = req->request.data, *name, *error_string = "";
Linus Torvalds1da177e2005-04-16 15:20:36 -0700474 int err;
475
476 ptr += strlen("config");
André Goddard Rosae7d28602009-12-14 18:01:06 -0800477 ptr = skip_spaces(ptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700478 dev = mconsole_find_dev(ptr);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700479 if (dev == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700480 mconsole_reply(req, "Bad configuration option", 1, 0);
481 return;
482 }
483
484 name = &ptr[strlen(dev->name)];
485 ptr = name;
Jeff Dikeae2587e2007-10-16 01:26:57 -0700486 while ((*ptr != '=') && (*ptr != '\0'))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700487 ptr++;
488
Jeff Dikeae2587e2007-10-16 01:26:57 -0700489 if (*ptr == '=') {
Jeff Dikef28169d2007-02-10 01:43:53 -0800490 err = (*dev->config)(name, &error_string);
491 mconsole_reply(req, error_string, err, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700492 }
493 else mconsole_get_config(dev->get_config, req, name);
494}
495
496void mconsole_remove(struct mc_request *req)
497{
Jeff Diked50084a2006-01-06 00:18:50 -0800498 struct mc_device *dev;
Jeff Dike29d56cf2005-06-25 14:55:25 -0700499 char *ptr = req->request.data, *err_msg = "";
Jeff Dike3a331a52006-01-06 00:19:05 -0800500 char error[256];
Jeff Dike29d56cf2005-06-25 14:55:25 -0700501 int err, start, end, n;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700502
503 ptr += strlen("remove");
André Goddard Rosae7d28602009-12-14 18:01:06 -0800504 ptr = skip_spaces(ptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700505 dev = mconsole_find_dev(ptr);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700506 if (dev == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700507 mconsole_reply(req, "Bad remove option", 1, 0);
508 return;
509 }
Jeff Dike29d56cf2005-06-25 14:55:25 -0700510
Jeff Dike3a331a52006-01-06 00:19:05 -0800511 ptr = &ptr[strlen(dev->name)];
Jeff Dike29d56cf2005-06-25 14:55:25 -0700512
Jeff Dike3a331a52006-01-06 00:19:05 -0800513 err = 1;
514 n = (*dev->id)(&ptr, &start, &end);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700515 if (n < 0) {
Jeff Dike3a331a52006-01-06 00:19:05 -0800516 err_msg = "Couldn't parse device number";
517 goto out;
518 }
Jeff Dikeae2587e2007-10-16 01:26:57 -0700519 else if ((n < start) || (n > end)) {
Jeff Dike3a331a52006-01-06 00:19:05 -0800520 sprintf(error, "Invalid device number - must be between "
521 "%d and %d", start, end);
522 err_msg = error;
523 goto out;
524 }
Jeff Dike29d56cf2005-06-25 14:55:25 -0700525
Jeff Dikef28169d2007-02-10 01:43:53 -0800526 err_msg = NULL;
527 err = (*dev->remove)(n, &err_msg);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700528 switch(err) {
Jeff Diked40f6d72007-03-29 01:20:28 -0700529 case 0:
530 err_msg = "";
531 break;
Jeff Dike3a331a52006-01-06 00:19:05 -0800532 case -ENODEV:
Jeff Dikeae2587e2007-10-16 01:26:57 -0700533 if (err_msg == NULL)
Jeff Dikef28169d2007-02-10 01:43:53 -0800534 err_msg = "Device doesn't exist";
Jeff Dike3a331a52006-01-06 00:19:05 -0800535 break;
536 case -EBUSY:
Jeff Dikeae2587e2007-10-16 01:26:57 -0700537 if (err_msg == NULL)
Jeff Dikef28169d2007-02-10 01:43:53 -0800538 err_msg = "Device is currently open";
Jeff Dike3a331a52006-01-06 00:19:05 -0800539 break;
540 default:
541 break;
542 }
543out:
Jeff Dike29d56cf2005-06-25 14:55:25 -0700544 mconsole_reply(req, err_msg, err, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545}
546
Jeff Dikef92afe52006-09-29 01:58:52 -0700547struct mconsole_output {
548 struct list_head list;
549 struct mc_request *req;
550};
551
Jeff Dike84f48d42007-02-10 01:44:01 -0800552static DEFINE_SPINLOCK(client_lock);
Jeff Dike6f517d32006-01-06 00:19:04 -0800553static LIST_HEAD(clients);
554static char console_buf[MCONSOLE_MAX_DATA];
Jeff Dike6f517d32006-01-06 00:19:04 -0800555
556static void console_write(struct console *console, const char *string,
Jeff Dike54fa0ba2007-10-16 01:27:17 -0700557 unsigned int len)
Jeff Dike6f517d32006-01-06 00:19:04 -0800558{
559 struct list_head *ele;
560 int n;
561
Jeff Dikeae2587e2007-10-16 01:26:57 -0700562 if (list_empty(&clients))
Jeff Dike6f517d32006-01-06 00:19:04 -0800563 return;
564
Jeff Dike54fa0ba2007-10-16 01:27:17 -0700565 while (len > 0) {
566 n = min((size_t) len, ARRAY_SIZE(console_buf));
567 strncpy(console_buf, string, n);
Jeff Dike6f517d32006-01-06 00:19:04 -0800568 string += n;
569 len -= n;
Jeff Dike6f517d32006-01-06 00:19:04 -0800570
Jeff Dikeae2587e2007-10-16 01:26:57 -0700571 list_for_each(ele, &clients) {
Jeff Dikef92afe52006-09-29 01:58:52 -0700572 struct mconsole_output *entry;
Jeff Dike6f517d32006-01-06 00:19:04 -0800573
Jeff Dikef92afe52006-09-29 01:58:52 -0700574 entry = list_entry(ele, struct mconsole_output, list);
Jeff Dike54fa0ba2007-10-16 01:27:17 -0700575 mconsole_reply_len(entry->req, console_buf, n, 0, 1);
Jeff Dike6f517d32006-01-06 00:19:04 -0800576 }
Jeff Dike6f517d32006-01-06 00:19:04 -0800577 }
578}
579
580static struct console mc_console = { .name = "mc",
581 .write = console_write,
Jeff Dikea174b302006-01-11 12:17:28 -0800582 .flags = CON_ENABLED,
Jeff Dike6f517d32006-01-06 00:19:04 -0800583 .index = -1 };
584
585static int mc_add_console(void)
586{
587 register_console(&mc_console);
588 return 0;
589}
590
591late_initcall(mc_add_console);
592
593static void with_console(struct mc_request *req, void (*proc)(void *),
594 void *arg)
595{
Jeff Dikef92afe52006-09-29 01:58:52 -0700596 struct mconsole_output entry;
Jeff Dike6f517d32006-01-06 00:19:04 -0800597 unsigned long flags;
598
Jeff Dikef92afe52006-09-29 01:58:52 -0700599 entry.req = req;
Jeff Dike84f48d42007-02-10 01:44:01 -0800600 spin_lock_irqsave(&client_lock, flags);
Jeff Dike6f517d32006-01-06 00:19:04 -0800601 list_add(&entry.list, &clients);
Jeff Dike84f48d42007-02-10 01:44:01 -0800602 spin_unlock_irqrestore(&client_lock, flags);
Jeff Dike6f517d32006-01-06 00:19:04 -0800603
604 (*proc)(arg);
605
Jeff Dike54fa0ba2007-10-16 01:27:17 -0700606 mconsole_reply_len(req, "", 0, 0, 0);
Jeff Dike6f517d32006-01-06 00:19:04 -0800607
Jeff Dike84f48d42007-02-10 01:44:01 -0800608 spin_lock_irqsave(&client_lock, flags);
Jeff Dike6f517d32006-01-06 00:19:04 -0800609 list_del(&entry.list);
Jeff Dike84f48d42007-02-10 01:44:01 -0800610 spin_unlock_irqrestore(&client_lock, flags);
Jeff Dike6f517d32006-01-06 00:19:04 -0800611}
612
Jeff Dike4111b022006-01-06 00:19:05 -0800613#ifdef CONFIG_MAGIC_SYSRQ
Jeff Dike54fa0ba2007-10-16 01:27:17 -0700614
615#include <linux/sysrq.h>
616
Jeff Dike4111b022006-01-06 00:19:05 -0800617static void sysrq_proc(void *arg)
618{
619 char *op = arg;
Dmitry Torokhovf3353972010-08-17 21:15:47 -0700620 handle_sysrq(*op);
Jeff Dike4111b022006-01-06 00:19:05 -0800621}
622
623void mconsole_sysrq(struct mc_request *req)
624{
625 char *ptr = req->request.data;
626
627 ptr += strlen("sysrq");
André Goddard Rosae7d28602009-12-14 18:01:06 -0800628 ptr = skip_spaces(ptr);
Jeff Dike4111b022006-01-06 00:19:05 -0800629
Jeff Dikeae2587e2007-10-16 01:26:57 -0700630 /*
631 * With 'b', the system will shut down without a chance to reply,
Jeff Dike4111b022006-01-06 00:19:05 -0800632 * so in this case, we reply first.
633 */
Jeff Dikeae2587e2007-10-16 01:26:57 -0700634 if (*ptr == 'b')
Jeff Dike4111b022006-01-06 00:19:05 -0800635 mconsole_reply(req, "", 0, 0);
636
637 with_console(req, sysrq_proc, ptr);
638}
639#else
640void mconsole_sysrq(struct mc_request *req)
641{
642 mconsole_reply(req, "Sysrq not compiled in", 1, 0);
643}
644#endif
645
Jeff Dike6f517d32006-01-06 00:19:04 -0800646static void stack_proc(void *arg)
647{
648 struct task_struct *from = current, *to = arg;
649
650 to->thread.saved_task = from;
Frederic Weisbecker4d9a5d42012-10-11 01:47:16 +0200651 rcu_user_hooks_switch(from, to);
Jeff Dike6f517d32006-01-06 00:19:04 -0800652 switch_to(from, to, from);
653}
654
Jeff Dikeae2587e2007-10-16 01:26:57 -0700655/*
656 * Mconsole stack trace
Jeff Dike3eddddc2005-09-16 19:27:46 -0700657 * Added by Allan Graves, Jeff Dike
658 * Dumps a stacks registers to the linux console.
659 * Usage stack <pid>.
660 */
Jeff Dike42fda662007-10-16 01:26:50 -0700661void mconsole_stack(struct mc_request *req)
Jeff Dike3eddddc2005-09-16 19:27:46 -0700662{
Jeff Dike3a331a52006-01-06 00:19:05 -0800663 char *ptr = req->request.data;
664 int pid_requested= -1;
Jeff Dike3eddddc2005-09-16 19:27:46 -0700665 struct task_struct *to = NULL;
666
Jeff Dikeae2587e2007-10-16 01:26:57 -0700667 /*
668 * Would be nice:
Jeff Dike3a331a52006-01-06 00:19:05 -0800669 * 1) Send showregs output to mconsole.
Jeff Dike3eddddc2005-09-16 19:27:46 -0700670 * 2) Add a way to stack dump all pids.
671 */
672
Jeff Dike3a331a52006-01-06 00:19:05 -0800673 ptr += strlen("stack");
André Goddard Rosae7d28602009-12-14 18:01:06 -0800674 ptr = skip_spaces(ptr);
Jeff Dike3eddddc2005-09-16 19:27:46 -0700675
Jeff Dikeae2587e2007-10-16 01:26:57 -0700676 /*
677 * Should really check for multiple pids or reject bad args here
678 */
Jeff Dike3a331a52006-01-06 00:19:05 -0800679 /* What do the arguments in mconsole_reply mean? */
Jeff Dikeae2587e2007-10-16 01:26:57 -0700680 if (sscanf(ptr, "%d", &pid_requested) == 0) {
Jeff Dike3a331a52006-01-06 00:19:05 -0800681 mconsole_reply(req, "Please specify a pid", 1, 0);
682 return;
683 }
Jeff Dike3eddddc2005-09-16 19:27:46 -0700684
Jeff Dike827b3f62008-02-04 22:31:29 -0800685 to = find_task_by_pid_ns(pid_requested, &init_pid_ns);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700686 if ((to == NULL) || (pid_requested == 0)) {
Jeff Dike3a331a52006-01-06 00:19:05 -0800687 mconsole_reply(req, "Couldn't find that pid", 1, 0);
688 return;
689 }
Jeff Dike6f517d32006-01-06 00:19:04 -0800690 with_console(req, stack_proc, to);
Jeff Dike3eddddc2005-09-16 19:27:46 -0700691}
Jeff Dike3eddddc2005-09-16 19:27:46 -0700692
Jeff Dikeae2587e2007-10-16 01:26:57 -0700693/*
694 * Changed by mconsole_setup, which is __setup, and called before SMP is
Linus Torvalds1da177e2005-04-16 15:20:36 -0700695 * active.
696 */
Jeff Diked50084a2006-01-06 00:18:50 -0800697static char *notify_socket = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700698
Jeff Dike97a1fcb2007-07-23 18:43:48 -0700699static int __init mconsole_init(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700700{
701 /* long to avoid size mismatch warnings from gcc */
702 long sock;
703 int err;
Balbir Singh36137122008-12-09 13:14:07 -0800704 char file[UNIX_PATH_MAX];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700705
Jeff Dikeae2587e2007-10-16 01:26:57 -0700706 if (umid_file_name("mconsole", file, sizeof(file)))
707 return -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700708 snprintf(mconsole_socket_name, sizeof(file), "%s", file);
709
710 sock = os_create_unix_socket(file, sizeof(file), 1);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700711 if (sock < 0) {
712 printk(KERN_ERR "Failed to initialize management console\n");
713 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700714 }
Jeff Dike438ee672008-02-04 22:31:19 -0800715 if (os_set_fd_block(sock, 0))
716 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700717
718 register_reboot_notifier(&reboot_notifier);
719
720 err = um_request_irq(MCONSOLE_IRQ, sock, IRQ_READ, mconsole_interrupt,
Theodore Ts'oaab94462012-07-17 14:18:23 -0400721 IRQF_SHARED, "mconsole", (void *)sock);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700722 if (err) {
723 printk(KERN_ERR "Failed to get IRQ for management console\n");
Jeff Dike438ee672008-02-04 22:31:19 -0800724 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700725 }
726
Jeff Dikeae2587e2007-10-16 01:26:57 -0700727 if (notify_socket != NULL) {
Jeff Dike970d6e32006-01-06 00:18:48 -0800728 notify_socket = kstrdup(notify_socket, GFP_KERNEL);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700729 if (notify_socket != NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700730 mconsole_notify(notify_socket, MCONSOLE_SOCKET,
Jeff Diked50084a2006-01-06 00:18:50 -0800731 mconsole_socket_name,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700732 strlen(mconsole_socket_name) + 1);
733 else printk(KERN_ERR "mconsole_setup failed to strdup "
734 "string\n");
735 }
736
Jeff Dikeae2587e2007-10-16 01:26:57 -0700737 printk(KERN_INFO "mconsole (version %d) initialized on %s\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700738 MCONSOLE_VERSION, mconsole_socket_name);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700739 return 0;
Jeff Dike438ee672008-02-04 22:31:19 -0800740
741 out:
742 os_close_file(sock);
743 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700744}
745
746__initcall(mconsole_init);
747
Alexey Dobriyan6613c5e2009-12-14 18:00:11 -0800748static ssize_t mconsole_proc_write(struct file *file,
749 const char __user *buffer, size_t count, loff_t *pos)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700750{
751 char *buf;
752
753 buf = kmalloc(count + 1, GFP_KERNEL);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700754 if (buf == NULL)
755 return -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756
Jeff Dikeae2587e2007-10-16 01:26:57 -0700757 if (copy_from_user(buf, buffer, count)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700758 count = -EFAULT;
759 goto out;
760 }
761
762 buf[count] = '\0';
763
764 mconsole_notify(notify_socket, MCONSOLE_USER_NOTIFY, buf, count);
765 out:
766 kfree(buf);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700767 return count;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700768}
769
Alexey Dobriyan6613c5e2009-12-14 18:00:11 -0800770static const struct file_operations mconsole_proc_fops = {
771 .owner = THIS_MODULE,
772 .write = mconsole_proc_write,
Arnd Bergmann6038f372010-08-15 18:52:59 +0200773 .llseek = noop_llseek,
Alexey Dobriyan6613c5e2009-12-14 18:00:11 -0800774};
775
Linus Torvalds1da177e2005-04-16 15:20:36 -0700776static int create_proc_mconsole(void)
777{
778 struct proc_dir_entry *ent;
779
Jeff Dikeae2587e2007-10-16 01:26:57 -0700780 if (notify_socket == NULL)
781 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700782
Alexey Dobriyan6613c5e2009-12-14 18:00:11 -0800783 ent = proc_create("mconsole", 0200, NULL, &mconsole_proc_fops);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700784 if (ent == NULL) {
785 printk(KERN_INFO "create_proc_mconsole : create_proc_entry "
786 "failed\n");
787 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700788 }
Jeff Dikeae2587e2007-10-16 01:26:57 -0700789 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700790}
791
792static DEFINE_SPINLOCK(notify_spinlock);
793
794void lock_notify(void)
795{
796 spin_lock(&notify_spinlock);
797}
798
799void unlock_notify(void)
800{
801 spin_unlock(&notify_spinlock);
802}
803
804__initcall(create_proc_mconsole);
805
Jeff Dike088bec42007-10-16 01:27:20 -0700806#define NOTIFY "notify:"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700807
808static int mconsole_setup(char *str)
809{
Jeff Dikeae2587e2007-10-16 01:26:57 -0700810 if (!strncmp(str, NOTIFY, strlen(NOTIFY))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700811 str += strlen(NOTIFY);
812 notify_socket = str;
813 }
814 else printk(KERN_ERR "mconsole_setup : Unknown option - '%s'\n", str);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700815 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700816}
817
Jeff Dike088bec42007-10-16 01:27:20 -0700818__setup("mconsole=", mconsole_setup);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700819
820__uml_help(mconsole_setup,
821"mconsole=notify:<socket>\n"
822" Requests that the mconsole driver send a message to the named Unix\n"
823" socket containing the name of the mconsole socket. This also serves\n"
824" to notify outside processes when UML has booted far enough to respond\n"
825" to mconsole requests.\n\n"
826);
827
828static int notify_panic(struct notifier_block *self, unsigned long unused1,
829 void *ptr)
830{
831 char *message = ptr;
832
Jeff Dikeae2587e2007-10-16 01:26:57 -0700833 if (notify_socket == NULL)
834 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700835
Jeff Diked50084a2006-01-06 00:18:50 -0800836 mconsole_notify(notify_socket, MCONSOLE_PANIC, message,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700837 strlen(message) + 1);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700838 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700839}
840
841static struct notifier_block panic_exit_notifier = {
842 .notifier_call = notify_panic,
843 .next = NULL,
844 .priority = 1
845};
846
847static int add_notifier(void)
848{
Alan Sterne041c682006-03-27 01:16:30 -0800849 atomic_notifier_chain_register(&panic_notifier_list,
850 &panic_exit_notifier);
Jeff Dikeae2587e2007-10-16 01:26:57 -0700851 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700852}
853
854__initcall(add_notifier);
855
856char *mconsole_notify_socket(void)
857{
Jeff Dikeae2587e2007-10-16 01:26:57 -0700858 return notify_socket;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700859}
860
861EXPORT_SYMBOL(mconsole_notify_socket);