static int __init imx_pcie_drv_init ( void ){return platform_driver_register (& imx_pcie_pltfm_driver );}static void __exit imx_pcie_drv_exit ( void ){platform_driver_unregister (& imx_pcie_pltfm_driver );}module_init ( imx_pcie_drv_init );module_exit ( imx_pcie_drv_exit );
static struct platform_driver imx_pcie_pltfm_driver = {. driver = {. name = "imx-pcie" ,. owner = THIS_MODULE ,},. probe = imx_pcie_pltfm_probe ,};
/* Add PCIe RC interface support */imx6q_add_pcie (& mx6_sabresd_pcie_data );
extern const struct imx_pcie_data imx6q_pcie_data __initconst ;#define imx6q_add_pcie ( pdata ) imx_add_pcie (& imx6q_pcie_data , pdata )
struct platform_device * __init imx_add_pcie (const struct imx_pcie_data * data ,const struct imx_pcie_platform_data * pdata ){struct resource res [] = {{. start = data -> iobase ,. end = data -> iobase + data -> iosize - 1 ,. flags = IORESOURCE_MEM ,}, {. start = data -> irq ,. end = data -> irq ,. flags = IORESOURCE_IRQ ,},};if (! fuse_dev_is_available ( MXC_DEV_PCIE ))return ERR_PTR (- ENODEV );return imx_add_platform_device ( "imx-pcie" , - 1 ,res , ARRAY_SIZE ( res ),pdata , sizeof (* pdata ));}
/*** struct imx_pcie_platform_data - optional platform data for pcie on i.MX** @pcie_pwr_en: used for enable/disable pcie power (-EINVAL if unused)* @pcie_rst: used for reset pcie ep (-EINVAL if unused)* @pcie_wake_up: used for wake up (-EINVAL if unused)* @pcie_dis: used for disable pcie ep (-EINVAL if unused)*/struct imx_pcie_platform_data {unsigned int pcie_pwr_en ;unsigned int pcie_rst ;unsigned int pcie_wake_up ;unsigned int pcie_dis ;unsigned int type_ep ; /* 1 EP, 0 RC */};#endif /* __ASM_ARCH_IMX_PCIE_H */
/** initialize __mach_desc_MX6Q_SABRESD data structure.*/MACHINE_START ( MX6Q_SABRESD , "Freescale i.MX 6Quad/DualLite/Solo Sabre-SD Board" )/* Maintainer: Freescale Semiconductor, Inc. */. boot_params = MX6_PHYS_OFFSET + 0x100 ,. fixup = fixup_mxc_board ,. map_io = mx6_map_io ,. init_irq = mx6_init_irq ,. init_machine = mx6_sabresd_board_init ,. timer = & mx6_sabresd_timer ,. reserve = mx6q_sabresd_reserve ,MACHINE_END
mem = platform_get_resource ( pdev , IORESOURCE_MEM , 0 );if (! mem ) {dev_err ( dev , "no mmio space\n" );return - EINVAL ;}
/* Added for PCI abort handling */hook_fault_code ( 16 + 6 , imx6q_pcie_abort_handler , SIGBUS , 0 ,"imprecise external abort" );
base = ioremap_nocache ( PCIE_ARB_END_ADDR - SZ_1M + 1 , SZ_1M - SZ_16K );if (! base ) {pr_err ( "error with ioremap in function %s\n" , __func__ );ret = PTR_ERR ( base );return ret ;}
#define PCIE_ARB_BASE_ADDR 0x01000000#define PCIE_ARB_END_ADDR 0x01FFFFFF
dbi_base = devm_ioremap ( dev , mem -> start , resource_size ( mem ));if (! dbi_base ) {dev_err ( dev , "can't map %pR\n" , mem );ret = PTR_ERR ( dbi_base );goto err_base ;}
/* FIXME the field name should be aligned to RM */imx_pcie_clrset ( IOMUXC_GPR12_APP_LTSSM_ENABLE , 0 << 10 , IOMUXC_GPR12 );/* configure constant input signal to the pcie ctrl and phy */if ( pdata -> type_ep & 1 )/* EP */imx_pcie_clrset ( IOMUXC_GPR12_DEVICE_TYPE ,PCI_EXP_TYPE_ENDPOINT << 12 , IOMUXC_GPR12 );else/* RC */imx_pcie_clrset ( IOMUXC_GPR12_DEVICE_TYPE ,PCI_EXP_TYPE_ROOT_PORT << 12 , IOMUXC_GPR12 );imx_pcie_clrset ( IOMUXC_GPR12_LOS_LEVEL , 9 << 4 , IOMUXC_GPR12 );imx_pcie_clrset ( IOMUXC_GPR8_TX_DEEMPH_GEN1 , 0 << 0 , IOMUXC_GPR8 );imx_pcie_clrset ( IOMUXC_GPR8_TX_DEEMPH_GEN2_3P5DB , 0 << 6 , IOMUXC_GPR8 );imx_pcie_clrset ( IOMUXC_GPR8_TX_DEEMPH_GEN2_6DB , 20 << 12 , IOMUXC_GPR8 );imx_pcie_clrset ( IOMUXC_GPR8_TX_SWING_FULL , 127 << 18 , IOMUXC_GPR8 );imx_pcie_clrset ( IOMUXC_GPR8_TX_SWING_LOW , 127 << 25 , IOMUXC_GPR8 );
/* Enable the pwr, clks and so on */imx_pcie_enable_controller ( dev );
gpio_request ( pdata -> pcie_pwr_en , "PCIE POWER_EN" );/* activate PCIE_PWR_EN */gpio_direction_output ( pdata -> pcie_pwr_en , 1 );imx_pcie_clrset ( IOMUXC_GPR1_TEST_POWERDOWN , 0 << 18 , IOMUXC_GPR1 );
/* enable the clks */if ( pdata -> type_ep ) {pcie_clk = clk_get ( NULL , "pcie_ep_clk" );if ( IS_ERR ( pcie_clk ))pr_err ( "no pcie_ep clock.\n" );if ( clk_enable ( pcie_clk )) {pr_err ( "can't enable pcie_ep clock.\n" );clk_put ( pcie_clk );}} else {pcie_clk = clk_get ( NULL , "pcie_clk" );if ( IS_ERR ( pcie_clk ))pr_err ( "no pcie clock.\n" );if ( clk_enable ( pcie_clk )) {pr_err ( "can't enable pcie clock.\n" );clk_put ( pcie_clk );}}
imx_pcie_clrset ( IOMUXC_GPR1_PCIE_REF_CLK_EN , 1 << 16 , IOMUXC_GPR1 );
/* start link up */imx_pcie_clrset ( IOMUXC_GPR12_APP_LTSSM_ENABLE , 1 << 10 , IOMUXC_GPR12 );
/* add the pcie port */add_pcie_port ( base , dbi_base , pdata );
if ( imx_pcie_link_up ( dbi_base )) {struct imx_pcie_port * pp = & imx_pcie_port [ num_pcie_ports ++];pr_info ( "IMX PCIe port: link up.\n" );pp -> index = 0 ;pp -> root_bus_nr = - 1 ;pp -> base = base ;pp -> dbi_base = dbi_base ;spin_lock_init (& pp -> conf_lock );memset ( pp -> res , 0 , sizeof ( pp -> res ));}
else {pr_info ( "IMX PCIe port: link down!\n" );/* Release the clocks, and disable the power */pcie_clk = clk_get ( NULL , "pcie_clk" );if ( IS_ERR ( pcie_clk ))pr_err ( "no pcie clock.\n" );clk_disable ( pcie_clk );clk_put ( pcie_clk );imx_pcie_clrset ( IOMUXC_GPR1_PCIE_REF_CLK_EN , 0 << 16 ,IOMUXC_GPR1 );/* Disable PCIE power */gpio_request ( pdata -> pcie_pwr_en , "PCIE POWER_EN" );/* activate PCIE_PWR_EN */gpio_direction_output ( pdata -> pcie_pwr_en , 0 );imx_pcie_clrset ( IOMUXC_GPR1_TEST_POWERDOWN , 1 << 18 ,IOMUXC_GPR1 );}
pci_common_init(&imx_pci);