diff options
author | Ingo Molnar <mingo@elte.hu> | 2007-02-10 04:46:36 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-02-11 14:18:07 -0500 |
commit | 656dad312fb41ed95ef08325e9df9bece3aacbbb (patch) | |
tree | 34d0a2c1e4b78104148850abc8a1359ce06c3f13 /arch | |
parent | 3e4fdaf8aebe489e8e59826fdf78cb64356d2ad0 (diff) |
[PATCH] highmem: catch illegal nesting
Catch illegally nested kmap_atomic()s even if the page that is mapped by
the 'inner' instance is from lowmem.
This avoids spuriously zapped kmap-atomic ptes and turns hard to find
crashes into clear asserts at the bug site.
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'arch')
-rw-r--r-- | arch/i386/mm/highmem.c | 7 |
1 files changed, 4 insertions, 3 deletions
diff --git a/arch/i386/mm/highmem.c b/arch/i386/mm/highmem.c index e0fa6cb655a8..bb2de1089add 100644 --- a/arch/i386/mm/highmem.c +++ b/arch/i386/mm/highmem.c | |||
@@ -33,13 +33,14 @@ void *kmap_atomic(struct page *page, enum km_type type) | |||
33 | 33 | ||
34 | /* even !CONFIG_PREEMPT needs this, for in_atomic in do_page_fault */ | 34 | /* even !CONFIG_PREEMPT needs this, for in_atomic in do_page_fault */ |
35 | pagefault_disable(); | 35 | pagefault_disable(); |
36 | |||
37 | idx = type + KM_TYPE_NR*smp_processor_id(); | ||
38 | BUG_ON(!pte_none(*(kmap_pte-idx))); | ||
39 | |||
36 | if (!PageHighMem(page)) | 40 | if (!PageHighMem(page)) |
37 | return page_address(page); | 41 | return page_address(page); |
38 | 42 | ||
39 | idx = type + KM_TYPE_NR*smp_processor_id(); | ||
40 | vaddr = __fix_to_virt(FIX_KMAP_BEGIN + idx); | 43 | vaddr = __fix_to_virt(FIX_KMAP_BEGIN + idx); |
41 | if (!pte_none(*(kmap_pte-idx))) | ||
42 | BUG(); | ||
43 | set_pte(kmap_pte-idx, mk_pte(page, kmap_prot)); | 44 | set_pte(kmap_pte-idx, mk_pte(page, kmap_prot)); |
44 | 45 | ||
45 | return (void*) vaddr; | 46 | return (void*) vaddr; |