integrator_cp.c 8.8 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 *  linux/arch/arm/mach-integrator/integrator_cp.c
 *
 *  Copyright (C) 2003 Deep Blue Solutions Ltd
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License.
 */
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/list.h>
14
#include <linux/platform_device.h>
L
Linus Torvalds 已提交
15 16
#include <linux/dma-mapping.h>
#include <linux/string.h>
K
Kay Sievers 已提交
17
#include <linux/device.h>
18 19 20
#include <linux/amba/bus.h>
#include <linux/amba/kmi.h>
#include <linux/amba/clcd.h>
21
#include <linux/amba/mmci.h>
22
#include <linux/io.h>
23
#include <linux/irqchip/versatile-fpga.h>
24
#include <linux/gfp.h>
25
#include <linux/mtd/physmap.h>
26
#include <linux/platform_data/clk-integrator.h>
27 28
#include <linux/of_irq.h>
#include <linux/of_address.h>
29
#include <linux/of_platform.h>
30
#include <linux/sys_soc.h>
L
Linus Torvalds 已提交
31

32
#include <mach/hardware.h>
33
#include <mach/platform.h>
L
Linus Torvalds 已提交
34 35
#include <asm/setup.h>
#include <asm/mach-types.h>
36
#include <asm/hardware/arm_timer.h>
37
#include <asm/hardware/icst.h>
L
Linus Torvalds 已提交
38

39
#include <mach/lm.h>
L
Linus Torvalds 已提交
40 41 42 43 44 45

#include <asm/mach/arch.h>
#include <asm/mach/irq.h>
#include <asm/mach/map.h>
#include <asm/mach/time.h>

46
#include <asm/hardware/timer-sp.h>
47

48
#include <plat/clcd.h>
49
#include <plat/sched_clock.h>
50

51
#include "cm.h"
52 53
#include "common.h"

54 55 56
/* Base address to the CP controller */
static void __iomem *intcp_con_base;

L
Linus Torvalds 已提交
57 58 59 60 61 62 63 64 65 66
#define INTCP_PA_FLASH_BASE		0x24000000

#define INTCP_PA_CLCD_BASE		0xc0000000

#define INTCP_FLASHPROG			0x04
#define CINTEGRATOR_FLASHPROG_FLVPPEN	(1 << 0)
#define CINTEGRATOR_FLASHPROG_FLWREN	(1 << 1)

/*
 * Logical      Physical
67
 * f1000000	10000000	Core module registers
L
Linus Torvalds 已提交
68 69 70 71 72
 * f1300000	13000000	Counter/Timer
 * f1400000	14000000	Interrupt controller
 * f1600000	16000000	UART 0
 * f1700000	17000000	UART 1
 * f1a00000	1a000000	Debug LEDs
73 74
 * fc900000	c9000000	GPIO
 * fca00000	ca000000	SIC
L
Linus Torvalds 已提交
75 76
 */

77
static struct map_desc intcp_io_desc[] __initdata __maybe_unused = {
78
	{
79 80 81 82 83
		.virtual	= IO_ADDRESS(INTEGRATOR_HDR_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_HDR_BASE),
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
		.virtual	= IO_ADDRESS(INTEGRATOR_CT_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_CT_BASE),
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
		.virtual	= IO_ADDRESS(INTEGRATOR_IC_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_IC_BASE),
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
		.virtual	= IO_ADDRESS(INTEGRATOR_UART0_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_UART0_BASE),
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
		.virtual	= IO_ADDRESS(INTEGRATOR_DBG_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_DBG_BASE),
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
104 105
		.virtual	= IO_ADDRESS(INTEGRATOR_CP_GPIO_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_CP_GPIO_BASE),
106 107 108
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
109 110
		.virtual	= IO_ADDRESS(INTEGRATOR_CP_SIC_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_CP_SIC_BASE),
111 112 113
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}
L
Linus Torvalds 已提交
114 115 116 117 118 119 120 121 122 123
};

static void __init intcp_map_io(void)
{
	iotable_init(intcp_io_desc, ARRAY_SIZE(intcp_io_desc));
}

/*
 * Flash handling.
 */
