mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-31 09:36:25 +00:00 
			
		
		
		
	 284901a90a
			
		
	
	
		284901a90a
		
	
	
	
	
		
			
			Replace all DMA_32BIT_MASK macro with DMA_BIT_MASK(32) Signed-off-by: Yang Hongyang<yanghy@cn.fujitsu.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
			
				
	
	
		
			377 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			377 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Hammerhead board-specific flash initialization
 | |
|  *
 | |
|  * Copyright (C) 2008 Miromico AG
 | |
|  *
 | |
|  * 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.
 | |
|  */
 | |
| 
 | |
| #include <linux/init.h>
 | |
| #include <linux/platform_device.h>
 | |
| #include <linux/mtd/mtd.h>
 | |
| #include <linux/mtd/partitions.h>
 | |
| #include <linux/mtd/physmap.h>
 | |
| #include <linux/usb/isp116x.h>
 | |
| #include <linux/dma-mapping.h>
 | |
| #include <linux/delay.h>
 | |
| 
 | |
| #include <mach/portmux.h>
 | |
| #include <mach/at32ap700x.h>
 | |
| #include <mach/smc.h>
 | |
| 
 | |
| #include "../../mach-at32ap/clock.h"
 | |
| #include "flash.h"
 | |
| 
 | |
| 
 | |
| #define HAMMERHEAD_USB_PERIPH_GCLK0	0x40000000
 | |
| #define HAMMERHEAD_USB_PERIPH_CS2	0x02000000
 | |
| #define HAMMERHEAD_USB_PERIPH_EXTINT0	0x02000000
 | |
| 
 | |
| #define HAMMERHEAD_FPGA_PERIPH_MOSI	0x00000002
 | |
| #define HAMMERHEAD_FPGA_PERIPH_SCK	0x00000020
 | |
| #define HAMMERHEAD_FPGA_PERIPH_EXTINT3	0x10000000
 | |
| 
 | |
| static struct smc_timing flash_timing __initdata = {
 | |
| 	.ncs_read_setup		= 0,
 | |
| 	.nrd_setup		= 40,
 | |
| 	.ncs_write_setup	= 0,
 | |
| 	.nwe_setup		= 10,
 | |
| 
 | |
| 	.ncs_read_pulse		= 80,
 | |
| 	.nrd_pulse		= 40,
 | |
| 	.ncs_write_pulse	= 65,
 | |
| 	.nwe_pulse		= 55,
 | |
| 
 | |
| 	.read_cycle		= 120,
 | |
| 	.write_cycle		= 120,
 | |
| };
 | |
| 
 | |
| static struct smc_config flash_config __initdata = {
 | |
| 	.bus_width		= 2,
 | |
| 	.nrd_controlled		= 1,
 | |
| 	.nwe_controlled		= 1,
 | |
| 	.byte_write		= 1,
 | |
| };
 | |
| 
 | |
| static struct mtd_partition flash_parts[] = {
 | |
| 	{
 | |
| 		.name		= "u-boot",
 | |
| 		.offset		= 0x00000000,
 | |
| 		.size		= 0x00020000,           /* 128 KiB */
 | |
| 		.mask_flags	= MTD_WRITEABLE,
 | |
| 	},
 | |
| 	{
 | |
| 		.name		= "root",
 | |
| 		.offset		= 0x00020000,
 | |
| 		.size		= 0x007d0000,
 | |
| 	},
 | |
| 	{
 | |
| 		.name		= "env",
 | |
| 		.offset		= 0x007f0000,
 | |
| 		.size		= 0x00010000,
 | |
| 		.mask_flags	= MTD_WRITEABLE,
 | |
| 	},
 | |
| };
 | |
| 
 | |
| static struct physmap_flash_data flash_data = {
 | |
| 	.width		= 2,
 | |
| 	.nr_parts	= ARRAY_SIZE(flash_parts),
 | |
| 	.parts		= flash_parts,
 | |
| };
 | |
| 
 | |
| static struct resource flash_resource = {
 | |
| 	.start		= 0x00000000,
 | |
| 	.end		= 0x007fffff,
 | |
| 	.flags		= IORESOURCE_MEM,
 | |
| };
 | |
| 
 | |
