Zephyr API Documentation  3.5.0
A Scalable Open Source RTOS
3.5.0
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Modules Pages
cache.h
Go to the documentation of this file.
1/*
2 * Copyright 2021 Intel Corporation
3 * SPDX-License-Identifier: Apache-2.0
4 */
5#ifndef ZEPHYR_INCLUDE_ARCH_XTENSA_CACHE_H_
6#define ZEPHYR_INCLUDE_ARCH_XTENSA_CACHE_H_
7
8#include <xtensa/config/core-isa.h>
9#include <zephyr/toolchain.h>
10#include <zephyr/sys/util.h>
11#include <zephyr/debug/sparse.h>
12#include <xtensa/hal.h>
13
14#ifdef __cplusplus
15extern "C" {
16#endif
17
18#define Z_DCACHE_MAX (XCHAL_DCACHE_SIZE / XCHAL_DCACHE_WAYS)
19
20#if XCHAL_DCACHE_SIZE
21BUILD_ASSERT(Z_IS_POW2(XCHAL_DCACHE_LINESIZE));
22BUILD_ASSERT(Z_IS_POW2(Z_DCACHE_MAX));
23#endif
24
25#if defined(CONFIG_DCACHE)
26static ALWAYS_INLINE int arch_dcache_flush_range(void *addr, size_t bytes)
27{
28#if XCHAL_DCACHE_SIZE
29 size_t step = XCHAL_DCACHE_LINESIZE;
30 size_t first = ROUND_DOWN(addr, step);
31 size_t last = ROUND_UP(((long)addr) + bytes, step);
32 size_t line;
33
34 for (line = first; bytes && line < last; line += step) {
35 __asm__ volatile("dhwb %0, 0" :: "r"(line));
36 }
37#endif
38 return 0;
39}
40
41static ALWAYS_INLINE int arch_dcache_flush_and_invd_range(void *addr, size_t bytes)
42{
43#if XCHAL_DCACHE_SIZE
44 size_t step = XCHAL_DCACHE_LINESIZE;
45 size_t first = ROUND_DOWN(addr, step);
46 size_t last = ROUND_UP(((long)addr) + bytes, step);
47 size_t line;
48
49 for (line = first; bytes && line < last; line += step) {
50 __asm__ volatile("dhwbi %0, 0" :: "r"(line));
51 }
52#endif
53 return 0;
54}
55
56static ALWAYS_INLINE int arch_dcache_invd_range(void *addr, size_t bytes)
57{
58#if XCHAL_DCACHE_SIZE
59 size_t step = XCHAL_DCACHE_LINESIZE;
60 size_t first = ROUND_DOWN(addr, step);
61 size_t last = ROUND_UP(((long)addr) + bytes, step);
62 size_t line;
63
64 for (line = first; bytes && line < last; line += step) {
65 __asm__ volatile("dhi %0, 0" :: "r"(line));
66 }
67#endif
68 return 0;
69}
70
71static ALWAYS_INLINE int arch_dcache_invd_all(void)
72{
73#if XCHAL_DCACHE_SIZE
74 size_t step = XCHAL_DCACHE_LINESIZE;
75 size_t line;
76
77 for (line = 0; line < XCHAL_DCACHE_SIZE; line += step) {
78 __asm__ volatile("dii %0, 0" :: "r"(line));
79 }
80#endif
81 return 0;
82}
83
84static ALWAYS_INLINE int arch_dcache_flush_all(void)
85{
86#if XCHAL_DCACHE_SIZE
87 size_t step = XCHAL_DCACHE_LINESIZE;
88 size_t line;
89
90 for (line = 0; line < XCHAL_DCACHE_SIZE; line += step) {
91 __asm__ volatile("diwb %0, 0" :: "r"(line));
92 }
93#endif
94 return 0;
95}
96
97static ALWAYS_INLINE int arch_dcache_flush_and_invd_all(void)
98{
99#if XCHAL_DCACHE_SIZE
100 size_t step = XCHAL_DCACHE_LINESIZE;
101 size_t line;
102
103 for (line = 0; line < XCHAL_DCACHE_SIZE; line += step) {
104 __asm__ volatile("diwbi %0, 0" :: "r"(line));
105 }
106#endif
107 return 0;
108}
109
110static ALWAYS_INLINE void arch_dcache_enable(void)
111{
112 /* nothing */
113}
114
115static ALWAYS_INLINE void arch_dcache_disable(void)
116{
117 /* nothing */
118}
119
120#endif /* CONFIG_DCACHE */
121
122#if defined(CONFIG_ICACHE)
123
124static size_t arch_icache_line_size_get(void)
125{
126 return -ENOTSUP;
127}
128
129static ALWAYS_INLINE int arch_icache_flush_all(void)
130{
131#if XCHAL_ICACHE_SIZE
132 xthal_icache_all_writeback();
133#endif
134 return 0;
135}
136
137static ALWAYS_INLINE int arch_icache_invd_all(void)
138{
139#if XCHAL_ICACHE_SIZE
140 xthal_icache_all_invalidate();
141#endif
142 return 0;
143}
144
145static ALWAYS_INLINE int arch_icache_flush_and_invd_all(void)
146{
147 return -ENOTSUP;
148}
149
150static ALWAYS_INLINE int arch_icache_flush_range(void *addr, size_t size)
151{
152 return -ENOTSUP;
153}
154
155static ALWAYS_INLINE int arch_icache_invd_range(void *addr, size_t size)
156{
157#if XCHAL_ICACHE_SIZE
158 xthal_icache_region_invalidate(addr, size);
159#endif
160 return 0;
161}
162
163static ALWAYS_INLINE int arch_icache_flush_and_invd_range(void *addr, size_t size)
164{
165 return -ENOTSUP;
166}
167
168static ALWAYS_INLINE void arch_icache_enable(void)
169{
170 /* nothing */
171}
172
173static ALWAYS_INLINE vid arch_icache_disable(void)
174{
175 /* nothing */
176}
177
178#endif /* CONFIG_ICACHE */
179
180
181
182#ifdef __cplusplus
183} /* extern "C" */
184#endif
185
186#endif /* ZEPHYR_INCLUDE_ARCH_XTENSA_CACHE_H_ */
#define ALWAYS_INLINE
Definition: common.h:129
#define ROUND_UP(x, align)
Value of x rounded up to the next multiple of align.
Definition: util.h:256
#define ROUND_DOWN(x, align)
Value of x rounded down to the previous multiple of align.
Definition: util.h:263
#define ENOTSUP
Unsupported value.
Definition: errno.h:115
Macros to abstract toolchain specific capabilities.
Misc utilities.