pgalloc.h 3.9 KB
Newer Older
1 2
#ifndef _ASM_POWERPC_PGALLOC_H
#define _ASM_POWERPC_PGALLOC_H
3
#ifdef __KERNEL__
4 5 6 7

#ifndef CONFIG_PPC64
#include <asm-ppc/pgalloc.h>
#else
L
Linus Torvalds 已提交
8 9 10 11 12 13

#include <linux/mm.h>
#include <linux/slab.h>
#include <linux/cpumask.h>
#include <linux/percpu.h>

14 15
extern kmem_cache_t *pgtable_cache[];

16 17
#ifdef CONFIG_PPC_64K_PAGES
#define PTE_CACHE_NUM	0
18 19
#define PMD_CACHE_NUM	1
#define PGD_CACHE_NUM	2
20
#define HUGEPTE_CACHE_NUM 3
21
#else
22 23 24 25
#define PTE_CACHE_NUM	0
#define PMD_CACHE_NUM	1
#define PUD_CACHE_NUM	1
#define PGD_CACHE_NUM	0
26
#define HUGEPTE_CACHE_NUM 2
27
#endif
L
Linus Torvalds 已提交
28 29 30 31 32 33 34 35

/*
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version
 * 2 of the License, or (at your option) any later version.
 */

36
static inline pgd_t *pgd_alloc(struct mm_struct *mm)
L
Linus Torvalds 已提交
37
{
38
	return kmem_cache_alloc(pgtable_cache[PGD_CACHE_NUM], GFP_KERNEL);
L
Linus Torvalds 已提交
39 40
}

41
static inline void pgd_free(pgd_t *pgd)
L
Linus Torvalds 已提交
42
{
43 44 45
	kmem_cache_free(pgtable_cache[PGD_CACHE_NUM], pgd);
}

46 47
#ifndef CONFIG_PPC_64K_PAGES

48 49 50 51 52 53 54 55 56 57 58
#define pgd_populate(MM, PGD, PUD)	pgd_set(PGD, PUD)

static inline pud_t *pud_alloc_one(struct mm_struct *mm, unsigned long addr)
{
	return kmem_cache_alloc(pgtable_cache[PUD_CACHE_NUM],
				GFP_KERNEL|__GFP_REPEAT);
}

static inline void pud_free(pud_t *pud)
{
	kmem_cache_free(pgtable_cache[PUD_CACHE_NUM], pud);
L
Linus Torvalds 已提交
59 60
}

61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
static inline void pud_populate(struct mm_struct *mm, pud_t *pud, pmd_t *pmd)
{
	pud_set(pud, (unsigned long)pmd);
}

#define pmd_populate(mm, pmd, pte_page) \
	pmd_populate_kernel(mm, pmd, page_address(pte_page))
#define pmd_populate_kernel(mm, pmd, pte) pmd_set(pmd, (unsigned long)(pte))


#else /* CONFIG_PPC_64K_PAGES */

#define pud_populate(mm, pud, pmd)	pud_set(pud, (unsigned long)pmd)

static inline void pmd_populate_kernel(struct mm_struct *mm, pmd_t *pmd,
				       pte_t *pte)
{
	pmd_set(pmd, (unsigned long)pte);
}

#define pmd_populate(mm, pmd, pte_page) \
	pmd_populate_kernel(mm, pmd, page_address(pte_page))

#endif /* CONFIG_PPC_64K_PAGES */
L
Linus Torvalds 已提交
85

86
static inline pmd_t *pmd_alloc_one(struct mm_struct *mm, unsigned long addr)
L
Linus Torvalds 已提交
87
{
88 89
	return kmem_cache_alloc(pgtable_cache[PMD_CACHE_NUM],
				GFP_KERNEL|__GFP_REPEAT);
L
Linus Torvalds 已提交
90 91
}

92
static inline void pmd_free(pmd_t *pmd)
L
Linus Torvalds 已提交
93
{
94
	kmem_cache_free(pgtable_cache[PMD_CACHE_NUM], pmd);
L
Linus Torvalds 已提交
95 96
}

97 98
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
					  unsigned long address)
L
Linus Torvalds 已提交
99
{
100 101
	return kmem_cache_alloc(pgtable_cache[PTE_CACHE_NUM],
				GFP_KERNEL|__GFP_REPEAT);
L
Linus Torvalds 已提交
102 103
}

104 105
static inline struct page *pte_alloc_one(struct mm_struct *mm,
					 unsigned long address)
L
Linus Torvalds 已提交
106
{
107
	return virt_to_page(pte_alloc_one_kernel(mm, address));
L
Linus Torvalds 已提交
108 109 110 111
}
		
static inline void pte_free_kernel(pte_t *pte)
{
112
	kmem_cache_free(pgtable_cache[PTE_CACHE_NUM], pte);
L
Linus Torvalds 已提交
113 114 115 116
}

static inline void pte_free(struct page *ptepage)
{
117
	pte_free_kernel(page_address(ptepage));
L
Linus Torvalds 已提交
118 119
}

120 121 122 123 124 125 126 127
#define PGF_CACHENUM_MASK	0xf

typedef struct pgtable_free {
	unsigned long val;
} pgtable_free_t;

static inline pgtable_free_t pgtable_free_cache(void *p, int cachenum,
						unsigned long mask)
L
Linus Torvalds 已提交
128
{
129
	BUG_ON(cachenum > PGF_CACHENUM_MASK);
L
Linus Torvalds 已提交
130

131 132
	return (pgtable_free_t){.val = ((unsigned long) p & ~mask) | cachenum};
}
L
Linus Torvalds 已提交
133

134 135 136 137
static inline void pgtable_free(pgtable_free_t pgf)
{
	void *p = (void *)(pgf.val & ~PGF_CACHENUM_MASK);
	int cachenum = pgf.val & PGF_CACHENUM_MASK;
L
Linus Torvalds 已提交
138

139 140
	kmem_cache_free(pgtable_cache[cachenum], p);
}
L
Linus Torvalds 已提交
141

142
extern void pgtable_free_tlb(struct mmu_gather *tlb, pgtable_free_t pgf);
143 144 145 146 147 148 149

#define __pte_free_tlb(tlb, ptepage)	\
	pgtable_free_tlb(tlb, pgtable_free_cache(page_address(ptepage), \
		PTE_CACHE_NUM, PTE_TABLE_SIZE-1))
#define __pmd_free_tlb(tlb, pmd) 	\
	pgtable_free_tlb(tlb, pgtable_free_cache(pmd, \
		PMD_CACHE_NUM, PMD_TABLE_SIZE-1))
150
#ifndef CONFIG_PPC_64K_PAGES
151
#define __pud_free_tlb(tlb, pud)	\
152 153
	pgtable_free_tlb(tlb, pgtable_free_cache(pud, \
		PUD_CACHE_NUM, PUD_TABLE_SIZE-1))
154
#endif /* CONFIG_PPC_64K_PAGES */
L
Linus Torvalds 已提交
155 156 157

#define check_pgt_cache()	do { } while (0)

158
#endif /* CONFIG_PPC64 */
159
#endif /* __KERNEL__ */
160
#endif /* _ASM_POWERPC_PGALLOC_H */