sram.c 12.6 KB
Newer Older
1 2 3 4 5 6 7 8
/*
 * linux/arch/arm/plat-omap/sram.c
 *
 * OMAP SRAM detection and management
 *
 * Copyright (C) 2005 Nokia Corporation
 * Written by Tony Lindgren <tony@atomide.com>
 *
9 10 11
 * Copyright (C) 2009 Texas Instruments
 * Added OMAP4 support - Santosh Shilimkar <santosh.shilimkar@ti.com>
 *
12 13 14 15
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */
16
#undef DEBUG
17 18 19 20

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
21
#include <linux/io.h>
22

23
#include <asm/tlb.h>
24 25
#include <asm/cacheflush.h>

26 27
#include <asm/mach/map.h>

28 29 30
#include <plat/sram.h>
#include <plat/board.h>
#include <plat/cpu.h>
T
Tomi Valkeinen 已提交
31
#include <plat/vram.h>
32

33
#include <plat/control.h>
34 35 36 37 38 39 40

#if defined(CONFIG_ARCH_OMAP2) || defined(CONFIG_ARCH_OMAP3)
# include "../mach-omap2/prm.h"
# include "../mach-omap2/cm.h"
# include "../mach-omap2/sdrc.h"
#endif

41
#define OMAP1_SRAM_PA		0x20000000
42
#define OMAP1_SRAM_VA		VMALLOC_END
43
#define OMAP2_SRAM_PA		0x40200000
44
#define OMAP2_SRAM_PUB_PA	0x4020f800
45
#define OMAP2_SRAM_VA		0xfe400000
46
#define OMAP2_SRAM_PUB_VA	(OMAP2_SRAM_VA + 0x800)
47
#define OMAP3_SRAM_PA           0x40200000
48
#define OMAP3_SRAM_VA           0xfe400000
49
#define OMAP3_SRAM_PUB_PA       0x40208000
50
#define OMAP3_SRAM_PUB_VA       (OMAP3_SRAM_VA + 0x8000)
S
Santosh Shilimkar 已提交
51 52 53 54
#define OMAP4_SRAM_PA		0x40300000
#define OMAP4_SRAM_VA		0xfe400000
#define OMAP4_SRAM_PUB_PA	(OMAP4_SRAM_PA + 0x4000)
#define OMAP4_SRAM_PUB_VA	(OMAP4_SRAM_VA + 0x4000)
55

56
#if defined(CONFIG_ARCH_OMAP2) || defined(CONFIG_ARCH_OMAP3)
57 58
#define SRAM_BOOTLOADER_SZ	0x00
#else
59
#define SRAM_BOOTLOADER_SZ	0x80
60 61
#endif

62 63 64
#define OMAP24XX_VA_REQINFOPERM0	OMAP2_L3_IO_ADDRESS(0x68005048)
#define OMAP24XX_VA_READPERM0		OMAP2_L3_IO_ADDRESS(0x68005050)
#define OMAP24XX_VA_WRITEPERM0		OMAP2_L3_IO_ADDRESS(0x68005058)
65

66 67 68 69 70 71
#define OMAP34XX_VA_REQINFOPERM0	OMAP2_L3_IO_ADDRESS(0x68012848)
#define OMAP34XX_VA_READPERM0		OMAP2_L3_IO_ADDRESS(0x68012850)
#define OMAP34XX_VA_WRITEPERM0		OMAP2_L3_IO_ADDRESS(0x68012858)
#define OMAP34XX_VA_ADDR_MATCH2		OMAP2_L3_IO_ADDRESS(0x68012880)
#define OMAP34XX_VA_SMS_RG_ATT0		OMAP2_L3_IO_ADDRESS(0x6C000048)
#define OMAP34XX_VA_CONTROL_STAT	OMAP2_L4_IO_ADDRESS(0x480022F0)
72

73 74 75
#define GP_DEVICE		0x300

#define ROUND_DOWN(value,boundary)	((value) & (~((boundary)-1)))
76

77
static unsigned long omap_sram_start;
78 79 80 81
static unsigned long omap_sram_base;
static unsigned long omap_sram_size;
static unsigned long omap_sram_ceil;

82 83 84 85 86
extern unsigned long omapfb_reserve_sram(unsigned long sram_pstart,
					 unsigned long sram_vstart,
					 unsigned long sram_size,
					 unsigned long pstart_avail,
					 unsigned long size_avail);
