Commit 84f48d4f authored by Jeff Dike's avatar Jeff Dike Committed by Linus Torvalds

[PATCH] uml: mconsole locking

Locking fixes.  Locking was totally lacking for the mconsole_devices, which
got a spin lock, and the unplugged pages data, which got a mutex.

The locking of the mconsole console output code was confused.  Now, the
console_lock (renamed to client_lock) protects the clients list.
Signed-off-by: default avatarJeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent d471c0fc
...@@ -337,13 +337,15 @@ void mconsole_stop(struct mc_request *req) ...@@ -337,13 +337,15 @@ void mconsole_stop(struct mc_request *req)
mconsole_reply(req, "", 0, 0); mconsole_reply(req, "", 0, 0);
} }
/* This list is populated by __initcall routines. */ static DEFINE_SPINLOCK(mc_devices_lock);
static LIST_HEAD(mconsole_devices); static LIST_HEAD(mconsole_devices);
void mconsole_register_dev(struct mc_device *new) void mconsole_register_dev(struct mc_device *new)
{ {
spin_lock(&mc_devices_lock);
BUG_ON(!list_empty(&new->list));
list_add(&new->list, &mconsole_devices); list_add(&new->list, &mconsole_devices);
spin_unlock(&mc_devices_lock);
} }
static struct mc_device *mconsole_find_dev(char *name) static struct mc_device *mconsole_find_dev(char *name)
...@@ -367,6 +369,7 @@ struct unplugged_pages { ...@@ -367,6 +369,7 @@ struct unplugged_pages {
void *pages[UNPLUGGED_PER_PAGE]; void *pages[UNPLUGGED_PER_PAGE];
}; };
static DECLARE_MUTEX(plug_mem_mutex);
static unsigned long long unplugged_pages_count = 0; static unsigned long long unplugged_pages_count = 0;
static struct list_head unplugged_pages = LIST_HEAD_INIT(unplugged_pages); static struct list_head unplugged_pages = LIST_HEAD_INIT(unplugged_pages);
static int unplug_index = UNPLUGGED_PER_PAGE; static int unplug_index = UNPLUGGED_PER_PAGE;
...@@ -402,6 +405,7 @@ static int mem_config(char *str, char **error_out) ...@@ -402,6 +405,7 @@ static int mem_config(char *str, char **error_out)
diff /= PAGE_SIZE; diff /= PAGE_SIZE;
down(&plug_mem_mutex);
for(i = 0; i < diff; i++){ for(i = 0; i < diff; i++){
struct unplugged_pages *unplugged; struct unplugged_pages *unplugged;
void *addr; void *addr;
...@@ -447,7 +451,7 @@ static int mem_config(char *str, char **error_out) ...@@ -447,7 +451,7 @@ static int mem_config(char *str, char **error_out)
printk("Failed to release memory - " printk("Failed to release memory - "
"errno = %d\n", err); "errno = %d\n", err);
*error_out = "Failed to release memory"; *error_out = "Failed to release memory";
goto out; goto out_unlock;
} }
unplugged->pages[unplug_index++] = addr; unplugged->pages[unplug_index++] = addr;
} }
...@@ -457,6 +461,8 @@ static int mem_config(char *str, char **error_out) ...@@ -457,6 +461,8 @@ static int mem_config(char *str, char **error_out)
} }
err = 0; err = 0;
out_unlock:
up(&plug_mem_mutex);
out: out:
return err; return err;
} }
...@@ -487,6 +493,7 @@ static int mem_remove(int n, char **error_out) ...@@ -487,6 +493,7 @@ static int mem_remove(int n, char **error_out)
} }
static struct mc_device mem_mc = { static struct mc_device mem_mc = {
.list = LIST_HEAD_INIT(mem_mc.list),
.name = "mem", .name = "mem",
.config = mem_config, .config = mem_config,
.get_config = mem_get_config, .get_config = mem_get_config,
...@@ -629,7 +636,7 @@ struct mconsole_output { ...@@ -629,7 +636,7 @@ struct mconsole_output {
struct mc_request *req; struct mc_request *req;
}; };
static DEFINE_SPINLOCK(console_lock); static DEFINE_SPINLOCK(client_lock);
static LIST_HEAD(clients); static LIST_HEAD(clients);
static char console_buf[MCONSOLE_MAX_DATA]; static char console_buf[MCONSOLE_MAX_DATA];
static int console_index = 0; static int console_index = 0;
...@@ -684,16 +691,18 @@ static void with_console(struct mc_request *req, void (*proc)(void *), ...@@ -684,16 +691,18 @@ static void with_console(struct mc_request *req, void (*proc)(void *),
unsigned long flags; unsigned long flags;
entry.req = req; entry.req = req;
spin_lock_irqsave(&client_lock, flags);
list_add(&entry.list, &clients); list_add(&entry.list, &clients);
spin_lock_irqsave(&console_lock, flags); spin_unlock_irqrestore(&client_lock, flags);
(*proc)(arg); (*proc)(arg);
mconsole_reply_len(req, console_buf, console_index, 0, 0); mconsole_reply_len(req, console_buf, console_index, 0, 0);
console_index = 0; console_index = 0;
spin_unlock_irqrestore(&console_lock, flags); spin_lock_irqsave(&client_lock, flags);
list_del(&entry.list); list_del(&entry.list);
spin_unlock_irqrestore(&client_lock, flags);
} }
#ifdef CONFIG_MAGIC_SYSRQ #ifdef CONFIG_MAGIC_SYSRQ
......
...@@ -690,6 +690,7 @@ static int net_remove(int n, char **error_out) ...@@ -690,6 +690,7 @@ static int net_remove(int n, char **error_out)
} }
static struct mc_device net_mc = { static struct mc_device net_mc = {
.list = LIST_HEAD_INIT(net_mc.list),
.name = "eth", .name = "eth",
.config = net_config, .config = net_config,
.get_config = NULL, .get_config = NULL,
......
...@@ -64,6 +64,7 @@ static struct line_driver driver = { ...@@ -64,6 +64,7 @@ static struct line_driver driver = {
.symlink_from = "serial", .symlink_from = "serial",
.symlink_to = "tts", .symlink_to = "tts",
.mc = { .mc = {
.list = LIST_HEAD_INIT(driver.mc.list),
.name = "ssl", .name = "ssl",
.config = ssl_config, .config = ssl_config,
.get_config = ssl_get_config, .get_config = ssl_get_config,
......
...@@ -828,6 +828,7 @@ static int ubd_remove(int n, char **error_out) ...@@ -828,6 +828,7 @@ static int ubd_remove(int n, char **error_out)
* ubd-specific locks. * ubd-specific locks.
*/ */
static struct mc_device ubd_mc = { static struct mc_device ubd_mc = {
.list = LIST_HEAD_INIT(ubd_mc.list),
.name = "ubd", .name = "ubd",
.config = ubd_config, .config = ubd_config,
.get_config = ubd_get_config, .get_config = ubd_get_config,
......
...@@ -12,6 +12,7 @@ extern int gdb_config(char *str, char **error_out); ...@@ -12,6 +12,7 @@ extern int gdb_config(char *str, char **error_out);
extern int gdb_remove(int n, char **error_out); extern int gdb_remove(int n, char **error_out);
static struct mc_device gdb_mc = { static struct mc_device gdb_mc = {
.list = INIT_LIST_HEAD(gdb_mc.list),
.name = "gdb", .name = "gdb",
.config = gdb_config, .config = gdb_config,
.remove = gdb_remove, .remove = gdb_remove,
......
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