flush.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Based on arch/arm/mm/flush.c
  4. *
  5. * Copyright (C) 1995-2002 Russell King
  6. * Copyright (C) 2012 ARM Ltd.
  7. */
  8. #include <linux/export.h>
  9. #include <linux/mm.h>
  10. #include <linux/libnvdimm.h>
  11. #include <linux/pagemap.h>
  12. #include <asm/cacheflush.h>
  13. #include <asm/cache.h>
  14. #include <asm/tlbflush.h>
  15. void sync_icache_aliases(unsigned long start, unsigned long end)
  16. {
  17. if (icache_is_aliasing()) {
  18. dcache_clean_pou(start, end);
  19. icache_inval_all_pou();
  20. } else {
  21. /*
  22. * Don't issue kick_all_cpus_sync() after I-cache invalidation
  23. * for user mappings.
  24. */
  25. caches_clean_inval_pou(start, end);
  26. }
  27. }
  28. static void flush_ptrace_access(struct vm_area_struct *vma, unsigned long start,
  29. unsigned long end)
  30. {
  31. if (vma->vm_flags & VM_EXEC)
  32. sync_icache_aliases(start, end);
  33. }
  34. /*
  35. * Copy user data from/to a page which is mapped into a different processes
  36. * address space. Really, we want to allow our "user space" model to handle
  37. * this.
  38. */
  39. void copy_to_user_page(struct vm_area_struct *vma, struct page *page,
  40. unsigned long uaddr, void *dst, const void *src,
  41. unsigned long len)
  42. {
  43. memcpy(dst, src, len);
  44. flush_ptrace_access(vma, (unsigned long)dst, (unsigned long)dst + len);
  45. }
  46. void __sync_icache_dcache(pte_t pte)
  47. {
  48. struct folio *folio = page_folio(pte_page(pte));
  49. if (!test_bit(PG_dcache_clean, &folio->flags)) {
  50. sync_icache_aliases((unsigned long)folio_address(folio),
  51. (unsigned long)folio_address(folio) +
  52. folio_size(folio));
  53. set_bit(PG_dcache_clean, &folio->flags);
  54. }
  55. }
  56. EXPORT_SYMBOL_GPL(__sync_icache_dcache);
  57. /*
  58. * This function is called when a page has been modified by the kernel. Mark
  59. * it as dirty for later flushing when mapped in user space (if executable,
  60. * see __sync_icache_dcache).
  61. */
  62. void flush_dcache_folio(struct folio *folio)
  63. {
  64. if (test_bit(PG_dcache_clean, &folio->flags))
  65. clear_bit(PG_dcache_clean, &folio->flags);
  66. }
  67. EXPORT_SYMBOL(flush_dcache_folio);
  68. void flush_dcache_page(struct page *page)
  69. {
  70. flush_dcache_folio(page_folio(page));
  71. }
  72. EXPORT_SYMBOL(flush_dcache_page);
  73. /*
  74. * Additional functions defined in assembly.
  75. */
  76. EXPORT_SYMBOL(caches_clean_inval_pou);
  77. #ifdef CONFIG_ARCH_HAS_PMEM_API
  78. void arch_wb_cache_pmem(void *addr, size_t size)
  79. {
  80. /* Ensure order against any prior non-cacheable writes */
  81. dmb(osh);
  82. dcache_clean_pop((unsigned long)addr, (unsigned long)addr + size);
  83. }
  84. EXPORT_SYMBOL_GPL(arch_wb_cache_pmem);
  85. void arch_invalidate_pmem(void *addr, size_t size)
  86. {
  87. dcache_inval_poc((unsigned long)addr, (unsigned long)addr + size);
  88. }
  89. EXPORT_SYMBOL_GPL(arch_invalidate_pmem);
  90. #endif