diff options
author | Carsten Otte <cotte@de.ibm.com> | 2011-10-30 10:17:00 -0400 |
---|---|---|
committer | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2011-10-30 10:16:44 -0400 |
commit | a9162f238a84ee05b09ea4b0ebd97fb20448c28c (patch) | |
tree | 06364148f57e6833bf7c7e66fa69181e141792ba /arch/s390 | |
parent | 69ba97436647f3b793f8e0784d1cde63adf241ea (diff) |
[S390] fix possible deadlock in gmap_map_segment
Fix possible deadlock reported by lockdep:
qemu-system-s39/2963 is trying to acquire lock:
(&mm->mmap_sem){++++++}, at: gmap_alloc_table+0x9c/0x120
but task is already holding lock:
(&mm->mmap_sem){++++++}, at: gmap_map_segment+0xa6/0x27c
Actually gmap_alloc_table is the only called in gmap_map_segment with
mmap_sem held, thus it's safe to simply remove the inner lock.
Signed-off-by: Carsten Otte <cotte@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Diffstat (limited to 'arch/s390')
-rw-r--r-- | arch/s390/mm/pgtable.c | 5 |
1 files changed, 3 insertions, 2 deletions
diff --git a/arch/s390/mm/pgtable.c b/arch/s390/mm/pgtable.c index 529a08838376..e4a4cefb92b3 100644 --- a/arch/s390/mm/pgtable.c +++ b/arch/s390/mm/pgtable.c | |||
@@ -256,6 +256,9 @@ void gmap_disable(struct gmap *gmap) | |||
256 | } | 256 | } |
257 | EXPORT_SYMBOL_GPL(gmap_disable); | 257 | EXPORT_SYMBOL_GPL(gmap_disable); |
258 | 258 | ||
259 | /* | ||
260 | * gmap_alloc_table is assumed to be called with mmap_sem held | ||
261 | */ | ||
259 | static int gmap_alloc_table(struct gmap *gmap, | 262 | static int gmap_alloc_table(struct gmap *gmap, |
260 | unsigned long *table, unsigned long init) | 263 | unsigned long *table, unsigned long init) |
261 | { | 264 | { |
@@ -267,14 +270,12 @@ static int gmap_alloc_table(struct gmap *gmap, | |||
267 | return -ENOMEM; | 270 | return -ENOMEM; |
268 | new = (unsigned long *) page_to_phys(page); | 271 | new = (unsigned long *) page_to_phys(page); |
269 | crst_table_init(new, init); | 272 | crst_table_init(new, init); |
270 | down_read(&gmap->mm->mmap_sem); | ||
271 | if (*table & _REGION_ENTRY_INV) { | 273 | if (*table & _REGION_ENTRY_INV) { |
272 | list_add(&page->lru, &gmap->crst_list); | 274 | list_add(&page->lru, &gmap->crst_list); |
273 | *table = (unsigned long) new | _REGION_ENTRY_LENGTH | | 275 | *table = (unsigned long) new | _REGION_ENTRY_LENGTH | |
274 | (*table & _REGION_ENTRY_TYPE_MASK); | 276 | (*table & _REGION_ENTRY_TYPE_MASK); |
275 | } else | 277 | } else |
276 | __free_pages(page, ALLOC_ORDER); | 278 | __free_pages(page, ALLOC_ORDER); |
277 | up_read(&gmap->mm->mmap_sem); | ||
278 | return 0; | 279 | return 0; |
279 | } | 280 | } |
280 | 281 | ||