| static struct platform_device flash_device = {
 | |
| 	.name		= "physmap-flash",
 | |
| 	.id		= 0,
 | |
| 	.resource	= &flash_resource,
 | |
| 	.num_resources	= 1,
 | |
| 	.dev		= { .platform_data = &flash_data, },
 | |
| };
 | |
| 
 | |
| #ifdef CONFIG_BOARD_HAMMERHEAD_USB
 | |
| 
 | |
| static struct smc_timing isp1160_timing __initdata = {
 | |
| 	.ncs_read_setup		= 75,
 | |
| 	.nrd_setup		= 75,
 | |
| 	.ncs_write_setup	= 75,
 | |
| 	.nwe_setup		= 75,
 | |
| 
 | |
| 
 | |
| 	/* We use conservative timing settings, as the minimal settings aren't
 | |
| 	   stable. There may be room for tweaking. */
 | |
| 	.ncs_read_pulse		= 75,  /* min. 33ns */
 | |
| 	.nrd_pulse		= 75,  /* min. 33ns */
 | |
| 	.ncs_write_pulse	= 75,  /* min. 26ns */
 | |
| 	.nwe_pulse		= 75,  /* min. 26ns */
 | |
| 
 | |
| 	.read_cycle		= 225, /* min. 143ns */
 | |
| 	.write_cycle		= 225, /* min. 136ns */
 | |
| };
 | |
| 
 | |
