aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/kernel/tcm.c
diff options
context:
space:
mode:
authorLinus Walleij <linus.walleij@stericsson.com>2010-07-12 16:53:28 -0400
committerRussell King <rmk+kernel@arm.linux.org.uk>2010-07-27 05:42:38 -0400
commit1dbd30e9890fd69e50b17edd70ca583546b0fe4e (patch)
tree5b5ab74c1792a81340478f7bbccd053e60a23a5e /arch/arm/kernel/tcm.c
parent07d2a5c721c6aa2bd69812a74c8b3b116abf3e56 (diff)
ARM: 6225/1: make TCM allocation static and common for all archs
This changes the TCM handling so that a fixed area is reserved at 0xfffe0000-0xfffeffff for TCM. This areas is used by XScale but XScale does not have TCM so the mechanisms are mutually exclusive. This change is needed to make TCM detection more dynamic while still being able to compile code into it, and is a must for the unified ARM goals: the current TCM allocation at different places in memory for each machine would be a nightmare if you want to compile a single image for more than one machine with TCM so it has to be nailed down in one place. Signed-off-by: Linus Walleij <linus.walleij@stericsson.com> Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Diffstat (limited to 'arch/arm/kernel/tcm.c')
-rw-r--r--arch/arm/kernel/tcm.c91
1 files changed, 48 insertions, 43 deletions
diff --git a/arch/arm/kernel/tcm.c b/arch/arm/kernel/tcm.c
index f2ead32c1aa0..26685c2f7a49 100644
--- a/arch/arm/kernel/tcm.c
+++ b/arch/arm/kernel/tcm.c
@@ -18,32 +18,30 @@
18#include <mach/memory.h> 18#include <mach/memory.h>
19#include "tcm.h" 19#include "tcm.h"
20 20
21/* Scream and warn about misuse */
22#if !defined(ITCM_OFFSET) || !defined(ITCM_END) || \
23 !defined(DTCM_OFFSET) || !defined(DTCM_END)
24#error "TCM support selected but offsets not defined!"
25#endif
26
27static struct gen_pool *tcm_pool; 21static struct gen_pool *tcm_pool;
28 22
29/* TCM section definitions from the linker */ 23/* TCM section definitions from the linker */
30extern char __itcm_start, __sitcm_text, __eitcm_text; 24extern char __itcm_start, __sitcm_text, __eitcm_text;
31extern char __dtcm_start, __sdtcm_data, __edtcm_data; 25extern char __dtcm_start, __sdtcm_data, __edtcm_data;
32 26
27/* These will be increased as we run */
28u32 dtcm_end = DTCM_OFFSET;
29u32 itcm_end = ITCM_OFFSET;
30
33/* 31/*
34 * TCM memory resources 32 * TCM memory resources
35 */ 33 */
36static struct resource dtcm_res = { 34static struct resource dtcm_res = {
37 .name = "DTCM RAM", 35 .name = "DTCM RAM",
38 .start = DTCM_OFFSET, 36 .start = DTCM_OFFSET,
39 .end = DTCM_END, 37 .end = DTCM_OFFSET,
40 .flags = IORESOURCE_MEM 38 .flags = IORESOURCE_MEM
41}; 39};
42 40
43static struct resource itcm_res = { 41static struct resource itcm_res = {
44 .name = "ITCM RAM", 42 .name = "ITCM RAM",
45 .start = ITCM_OFFSET, 43 .start = ITCM_OFFSET,
46 .end = ITCM_END, 44 .end = ITCM_OFFSET,
47 .flags = IORESOURCE_MEM 45 .flags = IORESOURCE_MEM
48}; 46};
49 47
@@ -51,7 +49,7 @@ static struct map_desc dtcm_iomap[] __initdata = {
51 { 49 {
52 .virtual = DTCM_OFFSET, 50 .virtual = DTCM_OFFSET,
53 .pfn = __phys_to_pfn(DTCM_OFFSET), 51 .pfn = __phys_to_pfn(DTCM_OFFSET),
54 .length = (DTCM_END - DTCM_OFFSET + 1), 52 .length = 0,
55 .type = MT_MEMORY_DTCM 53 .type = MT_MEMORY_DTCM
56 } 54 }
57}; 55};
@@ -60,7 +58,7 @@ static struct map_desc itcm_iomap[] __initdata = {
60 { 58 {
61 .virtual = ITCM_OFFSET, 59 .virtual = ITCM_OFFSET,
62 .pfn = __phys_to_pfn(ITCM_OFFSET), 60 .pfn = __phys_to_pfn(ITCM_OFFSET),
63 .length = (ITCM_END - ITCM_OFFSET + 1), 61 .length = 0,
64 .type = MT_MEMORY_ITCM 62 .type = MT_MEMORY_ITCM
65 } 63 }
66}; 64};
@@ -92,8 +90,8 @@ void tcm_free(void *addr, size_t len)
92} 90}
93EXPORT_SYMBOL(tcm_free); 91EXPORT_SYMBOL(tcm_free);
94 92
95static void __init setup_tcm_bank(u8 type, u8 bank, u8 banks, 93static int __init setup_tcm_bank(u8 type, u8 bank, u8 banks,
96 u32 offset, u32 expected_size) 94 u32 *offset)
97{ 95{
98 const int tcm_sizes[16] = { 0, -1, -1, 4, 8, 16, 32, 64, 128, 96 const int tcm_sizes[16] = { 0, -1, -1, 4, 8, 16, 32, 64, 128,
99 256, 512, 1024, -1, -1, -1, -1 }; 97 256, 512, 1024, -1, -1, -1, -1 };
@@ -120,8 +118,13 @@ static void __init setup_tcm_bank(u8 type, u8 bank, u8 banks,
120 118
121 tcm_size = tcm_sizes[(tcm_region >> 2) & 0x0f]; 119 tcm_size = tcm_sizes[(tcm_region >> 2) & 0x0f];
122 if (tcm_size < 0) { 120 if (tcm_size < 0) {
123 pr_err("CPU: %sTCM%d of unknown size!\n", 121 pr_err("CPU: %sTCM%d of unknown size\n",
124 type ? "I" : "D", bank); 122 type ? "I" : "D", bank);
123 return -EINVAL;
124 } else if (tcm_size > 32) {
125 pr_err("CPU: %sTCM%d larger than 32k found\n",
126 type ? "I" : "D", bank);
127 return -EINVAL;
125 } else { 128 } else {
126 pr_info("CPU: found %sTCM%d %dk @ %08x, %senabled\n", 129 pr_info("CPU: found %sTCM%d %dk @ %08x, %senabled\n",
127 type ? "I" : "D", 130 type ? "I" : "D",
@@ -131,17 +134,8 @@ static void __init setup_tcm_bank(u8 type, u8 bank, u8 banks,
131 (tcm_region & 1) ? "" : "not "); 134 (tcm_region & 1) ? "" : "not ");
132 } 135 }
133 136
134 if (tcm_size != (expected_size >> 10)) {
135 pr_crit("CPU: %sTCM%d was detected %dk but expected %dk!\n",
136 type ? "I" : "D",
137 bank,
138 tcm_size,
139 (expected_size >> 10));
140 /* Adjust to the expected size? what can we do... */
141 }
142
143 /* Force move the TCM bank to where we want it, enable */ 137 /* Force move the TCM bank to where we want it, enable */
144 tcm_region = offset | (tcm_region & 0x00000ffeU) | 1; 138 tcm_region = *offset | (tcm_region & 0x00000ffeU) | 1;
145 139
146 if (!type) 140 if (!type)
147 asm("mcr p15, 0, %0, c9, c1, 0" 141 asm("mcr p15, 0, %0, c9, c1, 0"
@@ -152,17 +146,17 @@ static void __init setup_tcm_bank(u8 type, u8 bank, u8 banks,
152 : /* No output operands */ 146 : /* No output operands */
153 : "r" (tcm_region)); 147 : "r" (tcm_region));
154 148
149 /* Increase offset */
150 *offset += (tcm_size << 10);
151
155 pr_info("CPU: moved %sTCM%d %dk to %08x, enabled\n", 152 pr_info("CPU: moved %sTCM%d %dk to %08x, enabled\n",
156 type ? "I" : "D", 153 type ? "I" : "D",
157 bank, 154 bank,
158 tcm_size, 155 tcm_size,
159 (tcm_region & 0xfffff000U)); 156 (tcm_region & 0xfffff000U));
157 return 0;
160} 158}
161 159
162/* We expect to find what is configured for the platform */
163#define DTCM_EXPECTED (DTCM_END - DTCM_OFFSET + 1)
164#define ITCM_EXPECTED (ITCM_END - ITCM_OFFSET + 1)
165
166/* 160/*
167 * This initializes the TCM memory 161 * This initializes the TCM memory
168 */ 162 */
@@ -170,40 +164,51 @@ void __init tcm_init(void)
170{ 164{
171 u32 tcm_status = read_cpuid_tcmstatus(); 165 u32 tcm_status = read_cpuid_tcmstatus();
172 u8 dtcm_banks = (tcm_status >> 16) & 0x03; 166 u8 dtcm_banks = (tcm_status >> 16) & 0x03;
173 u32 dtcm_banksize = DTCM_EXPECTED / dtcm_banks;
174 u8 itcm_banks = (tcm_status & 0x03); 167 u8 itcm_banks = (tcm_status & 0x03);
175 u32 itcm_banksize = ITCM_EXPECTED / itcm_banks;
176 char *start; 168 char *start;
177 char *end; 169 char *end;
178 char *ram; 170 char *ram;
171 int ret;
179 int i; 172 int i;
180 173
181 /* Setup DTCM if present */ 174 /* Setup DTCM if present */
182 for (i = 0; i < dtcm_banks; i++) { 175 if (dtcm_banks > 0) {
183 setup_tcm_bank(0, i, dtcm_banks, 176 for (i = 0; i < dtcm_banks; i++) {
184 DTCM_OFFSET + (i * dtcm_banksize), 177 ret = setup_tcm_bank(0, i, dtcm_banks, &dtcm_end);
185 dtcm_banksize); 178 if (ret)
179 return;
180 }
181 dtcm_res.end = dtcm_end - 1;
186 request_resource(&iomem_resource, &dtcm_res); 182 request_resource(&iomem_resource, &dtcm_res);
183 dtcm_iomap[0].length = dtcm_end - DTCM_OFFSET;
187 iotable_init(dtcm_iomap, 1); 184 iotable_init(dtcm_iomap, 1);
188 /* Copy data from RAM to DTCM */ 185 /* Copy data from RAM to DTCM */
189 start = &__sdtcm_data; 186 start = &__sdtcm_data;
190 end = &__edtcm_data; 187 end = &__edtcm_data;
191 ram = &__dtcm_start; 188 ram = &__dtcm_start;
189 /* This means you compiled more code than fits into DTCM */
190 BUG_ON((end - start) > (dtcm_end - DTCM_OFFSET));
192 memcpy(start, ram, (end-start)); 191 memcpy(start, ram, (end-start));
193 pr_debug("CPU DTCM: copied data from %p - %p\n", start, end); 192 pr_debug("CPU DTCM: copied data from %p - %p\n", start, end);
194 } 193 }
195 194
196 /* Setup ITCM if present */ 195 /* Setup ITCM if present */
197 for (i = 0; i < itcm_banks; i++) { 196 if (itcm_banks > 0) {
198 setup_tcm_bank(1, i, itcm_banks, 197 for (i = 0; i < itcm_banks; i++) {
199 ITCM_OFFSET + (i * itcm_banksize), 198 ret = setup_tcm_bank(1, i, itcm_banks, &itcm_end);
200 itcm_banksize); 199 if (ret)
200 return;
201 }
202 itcm_res.end = itcm_end - 1;
201 request_resource(&iomem_resource, &itcm_res); 203 request_resource(&iomem_resource, &itcm_res);
204 itcm_iomap[0].length = itcm_end - ITCM_OFFSET;
202 iotable_init(itcm_iomap, 1); 205 iotable_init(itcm_iomap, 1);
203 /* Copy code from RAM to ITCM */ 206 /* Copy code from RAM to ITCM */
204 start = &__sitcm_text; 207 start = &__sitcm_text;
205 end = &__eitcm_text; 208 end = &__eitcm_text;
206 ram = &__itcm_start; 209 ram = &__itcm_start;
210 /* This means you compiled more code than fits into ITCM */
211 BUG_ON((end - start) > (itcm_end - ITCM_OFFSET));
207 memcpy(start, ram, (end-start)); 212 memcpy(start, ram, (end-start));
208 pr_debug("CPU ITCM: copied code from %p - %p\n", start, end); 213 pr_debug("CPU ITCM: copied code from %p - %p\n", start, end);
209 } 214 }
@@ -232,9 +237,9 @@ static int __init setup_tcm_pool(void)
232 237
233 /* Add the rest of DTCM to the TCM pool */ 238 /* Add the rest of DTCM to the TCM pool */
234 if (tcm_status & (0x03 << 16)) { 239 if (tcm_status & (0x03 << 16)) {
235 if (dtcm_pool_start < DTCM_END) { 240 if (dtcm_pool_start < dtcm_end) {
236 ret = gen_pool_add(tcm_pool, dtcm_pool_start, 241 ret = gen_pool_add(tcm_pool, dtcm_pool_start,
237 DTCM_END - dtcm_pool_start + 1, -1); 242 dtcm_end - dtcm_pool_start, -1);
238 if (ret) { 243 if (ret) {
239 pr_err("CPU DTCM: could not add DTCM " \ 244 pr_err("CPU DTCM: could not add DTCM " \
240 "remainder to pool!\n"); 245 "remainder to pool!\n");
@@ -242,16 +247,16 @@ static int __init setup_tcm_pool(void)
242 } 247 }
243 pr_debug("CPU DTCM: Added %08x bytes @ %08x to " \ 248 pr_debug("CPU DTCM: Added %08x bytes @ %08x to " \
244 "the TCM memory pool\n", 249 "the TCM memory pool\n",
245 DTCM_END - dtcm_pool_start + 1, 250 dtcm_end - dtcm_pool_start,
246 dtcm_pool_start); 251 dtcm_pool_start);
247 } 252 }
248 } 253 }
249 254
250 /* Add the rest of ITCM to the TCM pool */ 255 /* Add the rest of ITCM to the TCM pool */
251 if (tcm_status & 0x03) { 256 if (tcm_status & 0x03) {
252 if (itcm_pool_start < ITCM_END) { 257 if (itcm_pool_start < itcm_end) {
253 ret = gen_pool_add(tcm_pool, itcm_pool_start, 258 ret = gen_pool_add(tcm_pool, itcm_pool_start,
254 ITCM_END - itcm_pool_start + 1, -1); 259 itcm_end - itcm_pool_start, -1);
255 if (ret) { 260 if (ret) {
256 pr_err("CPU ITCM: could not add ITCM " \ 261 pr_err("CPU ITCM: could not add ITCM " \
257 "remainder to pool!\n"); 262 "remainder to pool!\n");
@@ -259,7 +264,7 @@ static int __init setup_tcm_pool(void)
259 } 264 }
260 pr_debug("CPU ITCM: Added %08x bytes @ %08x to " \ 265 pr_debug("CPU ITCM: Added %08x bytes @ %08x to " \
261 "the TCM memory pool\n", 266 "the TCM memory pool\n",
262 ITCM_END - itcm_pool_start + 1, 267 itcm_end - itcm_pool_start,
263 itcm_pool_start); 268 itcm_pool_start);
264 } 269 }
265 } 270 }