2007-12-15 23:05:49 +03:00
/*
* QNAP TS - 109 / TS - 209 Board Setup
*
* Maintainer : Byron Bradley < byron . bbradley @ gmail . com >
*
* 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 , or ( at your option ) any later version .
*/
2011-07-26 13:53:52 +04:00
# include <linux/gpio.h>
2007-12-15 23:05:49 +03:00
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/platform_device.h>
# include <linux/pci.h>
# include <linux/irq.h>
# include <linux/mtd/physmap.h>
# include <linux/mtd/nand.h>
# include <linux/mv643xx_eth.h>
# include <linux/gpio_keys.h>
# include <linux/input.h>
# include <linux/i2c.h>
2007-12-16 19:42:31 +03:00
# include <linux/serial_reg.h>
2008-02-08 21:20:23 +03:00
# include <linux/ata_platform.h>
2007-12-15 23:05:49 +03:00
# include <asm/mach-types.h>
# include <asm/mach/arch.h>
# include <asm/mach/pci.h>
2008-08-05 19:14:15 +04:00
# include <mach/orion5x.h>
2007-12-15 23:05:49 +03:00
# include "common.h"
2008-05-11 01:25:46 +04:00
# include "mpp.h"
2008-05-31 20:21:49 +04:00
# include "tsx09-common.h"
2007-12-15 23:05:49 +03:00
# define QNAP_TS209_NOR_BOOT_BASE 0xf4000000
# define QNAP_TS209_NOR_BOOT_SIZE SZ_8M
/****************************************************************************
* 8 MiB NOR flash . The struct mtd_partition is not in the same order as the
2011-03-31 05:57:33 +04:00
* partitions on the device because we want to keep compatibility with
2007-12-15 23:05:49 +03:00
* existing QNAP firmware .
*
* Layout as used by QNAP :
* [ 2 ] 0x00000000 - 0x00200000 : " Kernel "
* [ 3 ] 0x00200000 - 0x00600000 : " RootFS1 "
* [ 4 ] 0x00600000 - 0x00700000 : " RootFS2 "
* [ 6 ] 0x00700000 - 0x00760000 : " NAS Config " ( read - only )
* [ 5 ] 0x00760000 - 0x00780000 : " U-Boot Config "
* [ 1 ] 0x00780000 - 0x00800000 : " U-Boot " ( read - only )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static struct mtd_partition qnap_ts209_partitions [ ] = {
{
2008-05-10 18:30:01 +04:00
. name = " U-Boot " ,
. size = 0x00080000 ,
. offset = 0x00780000 ,
. mask_flags = MTD_WRITEABLE ,
2007-12-15 23:05:49 +03:00
} , {
2008-05-10 18:30:01 +04:00
. name = " Kernel " ,
. size = 0x00200000 ,
. offset = 0 ,
2007-12-15 23:05:49 +03:00
} , {
2008-05-10 18:30:01 +04:00
. name = " RootFS1 " ,
. size = 0x00400000 ,
. offset = 0x00200000 ,
2007-12-15 23:05:49 +03:00
} , {
2008-05-10 18:30:01 +04:00
. name = " RootFS2 " ,
. size = 0x00100000 ,
. offset = 0x00600000 ,
2007-12-15 23:05:49 +03:00
} , {
2008-05-10 18:30:01 +04:00
. name = " U-Boot Config " ,
. size = 0x00020000 ,
. offset = 0x00760000 ,
2007-12-15 23:05:49 +03:00
} , {
2008-05-10 18:30:01 +04:00
. name = " NAS Config " ,
. size = 0x00060000 ,
. offset = 0x00700000 ,
. mask_flags = MTD_WRITEABLE ,
} ,
2007-12-15 23:05:49 +03:00
} ;
static struct physmap_flash_data qnap_ts209_nor_flash_data = {
2008-05-10 18:30:01 +04:00
. width = 1 ,
. parts = qnap_ts209_partitions ,
. nr_parts = ARRAY_SIZE ( qnap_ts209_partitions )
2007-12-15 23:05:49 +03:00
} ;
static struct resource qnap_ts209_nor_flash_resource = {
2008-05-10 18:30:01 +04:00
. flags = IORESOURCE_MEM ,
. start = QNAP_TS209_NOR_BOOT_BASE ,
. end = QNAP_TS209_NOR_BOOT_BASE + QNAP_TS209_NOR_BOOT_SIZE - 1 ,
2007-12-15 23:05:49 +03:00
} ;
static struct platform_device qnap_ts209_nor_flash = {
2008-05-10 18:30:01 +04:00
. name = " physmap-flash " ,
. id = 0 ,
. dev = {
. platform_data = & qnap_ts209_nor_flash_data ,
} ,
. resource = & qnap_ts209_nor_flash_resource ,
. num_resources = 1 ,
2007-12-15 23:05:49 +03:00
} ;
/*****************************************************************************
* PCI
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define QNAP_TS209_PCI_SLOT0_OFFS 7
# define QNAP_TS209_PCI_SLOT0_IRQ_PIN 6
# define QNAP_TS209_PCI_SLOT1_IRQ_PIN 7
void __init qnap_ts209_pci_preinit ( void )
{
int pin ;
/*
* Configure PCI GPIO IRQ pins
*/
pin = QNAP_TS209_PCI_SLOT0_IRQ_PIN ;
if ( gpio_request ( pin , " PCI Int1 " ) = = 0 ) {
if ( gpio_direction_input ( pin ) = = 0 ) {
2011-03-24 15:25:22 +03:00
irq_set_irq_type ( gpio_to_irq ( pin ) , IRQ_TYPE_LEVEL_LOW ) ;
2007-12-15 23:05:49 +03:00
} else {
printk ( KERN_ERR " qnap_ts209_pci_preinit failed to "
" set_irq_type pin %d \n " , pin ) ;
gpio_free ( pin ) ;
}
} else {
printk ( KERN_ERR " qnap_ts209_pci_preinit failed to gpio_request "
" %d \n " , pin ) ;
}
pin = QNAP_TS209_PCI_SLOT1_IRQ_PIN ;
if ( gpio_request ( pin , " PCI Int2 " ) = = 0 ) {
if ( gpio_direction_input ( pin ) = = 0 ) {
2011-03-24 15:25:22 +03:00
irq_set_irq_type ( gpio_to_irq ( pin ) , IRQ_TYPE_LEVEL_LOW ) ;
2007-12-15 23:05:49 +03:00
} else {
printk ( KERN_ERR " qnap_ts209_pci_preinit failed "
" to set_irq_type pin %d \n " , pin ) ;
gpio_free ( pin ) ;
}
} else {
printk ( KERN_ERR " qnap_ts209_pci_preinit failed to gpio_request "
" %d \n " , pin ) ;
}
}
2011-06-10 18:30:21 +04:00
static int __init qnap_ts209_pci_map_irq ( const struct pci_dev * dev , u8 slot ,
u8 pin )
2007-12-15 23:05:49 +03:00
{
2008-04-26 00:28:33 +04:00
int irq ;
2007-12-15 23:05:49 +03:00
/*
2008-04-26 00:28:33 +04:00
* Check for devices with hard - wired IRQs .
2007-12-15 23:05:49 +03:00
*/
2008-04-26 00:28:33 +04:00
irq = orion5x_pci_map_irq ( dev , slot , pin ) ;
if ( irq ! = - 1 )
return irq ;
2007-12-15 23:05:49 +03:00
/*
2008-04-26 00:28:33 +04:00
* PCI IRQs are connected via GPIOs .
2007-12-15 23:05:49 +03:00
*/
switch ( slot - QNAP_TS209_PCI_SLOT0_OFFS ) {
case 0 :
return gpio_to_irq ( QNAP_TS209_PCI_SLOT0_IRQ_PIN ) ;
case 1 :
return gpio_to_irq ( QNAP_TS209_PCI_SLOT1_IRQ_PIN ) ;
default :
return - 1 ;
}
}
static struct hw_pci qnap_ts209_pci __initdata = {
2008-05-10 18:30:01 +04:00
. nr_controllers = 2 ,
. preinit = qnap_ts209_pci_preinit ,
. swizzle = pci_std_swizzle ,
. setup = orion5x_pci_sys_setup ,
. scan = orion5x_pci_sys_scan_bus ,
. map_irq = qnap_ts209_pci_map_irq ,
2007-12-15 23:05:49 +03:00
} ;
static int __init qnap_ts209_pci_init ( void )
{
2011-12-06 12:59:38 +04:00
if ( machine_is_ts209 ( ) )
2007-12-15 23:05:49 +03:00
pci_common_init ( & qnap_ts209_pci ) ;
return 0 ;
}
subsys_initcall ( qnap_ts209_pci_init ) ;
/*****************************************************************************
* RTC S35390A on I2C bus
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-02-11 00:31:09 +03:00
# define TS209_RTC_GPIO 3
2007-12-15 23:05:49 +03:00
static struct i2c_board_info __initdata qnap_ts209_i2c_rtc = {
2008-04-30 01:11:40 +04:00
I2C_BOARD_INFO ( " s35390a " , 0x30 ) ,
2008-05-10 18:30:01 +04:00
. irq = 0 ,
2007-12-15 23:05:49 +03:00
} ;
/****************************************************************************
* GPIO Attached Keys
* Power button is attached to the PIC microcontroller
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define QNAP_TS209_GPIO_KEY_MEDIA 1
# define QNAP_TS209_GPIO_KEY_RESET 2
static struct gpio_keys_button qnap_ts209_buttons [ ] = {
{
2008-07-22 02:49:22 +04:00
. code = KEY_COPY ,
2007-12-15 23:05:49 +03:00
. gpio = QNAP_TS209_GPIO_KEY_MEDIA ,
. desc = " USB Copy Button " ,
. active_low = 1 ,
2008-05-10 18:30:01 +04:00
} , {
2008-07-22 02:49:22 +04:00
. code = KEY_RESTART ,
2007-12-15 23:05:49 +03:00
. gpio = QNAP_TS209_GPIO_KEY_RESET ,
. desc = " Reset Button " ,
. active_low = 1 ,
2008-05-10 18:30:01 +04:00
} ,
2007-12-15 23:05:49 +03:00
} ;
static struct gpio_keys_platform_data qnap_ts209_button_data = {
. buttons = qnap_ts209_buttons ,
2008-05-10 18:30:01 +04:00
. nbuttons = ARRAY_SIZE ( qnap_ts209_buttons ) ,
2007-12-15 23:05:49 +03:00
} ;
static struct platform_device qnap_ts209_button_device = {
. name = " gpio-keys " ,
. id = - 1 ,
. num_resources = 0 ,
2008-05-10 18:30:01 +04:00
. dev = {
. platform_data = & qnap_ts209_button_data ,
} ,
2007-12-15 23:05:49 +03:00
} ;
/*****************************************************************************
2008-02-08 21:20:23 +03:00
* SATA
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static struct mv_sata_platform_data qnap_ts209_sata_data = {
2008-05-10 18:30:01 +04:00
. n_ports = 2 ,
2008-02-08 21:20:23 +03:00
} ;
/*****************************************************************************
2007-12-15 23:05:49 +03:00
* General Setup
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2011-05-15 15:32:53 +04:00
static unsigned int ts209_mpp_modes [ ] __initdata = {
MPP0_UNUSED ,
MPP1_GPIO , /* USB copy button */
MPP2_GPIO , /* Load defaults button */
MPP3_GPIO , /* GPIO RTC */
MPP4_UNUSED ,
MPP5_UNUSED ,
MPP6_GPIO , /* PCI Int A */
MPP7_GPIO , /* PCI Int B */
MPP8_UNUSED ,
MPP9_UNUSED ,
MPP10_UNUSED ,
MPP11_UNUSED ,
MPP12_SATA_LED , /* SATA 0 presence */
MPP13_SATA_LED , /* SATA 1 presence */
MPP14_SATA_LED , /* SATA 0 active */
MPP15_SATA_LED , /* SATA 1 active */
MPP16_UART , /* UART1 RXD */
MPP17_UART , /* UART1 TXD */
MPP18_GPIO , /* SW_RST */
MPP19_UNUSED ,
0 ,
2008-05-11 01:25:46 +04:00
} ;
2007-12-15 23:05:49 +03:00
static void __init qnap_ts209_init ( void )
{
/*
* Setup basic Orion functions . Need to be called early .
*/
2008-03-27 21:51:41 +03:00
orion5x_init ( ) ;
2007-12-15 23:05:49 +03:00
2008-05-11 01:25:46 +04:00
orion5x_mpp_conf ( ts209_mpp_modes ) ;
2007-12-15 23:05:49 +03:00
/*
* MPP [ 20 ] PCI clock 0
* MPP [ 21 ] PCI clock 1
* MPP [ 22 ] USB 0 over current
* MPP [ 23 - 25 ] Reserved
*/
2008-04-22 07:37:12 +04:00
/*
* Configure peripherals .
*/
2008-08-09 19:13:33 +04:00
orion5x_setup_dev_boot_win ( QNAP_TS209_NOR_BOOT_BASE ,
QNAP_TS209_NOR_BOOT_SIZE ) ;
platform_device_register ( & qnap_ts209_nor_flash ) ;
2008-04-22 07:37:12 +04:00
orion5x_ehci0_init ( ) ;
orion5x_ehci1_init ( ) ;
2008-05-31 20:21:49 +04:00
qnap_tsx09_find_mac_addr ( QNAP_TS209_NOR_BOOT_BASE +
qnap_ts209_partitions [ 5 ] . offset ,
qnap_ts209_partitions [ 5 ] . size ) ;
orion5x_eth_init ( & qnap_tsx09_eth_data ) ;
2008-04-22 07:37:12 +04:00
orion5x_i2c_init ( ) ;
orion5x_sata_init ( & qnap_ts209_sata_data ) ;
orion5x_uart0_init ( ) ;
2008-08-13 11:46:11 +04:00
orion5x_uart1_init ( ) ;
2008-06-17 14:25:12 +04:00
orion5x_xor_init ( ) ;
2008-04-22 07:37:12 +04:00
platform_device_register ( & qnap_ts209_button_device ) ;
2008-02-11 00:31:09 +03:00
/* Get RTC IRQ and register the chip */
if ( gpio_request ( TS209_RTC_GPIO , " rtc " ) = = 0 ) {
if ( gpio_direction_input ( TS209_RTC_GPIO ) = = 0 )
qnap_ts209_i2c_rtc . irq = gpio_to_irq ( TS209_RTC_GPIO ) ;
else
gpio_free ( TS209_RTC_GPIO ) ;
}
if ( qnap_ts209_i2c_rtc . irq = = 0 )
pr_warning ( " qnap_ts209_init: failed to get RTC IRQ \n " ) ;
2007-12-15 23:05:49 +03:00
i2c_register_board_info ( 0 , & qnap_ts209_i2c_rtc , 1 ) ;
2008-02-11 00:31:09 +03:00
2008-05-31 20:21:49 +04:00
/* register tsx09 specific power-off method */
pm_power_off = qnap_tsx09_power_off ;
2007-12-15 23:05:49 +03:00
}
MACHINE_START ( TS209 , " QNAP TS-109/TS-209 " )
2008-05-10 18:30:01 +04:00
/* Maintainer: Byron Bradley <byron.bbradley@gmail.com> */
2011-07-06 06:38:15 +04:00
. atag_offset = 0x100 ,
2007-12-15 23:05:49 +03:00
. init_machine = qnap_ts209_init ,
2008-03-27 21:51:41 +03:00
. map_io = orion5x_map_io ,
2010-10-15 18:50:26 +04:00
. init_early = orion5x_init_early ,
2008-03-27 21:51:41 +03:00
. init_irq = orion5x_init_irq ,
. timer = & orion5x_timer ,
2008-02-29 23:12:57 +03:00
. fixup = tag_fixup_mem32 ,
2011-11-05 14:13:41 +04:00
. restart = orion5x_restart ,
2007-12-15 23:05:49 +03:00
MACHINE_END