提交 3a51237d 编写于 作者: A Al Viro 提交者: Linus Torvalds

[PATCH] uml: mconsole fixes

 * when we have stop/sysrq/go, we get pt_regs of whatever executes
   mc_work_proc().  Would be better to see what we had at the time of
   interrupt that got us stop.

 * stop/stop/stop.....  will give stack overflow.  Shouldn't allow stop
   from mconsole_stop().

 * stop/stop/go leaves us inside mconsole_stop() with
	os_set_fd_block(req->originating_fd, 0);
	reactivate_fd(req->originating_fd, MCONSOLE_IRQ);
   just done by nested mconsole_stop().  Ditto.

 * once we'd seen stop, there's a period when INTR commands are executed
   out of order (as they should; we might have the things stuck badly
   enough to never reach mconsole_stop(), but still not badly enough to
   block mconsole_interrupt(); in that situation we _want_ things like
   "cad" to be executed immediately).  Once we enter monsole_stop(), all
   INTR commands will be executed in order, mixed with PROC ones.  We'd
   better let user see that such change of behaviour has happened.
   (Suggested by lennert).

 * stack footprint of monsole_interrupt() is an atrocity; AFAICS we can
   safely make struct mc_request req; static in function there.
Signed-off-by: NAl Viro <viro@zeniv.linux.org.uk>
Acked-by: NJeff Dike <jdike@addtoit.com>
Signed-off-by: NLinus Torvalds <torvalds@osdl.org>
上级 6c504447
...@@ -79,7 +79,7 @@ static irqreturn_t mconsole_interrupt(int irq, void *dev_id) ...@@ -79,7 +79,7 @@ static irqreturn_t mconsole_interrupt(int irq, void *dev_id)
/* long to avoid size mismatch warnings from gcc */ /* long to avoid size mismatch warnings from gcc */
long fd; long fd;
struct mconsole_entry *new; struct mconsole_entry *new;
struct mc_request req; static struct mc_request req; /* that's OK */
fd = (long) dev_id; fd = (long) dev_id;
while (mconsole_get_request(fd, &req)){ while (mconsole_get_request(fd, &req)){
...@@ -91,6 +91,7 @@ static irqreturn_t mconsole_interrupt(int irq, void *dev_id) ...@@ -91,6 +91,7 @@ static irqreturn_t mconsole_interrupt(int irq, void *dev_id)
mconsole_reply(&req, "Out of memory", 1, 0); mconsole_reply(&req, "Out of memory", 1, 0);
else { else {
new->request = req; new->request = req;
new->request.regs = get_irq_regs()->regs;
list_add(&new->list, &mc_requests); list_add(&new->list, &mc_requests);
} }
} }
...@@ -314,9 +315,21 @@ void mconsole_stop(struct mc_request *req) ...@@ -314,9 +315,21 @@ void mconsole_stop(struct mc_request *req)
{ {
deactivate_fd(req->originating_fd, MCONSOLE_IRQ); deactivate_fd(req->originating_fd, MCONSOLE_IRQ);
os_set_fd_block(req->originating_fd, 1); os_set_fd_block(req->originating_fd, 1);
mconsole_reply(req, "", 0, 0); mconsole_reply(req, "stopped", 0, 0);
while(mconsole_get_request(req->originating_fd, req)){ while (mconsole_get_request(req->originating_fd, req)) {
if(req->cmd->handler == mconsole_go) break; if (req->cmd->handler == mconsole_go)
break;
if (req->cmd->handler == mconsole_stop) {
mconsole_reply(req, "Already stopped", 1, 0);
continue;
}
if (req->cmd->handler == mconsole_sysrq) {
struct pt_regs *old_regs;
old_regs = set_irq_regs((struct pt_regs *)&req->regs);
mconsole_sysrq(req);
set_irq_regs(old_regs);
continue;
}
(*req->cmd->handler)(req); (*req->cmd->handler)(req);
} }
os_set_fd_block(req->originating_fd, 0); os_set_fd_block(req->originating_fd, 0);
...@@ -673,9 +686,7 @@ static void with_console(struct mc_request *req, void (*proc)(void *), ...@@ -673,9 +686,7 @@ static void with_console(struct mc_request *req, void (*proc)(void *),
static void sysrq_proc(void *arg) static void sysrq_proc(void *arg)
{ {
char *op = arg; char *op = arg;
struct pt_regs *old_regs = set_irq_regs(&current->thread.regs);
handle_sysrq(*op, NULL); handle_sysrq(*op, NULL);
set_irq_regs(old_regs);
} }
void mconsole_sysrq(struct mc_request *req) void mconsole_sysrq(struct mc_request *req)
......
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
#include <sys/un.h> #include <sys/un.h>
#include <unistd.h> #include <unistd.h>
#include "user.h" #include "user.h"
#include "sysdep/ptrace.h"
#include "mconsole.h" #include "mconsole.h"
#include "umid.h" #include "umid.h"
#include "user_util.h" #include "user_util.h"
......
...@@ -61,6 +61,7 @@ struct mc_request ...@@ -61,6 +61,7 @@ struct mc_request
struct mconsole_request request; struct mconsole_request request;
struct mconsole_command *cmd; struct mconsole_command *cmd;
union uml_pt_regs regs;
}; };
extern char mconsole_socket_name[]; extern char mconsole_socket_name[];
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册