87

88 89
/*
 * Depending on the target RAMFS firewall setup, the public usable amount of
S
Simon Arlott 已提交
90 91
 * SRAM varies.  The default accessible size for all device types is 2k. A GP
 * device allows ARM11 but not other initiators for full size. This
92 93 94 95
 * functionality seems ok until some nice security API happens.
 */
static int is_sram_locked(void)
{
96
	if (OMAP2_DEVICE_TYPE_GP == omap_type()) {
S
Simon Arlott 已提交
97
		/* RAMFW: R/W access to all initiators for all qualifier sets */
98
		if (cpu_is_omap242x()) {
99 100 101 102 103 104 105 106 107 108
			__raw_writel(0xFF, OMAP24XX_VA_REQINFOPERM0); /* all q-vects */
			__raw_writel(0xCFDE, OMAP24XX_VA_READPERM0);  /* all i-read */
			__raw_writel(0xCFDE, OMAP24XX_VA_WRITEPERM0); /* all i-write */
		}
		if (cpu_is_omap34xx()) {
			__raw_writel(0xFFFF, OMAP34XX_VA_REQINFOPERM0); /* all q-vects */
			__raw_writel(0xFFFF, OMAP34XX_VA_READPERM0);  /* all i-read */
			__raw_writel(0xFFFF, OMAP34XX_VA_WRITEPERM0); /* all i-write */
			__raw_writel(0x0, OMAP34XX_VA_ADDR_MATCH2);
			__raw_writel(0xFFFFFFFF, OMAP34XX_VA_SMS_RG_ATT0);
109 110 111 112 113 114
		}
		return 0;
	} else
		return 1; /* assume locked with no PPA or security driver */
}

115
/*
116
 * The amount of SRAM depends on the core type.
117 118 119 120 121 122
 * Note that we cannot try to test for SRAM here because writes
 * to secure SRAM will hang the system. Also the SRAM is not
 * yet mapped at this point.
 */
void __init omap_detect_sram(void)
{
123
	unsigned long reserved;
124

125
	if (cpu_class_is_omap2()) {
126
		if (is_sram_locked()) {
127 128 129
			if (cpu_is_omap34xx()) {
				omap_sram_base = OMAP3_SRAM_PUB_VA;
				omap_sram_start = OMAP3_SRAM_PUB_PA;
130 131 132 133 134 135
				if ((omap_type() == OMAP2_DEVICE_TYPE_EMU) ||
				    (omap_type() == OMAP2_DEVICE_TYPE_SEC)) {
					omap_sram_size = 0x7000; /* 28K */
				} else {
					omap_sram_size = 0x8000; /* 32K */
				}
S
Santosh Shilimkar 已提交
136 137 138 139
			} else if (cpu_is_omap44xx()) {
				omap_sram_base = OMAP4_SRAM_PUB_VA;
				omap_sram_start = OMAP4_SRAM_PUB_PA;
				omap_sram_size = 0xa000; /* 40K */
140 141 142 143 144
			} else {
				omap_sram_base = OMAP2_SRAM_PUB_VA;
				omap_sram_start = OMAP2_SRAM_PUB_PA;
				omap_sram_size = 0x800; /* 2K */
			}
145
		} else {
146 147 148
			if (cpu_is_omap34xx()) {
				omap_sram_base = OMAP3_SRAM_VA;
				omap_sram_start = OMAP3_SRAM_PA;
149
				omap_sram_size = 0x10000; /* 64K */
150 151 152
			} else if (cpu_is_omap44xx()) {
				omap_sram_base = OMAP4_SRAM_VA;
				omap_sram_start = OMAP4_SRAM_PA;
S
Santosh Shilimkar 已提交
153
				omap_sram_size = 0xe000; /* 56K */
154 155 156 157 158 159 160 161
			} else {
				omap_sram_base = OMAP2_SRAM_VA;
				omap_sram_start = OMAP2_SRAM_PA;
				if (cpu_is_omap242x())
					omap_sram_size = 0xa0000; /* 640K */
				else if (cpu_is_omap243x())
					omap_sram_size = 0x10000; /* 64K */
			}
162 163
		}
	} else {
164
		omap_sram_base = OMAP1_SRAM_VA;
165
		omap_sram_start = OMAP1_SRAM_PA;
166

167
		if (cpu_is_omap7xx())
168 169 170 171 172 173 174 175 176 177 178 179
			omap_sram_size = 0x32000;	/* 200K */
		else if (cpu_is_omap15xx())
			omap_sram_size = 0x30000;	/* 192K */
		else if (cpu_is_omap1610() || cpu_is_omap1621() ||
		     cpu_is_omap1710())
			omap_sram_size = 0x4000;	/* 16K */
		else if (cpu_is_omap1611())
			omap_sram_size = 0x3e800;	/* 250K */
		else {
			printk(KERN_ERR "Could not detect SRAM size\n");
			omap_sram_size = 0x4000;
		}
180
	}
181 182 183 184 185
	reserved = omapfb_reserve_sram(omap_sram_start, omap_sram_base,
				       omap_sram_size,
				       omap_sram_start + SRAM_BOOTLOADER_SZ,
				       omap_sram_size - SRAM_BOOTLOADER_SZ);
	omap_sram_size -= reserved;
T
Tomi Valkeinen 已提交
186 187 188 189 190 191 192

	reserved = omap_vram_reserve_sram(omap_sram_start, omap_sram_base,
			omap_sram_size,
			omap_sram_start + SRAM_BOOTLOADER_SZ,
			omap_sram_size - SRAM_BOOTLOADER_SZ);
	omap_sram_size -= reserved;

193 194 195 196
	omap_sram_ceil = omap_sram_base + omap_sram_size;
}

