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_OMAP24XX) || defined(CONFIG_ARCH_OMAP34XX)
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 96 97
 * functionality seems ok until some nice security API happens.
 */
static int is_sram_locked(void)
{
	int type = 0;

98 99 100 101
	if (cpu_is_omap44xx())
		/* Not yet supported */
		return 0;

102
	if (cpu_is_omap242x())
103
		type = omap_rev() & OMAP2_DEVICETYPE_MASK;
104 105

	if (type == GP_DEVICE) {
S
Simon Arlott 已提交
106
		/* RAMFW: R/W access to all initiators for all qualifier sets */
107
		if (cpu_is_omap242x()) {
108 109 110 111 112 113 114 115 116 117
			__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);
118 119 120 121 122 123
		}
		return 0;
	} else
		return 1; /* assume locked with no PPA or security driver */
}

124
/*
125
 * The amount of SRAM depends on the core type.
126 127 128 129 130 131
 * 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)
{
132
	unsigned long reserved;
133

134
	if (cpu_class_is_omap2()) {
135
		if (is_sram_locked()) {
136 137 138
			if (cpu_is_omap34xx()) {
				omap_sram_base = OMAP3_SRAM_PUB_VA;
				omap_sram_start = OMAP3_SRAM_PUB_PA;
139 140 141 142 143 144
				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 已提交
145 146 147 148
			} else if (cpu_is_omap44xx()) {
				omap_sram_base = OMAP4_SRAM_PUB_VA;
				omap_sram_start = OMAP4_SRAM_PUB_PA;
				omap_sram_size = 0xa000; /* 40K */
149 150 151 152 153
			} else {
				omap_sram_base = OMAP2_SRAM_PUB_VA;
				omap_sram_start = OMAP2_SRAM_PUB_PA;
				omap_sram_size = 0x800; /* 2K */
			}
154
		} else {
155 156 157
			if (cpu_is_omap34xx()) {
				omap_sram_base = OMAP3_SRAM_VA;
				omap_sram_start = OMAP3_SRAM_PA;
158
				omap_sram_size = 0x10000; /* 64K */
159 160 161
			} else if (cpu_is_omap44xx()) {
				omap_sram_base = OMAP4_SRAM_VA;
				omap_sram_start = OMAP4_SRAM_PA;
S
Santosh Shilimkar 已提交
162
				omap_sram_size = 0xe000; /* 56K */
163 164 165 166 167 168 169 170
			} 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 */
			}
171 172
		}
	} else {
173
		omap_sram_base = OMAP1_SRAM_VA;
174
		omap_sram_start = OMAP1_SRAM_PA;
175

176
		if (cpu_is_omap7xx())
177 178 179 180 181 182 183 184 185 186 187 188
			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;
		}
189
	}
190 191 192 193 194
	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 已提交
195 196 197 198 199 200 201

	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;

202 203 204 205
	omap_sram_ceil = omap_sram_base + omap_sram_size;
}

static struct map_desc omap_sram_io_desc[] __initdata = {
206
	{	/* .length gets filled in at runtime */
207 208
		.virtual	= OMAP1_SRAM_VA,
		.pfn		= __phys_to_pfn(OMAP1_SRAM_PA),
209
		.type		= MT_MEMORY
210
	}
211 212 213
};

/*
214
 * Note that we cannot use ioremap for SRAM, as clock init needs SRAM early.
215 216 217
 */
void __init omap_map_sram(void)
{
218 219
	unsigned long base;

220 221 222
	if (omap_sram_size == 0)
		return;

223 224
	if (cpu_is_omap24xx()) {
		omap_sram_io_desc[0].virtual = OMAP2_SRAM_VA;
225

226
		base = OMAP2_SRAM_PA;
227 228
		base = ROUND_DOWN(base, PAGE_SIZE);
		omap_sram_io_desc[0].pfn = __phys_to_pfn(base);
229 230
	}

231 232 233 234 235
	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);
236 237 238 239 240 241 242 243 244

		/*
		 * 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;
245 246
	}

247 248 249 250 251 252
	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);
	}
253
	omap_sram_io_desc[0].length = 1024 * 1024;	/* Use section desc */