124
static int intcp_flash_init(struct platform_device *dev)
L
Linus Torvalds 已提交
125 126 127
{
	u32 val;

128
	val = readl(intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
129
	val |= CINTEGRATOR_FLASHPROG_FLWREN;
130
	writel(val, intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
131 132 133 134

	return 0;
}

135
static void intcp_flash_exit(struct platform_device *dev)
L
Linus Torvalds 已提交
136 137 138
{
	u32 val;

139
	val = readl(intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
140
	val &= ~(CINTEGRATOR_FLASHPROG_FLVPPEN|CINTEGRATOR_FLASHPROG_FLWREN);
141
	writel(val, intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
142 143
}

144
static void intcp_flash_set_vpp(struct platform_device *pdev, int on)
L
Linus Torvalds 已提交
145 146 147
{
	u32 val;

148
	val = readl(intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
149 150 151 152
	if (on)
		val |= CINTEGRATOR_FLASHPROG_FLVPPEN;
	else
		val &= ~CINTEGRATOR_FLASHPROG_FLVPPEN;
153
	writel(val, intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
154 155
}

156
static struct physmap_flash_data intcp_flash_data = {
L
Linus Torvalds 已提交
157 158 159 160 161 162 163 164 165 166 167 168 169 170
	.width		= 4,
	.init		= intcp_flash_init,
	.exit		= intcp_flash_exit,
	.set_vpp	= intcp_flash_set_vpp,
};

/*
 * It seems that the card insertion interrupt remains active after
 * we've acknowledged it.  We therefore ignore the interrupt, and
 * rely on reading it from the SIC.  This also means that we must
 * clear the latched interrupt.
 */
static unsigned int mmc_status(struct device *dev)
{
171
	unsigned int status = readl(__io_address(0xca000000 + 4));
172
	writel(8, intcp_con_base + 8);
L
Linus Torvalds 已提交
173 174 175 176

	return status & 8;
}

177
static struct mmci_platform_data mmc_data = {
L
Linus Torvalds 已提交
178 179
	.ocr_mask	= MMC_VDD_32_33|MMC_VDD_33_34,
	.status		= mmc_status,
180 181
	.gpio_wp	= -1,
	.gpio_cd	= -1,
L
Linus Torvalds 已提交
182 183 184 185 186 187 188 189 190 191
};

/*
 * CLCD support
 */
/*
 * Ensure VGA is selected.
 */
static void cp_clcd_enable(struct clcd_fb *fb)
{
192
	struct fb_var_screeninfo *var = &fb->fb.var;
193 194
	u32 val = CM_CTRL_STATIC1 | CM_CTRL_STATIC2
			| CM_CTRL_LCDEN0 | CM_CTRL_LCDEN1;
195

196 197 198 199
	if (var->bits_per_pixel <= 8 ||
	    (var->bits_per_pixel == 16 && var->green.length == 5))
		/* Pseudocolor, RGB555, BGR555 */
		val |= CM_CTRL_LCDMUXSEL_VGA555_TFT555;
200
	else if (fb->fb.var.bits_per_pixel <= 16)
201 202
		/* truecolor RGB565 */
		val |= CM_CTRL_LCDMUXSEL_VGA565_TFT555;
203 204 205 206 207 208 209 210 211 212
	else
		val = 0; /* no idea for this, don't trust the docs */

	cm_control(CM_CTRL_LCDMUXSEL_MASK|
		   CM_CTRL_LCDEN0|
		   CM_CTRL_LCDEN1|
		   CM_CTRL_STATIC1|
		   CM_CTRL_STATIC2|
		   CM_CTRL_STATIC|
		   CM_CTRL_n24BITEN, val);
L
Linus Torvalds 已提交
213 214 215 216
}

static int cp_clcd_setup(struct clcd_fb *fb)
{
217 218 219
	fb->panel = versatile_clcd_get_panel("VGA");
	if (!fb->panel)
		return -EINVAL;
L
Linus Torvalds 已提交
220

221
	return versatile_clcd_setup_dma(fb, SZ_1M);
L
Linus Torvalds 已提交
222 223 224 225
}

static struct clcd_board clcd_data = {
	.name		= "Integrator/CP",
226
	.caps		= CLCD_CAP_5551 | CLCD_CAP_RGB565 | CLCD_CAP_888,
L
Linus Torvalds 已提交
227 228 229 230
	.check		= clcdfb_check,
	.decode		= clcdfb_decode,
	.enable		= cp_clcd_enable,
	.setup		= cp_clcd_setup,
231 232
	.mmap		= versatile_clcd_mmap_dma,
	.remove		= versatile_clcd_remove_dma,
L
Linus Torvalds 已提交
233 234
};

235 236
#define REFCOUNTER (__io_address(INTEGRATOR_HDR_BASE) + 0x28)

237 238
static void __init intcp_init_early(void)
{
239 240 241
#ifdef CONFIG_PLAT_VERSATILE_SCHED_CLOCK
	versatile_sched_clock_init(REFCOUNTER, 24000000);
#endif
242 243
}

244 245 246 247 248 249
static const struct of_device_id fpga_irq_of_match[] __initconst = {
	{ .compatible = "arm,versatile-fpga-irq", .data = fpga_irq_of_init, },
	{ /* Sentinel */ }
};

static void __init intcp_init_irq_of(void)
L
Linus Torvalds 已提交
250
{
251
	cm_init();
252 253 254
	of_irq_init(fpga_irq_of_match);
	integrator_clk_init(true);
}
L
Linus Torvalds 已提交
255

256 257 258 259 260 261 262 263
/*
 * For the Device Tree, add in the UART, MMC and CLCD specifics as AUXDATA
 * and enforce the bus names since these are used for clock lookups.
 */
static struct of_dev_auxdata intcp_auxdata_lookup[] __initdata = {
	OF_DEV_AUXDATA("arm,primecell", INTEGRATOR_RTC_BASE,
		"rtc", NULL),
	OF_DEV_AUXDATA("arm,primecell", INTEGRATOR_UART0_BASE,
264
		"uart0", NULL),
265
	OF_DEV_AUXDATA("arm,primecell", INTEGRATOR_UART1_BASE,
266
		"uart1", NULL),
267 268 269 270 271 272 273 274 275 276
	OF_DEV_AUXDATA("arm,primecell", KMI0_BASE,
		"kmi0", NULL),
	OF_DEV_AUXDATA("arm,primecell", KMI1_BASE,
		"kmi1", NULL),
	OF_DEV_AUXDATA("arm,primecell", INTEGRATOR_CP_MMC_BASE,
		"mmci", &mmc_data),
	OF_DEV_AUXDATA("arm,primecell", INTEGRATOR_CP_AACI_BASE,
		"aaci", &mmc_data),
	OF_DEV_AUXDATA("arm,primecell", INTCP_PA_CLCD_BASE,
		"clcd", &clcd_data),
277 278
	OF_DEV_AUXDATA("cfi-flash", INTCP_PA_FLASH_BASE,
		"physmap-flash", &intcp_flash_data),
279 280 281
	{ /* sentinel */ },
};

282 283 284 285 286
static const struct of_device_id intcp_syscon_match[] = {
	{ .compatible = "arm,integrator-cp-syscon"},
	{ },
};

287 288
static void __init intcp_init_of(void)
{
289 290 291 292 293 294 295 296 297 298 299 300
	struct device_node *root;
	struct device_node *cpcon;
	struct device *parent;
	struct soc_device *soc_dev;
	struct soc_device_attribute *soc_dev_attr;
	u32 intcp_sc_id;
	int err;

	/* Here we create an SoC device for the root node */
	root = of_find_node_by_path("/");
	if (!root)
		return;
301 302

	cpcon = of_find_matching_node(root, intcp_syscon_match);
303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327
	if (!cpcon)
		return;

	intcp_con_base = of_iomap(cpcon, 0);
	if (!intcp_con_base)
		return;

	intcp_sc_id = readl(intcp_con_base);

	soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL);
	if (!soc_dev_attr)
		return;

	err = of_property_read_string(root, "compatible",
				      &soc_dev_attr->soc_id);
	if (err)
		return;
	err = of_property_read_string(root, "model", &soc_dev_attr->machine);
	if (err)
		return;
	soc_dev_attr->family = "Integrator";
	soc_dev_attr->revision = kasprintf(GFP_KERNEL, "%c",
					   'A' + (intcp_sc_id & 0x0f));

	soc_dev = soc_device_register(soc_dev_attr);
328
	if (IS_ERR(soc_dev)) {
329 330 331 332 333 334
		kfree(soc_dev_attr->revision);
		kfree(soc_dev_attr);
		return;
	}

	parent = soc_device_to_device(soc_dev);
335
	integrator_init_sysfs(parent, intcp_sc_id);
336 337
	of_platform_populate(root, of_default_bus_match_table,
			intcp_auxdata_lookup, parent);
338 339
}

340 341 342 343
static const char * intcp_dt_board_compat[] = {
	"arm,integrator-cp",
	NULL,
};
L
Linus Torvalds 已提交
344

345 346 347 348 349 350
DT_MACHINE_START(INTEGRATOR_CP_DT, "ARM Integrator/CP (Device Tree)")
	.reserve	= integrator_reserve,
	.map_io		= intcp_map_io,
	.init_early	= intcp_init_early,
	.init_irq	= intcp_init_irq_of,
	.handle_irq	= fpga_handle_irq,
351
	.init_machine	= intcp_init_of,
352 353 354
	.restart	= integrator_restart,
	.dt_compat      = intcp_dt_board_compat,
MACHINE_END