static struct map_desc omap_sram_io_desc[] __initdata = {
197
	{	/* .length gets filled in at runtime */
198 199
		.virtual	= OMAP1_SRAM_VA,
		.pfn		= __phys_to_pfn(OMAP1_SRAM_PA),
200
		.type		= MT_MEMORY
201
	}
202 203 204
};

/*
205
 * Note that we cannot use ioremap for SRAM, as clock init needs SRAM early.
206 207 208
 */
void __init omap_map_sram(void)
{
209 210
	unsigned long base;

211 212 213
	if (omap_sram_size == 0)
		return;

214 215
	if (cpu_is_omap24xx()) {
		omap_sram_io_desc[0].virtual = OMAP2_SRAM_VA;
216

217
		base = OMAP2_SRAM_PA;
218 219
		base = ROUND_DOWN(base, PAGE_SIZE);
		omap_sram_io_desc[0].pfn = __phys_to_pfn(base);
220 221
	}

222 223 224 225 226
	if (cpu_is_omap34xx()) {
		omap_sram_io_desc[0].virtual = OMAP3_SRAM_VA;
		base = OMAP3_SRAM_PA;
		base = ROUND_DOWN(base, PAGE_SIZE);
		omap_sram_io_desc[0].pfn = __phys_to_pfn(base);
227 228 229 230 231 232 233 234 235

		/*
		 * SRAM must be marked as non-cached on OMAP3 since the
		 * CORE DPLL M2 divider change code (in SRAM) runs with the
		 * SDRAM controller disabled, and if it is marked cached,
		 * the ARM may attempt to write cache lines back to SDRAM
		 * which will cause the system to hang.
		 */
		omap_sram_io_desc[0].type = MT_MEMORY_NONCACHED;
236 237
	}

238 239 240 241 242 243
	if (cpu_is_omap44xx()) {
		omap_sram_io_desc[0].virtual = OMAP4_SRAM_VA;
		base = OMAP4_SRAM_PA;
		base = ROUND_DOWN(base, PAGE_SIZE);
		omap_sram_io_desc[0].pfn = __phys_to_pfn(base);
	}
244
	omap_sram_io_desc[0].length = 1024 * 1024;	/* Use section desc */
245 246
	iotable_init(omap_sram_io_desc, ARRAY_SIZE(omap_sram_io_desc));

247
	printk(KERN_INFO "SRAM: Mapped pa 0x%08lx to va 0x%08lx size: 0x%lx\n",
248 249
	__pfn_to_phys(omap_sram_io_desc[0].pfn),
	omap_sram_io_desc[0].virtual,
250 251
	       omap_sram_io_desc[0].length);

252 253 254 255 256 257 258 259
	/*
	 * Normally devicemaps_init() would flush caches and tlb after
	 * mdesc->map_io(), but since we're called from map_io(), we
	 * must do it here.
	 */
	local_flush_tlb_all();
	flush_cache_all();

260 261 262 263 264 265 266 267 268 269 270 271 272 273
	/*
	 * Looks like we need to preserve some bootloader code at the
	 * beginning of SRAM for jumping to flash for reboot to work...
	 */
	memset((void *)omap_sram_base + SRAM_BOOTLOADER_SZ, 0,
	       omap_sram_size - SRAM_BOOTLOADER_SZ);
}

