highmem.c 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
  1. /*
  2. * High memory support for Xtensa architecture
  3. *
  4. * This file is subject to the terms and conditions of the GNU General
  5. * Public License. See the file "COPYING" in the main directory of
  6. * this archive for more details.
  7. *
  8. * Copyright (C) 2014 Cadence Design Systems Inc.
  9. */
  10. #include <linux/export.h>
  11. #include <linux/highmem.h>
  12. #include <asm/tlbflush.h>
  13. static pte_t *kmap_pte;
  14. void *kmap_atomic(struct page *page)
  15. {
  16. enum fixed_addresses idx;
  17. unsigned long vaddr;
  18. int type;
  19. pagefault_disable();
  20. if (!PageHighMem(page))
  21. return page_address(page);
  22. type = kmap_atomic_idx_push();
  23. idx = type + KM_TYPE_NR * smp_processor_id();
  24. vaddr = __fix_to_virt(FIX_KMAP_BEGIN + idx);
  25. #ifdef CONFIG_DEBUG_HIGHMEM
  26. BUG_ON(!pte_none(*(kmap_pte - idx)));
  27. #endif
  28. set_pte(kmap_pte - idx, mk_pte(page, PAGE_KERNEL_EXEC));
  29. return (void *)vaddr;
  30. }
  31. EXPORT_SYMBOL(kmap_atomic);
  32. void __kunmap_atomic(void *kvaddr)
  33. {
  34. int idx, type;
  35. if (kvaddr >= (void *)FIXADDR_START &&
  36. kvaddr < (void *)FIXADDR_TOP) {
  37. type = kmap_atomic_idx();
  38. idx = type + KM_TYPE_NR * smp_processor_id();
  39. /*
  40. * Force other mappings to Oops if they'll try to access this
  41. * pte without first remap it. Keeping stale mappings around
  42. * is a bad idea also, in case the page changes cacheability
  43. * attributes or becomes a protected page in a hypervisor.
  44. */
  45. pte_clear(&init_mm, kvaddr, kmap_pte - idx);
  46. local_flush_tlb_kernel_range((unsigned long)kvaddr,
  47. (unsigned long)kvaddr + PAGE_SIZE);
  48. kmap_atomic_idx_pop();
  49. }
  50. pagefault_enable();
  51. }
  52. EXPORT_SYMBOL(__kunmap_atomic);
  53. void __init kmap_init(void)
  54. {
  55. unsigned long kmap_vstart;
  56. /* cache the first kmap pte */
  57. kmap_vstart = __fix_to_virt(FIX_KMAP_BEGIN);
  58. kmap_pte = kmap_get_fixmap_pte(kmap_vstart);
  59. }