diff options
| author | Naoya Horiguchi <n-horiguchi@ah.jp.nec.com> | 2015-02-11 18:25:15 -0500 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2015-02-11 20:06:01 -0500 |
| commit | 61f77eda9bbf0d2e922197ed2dcf88638a639ce5 (patch) | |
| tree | 4ca42137ea20cf5c48379a87cc0811050f124ee8 /arch/tile | |
| parent | cfc511557945812280699a92f171ddd2d254aca6 (diff) | |
mm/hugetlb: reduce arch dependent code around follow_huge_*
Currently we have many duplicates in definitions around
follow_huge_addr(), follow_huge_pmd(), and follow_huge_pud(), so this
patch tries to remove the m. The basic idea is to put the default
implementation for these functions in mm/hugetlb.c as weak symbols
(regardless of CONFIG_ARCH_WANT_GENERAL_HUGETL B), and to implement
arch-specific code only when the arch needs it.
For follow_huge_addr(), only powerpc and ia64 have their own
implementation, and in all other architectures this function just returns
ERR_PTR(-EINVAL). So this patch sets returning ERR_PTR(-EINVAL) as
default.
As for follow_huge_(pmd|pud)(), if (pmd|pud)_huge() is implemented to
always return 0 in your architecture (like in ia64 or sparc,) it's never
called (the callsite is optimized away) no matter how implemented it is.
So in such architectures, we don't need arch-specific implementation.
In some architecture (like mips, s390 and tile,) their current
arch-specific follow_huge_(pmd|pud)() are effectively identical with the
common code, so this patch lets these architecture use the common code.
One exception is metag, where pmd_huge() could return non-zero but it
expects follow_huge_pmd() to always return NULL. This means that we need
arch-specific implementation which returns NULL. This behavior looks
strange to me (because non-zero pmd_huge() implies that the architecture
supports PMD-based hugepage, so follow_huge_pmd() can/should return some
relevant value,) but that's beyond this cleanup patch, so let's keep it.
Justification of non-trivial changes:
- in s390, follow_huge_pmd() checks !MACHINE_HAS_HPAGE at first, and this
patch removes the check. This is OK because we can assume MACHINE_HAS_HPAGE
is true when follow_huge_pmd() can be called (note that pmd_huge() has
the same check and always returns 0 for !MACHINE_HAS_HPAGE.)
- in s390 and mips, we use HPAGE_MASK instead of PMD_MASK as done in common
code. This patch forces these archs use PMD_MASK, but it's OK because
they are identical in both archs.
In s390, both of HPAGE_SHIFT and PMD_SHIFT are 20.
In mips, HPAGE_SHIFT is defined as (PAGE_SHIFT + PAGE_SHIFT - 3) and
PMD_SHIFT is define as (PAGE_SHIFT + PAGE_SHIFT + PTE_ORDER - 3), but
PTE_ORDER is always 0, so these are identical.
Signed-off-by: Naoya Horiguchi <n-horiguchi@ah.jp.nec.com>
Acked-by: Hugh Dickins <hughd@google.com>
Cc: James Hogan <james.hogan@imgtec.com>
Cc: David Rientjes <rientjes@google.com>
Cc: Mel Gorman <mel@csn.ul.ie>
Cc: Johannes Weiner <hannes@cmpxchg.org>
Cc: Michal Hocko <mhocko@suse.cz>
Cc: Rik van Riel <riel@redhat.com>
Cc: Andrea Arcangeli <aarcange@redhat.com>
Cc: Luiz Capitulino <lcapitulino@redhat.com>
Cc: Nishanth Aravamudan <nacc@linux.vnet.ibm.com>
Cc: Lee Schermerhorn <lee.schermerhorn@hp.com>
Cc: Steve Capper <steve.capper@linaro.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'arch/tile')
| -rw-r--r-- | arch/tile/mm/hugetlbpage.c | 28 |
1 files changed, 0 insertions, 28 deletions
diff --git a/arch/tile/mm/hugetlbpage.c b/arch/tile/mm/hugetlbpage.c index 3270e0019266..8416240c322c 100644 --- a/arch/tile/mm/hugetlbpage.c +++ b/arch/tile/mm/hugetlbpage.c | |||
| @@ -150,12 +150,6 @@ pte_t *huge_pte_offset(struct mm_struct *mm, unsigned long addr) | |||
| 150 | return NULL; | 150 | return NULL; |
| 151 | } | 151 | } |
| 152 | 152 | ||
| 153 | struct page *follow_huge_addr(struct mm_struct *mm, unsigned long address, | ||
| 154 | int write) | ||
| 155 | { | ||
| 156 | return ERR_PTR(-EINVAL); | ||
| 157 | } | ||
| 158 | |||
| 159 | int pmd_huge(pmd_t pmd) | 153 | int pmd_huge(pmd_t pmd) |
| 160 | { | 154 | { |
| 161 | return !!(pmd_val(pmd) & _PAGE_HUGE_PAGE); | 155 | return !!(pmd_val(pmd) & _PAGE_HUGE_PAGE); |
| @@ -166,28 +160,6 @@ int pud_huge(pud_t pud) | |||
| 166 | return !!(pud_val(pud) & _PAGE_HUGE_PAGE); | 160 | return !!(pud_val(pud) & _PAGE_HUGE_PAGE); |
| 167 | } | 161 | } |
| 168 | 162 | ||
| 169 | struct page *follow_huge_pmd(struct mm_struct *mm, unsigned long address, | ||
| 170 | pmd_t *pmd, int write) | ||
| 171 | { | ||
| 172 | struct page *page; | ||
| 173 | |||
| 174 | page = pte_page(*(pte_t *)pmd); | ||
| 175 | if (page) | ||
| 176 | page += ((address & ~PMD_MASK) >> PAGE_SHIFT); | ||
| 177 | return page; | ||
| 178 | } | ||
| 179 | |||
| 180 | struct page *follow_huge_pud(struct mm_struct *mm, unsigned long address, | ||
| 181 | pud_t *pud, int write) | ||
| 182 | { | ||
| 183 | struct page *page; | ||
| 184 | |||
| 185 | page = pte_page(*(pte_t *)pud); | ||
| 186 | if (page) | ||
| 187 | page += ((address & ~PUD_MASK) >> PAGE_SHIFT); | ||
| 188 | return page; | ||
| 189 | } | ||
| 190 | |||
| 191 | int huge_pmd_unshare(struct mm_struct *mm, unsigned long *addr, pte_t *ptep) | 163 | int huge_pmd_unshare(struct mm_struct *mm, unsigned long *addr, pte_t *ptep) |
| 192 | { | 164 | { |
| 193 | return 0; | 165 | return 0; |
