diff options
Diffstat (limited to 'lib/ioremap.c')
| -rw-r--r-- | lib/ioremap.c | 53 | 
1 files changed, 53 insertions, 0 deletions
| diff --git a/lib/ioremap.c b/lib/ioremap.c index 0c9216c48762..86c8911b0e3a 100644 --- a/lib/ioremap.c +++ b/lib/ioremap.c @@ -13,6 +13,43 @@  #include <asm/cacheflush.h>  #include <asm/pgtable.h> +#ifdef CONFIG_HAVE_ARCH_HUGE_VMAP +static int __read_mostly ioremap_pud_capable; +static int __read_mostly ioremap_pmd_capable; +static int __read_mostly ioremap_huge_disabled; + +static int __init set_nohugeiomap(char *str) +{ +	ioremap_huge_disabled = 1; +	return 0; +} +early_param("nohugeiomap", set_nohugeiomap); + +void __init ioremap_huge_init(void) +{ +	if (!ioremap_huge_disabled) { +		if (arch_ioremap_pud_supported()) +			ioremap_pud_capable = 1; +		if (arch_ioremap_pmd_supported()) +			ioremap_pmd_capable = 1; +	} +} + +static inline int ioremap_pud_enabled(void) +{ +	return ioremap_pud_capable; +} + +static inline int ioremap_pmd_enabled(void) +{ +	return ioremap_pmd_capable; +} + +#else	/* !CONFIG_HAVE_ARCH_HUGE_VMAP */ +static inline int ioremap_pud_enabled(void) { return 0; } +static inline int ioremap_pmd_enabled(void) { return 0; } +#endif	/* CONFIG_HAVE_ARCH_HUGE_VMAP */ +  static int ioremap_pte_range(pmd_t *pmd, unsigned long addr,  		unsigned long end, phys_addr_t phys_addr, pgprot_t prot)  { @@ -43,6 +80,14 @@ static inline int ioremap_pmd_range(pud_t *pud, unsigned long addr,  		return -ENOMEM;  	do {  		next = pmd_addr_end(addr, end); + +		if (ioremap_pmd_enabled() && +		    ((next - addr) == PMD_SIZE) && +		    IS_ALIGNED(phys_addr + addr, PMD_SIZE)) { +			if (pmd_set_huge(pmd, phys_addr + addr, prot)) +				continue; +		} +  		if (ioremap_pte_range(pmd, addr, next, phys_addr + addr, prot))  			return -ENOMEM;  	} while (pmd++, addr = next, addr != end); @@ -61,6 +106,14 @@ static inline int ioremap_pud_range(pgd_t *pgd, unsigned long addr,  		return -ENOMEM;  	do {  		next = pud_addr_end(addr, end); + +		if (ioremap_pud_enabled() && +		    ((next - addr) == PUD_SIZE) && +		    IS_ALIGNED(phys_addr + addr, PUD_SIZE)) { +			if (pud_set_huge(pud, phys_addr + addr, prot)) +				continue; +		} +  		if (ioremap_pmd_range(pud, addr, next, phys_addr + addr, prot))  			return -ENOMEM;  	} while (pud++, addr = next, addr != end); | 
