Commit f7750a79 authored by Tom Lendacky's avatar Tom Lendacky Committed by Ingo Molnar

x86, mpparse, x86/acpi, x86/PCI, x86/dmi, SFI: Use memremap() for RAM mappings

The ioremap() function is intended for mapping MMIO. For RAM, the
memremap() function should be used. Convert calls from ioremap() to
memremap() when re-mapping RAM.

This will be used later by SME to control how the encryption mask is
applied to memory mappings, with certain memory locations being mapped
decrypted vs encrypted.
Signed-off-by: default avatarTom Lendacky <thomas.lendacky@amd.com>
Reviewed-by: default avatarThomas Gleixner <tglx@linutronix.de>
Reviewed-by: default avatarBorislav Petkov <bp@suse.de>
Cc: Alexander Potapenko <glider@google.com>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Andy Lutomirski <luto@kernel.org>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Borislav Petkov <bp@alien8.de>
Cc: Brijesh Singh <brijesh.singh@amd.com>
Cc: Dave Young <dyoung@redhat.com>
Cc: Dmitry Vyukov <dvyukov@google.com>
Cc: Jonathan Corbet <corbet@lwn.net>
Cc: Konrad Rzeszutek Wilk <konrad.wilk@oracle.com>
Cc: Larry Woodman <lwoodman@redhat.com>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Matt Fleming <matt@codeblueprint.co.uk>
Cc: Michael S. Tsirkin <mst@redhat.com>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Radim Krčmář <rkrcmar@redhat.com>
Cc: Rik van Riel <riel@redhat.com>
Cc: Toshimitsu Kani <toshi.kani@hpe.com>
Cc: kasan-dev@googlegroups.com
Cc: kvm@vger.kernel.org
Cc: linux-arch@vger.kernel.org
Cc: linux-doc@vger.kernel.org
Cc: linux-efi@vger.kernel.org
Cc: linux-mm@kvack.org
Link: http://lkml.kernel.org/r/b13fccb9abbd547a7eef7b1fdfc223431b211c88.1500319216.git.thomas.lendacky@amd.comSigned-off-by: default avatarIngo Molnar <mingo@kernel.org>
parent aac7b79e
...@@ -13,9 +13,9 @@ static __always_inline __init void *dmi_alloc(unsigned len) ...@@ -13,9 +13,9 @@ static __always_inline __init void *dmi_alloc(unsigned len)
} }
/* Use early IO mappings for DMI because it's initialized early */ /* Use early IO mappings for DMI because it's initialized early */
#define dmi_early_remap early_ioremap #define dmi_early_remap early_memremap
#define dmi_early_unmap early_iounmap #define dmi_early_unmap early_memunmap
#define dmi_remap ioremap_cache #define dmi_remap(_x, _l) memremap(_x, _l, MEMREMAP_WB)
#define dmi_unmap iounmap #define dmi_unmap(_x) memunmap(_x)
#endif /* _ASM_X86_DMI_H */ #endif /* _ASM_X86_DMI_H */
...@@ -115,7 +115,7 @@ static u32 isa_irq_to_gsi[NR_IRQS_LEGACY] __read_mostly = { ...@@ -115,7 +115,7 @@ static u32 isa_irq_to_gsi[NR_IRQS_LEGACY] __read_mostly = {
#define ACPI_INVALID_GSI INT_MIN #define ACPI_INVALID_GSI INT_MIN
/* /*
* This is just a simple wrapper around early_ioremap(), * This is just a simple wrapper around early_memremap(),
* with sanity checks for phys == 0 and size == 0. * with sanity checks for phys == 0 and size == 0.
*/ */
char *__init __acpi_map_table(unsigned long phys, unsigned long size) char *__init __acpi_map_table(unsigned long phys, unsigned long size)
...@@ -124,7 +124,7 @@ char *__init __acpi_map_table(unsigned long phys, unsigned long size) ...@@ -124,7 +124,7 @@ char *__init __acpi_map_table(unsigned long phys, unsigned long size)
if (!phys || !size) if (!phys || !size)
return NULL; return NULL;
return early_ioremap(phys, size); return early_memremap(phys, size);
} }
void __init __acpi_unmap_table(char *map, unsigned long size) void __init __acpi_unmap_table(char *map, unsigned long size)
...@@ -132,7 +132,7 @@ void __init __acpi_unmap_table(char *map, unsigned long size) ...@@ -132,7 +132,7 @@ void __init __acpi_unmap_table(char *map, unsigned long size)
if (!map || !size) if (!map || !size)
return; return;
early_iounmap(map, size); early_memunmap(map, size);
} }
#ifdef CONFIG_X86_LOCAL_APIC #ifdef CONFIG_X86_LOCAL_APIC
......
...@@ -33,7 +33,6 @@ static ssize_t setup_data_read(struct file *file, char __user *user_buf, ...@@ -33,7 +33,6 @@ static ssize_t setup_data_read(struct file *file, char __user *user_buf,
struct setup_data_node *node = file->private_data; struct setup_data_node *node = file->private_data;
unsigned long remain; unsigned long remain;
loff_t pos = *ppos; loff_t pos = *ppos;
struct page *pg;
void *p; void *p;
u64 pa; u64 pa;
...@@ -47,18 +46,13 @@ static ssize_t setup_data_read(struct file *file, char __user *user_buf, ...@@ -47,18 +46,13 @@ static ssize_t setup_data_read(struct file *file, char __user *user_buf,
count = node->len - pos; count = node->len - pos;
pa = node->paddr + sizeof(struct setup_data) + pos; pa = node->paddr + sizeof(struct setup_data) + pos;
pg = pfn_to_page((pa + count - 1) >> PAGE_SHIFT); p = memremap(pa, count, MEMREMAP_WB);
if (PageHighMem(pg)) {
p = ioremap_cache(pa, count);
if (!p) if (!p)
return -ENXIO; return -ENOMEM;
} else
p = __va(pa);
remain = copy_to_user(user_buf, p, count); remain = copy_to_user(user_buf, p, count);
if (PageHighMem(pg)) memunmap(p);
iounmap(p);
if (remain) if (remain)
return -EFAULT; return -EFAULT;
...@@ -109,7 +103,6 @@ static int __init create_setup_data_nodes(struct dentry *parent) ...@@ -109,7 +103,6 @@ static int __init create_setup_data_nodes(struct dentry *parent)
struct setup_data *data; struct setup_data *data;
int error; int error;
struct dentry *d; struct dentry *d;
struct page *pg;
u64 pa_data; u64 pa_data;
int no = 0; int no = 0;
...@@ -126,16 +119,12 @@ static int __init create_setup_data_nodes(struct dentry *parent) ...@@ -126,16 +119,12 @@ static int __init create_setup_data_nodes(struct dentry *parent)
goto err_dir; goto err_dir;
} }
pg = pfn_to_page((pa_data+sizeof(*data)-1) >> PAGE_SHIFT); data = memremap(pa_data, sizeof(*data), MEMREMAP_WB);
if (PageHighMem(pg)) {
data = ioremap_cache(pa_data, sizeof(*data));
if (!data) { if (!data) {
kfree(node); kfree(node);
error = -ENXIO; error = -ENOMEM;
goto err_dir; goto err_dir;
} }
} else
data = __va(pa_data);
node->paddr = pa_data; node->paddr = pa_data;
node->type = data->type; node->type = data->type;
...@@ -143,8 +132,7 @@ static int __init create_setup_data_nodes(struct dentry *parent) ...@@ -143,8 +132,7 @@ static int __init create_setup_data_nodes(struct dentry *parent)
error = create_setup_data_node(d, no, node); error = create_setup_data_node(d, no, node);
pa_data = data->next; pa_data = data->next;
if (PageHighMem(pg)) memunmap(data);
iounmap(data);
if (error) if (error)
goto err_dir; goto err_dir;
no++; no++;
......
...@@ -16,8 +16,8 @@ ...@@ -16,8 +16,8 @@
#include <linux/stat.h> #include <linux/stat.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/mm.h> #include <linux/mm.h>
#include <linux/io.h>
#include <asm/io.h>
#include <asm/setup.h> #include <asm/setup.h>
static ssize_t version_show(struct kobject *kobj, static ssize_t version_show(struct kobject *kobj,
...@@ -79,12 +79,12 @@ static int get_setup_data_paddr(int nr, u64 *paddr) ...@@ -79,12 +79,12 @@ static int get_setup_data_paddr(int nr, u64 *paddr)
*paddr = pa_data; *paddr = pa_data;
return 0; return 0;
} }
data = ioremap_cache(pa_data, sizeof(*data)); data = memremap(pa_data, sizeof(*data), MEMREMAP_WB);
if (!data) if (!data)
return -ENOMEM; return -ENOMEM;
pa_data = data->next; pa_data = data->next;
iounmap(data); memunmap(data);
i++; i++;
} }
return -EINVAL; return -EINVAL;
...@@ -97,17 +97,17 @@ static int __init get_setup_data_size(int nr, size_t *size) ...@@ -97,17 +97,17 @@ static int __init get_setup_data_size(int nr, size_t *size)
u64 pa_data = boot_params.hdr.setup_data; u64 pa_data = boot_params.hdr.setup_data;
while (pa_data) { while (pa_data) {
data = ioremap_cache(pa_data, sizeof(*data)); data = memremap(pa_data, sizeof(*data), MEMREMAP_WB);
if (!data) if (!data)
return -ENOMEM; return -ENOMEM;
if (nr == i) { if (nr == i) {
*size = data->len; *size = data->len;
iounmap(data); memunmap(data);
return 0; return 0;
} }
pa_data = data->next; pa_data = data->next;
iounmap(data); memunmap(data);
i++; i++;
} }
return -EINVAL; return -EINVAL;
...@@ -127,12 +127,12 @@ static ssize_t type_show(struct kobject *kobj, ...@@ -127,12 +127,12 @@ static ssize_t type_show(struct kobject *kobj,
ret = get_setup_data_paddr(nr, &paddr); ret = get_setup_data_paddr(nr, &paddr);
if (ret) if (ret)
return ret; return ret;
data = ioremap_cache(paddr, sizeof(*data)); data = memremap(paddr, sizeof(*data), MEMREMAP_WB);
if (!data) if (!data)
return -ENOMEM; return -ENOMEM;
ret = sprintf(buf, "0x%x\n", data->type); ret = sprintf(buf, "0x%x\n", data->type);
iounmap(data); memunmap(data);
return ret; return ret;
} }
...@@ -154,7 +154,7 @@ static ssize_t setup_data_data_read(struct file *fp, ...@@ -154,7 +154,7 @@ static ssize_t setup_data_data_read(struct file *fp,
ret = get_setup_data_paddr(nr, &paddr); ret = get_setup_data_paddr(nr, &paddr);
if (ret) if (ret)
return ret; return ret;
data = ioremap_cache(paddr, sizeof(*data)); data = memremap(paddr, sizeof(*data), MEMREMAP_WB);
if (!data) if (!data)
return -ENOMEM; return -ENOMEM;
...@@ -170,15 +170,15 @@ static ssize_t setup_data_data_read(struct file *fp, ...@@ -170,15 +170,15 @@ static ssize_t setup_data_data_read(struct file *fp,
goto out; goto out;
ret = count; ret = count;
p = ioremap_cache(paddr + sizeof(*data), data->len); p = memremap(paddr + sizeof(*data), data->len, MEMREMAP_WB);
if (!p) { if (!p) {
ret = -ENOMEM; ret = -ENOMEM;
goto out; goto out;
} }
memcpy(buf, p + off, count); memcpy(buf, p + off, count);
iounmap(p); memunmap(p);
out: out:
iounmap(data); memunmap(data);
return ret; return ret;
} }
...@@ -250,13 +250,13 @@ static int __init get_setup_data_total_num(u64 pa_data, int *nr) ...@@ -250,13 +250,13 @@ static int __init get_setup_data_total_num(u64 pa_data, int *nr)
*nr = 0; *nr = 0;
while (pa_data) { while (pa_data) {
*nr += 1; *nr += 1;
data = ioremap_cache(pa_data, sizeof(*data)); data = memremap(pa_data, sizeof(*data), MEMREMAP_WB);
if (!data) { if (!data) {
ret = -ENOMEM; ret = -ENOMEM;
goto out; goto out;
} }
pa_data = data->next; pa_data = data->next;
iounmap(data); memunmap(data);
} }
out: out:
......
...@@ -436,9 +436,9 @@ static unsigned long __init get_mpc_size(unsigned long physptr) ...@@ -436,9 +436,9 @@ static unsigned long __init get_mpc_size(unsigned long physptr)
struct mpc_table *mpc; struct mpc_table *mpc;
unsigned long size; unsigned long size;
mpc = early_ioremap(physptr, PAGE_SIZE); mpc = early_memremap(physptr, PAGE_SIZE);
size = mpc->length; size = mpc->length;
early_iounmap(mpc, PAGE_SIZE); early_memunmap(mpc, PAGE_SIZE);
apic_printk(APIC_VERBOSE, " mpc: %lx-%lx\n", physptr, physptr + size); apic_printk(APIC_VERBOSE, " mpc: %lx-%lx\n", physptr, physptr + size);
return size; return size;
...@@ -450,7 +450,7 @@ static int __init check_physptr(struct mpf_intel *mpf, unsigned int early) ...@@ -450,7 +450,7 @@ static int __init check_physptr(struct mpf_intel *mpf, unsigned int early)
unsigned long size; unsigned long size;
size = get_mpc_size(mpf->physptr); size = get_mpc_size(mpf->physptr);
mpc = early_ioremap(mpf->physptr, size); mpc = early_memremap(mpf->physptr, size);
/* /*
* Read the physical hardware table. Anything here will * Read the physical hardware table. Anything here will
* override the defaults. * override the defaults.
...@@ -461,10 +461,10 @@ static int __init check_physptr(struct mpf_intel *mpf, unsigned int early) ...@@ -461,10 +461,10 @@ static int __init check_physptr(struct mpf_intel *mpf, unsigned int early)
#endif #endif
pr_err("BIOS bug, MP table errors detected!...\n"); pr_err("BIOS bug, MP table errors detected!...\n");
pr_cont("... disabling SMP support. (tell your hw vendor)\n"); pr_cont("... disabling SMP support. (tell your hw vendor)\n");
early_iounmap(mpc, size); early_memunmap(mpc, size);
return -1; return -1;
} }
early_iounmap(mpc, size); early_memunmap(mpc, size);
if (early) if (early)
return -1; return -1;
......
...@@ -674,7 +674,7 @@ int pcibios_add_device(struct pci_dev *dev) ...@@ -674,7 +674,7 @@ int pcibios_add_device(struct pci_dev *dev)
pa_data = boot_params.hdr.setup_data; pa_data = boot_params.hdr.setup_data;
while (pa_data) { while (pa_data) {
data = ioremap(pa_data, sizeof(*rom)); data = memremap(pa_data, sizeof(*rom), MEMREMAP_WB);
if (!data) if (!data)
return -ENOMEM; return -ENOMEM;
...@@ -693,7 +693,7 @@ int pcibios_add_device(struct pci_dev *dev) ...@@ -693,7 +693,7 @@ int pcibios_add_device(struct pci_dev *dev)
} }
} }
pa_data = data->next; pa_data = data->next;
iounmap(data); memunmap(data);
} }
set_dma_domain_ops(dev); set_dma_domain_ops(dev);
set_dev_domain_options(dev); set_dev_domain_options(dev);
......
...@@ -25,6 +25,7 @@ ...@@ -25,6 +25,7 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/list.h> #include <linux/list.h>
#include <linux/io.h> #include <linux/io.h>
#include <asm/dmi.h>
#define MAX_ENTRY_TYPE 255 /* Most of these aren't used, but we consider #define MAX_ENTRY_TYPE 255 /* Most of these aren't used, but we consider
the top entry type is only 8 bits */ the top entry type is only 8 bits */
...@@ -380,7 +381,7 @@ static ssize_t dmi_sel_raw_read_phys32(struct dmi_sysfs_entry *entry, ...@@ -380,7 +381,7 @@ static ssize_t dmi_sel_raw_read_phys32(struct dmi_sysfs_entry *entry,
u8 __iomem *mapped; u8 __iomem *mapped;
ssize_t wrote = 0; ssize_t wrote = 0;
mapped = ioremap(sel->access_method_address, sel->area_length); mapped = dmi_remap(sel->access_method_address, sel->area_length);
if (!mapped) if (!mapped)
return -EIO; return -EIO;
...@@ -390,7 +391,7 @@ static ssize_t dmi_sel_raw_read_phys32(struct dmi_sysfs_entry *entry, ...@@ -390,7 +391,7 @@ static ssize_t dmi_sel_raw_read_phys32(struct dmi_sysfs_entry *entry,
wrote++; wrote++;
} }
iounmap(mapped); dmi_unmap(mapped);
return wrote; return wrote;
} }
......
...@@ -95,7 +95,7 @@ efi_setup_pcdp_console(char *cmdline) ...@@ -95,7 +95,7 @@ efi_setup_pcdp_console(char *cmdline)
if (efi.hcdp == EFI_INVALID_TABLE_ADDR) if (efi.hcdp == EFI_INVALID_TABLE_ADDR)
return -ENODEV; return -ENODEV;
pcdp = early_ioremap(efi.hcdp, 4096); pcdp = early_memremap(efi.hcdp, 4096);
printk(KERN_INFO "PCDP: v%d at 0x%lx\n", pcdp->rev, efi.hcdp); printk(KERN_INFO "PCDP: v%d at 0x%lx\n", pcdp->rev, efi.hcdp);
if (strstr(cmdline, "console=hcdp")) { if (strstr(cmdline, "console=hcdp")) {
...@@ -131,6 +131,6 @@ efi_setup_pcdp_console(char *cmdline) ...@@ -131,6 +131,6 @@ efi_setup_pcdp_console(char *cmdline)
} }
out: out:
early_iounmap(pcdp, 4096); early_memunmap(pcdp, 4096);
return rc; return rc;
} }
...@@ -86,13 +86,13 @@ static struct sfi_table_simple *syst_va __read_mostly; ...@@ -86,13 +86,13 @@ static struct sfi_table_simple *syst_va __read_mostly;
/* /*
* FW creates and saves the SFI tables in memory. When these tables get * FW creates and saves the SFI tables in memory. When these tables get
* used, they may need to be mapped to virtual address space, and the mapping * used, they may need to be mapped to virtual address space, and the mapping
* can happen before or after the ioremap() is ready, so a flag is needed * can happen before or after the memremap() is ready, so a flag is needed
* to indicating this * to indicating this
*/ */
static u32 sfi_use_ioremap __read_mostly; static u32 sfi_use_memremap __read_mostly;
/* /*
* sfi_un/map_memory calls early_ioremap/iounmap which is a __init function * sfi_un/map_memory calls early_memremap/memunmap which is a __init function
* and introduces section mismatch. So use __ref to make it calm. * and introduces section mismatch. So use __ref to make it calm.
*/ */
static void __iomem * __ref sfi_map_memory(u64 phys, u32 size) static void __iomem * __ref sfi_map_memory(u64 phys, u32 size)
...@@ -100,10 +100,10 @@ static void __iomem * __ref sfi_map_memory(u64 phys, u32 size) ...@@ -100,10 +100,10 @@ static void __iomem * __ref sfi_map_memory(u64 phys, u32 size)
if (!phys || !size) if (!phys || !size)
return NULL; return NULL;
if (sfi_use_ioremap) if (sfi_use_memremap)
return ioremap_cache(phys, size); return memremap(phys, size, MEMREMAP_WB);
else else
return early_ioremap(phys, size); return early_memremap(phys, size);
} }
static void __ref sfi_unmap_memory(void __iomem *virt, u32 size) static void __ref sfi_unmap_memory(void __iomem *virt, u32 size)
...@@ -111,10 +111,10 @@ static void __ref sfi_unmap_memory(void __iomem *virt, u32 size) ...@@ -111,10 +111,10 @@ static void __ref sfi_unmap_memory(void __iomem *virt, u32 size)
if (!virt || !size) if (!virt || !size)
return; return;
if (sfi_use_ioremap) if (sfi_use_memremap)
iounmap(virt); memunmap(virt);
else else
early_iounmap(virt, size); early_memunmap(virt, size);
} }
static void sfi_print_table_header(unsigned long long pa, static void sfi_print_table_header(unsigned long long pa,
...@@ -507,8 +507,8 @@ void __init sfi_init_late(void) ...@@ -507,8 +507,8 @@ void __init sfi_init_late(void)
length = syst_va->header.len; length = syst_va->header.len;
sfi_unmap_memory(syst_va, sizeof(struct sfi_table_simple)); sfi_unmap_memory(syst_va, sizeof(struct sfi_table_simple));
/* Use ioremap now after it is ready */ /* Use memremap now after it is ready */
sfi_use_ioremap = 1; sfi_use_memremap = 1;
syst_va = sfi_map_memory(syst_pa, length); syst_va = sfi_map_memory(syst_pa, length);
sfi_acpi_init(); sfi_acpi_init();
......
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