void * omap_sram_push(void * start, unsigned long size)
{
	if (size > (omap_sram_ceil - (omap_sram_base + SRAM_BOOTLOADER_SZ))) {
		printk(KERN_ERR "Not enough space in SRAM\n");
		return NULL;
	}
274

275
	omap_sram_ceil -= size;
276
	omap_sram_ceil = ROUND_DOWN(omap_sram_ceil, sizeof(void *));
277
	memcpy((void *)omap_sram_ceil, start, size);
278 279
	flush_icache_range((unsigned long)omap_sram_ceil,
		(unsigned long)(omap_sram_ceil + size));
280 281 282 283

	return (void *)omap_sram_ceil;
}

284 285 286 287 288 289
#ifdef CONFIG_ARCH_OMAP1

static void (*_omap_sram_reprogram_clock)(u32 dpllctl, u32 ckctl);

void omap_sram_reprogram_clock(u32 dpllctl, u32 ckctl)
{
290
	BUG_ON(!_omap_sram_reprogram_clock);
R
Russell King 已提交
291
	_omap_sram_reprogram_clock(dpllctl, ckctl);
292 293 294
}

int __init omap1_sram_init(void)
295
{
296 297 298
	_omap_sram_reprogram_clock =
			omap_sram_push(omap1_sram_reprogram_clock,
					omap1_sram_reprogram_clock_sz);
299 300 301 302 303 304 305 306

	return 0;
}

#else
#define omap1_sram_init()	do {} while (0)
#endif

307
#if defined(CONFIG_ARCH_OMAP2)
308 309 310 311 312 313 314

static void (*_omap2_sram_ddr_init)(u32 *slow_dll_ctrl, u32 fast_dll_ctrl,
			      u32 base_cs, u32 force_unlock);

void omap2_sram_ddr_init(u32 *slow_dll_ctrl, u32 fast_dll_ctrl,
		   u32 base_cs, u32 force_unlock)
{
315
	BUG_ON(!_omap2_sram_ddr_init);
R
Russell King 已提交
316 317
	_omap2_sram_ddr_init(slow_dll_ctrl, fast_dll_ctrl,
			     base_cs, force_unlock);
318 319 320 321 322 323 324
}

static void (*_omap2_sram_reprogram_sdrc)(u32 perf_level, u32 dll_val,
					  u32 mem_type);

void omap2_sram_reprogram_sdrc(u32 perf_level, u32 dll_val, u32 mem_type)
{
325
	BUG_ON(!_omap2_sram_reprogram_sdrc);
R
Russell King 已提交
326
	_omap2_sram_reprogram_sdrc(perf_level, dll_val, mem_type);
327 328 329 330 331 332
}

static u32 (*_omap2_set_prcm)(u32 dpll_ctrl_val, u32 sdrc_rfr_val, int bypass);

u32 omap2_set_prcm(u32 dpll_ctrl_val, u32 sdrc_rfr_val, int bypass)
{
333
	BUG_ON(!_omap2_set_prcm);
334 335
	return _omap2_set_prcm(dpll_ctrl_val, sdrc_rfr_val, bypass);
}
336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381
#endif

#ifdef CONFIG_ARCH_OMAP2420
int __init omap242x_sram_init(void)
{
	_omap2_sram_ddr_init = omap_sram_push(omap242x_sram_ddr_init,
					omap242x_sram_ddr_init_sz);

	_omap2_sram_reprogram_sdrc = omap_sram_push(omap242x_sram_reprogram_sdrc,
					    omap242x_sram_reprogram_sdrc_sz);

	_omap2_set_prcm = omap_sram_push(omap242x_sram_set_prcm,
					 omap242x_sram_set_prcm_sz);

	return 0;
}
#else
static inline int omap242x_sram_init(void)
{
	return 0;
}
#endif