| static struct smc_config isp1160_config __initdata = {
 | |
| 	.bus_width		= 2,
 | |
| 	.nrd_controlled		= 1,
 | |
| 	.nwe_controlled		= 1,
 | |
| 	.byte_write		= 0,
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * The platform delay function is only used to enforce the strange
 | |
|  * read to write delay. This can not be configured in the SMC. All other
 | |
|  * timings are controlled by the SMC (see timings obove)
 | |
|  * So in isp116x-hcd.c we should comment out USE_PLATFORM_DELAY
 | |
|  */
 | |
| void isp116x_delay(struct device *dev, int delay)
 | |
| {
 | |
| 	if (delay > 150)
 | |
| 		ndelay(delay - 150);
 | |
| }
 | |
| 
 | |
| static struct  isp116x_platform_data isp1160_data = {
 | |
| 	.sel15Kres		= 1,	/* use internal downstream resistors */
 | |
| 	.oc_enable		= 0,	/* external overcurrent detection */
 | |
| 	.int_edge_triggered	= 0,	/* interrupt is level triggered */
 | |
| 	.int_act_high		= 0,	/* interrupt is active low */
 | |
| 	.delay = isp116x_delay,		/* platform delay function */
 | |
| };
 | |
| 
 | |
| static struct resource isp1160_resource[] = {
 | |
| 	{
 | |
| 		.start		= 0x08000000,
 | |
| 		.end		= 0x08000001,
 | |
| 		.flags		= IORESOURCE_MEM,
 | |
| 	},
 | |
| 	{
 | |
| 		.start		= 0x08000002,
 | |
| 		.end		= 0x08000003,
 | |
| 		.flags		= IORESOURCE_MEM,
 | |
| 	},
 | |
| 	{
 | |
| 		.start		= 64,
 | |
| 		.flags		= IORESOURCE_IRQ,
 | |
| 	},
 | |
| };
 | |
| 
 | |
| static struct platform_device isp1160_device = {
 | |
| 	.name		= "isp116x-hcd",
 | |
| 	.id		= 0,
 | |
| 	.resource	= isp1160_resource,
 | |
| 	.num_resources	= 3,
 | |
| 	.dev		= {
 | |
| 		.platform_data = &isp1160_data,
 | |
| 	},
 | |
| };
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_BOARD_HAMMERHEAD_USB
 | |
| static int __init hammerhead_usbh_init(void)
 | |
| {
 | |
| 	struct clk *gclk;
 | |
| 	struct clk *osc;
 | |
| 
 | |
| 	int ret;
 | |
| 
 | |
| 	/* setup smc for usbh */
 | |
| 	smc_set_timing(&isp1160_config, &isp1160_timing);
 | |
| 	ret = smc_set_configuration(2, &isp1160_config);
 | |
| 
 | |
| 	if (ret < 0) {
 | |
| 		printk(KERN_ERR
 | |
| 		       "hammerhead: failed to set ISP1160 USBH timing\n");
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	/* setup gclk0 to run from osc1 */
 | |
| 	gclk = clk_get(NULL, "gclk0");
 | |
| 	if (IS_ERR(gclk))
 | |
| 		goto err_gclk;
 | |
| 
 | |
| 	osc = clk_get(NULL, "osc1");
 | |
| 	if (IS_ERR(osc))
 | |
| 		goto err_osc;
 | |
| 
 | |
| 	if (clk_set_parent(gclk, osc)) {
 | |
| 		pr_debug("hammerhead: failed to set osc1 for USBH clock\n");
 | |
| 		goto err_set_clk;
 | |
| 	}
 | |
| 
 | |
| 	/* set clock to 6MHz */
 | |
| 	clk_set_rate(gclk, 6000000);
 | |
| 
 | |
| 	/* and enable */
 | |
| 	clk_enable(gclk);
 | |
| 
 | |
| 	/* select GCLK0 peripheral function */
 | |
| 	at32_select_periph(GPIO_PIOA_BASE, HAMMERHEAD_USB_PERIPH_GCLK0,
 | |
| 			   GPIO_PERIPH_A, 0);
 | |
| 
 | |
| 	/* enable CS2 peripheral function */
 | |
| 	at32_select_periph(GPIO_PIOE_BASE, HAMMERHEAD_USB_PERIPH_CS2,
 | |
| 			   GPIO_PERIPH_A, 0);
 | |
| 
 | |
| 	/* H_WAKEUP must be driven low */
 | |
| 	at32_select_gpio(GPIO_PIN_PA(8), AT32_GPIOF_OUTPUT);
 | |
| 
 | |
| 	/* Select EXTINT0 for PB25 */
 | |
| 	at32_select_periph(GPIO_PIOB_BASE, HAMMERHEAD_USB_PERIPH_EXTINT0,
 | |
| 			   GPIO_PERIPH_A, 0);
 | |
| 
 | |
| 	/* register usbh device driver */
 | |
| 	platform_device_register(&isp1160_device);
 | |
| 
 | |
|  err_set_clk:
 | |
| 	clk_put(osc);
 | |
|  err_osc:
 | |
| 	clk_put(gclk);
 | |
|  err_gclk:
 | |
| 	return ret;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_BOARD_HAMMERHEAD_FPGA
 | |
| static struct smc_timing fpga_timing __initdata = {
 | |
| 	.ncs_read_setup		= 16,
 | |
| 	.nrd_setup		= 32,
 | |
| 	.ncs_read_pulse		= 48,
 | |
| 	.nrd_pulse		= 32,
 | |
| 	.read_cycle		= 64,
 | |
| 
 | |
| 	.ncs_write_setup	= 16,
 | |
| 	.nwe_setup		= 16,
 | |
| 	.ncs_write_pulse	= 32,
 | |
| 	.nwe_pulse		= 32,
 | |
| 	.write_cycle		= 64,
 | |
| };
 | |
| 
 | |
| static struct smc_config fpga_config __initdata = {
 | |
| 	.bus_width		= 4,
 | |
| 	.nrd_controlled		= 1,
 | |
| 	.nwe_controlled		= 1,
 | |
| 	.byte_write		= 0,
 | |
| };
 | |
| 
 | |
| static struct resource hh_fpga0_resource[] = {
 | |
| 	{
 | |
| 		.start		= 0xffe00400,
 | |
| 		.end		= 0xffe00400 + 0x3ff,
 | |
| 		.flags		= IORESOURCE_MEM,
 | |
| 	},
 | |
| 	{
 | |
| 		.start		= 4,
 | |
| 		.end		= 4,
 | |
| 		.flags		= IORESOURCE_IRQ,
 | |
| 	},
 | |
| 	{
 | |
| 		.start		= 0x0c000000,
 | |
| 		.end		= 0x0c000100,
 | |
| 		.flags		= IORESOURCE_MEM,
 | |
| 	},
 | |
| 	{
 | |
| 		.start		= 67,
 | |
| 		.end		= 67,
 | |
| 		.flags		= IORESOURCE_IRQ,
 | |
| 	},
 | |
| };
 | |
| 
 | |
| static u64 hh_fpga0_dma_mask = DMA_BIT_MASK(32);
 | |
| static struct platform_device hh_fpga0_device = {
 | |
| 	.name		= "hh_fpga",
 | |
| 	.id		= 0,
 | |
| 	.dev		= {
 | |
| 		.dma_mask = &hh_fpga0_dma_mask,
 | |
| 		.coherent_dma_mask = DMA_BIT_MASK(32),
 | |
| 	},
 | |
| 	.resource	= hh_fpga0_resource,
 | |
| 	.num_resources	= ARRAY_SIZE(hh_fpga0_resource),
 | |
| };
 | |
| 
 | |
| static struct clk hh_fpga0_spi_clk = {
 | |
| 	.name		= "spi_clk",
 | |
| 	.dev		= &hh_fpga0_device.dev,
 | |
| 	.mode		= pba_clk_mode,
 | |
| 	.get_rate	= pba_clk_get_rate,
 | |
| 	.index		= 1,
 | |
| };
 | |
| 
 | |
| struct platform_device *__init at32_add_device_hh_fpga(void)
 | |
| {
 | |
| 	/* Select peripheral functionallity for SPI SCK and MOSI */
 | |
| 	at32_select_periph(GPIO_PIOB_BASE, HAMMERHEAD_FPGA_PERIPH_SCK,
 | |
| 			   GPIO_PERIPH_B, 0);
 | |
| 	at32_select_periph(GPIO_PIOB_BASE, HAMMERHEAD_FPGA_PERIPH_MOSI,
 | |
| 			   GPIO_PERIPH_B, 0);
 | |
| 
 | |
| 	/* reserve all other needed gpio
 | |
| 	 * We have on board pull ups, so there is no need
 | |
| 	 * to enable gpio pull ups */
 | |
| 	/* INIT_DONE (input) */
 | |
| 	at32_select_gpio(GPIO_PIN_PB(0), 0);
 | |
| 
 | |
| 	/* nSTATUS (input) */
 | |
| 	at32_select_gpio(GPIO_PIN_PB(2), 0);
 | |
| 
 | |
| 	/* nCONFIG (output, low) */
 | |
| 	at32_select_gpio(GPIO_PIN_PB(3), AT32_GPIOF_OUTPUT);
 | |
| 
 | |
| 	/* CONF_DONE (input) */
 | |
| 	at32_select_gpio(GPIO_PIN_PB(4), 0);
 | |
| 
 | |
| 	/* Select EXTINT3 for PB28 (Interrupt from FPGA) */
 | |
| 	at32_select_periph(GPIO_PIOB_BASE, HAMMERHEAD_FPGA_PERIPH_EXTINT3,
 | |
| 			   GPIO_PERIPH_A, 0);
 | |
| 
 | |
| 	/* Get our parent clock */
 | |
| 	hh_fpga0_spi_clk.parent = clk_get(NULL, "pba");
 | |
| 	clk_put(hh_fpga0_spi_clk.parent);
 | |
| 
 | |
| 	/* Register clock in at32 clock tree */
 | |
| 	at32_clk_register(&hh_fpga0_spi_clk);
 | |
| 
 | |
| 	platform_device_register(&hh_fpga0_device);
 | |
| 	return &hh_fpga0_device;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /* This needs to be called after the SMC has been initialized */
 | |
| static int __init hammerhead_flash_init(void)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	smc_set_timing(&flash_config, &flash_timing);
 | |
| 	ret = smc_set_configuration(0, &flash_config);
 | |
| 
 | |
| 	if (ret < 0) {
 | |
| 		printk(KERN_ERR "hammerhead: failed to set NOR flash timing\n");
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	platform_device_register(&flash_device);
 | |
| 
 | |
| #ifdef CONFIG_BOARD_HAMMERHEAD_USB
 | |
| 	hammerhead_usbh_init();
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_BOARD_HAMMERHEAD_FPGA
 | |
| 	/* Setup SMC for FPGA interface */
 | |
| 	smc_set_timing(&fpga_config, &fpga_timing);
 | |
| 	ret = smc_set_configuration(3, &fpga_config);
 | |
| #endif
 | |
| 
 | |
| 
 | |
| 	if (ret < 0) {
 | |
| 		printk(KERN_ERR "hammerhead: failed to set FPGA timing\n");
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| device_initcall(hammerhead_flash_init);
 |