integrator_cp.c 9.0 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 67 68 69 70 71 72 73 74
#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
 * f1000000	10000000	Core module registers
 * f1100000	11000000	System controller registers
 * f1200000	12000000	EBI registers
 * f1300000	13000000	Counter/Timer
 * f1400000	14000000	Interrupt controller
 * f1600000	16000000	UART 0
 * f1700000	17000000	UART 1
 * f1a00000	1a000000	Debug LEDs
75 76 77
 * fc900000	c9000000	GPIO
 * fca00000	ca000000	SIC
 * fcb00000	cb000000	CP system control
L
Linus Torvalds 已提交
78 79
 */

80
static struct map_desc intcp_io_desc[] __initdata __maybe_unused = {
81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
	{
		.virtual	= IO_ADDRESS(INTEGRATOR_HDR_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_HDR_BASE),
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
		.virtual	= IO_ADDRESS(INTEGRATOR_EBI_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_EBI_BASE),
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
		.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
	}, {
112 113
		.virtual	= IO_ADDRESS(INTEGRATOR_CP_GPIO_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_CP_GPIO_BASE),
114 115 116
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}, {
117 118
		.virtual	= IO_ADDRESS(INTEGRATOR_CP_SIC_BASE),
		.pfn		= __phys_to_pfn(INTEGRATOR_CP_SIC_BASE),
119 120 121
		.length		= SZ_4K,
		.type		= MT_DEVICE
	}
L
Linus Torvalds 已提交
122 123 124 125 126 127 128 129 130 131
};

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

/*
 * Flash handling.
 */
132
static int intcp_flash_init(struct platform_device *dev)
L
Linus Torvalds 已提交
133 134 135
{
	u32 val;

136
	val = readl(intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
137
	val |= CINTEGRATOR_FLASHPROG_FLWREN;
138
	writel(val, intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
139 140 141 142

	return 0;
}

143
static void intcp_flash_exit(struct platform_device *dev)
L
Linus Torvalds 已提交
144 145 146
{
	u32 val;

147
	val = readl(intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
148
	val &= ~(CINTEGRATOR_FLASHPROG_FLVPPEN|CINTEGRATOR_FLASHPROG_FLWREN);
149
	writel(val, intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
150 151
}

152
static void intcp_flash_set_vpp(struct platform_device *pdev, int on)
L
Linus Torvalds 已提交
153 154 155
{
	u32 val;

156
	val = readl(intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
157 158 159 160
	if (on)
		val |= CINTEGRATOR_FLASHPROG_FLVPPEN;
	else
		val &= ~CINTEGRATOR_FLASHPROG_FLVPPEN;
161
	writel(val, intcp_con_base + INTCP_FLASHPROG);
L
Linus Torvalds 已提交
162 163
}

164
static struct physmap_flash_data intcp_flash_data = {
L
Linus Torvalds 已提交
165 166 167 168 169 170 171 172 173 174 175 176 177 178
	.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)
{
179
	unsigned int status = readl(__io_address(0xca000000 + 4));
180
	writel(8, intcp_con_base + 8);
L
Linus Torvalds 已提交
181 182 183 184

	return status & 8;
}

185
static struct mmci_platform_data mmc_data = {
L
Linus Torvalds 已提交
186 187
	.ocr_mask	= MMC_VDD_32_33|MMC_VDD_33_34,
	.status		= mmc_status,
188 189
	.gpio_wp	= -1,
	.gpio_cd	= -1,
L
Linus Torvalds 已提交
190 191 192 193 194 195 196 197 198 199
};

/*
 * CLCD support
 */
/*
 * Ensure VGA is selected.
 */
static void cp_clcd_enable(struct clcd_fb *fb)
{
200 201
	struct fb_var_screeninfo *var = &fb->fb.var;
	u32 val = CM_CTRL_STATIC1 | CM_CTRL_STATIC2;
202

203 204 205 206
	if (var->bits_per_pixel <= 8 ||
	    (var->bits_per_pixel == 16 && var->green.length == 5))
		/* Pseudocolor, RGB555, BGR555 */
		val |= CM_CTRL_LCDMUXSEL_VGA555_TFT555;
207
	else if (fb->fb.var.bits_per_pixel <= 16)
208 209
		/* truecolor RGB565 */
		val |= CM_CTRL_LCDMUXSEL_VGA565_TFT555;
210 211 212 213 214 215 216 217 218 219
	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 已提交
220 221 222 223
}

static int cp_clcd_setup(struct clcd_fb *fb)
{
224 225 226
	fb->panel = versatile_clcd_get_panel("VGA");
	if (!fb->panel)
		return -EINVAL;
L
Linus Torvalds 已提交
227

228
	return versatile_clcd_setup_dma(fb, SZ_1M);
L
Linus Torvalds 已提交
229 230 231 232
}

static struct clcd_board clcd_data = {
	.name		= "Integrator/CP",
233
	.caps		= CLCD_CAP_5551 | CLCD_CAP_RGB565 | CLCD_CAP_888,
L
Linus Torvalds 已提交
234 235 236 237
	.check		= clcdfb_check,
	.decode		= clcdfb_decode,
	.enable		= cp_clcd_enable,
	.setup		= cp_clcd_setup,
238 239
	.mmap		= versatile_clcd_mmap_dma,
	.remove		= versatile_clcd_remove_dma,
L
Linus Torvalds 已提交
240 241
};

242 243
#define REFCOUNTER (__io_address(INTEGRATOR_HDR_BASE) + 0x28)

244 245
static void __init intcp_init_early(void)
{
246 247 248
#ifdef CONFIG_PLAT_VERSATILE_SCHED_CLOCK
	versatile_sched_clock_init(REFCOUNTER, 24000000);
#endif
249 250
}

251 252 253 254 255 256
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 已提交
257
{
258
	cm_init();
259 260 261
	of_irq_init(fpga_irq_of_match);
	integrator_clk_init(true);
}
L
Linus Torvalds 已提交
262

263 264 265 266 267 268 269 270
/*
 * 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,
271
		"uart0", NULL),
272
	OF_DEV_AUXDATA("arm,primecell", INTEGRATOR_UART1_BASE,
273
		"uart1", NULL),
274 275 276 277 278 279 280 281 282 283
	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),
284 285
	OF_DEV_AUXDATA("cfi-flash", INTCP_PA_FLASH_BASE,
		"physmap-flash", &intcp_flash_data),
286 287 288
	{ /* sentinel */ },
};

289 290 291 292 293
static const struct of_device_id intcp_syscon_match[] = {
	{ .compatible = "arm,integrator-cp-syscon"},
	{ },
};

294 295
static void __init intcp_init_of(void)
{
296 297 298 299 300 301 302 303 304 305 306 307
	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;
308 309

	cpcon = of_find_matching_node(root, intcp_syscon_match);
310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334
	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);
335
	if (IS_ERR(soc_dev)) {
336 337 338 339 340 341
		kfree(soc_dev_attr->revision);
		kfree(soc_dev_attr);
		return;
	}

	parent = soc_device_to_device(soc_dev);
342
	integrator_init_sysfs(parent, intcp_sc_id);
343 344
	of_platform_populate(root, of_default_bus_match_table,
			intcp_auxdata_lookup, parent);
345 346
}

347 348 349 350
static const char * intcp_dt_board_compat[] = {
	"arm,integrator-cp",
	NULL,
};
L
Linus Torvalds 已提交
351

352 353 354 355 356 357
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,
358
	.init_machine	= intcp_init_of,
359 360 361
	.restart	= integrator_restart,
	.dt_compat      = intcp_dt_board_compat,
MACHINE_END