1 // SPDX-License-Identifier: GPL-2.0-or-later
5 * Linux architectural port borrowing liberally from similar works of
6 * others. All original copyrights apply as per the original source
9 * Modifications for the OpenRISC architecture:
10 * Copyright (C) 2015 Jan Henrik Weinstock <jan.weinstock@rwth-aachen.de>
14 #include <asm/spr_defs.h>
15 #include <asm/cache.h>
16 #include <asm/cacheflush.h>
17 #include <asm/tlbflush.h>
19 static __always_inline
void cache_loop(struct page
*page
, const unsigned int reg
)
21 unsigned long paddr
= page_to_pfn(page
) << PAGE_SHIFT
;
22 unsigned long line
= paddr
& ~(L1_CACHE_BYTES
- 1);
24 while (line
< paddr
+ PAGE_SIZE
) {
26 line
+= L1_CACHE_BYTES
;
30 void local_dcache_page_flush(struct page
*page
)
32 cache_loop(page
, SPR_DCBFR
);
34 EXPORT_SYMBOL(local_dcache_page_flush
);
36 void local_icache_page_inv(struct page
*page
)
38 cache_loop(page
, SPR_ICBIR
);
40 EXPORT_SYMBOL(local_icache_page_inv
);
42 void update_cache(struct vm_area_struct
*vma
, unsigned long address
,
45 unsigned long pfn
= pte_val(*pte
) >> PAGE_SHIFT
;
46 struct folio
*folio
= page_folio(pfn_to_page(pfn
));
47 int dirty
= !test_and_set_bit(PG_dc_clean
, &folio
->flags
);
50 * Since icaches do not snoop for updated data on OpenRISC, we
51 * must write back and invalidate any dirty pages manually. We
52 * can skip data pages, since they will not end up in icaches.
54 if ((vma
->vm_flags
& VM_EXEC
) && dirty
) {
55 unsigned int nr
= folio_nr_pages(folio
);
58 sync_icache_dcache(folio_page(folio
, nr
));