#ifdef CONFIG_ARCH_OMAP2430
int __init omap243x_sram_init(void)
{
	_omap2_sram_ddr_init = omap_sram_push(omap243x_sram_ddr_init,
					omap243x_sram_ddr_init_sz);

	_omap2_sram_reprogram_sdrc = omap_sram_push(omap243x_sram_reprogram_sdrc,
					    omap243x_sram_reprogram_sdrc_sz);

	_omap2_set_prcm = omap_sram_push(omap243x_sram_set_prcm,
					 omap243x_sram_set_prcm_sz);

	return 0;
}
#else
static inline int omap243x_sram_init(void)
{
	return 0;
}
#endif

#ifdef CONFIG_ARCH_OMAP3

382 383 384 385 386 387 388 389 390 391 392 393
static u32 (*_omap3_sram_configure_core_dpll)(
			u32 m2, u32 unlock_dll, u32 f, u32 inc,
			u32 sdrc_rfr_ctrl_0, u32 sdrc_actim_ctrl_a_0,
			u32 sdrc_actim_ctrl_b_0, u32 sdrc_mr_0,
			u32 sdrc_rfr_ctrl_1, u32 sdrc_actim_ctrl_a_1,
			u32 sdrc_actim_ctrl_b_1, u32 sdrc_mr_1);

u32 omap3_configure_core_dpll(u32 m2, u32 unlock_dll, u32 f, u32 inc,
			u32 sdrc_rfr_ctrl_0, u32 sdrc_actim_ctrl_a_0,
			u32 sdrc_actim_ctrl_b_0, u32 sdrc_mr_0,
			u32 sdrc_rfr_ctrl_1, u32 sdrc_actim_ctrl_a_1,
			u32 sdrc_actim_ctrl_b_1, u32 sdrc_mr_1)
394
{
395
	BUG_ON(!_omap3_sram_configure_core_dpll);
396 397 398 399 400 401
	return _omap3_sram_configure_core_dpll(
			m2, unlock_dll, f, inc,
			sdrc_rfr_ctrl_0, sdrc_actim_ctrl_a_0,
			sdrc_actim_ctrl_b_0, sdrc_mr_0,
			sdrc_rfr_ctrl_1, sdrc_actim_ctrl_a_1,
			sdrc_actim_ctrl_b_1, sdrc_mr_1);
402
}
403

404 405
#ifdef CONFIG_PM
void omap3_sram_restore_context(void)
406
{
407
	omap_sram_ceil = omap_sram_base + omap_sram_size;
408

409 410 411
	_omap3_sram_configure_core_dpll =
		omap_sram_push(omap3_sram_configure_core_dpll,
			       omap3_sram_configure_core_dpll_sz);
412
	omap_push_sram_idle();
413
}
414
#endif /* CONFIG_PM */
415 416 417

int __init omap34xx_sram_init(void)
{
418 419 420
	_omap3_sram_configure_core_dpll =
		omap_sram_push(omap3_sram_configure_core_dpll,
			       omap3_sram_configure_core_dpll_sz);
421
	omap_push_sram_idle();
422 423 424
	return 0;
}
#else
425 426 427 428
static inline int omap34xx_sram_init(void)
{
	return 0;
}
429 430
#endif

431 432 433 434 435 436 437 438 439 440 441 442 443 444
#ifdef CONFIG_ARCH_OMAP4
int __init omap44xx_sram_init(void)
{
	printk(KERN_ERR "FIXME: %s not implemented\n", __func__);

	return -ENODEV;
}
#else
static inline int omap44xx_sram_init(void)
{
	return 0;
}
#endif

445 446 447 448 449
int __init omap_sram_init(void)
{
	omap_detect_sram();
	omap_map_sram();

450
	if (!(cpu_class_is_omap2()))
451
		omap1_sram_init();
452 453 454 455 456 457
	else if (cpu_is_omap242x())
		omap242x_sram_init();
	else if (cpu_is_omap2430())
		omap243x_sram_init();
	else if (cpu_is_omap34xx())
		omap34xx_sram_init();
458
	else if (cpu_is_omap44xx())
459
		omap44xx_sram_init();
460 461

	return 0;
462
}