pageattr.c 1.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
  1. /*
  2. * Copyright IBM Corp. 2011
  3. * Author(s): Jan Glauber <jang@linux.vnet.ibm.com>
  4. */
  5. #include <linux/module.h>
  6. #include <linux/mm.h>
  7. #include <linux/hugetlb.h>
  8. #include <asm/pgtable.h>
  9. static void change_page_attr(unsigned long addr, int numpages,
  10. pte_t (*set) (pte_t))
  11. {
  12. pte_t *ptep, pte;
  13. pmd_t *pmdp;
  14. pud_t *pudp;
  15. pgd_t *pgdp;
  16. int i;
  17. for (i = 0; i < numpages; i++) {
  18. pgdp = pgd_offset(&init_mm, addr);
  19. pudp = pud_offset(pgdp, addr);
  20. pmdp = pmd_offset(pudp, addr);
  21. if (pmd_huge(*pmdp)) {
  22. WARN_ON_ONCE(1);
  23. continue;
  24. }
  25. ptep = pte_offset_kernel(pmdp, addr + i * PAGE_SIZE);
  26. pte = *ptep;
  27. pte = set(pte);
  28. ptep_invalidate(&init_mm, addr + i * PAGE_SIZE, ptep);
  29. *ptep = pte;
  30. }
  31. }
  32. int set_memory_ro(unsigned long addr, int numpages)
  33. {
  34. change_page_attr(addr, numpages, pte_wrprotect);
  35. return 0;
  36. }
  37. EXPORT_SYMBOL_GPL(set_memory_ro);
  38. int set_memory_rw(unsigned long addr, int numpages)
  39. {
  40. change_page_attr(addr, numpages, pte_mkwrite);
  41. return 0;
  42. }
  43. EXPORT_SYMBOL_GPL(set_memory_rw);
  44. /* not possible */
  45. int set_memory_nx(unsigned long addr, int numpages)
  46. {
  47. return 0;
  48. }
  49. EXPORT_SYMBOL_GPL(set_memory_nx);