pgalloc.h 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. /* SPDX-License-Identifier: GPL-2.0 */
  2. /*
  3. * S390 version
  4. * Copyright IBM Corp. 1999, 2000
  5. * Author(s): Hartmut Penner ([email protected])
  6. * Martin Schwidefsky ([email protected])
  7. *
  8. * Derived from "include/asm-i386/pgalloc.h"
  9. * Copyright (C) 1994 Linus Torvalds
  10. */
  11. #ifndef _S390_PGALLOC_H
  12. #define _S390_PGALLOC_H
  13. #include <linux/threads.h>
  14. #include <linux/string.h>
  15. #include <linux/gfp.h>
  16. #include <linux/mm.h>
  17. #define CRST_ALLOC_ORDER 2
  18. unsigned long *crst_table_alloc(struct mm_struct *);
  19. void crst_table_free(struct mm_struct *, unsigned long *);
  20. unsigned long *page_table_alloc(struct mm_struct *);
  21. struct page *page_table_alloc_pgste(struct mm_struct *mm);
  22. void page_table_free(struct mm_struct *, unsigned long *);
  23. void page_table_free_rcu(struct mmu_gather *, unsigned long *, unsigned long);
  24. void page_table_free_pgste(struct page *page);
  25. extern int page_table_allocate_pgste;
  26. static inline void crst_table_init(unsigned long *crst, unsigned long entry)
  27. {
  28. memset64((u64 *)crst, entry, _CRST_ENTRIES);
  29. }
  30. int crst_table_upgrade(struct mm_struct *mm, unsigned long limit);
  31. static inline unsigned long check_asce_limit(struct mm_struct *mm, unsigned long addr,
  32. unsigned long len)
  33. {
  34. int rc;
  35. if (addr + len > mm->context.asce_limit &&
  36. addr + len <= TASK_SIZE) {
  37. rc = crst_table_upgrade(mm, addr + len);
  38. if (rc)
  39. return (unsigned long) rc;
  40. }
  41. return addr;
  42. }
  43. static inline p4d_t *p4d_alloc_one(struct mm_struct *mm, unsigned long address)
  44. {
  45. unsigned long *table = crst_table_alloc(mm);
  46. if (table)
  47. crst_table_init(table, _REGION2_ENTRY_EMPTY);
  48. return (p4d_t *) table;
  49. }
  50. static inline void p4d_free(struct mm_struct *mm, p4d_t *p4d)
  51. {
  52. if (!mm_p4d_folded(mm))
  53. crst_table_free(mm, (unsigned long *) p4d);
  54. }
  55. static inline pud_t *pud_alloc_one(struct mm_struct *mm, unsigned long address)
  56. {
  57. unsigned long *table = crst_table_alloc(mm);
  58. if (table)
  59. crst_table_init(table, _REGION3_ENTRY_EMPTY);
  60. return (pud_t *) table;
  61. }
  62. static inline void pud_free(struct mm_struct *mm, pud_t *pud)
  63. {
  64. if (!mm_pud_folded(mm))
  65. crst_table_free(mm, (unsigned long *) pud);
  66. }
  67. static inline pmd_t *pmd_alloc_one(struct mm_struct *mm, unsigned long vmaddr)
  68. {
  69. unsigned long *table = crst_table_alloc(mm);
  70. if (!table)
  71. return NULL;
  72. crst_table_init(table, _SEGMENT_ENTRY_EMPTY);
  73. if (!pgtable_pmd_page_ctor(virt_to_page(table))) {
  74. crst_table_free(mm, table);
  75. return NULL;
  76. }
  77. return (pmd_t *) table;
  78. }
  79. static inline void pmd_free(struct mm_struct *mm, pmd_t *pmd)
  80. {
  81. if (mm_pmd_folded(mm))
  82. return;
  83. pgtable_pmd_page_dtor(virt_to_page(pmd));
  84. crst_table_free(mm, (unsigned long *) pmd);
  85. }
  86. static inline void pgd_populate(struct mm_struct *mm, pgd_t *pgd, p4d_t *p4d)
  87. {
  88. set_pgd(pgd, __pgd(_REGION1_ENTRY | __pa(p4d)));
  89. }
  90. static inline void p4d_populate(struct mm_struct *mm, p4d_t *p4d, pud_t *pud)
  91. {
  92. set_p4d(p4d, __p4d(_REGION2_ENTRY | __pa(pud)));
  93. }
  94. static inline void pud_populate(struct mm_struct *mm, pud_t *pud, pmd_t *pmd)
  95. {
  96. set_pud(pud, __pud(_REGION3_ENTRY | __pa(pmd)));
  97. }
  98. static inline pgd_t *pgd_alloc(struct mm_struct *mm)
  99. {
  100. return (pgd_t *) crst_table_alloc(mm);
  101. }
  102. static inline void pgd_free(struct mm_struct *mm, pgd_t *pgd)
  103. {
  104. crst_table_free(mm, (unsigned long *) pgd);
  105. }
  106. static inline void pmd_populate(struct mm_struct *mm,
  107. pmd_t *pmd, pgtable_t pte)
  108. {
  109. set_pmd(pmd, __pmd(_SEGMENT_ENTRY | __pa(pte)));
  110. }
  111. #define pmd_populate_kernel(mm, pmd, pte) pmd_populate(mm, pmd, pte)
  112. /*
  113. * page table entry allocation/free routines.
  114. */
  115. #define pte_alloc_one_kernel(mm) ((pte_t *)page_table_alloc(mm))
  116. #define pte_alloc_one(mm) ((pte_t *)page_table_alloc(mm))
  117. #define pte_free_kernel(mm, pte) page_table_free(mm, (unsigned long *) pte)
  118. #define pte_free(mm, pte) page_table_free(mm, (unsigned long *) pte)
  119. void vmem_map_init(void);
  120. void *vmem_crst_alloc(unsigned long val);
  121. pte_t *vmem_pte_alloc(void);
  122. unsigned long base_asce_alloc(unsigned long addr, unsigned long num_pages);
  123. void base_asce_free(unsigned long asce);
  124. #endif /* _S390_PGALLOC_H */