254 255
	iotable_init(omap_sram_io_desc, ARRAY_SIZE(omap_sram_io_desc));

256
	printk(KERN_INFO "SRAM: Mapped pa 0x%08lx to va 0x%08lx size: 0x%lx\n",
257 258
	__pfn_to_phys(omap_sram_io_desc[0].pfn),
	omap_sram_io_desc[0].virtual,
259 260
	       omap_sram_io_desc[0].length);

261 262 263 264 265 266 267 268
	/*
	 * 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();

269 270 271 272 273 274 275 276 277 278 279 280 281 282
	/*
	 * 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;
	}
283

284
	omap_sram_ceil -= size;
285
	omap_sram_ceil = ROUND_DOWN(omap_sram_ceil, sizeof(void *));
286
	memcpy((void *)omap_sram_ceil, start, size);
287 288
	flush_icache_range((unsigned long)omap_sram_ceil,
		(unsigned long)(omap_sram_ceil + size));
289 290 291 292

	return (void *)omap_sram_ceil;
}

293 294 295 296 297 298
#ifdef CONFIG_ARCH_OMAP1

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

void omap_sram_reprogram_clock(u32 dpllctl, u32 ckctl)
{
299
	BUG_ON(!_omap_sram_reprogram_clock);
R
Russell King 已提交
300
	_omap_sram_reprogram_clock(dpllctl, ckctl);
301 302 303
}

int __init omap1_sram_init(void)
304
{
305 306 307
	_omap_sram_reprogram_clock =
			omap_sram_push(omap1_sram_reprogram_clock,
					omap1_sram_reprogram_clock_sz);
308 309 310 311 312 313 314 315

	return 0;
}

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

316
#if defined(CONFIG_ARCH_OMAP2)
317 318 319 320 321 322 323

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)
{
324
	BUG_ON(!_omap2_sram_ddr_init);
R
Russell King 已提交
325 326
	_omap2_sram_ddr_init(slow_dll_ctrl, fast_dll_ctrl,
			     base_cs, force_unlock);
327 328 329 330 331 332 333
}

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)
{
334
	BUG_ON(!_omap2_sram_reprogram_sdrc);
R
Russell King 已提交
335
	_omap2_sram_reprogram_sdrc(perf_level, dll_val, mem_type);
336 337 338 339 340 341
}

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)
{
342
	BUG_ON(!_omap2_set_prcm);
343 344
	return _omap2_set_prcm(dpll_ctrl_val, sdrc_rfr_val, bypass);
}
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 382 383 384 385 386 387 388 389 390
#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

391 392 393 394 395 396 397 398 399 400 401 402
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)
403
{
404
	BUG_ON(!_omap3_sram_configure_core_dpll);
405 406 407 408 409 410
	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);
411
}
412

413 414
#ifdef CONFIG_PM
void omap3_sram_restore_context(void)
415
{
416
	omap_sram_ceil = omap_sram_base + omap_sram_size;
417

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
#endif /* CONFIG_PM */
424 425 426

int __init omap34xx_sram_init(void)
{
427 428 429
	_omap3_sram_configure_core_dpll =
		omap_sram_push(omap3_sram_configure_core_dpll,
			       omap3_sram_configure_core_dpll_sz);
430
	omap_push_sram_idle();
431 432 433
	return 0;
}
#else
434 435 436 437
static inline int omap34xx_sram_init(void)
{
	return 0;
}
438 439 440 441 442 443 444
#endif

int __init omap_sram_init(void)
{
	omap_detect_sram();
	omap_map_sram();

445
	if (!(cpu_class_is_omap2()))
446
		omap1_sram_init();
447 448 449 450 451 452
	else if (cpu_is_omap242x())
		omap242x_sram_init();
	else if (cpu_is_omap2430())
		omap243x_sram_init();
	else if (cpu_is_omap34xx())
		omap34xx_sram_init();
453 454
	else if (cpu_is_omap44xx())
		omap34xx_sram_init(); /* FIXME: */
455 456

	return 0;
457
}