Skip to content

Commit 182475b

Browse files
committed
memremap: fix highmem support
Currently memremap checks if the range is "System RAM" and returns the kernel linear address. This is broken for highmem platforms where a range may be "System RAM", but is not part of the kernel linear mapping. Fallback to ioremap_cache() in these cases, to let the arch code attempt to handle it. Note that ARM ioremap will WARN when attempting to remap ram, and in that case the caller needs to be fixed. For this reason, existing ioremap_cache() usages for ARM are already trained to avoid attempts to remap ram. The impact of this bug is low for now since the pmem driver is the only user of memremap(), but this is important to fix before more conversions to memremap arrive in 4.4. Cc: Rafael J. Wysocki <rafael.j.wysocki@intel.com> Reported-by: Ard Biesheuvel <ard.biesheuvel@linaro.org> Acked-by: Ard Biesheuvel <ard.biesheuvel@linaro.org> Signed-off-by: Dan Williams <dan.j.williams@intel.com>
1 parent 25cb62b commit 182475b

File tree

1 file changed

+12
-2
lines changed

1 file changed

+12
-2
lines changed

kernel/memremap.c

Lines changed: 12 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,16 @@ __weak void __iomem *ioremap_cache(resource_size_t offset, unsigned long size)
2424
}
2525
#endif
2626

27+
static void *try_ram_remap(resource_size_t offset, size_t size)
28+
{
29+
struct page *page = pfn_to_page(offset >> PAGE_SHIFT);
30+
31+
/* In the simple case just return the existing linear address */
32+
if (!PageHighMem(page))
33+
return __va(offset);
34+
return NULL; /* fallback to ioremap_cache */
35+
}
36+
2737
/**
2838
* memremap() - remap an iomem_resource as cacheable memory
2939
* @offset: iomem resource start address
@@ -66,8 +76,8 @@ void *memremap(resource_size_t offset, size_t size, unsigned long flags)
6676
* the requested range is potentially in "System RAM"
6777
*/
6878
if (is_ram == REGION_INTERSECTS)
69-
addr = __va(offset);
70-
else
79+
addr = try_ram_remap(offset, size);
80+
if (!addr)
7181
addr = ioremap_cache(offset, size);
7282
}
7383

0 commit comments

Comments
 (0)