Path: blob/master/arch/mips/cavium-octeon/octeon-platform.c
10820 views
/*1* This file is subject to the terms and conditions of the GNU General Public2* License. See the file "COPYING" in the main directory of this archive3* for more details.4*5* Copyright (C) 2004-2010 Cavium Networks6* Copyright (C) 2008 Wind River Systems7*/89#include <linux/init.h>10#include <linux/irq.h>11#include <linux/i2c.h>12#include <linux/usb.h>13#include <linux/dma-mapping.h>14#include <linux/module.h>15#include <linux/platform_device.h>1617#include <asm/octeon/octeon.h>18#include <asm/octeon/cvmx-rnm-defs.h>1920static struct octeon_cf_data octeon_cf_data;2122static int __init octeon_cf_device_init(void)23{24union cvmx_mio_boot_reg_cfgx mio_boot_reg_cfg;25unsigned long base_ptr, region_base, region_size;26struct platform_device *pd;27struct resource cf_resources[3];28unsigned int num_resources;29int i;30int ret = 0;3132/* Setup octeon-cf platform device if present. */33base_ptr = 0;34if (octeon_bootinfo->major_version == 135&& octeon_bootinfo->minor_version >= 1) {36if (octeon_bootinfo->compact_flash_common_base_addr)37base_ptr =38octeon_bootinfo->compact_flash_common_base_addr;39} else {40base_ptr = 0x1d000800;41}4243if (!base_ptr)44return ret;4546/* Find CS0 region. */47for (i = 0; i < 8; i++) {48mio_boot_reg_cfg.u64 = cvmx_read_csr(CVMX_MIO_BOOT_REG_CFGX(i));49region_base = mio_boot_reg_cfg.s.base << 16;50region_size = (mio_boot_reg_cfg.s.size + 1) << 16;51if (mio_boot_reg_cfg.s.en && base_ptr >= region_base52&& base_ptr < region_base + region_size)53break;54}55if (i >= 7) {56/* i and i + 1 are CS0 and CS1, both must be less than 8. */57goto out;58}59octeon_cf_data.base_region = i;60octeon_cf_data.is16bit = mio_boot_reg_cfg.s.width;61octeon_cf_data.base_region_bias = base_ptr - region_base;62memset(cf_resources, 0, sizeof(cf_resources));63num_resources = 0;64cf_resources[num_resources].flags = IORESOURCE_MEM;65cf_resources[num_resources].start = region_base;66cf_resources[num_resources].end = region_base + region_size - 1;67num_resources++;686970if (!(base_ptr & 0xfffful)) {71/*72* Boot loader signals availability of DMA (true_ide73* mode) by setting low order bits of base_ptr to74* zero.75*/7677/* Assume that CS1 immediately follows. */78mio_boot_reg_cfg.u64 =79cvmx_read_csr(CVMX_MIO_BOOT_REG_CFGX(i + 1));80region_base = mio_boot_reg_cfg.s.base << 16;81region_size = (mio_boot_reg_cfg.s.size + 1) << 16;82if (!mio_boot_reg_cfg.s.en)83goto out;8485cf_resources[num_resources].flags = IORESOURCE_MEM;86cf_resources[num_resources].start = region_base;87cf_resources[num_resources].end = region_base + region_size - 1;88num_resources++;8990octeon_cf_data.dma_engine = 0;91cf_resources[num_resources].flags = IORESOURCE_IRQ;92cf_resources[num_resources].start = OCTEON_IRQ_BOOTDMA;93cf_resources[num_resources].end = OCTEON_IRQ_BOOTDMA;94num_resources++;95} else {96octeon_cf_data.dma_engine = -1;97}9899pd = platform_device_alloc("pata_octeon_cf", -1);100if (!pd) {101ret = -ENOMEM;102goto out;103}104pd->dev.platform_data = &octeon_cf_data;105106ret = platform_device_add_resources(pd, cf_resources, num_resources);107if (ret)108goto fail;109110ret = platform_device_add(pd);111if (ret)112goto fail;113114return ret;115fail:116platform_device_put(pd);117out:118return ret;119}120device_initcall(octeon_cf_device_init);121122/* Octeon Random Number Generator. */123static int __init octeon_rng_device_init(void)124{125struct platform_device *pd;126int ret = 0;127128struct resource rng_resources[] = {129{130.flags = IORESOURCE_MEM,131.start = XKPHYS_TO_PHYS(CVMX_RNM_CTL_STATUS),132.end = XKPHYS_TO_PHYS(CVMX_RNM_CTL_STATUS) + 0xf133}, {134.flags = IORESOURCE_MEM,135.start = cvmx_build_io_address(8, 0),136.end = cvmx_build_io_address(8, 0) + 0x7137}138};139140pd = platform_device_alloc("octeon_rng", -1);141if (!pd) {142ret = -ENOMEM;143goto out;144}145146ret = platform_device_add_resources(pd, rng_resources,147ARRAY_SIZE(rng_resources));148if (ret)149goto fail;150151ret = platform_device_add(pd);152if (ret)153goto fail;154155return ret;156fail:157platform_device_put(pd);158159out:160return ret;161}162device_initcall(octeon_rng_device_init);163164static struct i2c_board_info __initdata octeon_i2c_devices[] = {165{166I2C_BOARD_INFO("ds1337", 0x68),167},168};169170static int __init octeon_i2c_devices_init(void)171{172return i2c_register_board_info(0, octeon_i2c_devices,173ARRAY_SIZE(octeon_i2c_devices));174}175arch_initcall(octeon_i2c_devices_init);176177#define OCTEON_I2C_IO_BASE 0x1180000001000ull178#define OCTEON_I2C_IO_UNIT_OFFSET 0x200179180static struct octeon_i2c_data octeon_i2c_data[2];181182static int __init octeon_i2c_device_init(void)183{184struct platform_device *pd;185int ret = 0;186int port, num_ports;187188struct resource i2c_resources[] = {189{190.flags = IORESOURCE_MEM,191}, {192.flags = IORESOURCE_IRQ,193}194};195196if (OCTEON_IS_MODEL(OCTEON_CN56XX) || OCTEON_IS_MODEL(OCTEON_CN52XX))197num_ports = 2;198else199num_ports = 1;200201for (port = 0; port < num_ports; port++) {202octeon_i2c_data[port].sys_freq = octeon_get_io_clock_rate();203/*FIXME: should be examined. At the moment is set for 100Khz */204octeon_i2c_data[port].i2c_freq = 100000;205206pd = platform_device_alloc("i2c-octeon", port);207if (!pd) {208ret = -ENOMEM;209goto out;210}211212pd->dev.platform_data = octeon_i2c_data + port;213214i2c_resources[0].start =215OCTEON_I2C_IO_BASE + (port * OCTEON_I2C_IO_UNIT_OFFSET);216i2c_resources[0].end = i2c_resources[0].start + 0x1f;217switch (port) {218case 0:219i2c_resources[1].start = OCTEON_IRQ_TWSI;220i2c_resources[1].end = OCTEON_IRQ_TWSI;221break;222case 1:223i2c_resources[1].start = OCTEON_IRQ_TWSI2;224i2c_resources[1].end = OCTEON_IRQ_TWSI2;225break;226default:227BUG();228}229230ret = platform_device_add_resources(pd,231i2c_resources,232ARRAY_SIZE(i2c_resources));233if (ret)234goto fail;235236ret = platform_device_add(pd);237if (ret)238goto fail;239}240return ret;241fail:242platform_device_put(pd);243out:244return ret;245}246device_initcall(octeon_i2c_device_init);247248/* Octeon SMI/MDIO interface. */249static int __init octeon_mdiobus_device_init(void)250{251struct platform_device *pd;252int ret = 0;253254if (octeon_is_simulation())255return 0; /* No mdio in the simulator. */256257/* The bus number is the platform_device id. */258pd = platform_device_alloc("mdio-octeon", 0);259if (!pd) {260ret = -ENOMEM;261goto out;262}263264ret = platform_device_add(pd);265if (ret)266goto fail;267268return ret;269fail:270platform_device_put(pd);271272out:273return ret;274275}276device_initcall(octeon_mdiobus_device_init);277278/* Octeon mgmt port Ethernet interface. */279static int __init octeon_mgmt_device_init(void)280{281struct platform_device *pd;282int ret = 0;283int port, num_ports;284285struct resource mgmt_port_resource = {286.flags = IORESOURCE_IRQ,287.start = -1,288.end = -1289};290291if (!OCTEON_IS_MODEL(OCTEON_CN56XX) && !OCTEON_IS_MODEL(OCTEON_CN52XX))292return 0;293294if (OCTEON_IS_MODEL(OCTEON_CN56XX))295num_ports = 1;296else297num_ports = 2;298299for (port = 0; port < num_ports; port++) {300pd = platform_device_alloc("octeon_mgmt", port);301if (!pd) {302ret = -ENOMEM;303goto out;304}305/* No DMA restrictions */306pd->dev.coherent_dma_mask = DMA_BIT_MASK(64);307pd->dev.dma_mask = &pd->dev.coherent_dma_mask;308309switch (port) {310case 0:311mgmt_port_resource.start = OCTEON_IRQ_MII0;312break;313case 1:314mgmt_port_resource.start = OCTEON_IRQ_MII1;315break;316default:317BUG();318}319mgmt_port_resource.end = mgmt_port_resource.start;320321ret = platform_device_add_resources(pd, &mgmt_port_resource, 1);322323if (ret)324goto fail;325326ret = platform_device_add(pd);327if (ret)328goto fail;329}330return ret;331fail:332platform_device_put(pd);333334out:335return ret;336337}338device_initcall(octeon_mgmt_device_init);339340#ifdef CONFIG_USB341342static int __init octeon_ehci_device_init(void)343{344struct platform_device *pd;345int ret = 0;346347struct resource usb_resources[] = {348{349.flags = IORESOURCE_MEM,350}, {351.flags = IORESOURCE_IRQ,352}353};354355/* Only Octeon2 has ehci/ohci */356if (!OCTEON_IS_MODEL(OCTEON_CN63XX))357return 0;358359if (octeon_is_simulation() || usb_disabled())360return 0; /* No USB in the simulator. */361362pd = platform_device_alloc("octeon-ehci", 0);363if (!pd) {364ret = -ENOMEM;365goto out;366}367368usb_resources[0].start = 0x00016F0000000000ULL;369usb_resources[0].end = usb_resources[0].start + 0x100;370371usb_resources[1].start = OCTEON_IRQ_USB0;372usb_resources[1].end = OCTEON_IRQ_USB0;373374ret = platform_device_add_resources(pd, usb_resources,375ARRAY_SIZE(usb_resources));376if (ret)377goto fail;378379ret = platform_device_add(pd);380if (ret)381goto fail;382383return ret;384fail:385platform_device_put(pd);386out:387return ret;388}389device_initcall(octeon_ehci_device_init);390391static int __init octeon_ohci_device_init(void)392{393struct platform_device *pd;394int ret = 0;395396struct resource usb_resources[] = {397{398.flags = IORESOURCE_MEM,399}, {400.flags = IORESOURCE_IRQ,401}402};403404/* Only Octeon2 has ehci/ohci */405if (!OCTEON_IS_MODEL(OCTEON_CN63XX))406return 0;407408if (octeon_is_simulation() || usb_disabled())409return 0; /* No USB in the simulator. */410411pd = platform_device_alloc("octeon-ohci", 0);412if (!pd) {413ret = -ENOMEM;414goto out;415}416417usb_resources[0].start = 0x00016F0000000400ULL;418usb_resources[0].end = usb_resources[0].start + 0x100;419420usb_resources[1].start = OCTEON_IRQ_USB0;421usb_resources[1].end = OCTEON_IRQ_USB0;422423ret = platform_device_add_resources(pd, usb_resources,424ARRAY_SIZE(usb_resources));425if (ret)426goto fail;427428ret = platform_device_add(pd);429if (ret)430goto fail;431432return ret;433fail:434platform_device_put(pd);435out:436return ret;437}438device_initcall(octeon_ohci_device_init);439440#endif /* CONFIG_USB */441442MODULE_AUTHOR("David Daney <[email protected]>");443MODULE_LICENSE("GPL");444MODULE_DESCRIPTION("Platform driver for Octeon SOC");445446447