2015-05-20 17:18:40 -07:00
/*
* Broadcom SATA3 AHCI Controller PHY Driver
*
2016-03-28 10:18:26 +05:30
* Copyright ( C ) 2016 Broadcom
2015-05-20 17:18:40 -07:00
*
* 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 , or ( at your option )
* any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*/
2016-03-28 10:18:27 +05:30
# include <linux/delay.h>
2015-05-20 17:18:40 -07:00
# include <linux/device.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/phy/phy.h>
# include <linux/platform_device.h>
2016-03-28 10:18:27 +05:30
# define SATA_PCB_BANK_OFFSET 0x23c
# define SATA_PCB_REG_OFFSET(ofs) ((ofs) * 4)
2015-05-20 17:18:40 -07:00
# define MAX_PORTS 2
/* Register offset between PHYs in PCB space */
2016-03-28 10:18:27 +05:30
# define SATA_PCB_REG_28NM_SPACE_SIZE 0x1000
2015-11-26 11:56:34 +09:00
2015-11-26 11:56:35 +09:00
/* The older SATA PHY registers duplicated per port registers within the map,
* rather than having a separate map per port .
*/
2016-03-28 10:18:27 +05:30
# define SATA_PCB_REG_40NM_SPACE_SIZE 0x10
/* Register offset between PHYs in PHY control space */
# define SATA_PHY_CTRL_REG_28NM_SPACE_SIZE 0x8
2015-11-26 11:56:35 +09:00
2015-11-26 11:56:34 +09:00
enum brcm_sata_phy_version {
2016-03-28 10:18:27 +05:30
BRCM_SATA_PHY_STB_28NM ,
BRCM_SATA_PHY_STB_40NM ,
BRCM_SATA_PHY_IPROC_NS2 ,
2015-11-26 11:56:34 +09:00
} ;
2015-05-20 17:18:40 -07:00
struct brcm_sata_port {
int portnum ;
struct phy * phy ;
struct brcm_sata_phy * phy_priv ;
bool ssc_en ;
} ;
struct brcm_sata_phy {
struct device * dev ;
void __iomem * phy_base ;
2016-03-28 10:18:27 +05:30
void __iomem * ctrl_base ;
2015-11-26 11:56:34 +09:00
enum brcm_sata_phy_version version ;
2015-05-20 17:18:40 -07:00
struct brcm_sata_port phys [ MAX_PORTS ] ;
} ;
2016-03-28 10:18:27 +05:30
enum sata_phy_regs {
BLOCK0_REG_BANK = 0x000 ,
BLOCK0_XGXSSTATUS = 0x81 ,
BLOCK0_XGXSSTATUS_PLL_LOCK = BIT ( 12 ) ,
BLOCK0_SPARE = 0x8d ,
BLOCK0_SPARE_OOB_CLK_SEL_MASK = 0x3 ,
BLOCK0_SPARE_OOB_CLK_SEL_REFBY2 = 0x1 ,
PLL_REG_BANK_0 = 0x050 ,
2015-05-20 17:18:40 -07:00
PLL_REG_BANK_0_PLLCONTROL_0 = 0x81 ,
2016-03-28 10:18:27 +05:30
PLL1_REG_BANK = 0x060 ,
PLL1_ACTRL2 = 0x82 ,
PLL1_ACTRL3 = 0x83 ,
PLL1_ACTRL4 = 0x84 ,
OOB_REG_BANK = 0x150 ,
OOB_CTRL1 = 0x80 ,
OOB_CTRL1_BURST_MAX_MASK = 0xf ,
OOB_CTRL1_BURST_MAX_SHIFT = 12 ,
OOB_CTRL1_BURST_MIN_MASK = 0xf ,
OOB_CTRL1_BURST_MIN_SHIFT = 8 ,
OOB_CTRL1_WAKE_IDLE_MAX_MASK = 0xf ,
OOB_CTRL1_WAKE_IDLE_MAX_SHIFT = 4 ,
OOB_CTRL1_WAKE_IDLE_MIN_MASK = 0xf ,
OOB_CTRL1_WAKE_IDLE_MIN_SHIFT = 0 ,
OOB_CTRL2 = 0x81 ,
OOB_CTRL2_SEL_ENA_SHIFT = 15 ,
OOB_CTRL2_SEL_ENA_RC_SHIFT = 14 ,
OOB_CTRL2_RESET_IDLE_MAX_MASK = 0x3f ,
OOB_CTRL2_RESET_IDLE_MAX_SHIFT = 8 ,
OOB_CTRL2_BURST_CNT_MASK = 0x3 ,
OOB_CTRL2_BURST_CNT_SHIFT = 6 ,
OOB_CTRL2_RESET_IDLE_MIN_MASK = 0x3f ,
OOB_CTRL2_RESET_IDLE_MIN_SHIFT = 0 ,
2015-05-20 17:18:40 -07:00
TXPMD_REG_BANK = 0x1a0 ,
TXPMD_CONTROL1 = 0x81 ,
TXPMD_CONTROL1_TX_SSC_EN_FRC = BIT ( 0 ) ,
TXPMD_CONTROL1_TX_SSC_EN_FRC_VAL = BIT ( 1 ) ,
TXPMD_TX_FREQ_CTRL_CONTROL1 = 0x82 ,
TXPMD_TX_FREQ_CTRL_CONTROL2 = 0x83 ,
TXPMD_TX_FREQ_CTRL_CONTROL2_FMIN_MASK = 0x3ff ,
TXPMD_TX_FREQ_CTRL_CONTROL3 = 0x84 ,
TXPMD_TX_FREQ_CTRL_CONTROL3_FMAX_MASK = 0x3ff ,
} ;
2016-03-28 10:18:27 +05:30
enum sata_phy_ctrl_regs {
PHY_CTRL_1 = 0x0 ,
PHY_CTRL_1_RESET = BIT ( 0 ) ,
} ;
static inline void __iomem * brcm_sata_pcb_base ( struct brcm_sata_port * port )
2015-05-20 17:18:40 -07:00
{
struct brcm_sata_phy * priv = port - > phy_priv ;
2016-03-28 10:18:27 +05:30
u32 size = 0 ;
switch ( priv - > version ) {
case BRCM_SATA_PHY_STB_28NM :
case BRCM_SATA_PHY_IPROC_NS2 :
size = SATA_PCB_REG_28NM_SPACE_SIZE ;
break ;
case BRCM_SATA_PHY_STB_40NM :
size = SATA_PCB_REG_40NM_SPACE_SIZE ;
break ;
default :
dev_err ( priv - > dev , " invalid phy version \n " ) ;
break ;
} ;
2015-05-20 17:18:40 -07:00
2016-03-28 10:18:27 +05:30
return priv - > phy_base + ( port - > portnum * size ) ;
}
static inline void __iomem * brcm_sata_ctrl_base ( struct brcm_sata_port * port )
{
struct brcm_sata_phy * priv = port - > phy_priv ;
u32 size = 0 ;
switch ( priv - > version ) {
case BRCM_SATA_PHY_IPROC_NS2 :
size = SATA_PHY_CTRL_REG_28NM_SPACE_SIZE ;
break ;
default :
2015-11-26 11:56:35 +09:00
dev_err ( priv - > dev , " invalid phy version \n " ) ;
2016-03-28 10:18:27 +05:30
break ;
} ;
2015-11-26 11:56:34 +09:00
2016-03-28 10:18:27 +05:30
return priv - > ctrl_base + ( port - > portnum * size ) ;
2015-05-20 17:18:40 -07:00
}
2016-03-28 10:18:27 +05:30
static void brcm_sata_phy_wr ( void __iomem * pcb_base , u32 bank ,
u32 ofs , u32 msk , u32 value )
2015-05-20 17:18:40 -07:00
{
u32 tmp ;
2016-03-28 10:18:27 +05:30
writel ( bank , pcb_base + SATA_PCB_BANK_OFFSET ) ;
tmp = readl ( pcb_base + SATA_PCB_REG_OFFSET ( ofs ) ) ;
2015-05-20 17:18:40 -07:00
tmp = ( tmp & msk ) | value ;
2016-03-28 10:18:27 +05:30
writel ( tmp , pcb_base + SATA_PCB_REG_OFFSET ( ofs ) ) ;
}
static u32 brcm_sata_phy_rd ( void __iomem * pcb_base , u32 bank , u32 ofs )
{
writel ( bank , pcb_base + SATA_PCB_BANK_OFFSET ) ;
return readl ( pcb_base + SATA_PCB_REG_OFFSET ( ofs ) ) ;
2015-05-20 17:18:40 -07:00
}
/* These defaults were characterized by H/W group */
2016-03-28 10:18:27 +05:30
# define STB_FMIN_VAL_DEFAULT 0x3df
# define STB_FMAX_VAL_DEFAULT 0x3df
# define STB_FMAX_VAL_SSC 0x83
2015-05-20 17:18:40 -07:00
2016-03-28 10:18:27 +05:30
static int brcm_stb_sata_init ( struct brcm_sata_port * port )
2015-05-20 17:18:40 -07:00
{
2016-03-28 10:18:27 +05:30
void __iomem * base = brcm_sata_pcb_base ( port ) ;
2015-05-20 17:18:40 -07:00
struct brcm_sata_phy * priv = port - > phy_priv ;
u32 tmp ;
/* override the TX spread spectrum setting */
tmp = TXPMD_CONTROL1_TX_SSC_EN_FRC_VAL | TXPMD_CONTROL1_TX_SSC_EN_FRC ;
2016-03-28 10:18:27 +05:30
brcm_sata_phy_wr ( base , TXPMD_REG_BANK , TXPMD_CONTROL1 , ~ tmp , tmp ) ;
2015-05-20 17:18:40 -07:00
/* set fixed min freq */
2016-03-28 10:18:27 +05:30
brcm_sata_phy_wr ( base , TXPMD_REG_BANK , TXPMD_TX_FREQ_CTRL_CONTROL2 ,
~ TXPMD_TX_FREQ_CTRL_CONTROL2_FMIN_MASK ,
STB_FMIN_VAL_DEFAULT ) ;
2015-05-20 17:18:40 -07:00
/* set fixed max freq depending on SSC config */
if ( port - > ssc_en ) {
2016-03-28 10:18:27 +05:30
dev_info ( priv - > dev , " enabling SSC on port%d \n " , port - > portnum ) ;
tmp = STB_FMAX_VAL_SSC ;
2015-05-20 17:18:40 -07:00
} else {
2016-03-28 10:18:27 +05:30
tmp = STB_FMAX_VAL_DEFAULT ;
2015-05-20 17:18:40 -07:00
}
2016-03-28 10:18:27 +05:30
brcm_sata_phy_wr ( base , TXPMD_REG_BANK , TXPMD_TX_FREQ_CTRL_CONTROL3 ,
2015-05-20 17:18:40 -07:00
~ TXPMD_TX_FREQ_CTRL_CONTROL3_FMAX_MASK , tmp ) ;
2016-03-28 10:18:27 +05:30
return 0 ;
}
/* NS2 SATA PLL1 defaults were characterized by H/W group */
# define NS2_PLL1_ACTRL2_MAGIC 0x1df8
# define NS2_PLL1_ACTRL3_MAGIC 0x2b00
# define NS2_PLL1_ACTRL4_MAGIC 0x8824
static int brcm_ns2_sata_init ( struct brcm_sata_port * port )
{
int try ;
unsigned int val ;
void __iomem * base = brcm_sata_pcb_base ( port ) ;
void __iomem * ctrl_base = brcm_sata_ctrl_base ( port ) ;
struct device * dev = port - > phy_priv - > dev ;
/* Configure OOB control */
val = 0x0 ;
val | = ( 0xc < < OOB_CTRL1_BURST_MAX_SHIFT ) ;
val | = ( 0x4 < < OOB_CTRL1_BURST_MIN_SHIFT ) ;
val | = ( 0x9 < < OOB_CTRL1_WAKE_IDLE_MAX_SHIFT ) ;
val | = ( 0x3 < < OOB_CTRL1_WAKE_IDLE_MIN_SHIFT ) ;
brcm_sata_phy_wr ( base , OOB_REG_BANK , OOB_CTRL1 , 0x0 , val ) ;
val = 0x0 ;
val | = ( 0x1b < < OOB_CTRL2_RESET_IDLE_MAX_SHIFT ) ;
val | = ( 0x2 < < OOB_CTRL2_BURST_CNT_SHIFT ) ;
val | = ( 0x9 < < OOB_CTRL2_RESET_IDLE_MIN_SHIFT ) ;
brcm_sata_phy_wr ( base , OOB_REG_BANK , OOB_CTRL2 , 0x0 , val ) ;
/* Configure PHY PLL register bank 1 */
val = NS2_PLL1_ACTRL2_MAGIC ;
brcm_sata_phy_wr ( base , PLL1_REG_BANK , PLL1_ACTRL2 , 0x0 , val ) ;
val = NS2_PLL1_ACTRL3_MAGIC ;
brcm_sata_phy_wr ( base , PLL1_REG_BANK , PLL1_ACTRL3 , 0x0 , val ) ;
val = NS2_PLL1_ACTRL4_MAGIC ;
brcm_sata_phy_wr ( base , PLL1_REG_BANK , PLL1_ACTRL4 , 0x0 , val ) ;
/* Configure PHY BLOCK0 register bank */
/* Set oob_clk_sel to refclk/2 */
brcm_sata_phy_wr ( base , BLOCK0_REG_BANK , BLOCK0_SPARE ,
~ BLOCK0_SPARE_OOB_CLK_SEL_MASK ,
BLOCK0_SPARE_OOB_CLK_SEL_REFBY2 ) ;
/* Strobe PHY reset using PHY control register */
writel ( PHY_CTRL_1_RESET , ctrl_base + PHY_CTRL_1 ) ;
mdelay ( 1 ) ;
writel ( 0x0 , ctrl_base + PHY_CTRL_1 ) ;
mdelay ( 1 ) ;
/* Wait for PHY PLL lock by polling pll_lock bit */
try = 50 ;
while ( try ) {
val = brcm_sata_phy_rd ( base , BLOCK0_REG_BANK ,
BLOCK0_XGXSSTATUS ) ;
if ( val & BLOCK0_XGXSSTATUS_PLL_LOCK )
break ;
msleep ( 20 ) ;
try - - ;
}
if ( ! try ) {
/* PLL did not lock; give up */
dev_err ( dev , " port%d PLL did not lock \n " , port - > portnum ) ;
return - ETIMEDOUT ;
}
dev_dbg ( dev , " port%d initialized \n " , port - > portnum ) ;
return 0 ;
2015-05-20 17:18:40 -07:00
}
static int brcm_sata_phy_init ( struct phy * phy )
{
2016-03-28 10:18:27 +05:30
int rc ;
2015-05-20 17:18:40 -07:00
struct brcm_sata_port * port = phy_get_drvdata ( phy ) ;
2016-03-28 10:18:27 +05:30
switch ( port - > phy_priv - > version ) {
case BRCM_SATA_PHY_STB_28NM :
case BRCM_SATA_PHY_STB_40NM :
rc = brcm_stb_sata_init ( port ) ;
break ;
case BRCM_SATA_PHY_IPROC_NS2 :
rc = brcm_ns2_sata_init ( port ) ;
break ;
default :
rc = - ENODEV ;
} ;
2015-05-20 17:18:40 -07:00
return 0 ;
}
2015-11-26 11:56:35 +09:00
static const struct phy_ops phy_ops = {
2015-05-20 17:18:40 -07:00
. init = brcm_sata_phy_init ,
. owner = THIS_MODULE ,
} ;
static const struct of_device_id brcm_sata_phy_of_match [ ] = {
2015-11-26 11:56:34 +09:00
{ . compatible = " brcm,bcm7445-sata-phy " ,
2016-03-28 10:18:27 +05:30
. data = ( void * ) BRCM_SATA_PHY_STB_28NM } ,
2015-11-26 11:56:35 +09:00
{ . compatible = " brcm,bcm7425-sata-phy " ,
2016-03-28 10:18:27 +05:30
. data = ( void * ) BRCM_SATA_PHY_STB_40NM } ,
{ . compatible = " brcm,iproc-ns2-sata-phy " ,
. data = ( void * ) BRCM_SATA_PHY_IPROC_NS2 } ,
2015-05-20 17:18:40 -07:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , brcm_sata_phy_of_match ) ;
static int brcm_sata_phy_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct device_node * dn = dev - > of_node , * child ;
2015-11-26 11:56:34 +09:00
const struct of_device_id * of_id ;
2015-05-20 17:18:40 -07:00
struct brcm_sata_phy * priv ;
struct resource * res ;
struct phy_provider * provider ;
2015-11-16 12:33:14 +01:00
int ret , count = 0 ;
2015-05-20 17:18:40 -07:00
if ( of_get_child_count ( dn ) = = 0 )
return - ENODEV ;
priv = devm_kzalloc ( dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
dev_set_drvdata ( dev , priv ) ;
priv - > dev = dev ;
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " phy " ) ;
priv - > phy_base = devm_ioremap_resource ( dev , res ) ;
if ( IS_ERR ( priv - > phy_base ) )
return PTR_ERR ( priv - > phy_base ) ;
2015-11-26 11:56:34 +09:00
of_id = of_match_node ( brcm_sata_phy_of_match , dn ) ;
if ( of_id )
priv - > version = ( enum brcm_sata_phy_version ) of_id - > data ;
else
2016-03-28 10:18:27 +05:30
priv - > version = BRCM_SATA_PHY_STB_28NM ;
if ( priv - > version = = BRCM_SATA_PHY_IPROC_NS2 ) {
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM ,
" phy-ctrl " ) ;
priv - > ctrl_base = devm_ioremap_resource ( dev , res ) ;
if ( IS_ERR ( priv - > ctrl_base ) )
return PTR_ERR ( priv - > ctrl_base ) ;
}
2015-11-26 11:56:34 +09:00
2015-05-20 17:18:40 -07:00
for_each_available_child_of_node ( dn , child ) {
unsigned int id ;
struct brcm_sata_port * port ;
if ( of_property_read_u32 ( child , " reg " , & id ) ) {
dev_err ( dev , " missing reg property in node %s \n " ,
child - > name ) ;
2015-11-16 12:33:14 +01:00
ret = - EINVAL ;
goto put_child ;
2015-05-20 17:18:40 -07:00
}
if ( id > = MAX_PORTS ) {
dev_err ( dev , " invalid reg: %u \n " , id ) ;
2015-11-16 12:33:14 +01:00
ret = - EINVAL ;
goto put_child ;
2015-05-20 17:18:40 -07:00
}
if ( priv - > phys [ id ] . phy ) {
dev_err ( dev , " already registered port %u \n " , id ) ;
2015-11-16 12:33:14 +01:00
ret = - EINVAL ;
goto put_child ;
2015-05-20 17:18:40 -07:00
}
port = & priv - > phys [ id ] ;
port - > portnum = id ;
port - > phy_priv = priv ;
2015-11-26 11:56:35 +09:00
port - > phy = devm_phy_create ( dev , child , & phy_ops ) ;
2015-05-20 17:18:40 -07:00
port - > ssc_en = of_property_read_bool ( child , " brcm,enable-ssc " ) ;
if ( IS_ERR ( port - > phy ) ) {
dev_err ( dev , " failed to create PHY \n " ) ;
2015-11-16 12:33:14 +01:00
ret = PTR_ERR ( port - > phy ) ;
goto put_child ;
2015-05-20 17:18:40 -07:00
}
phy_set_drvdata ( port - > phy , port ) ;
count + + ;
}
provider = devm_of_phy_provider_register ( dev , of_phy_simple_xlate ) ;
if ( IS_ERR ( provider ) ) {
dev_err ( dev , " could not register PHY provider \n " ) ;
return PTR_ERR ( provider ) ;
}
dev_info ( dev , " registered %d port(s) \n " , count ) ;
return 0 ;
2015-11-16 12:33:14 +01:00
put_child :
of_node_put ( child ) ;
return ret ;
2015-05-20 17:18:40 -07:00
}
static struct platform_driver brcm_sata_phy_driver = {
. probe = brcm_sata_phy_probe ,
. driver = {
. of_match_table = brcm_sata_phy_of_match ,
2016-03-28 10:18:26 +05:30
. name = " brcm-sata-phy " ,
2015-05-20 17:18:40 -07:00
}
} ;
module_platform_driver ( brcm_sata_phy_driver ) ;
2016-03-28 10:18:26 +05:30
MODULE_DESCRIPTION ( " Broadcom SATA PHY driver " ) ;
2015-05-20 17:18:40 -07:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Marc Carino " ) ;
MODULE_AUTHOR ( " Brian Norris " ) ;
2016-03-28 10:18:26 +05:30
MODULE_ALIAS ( " platform:phy-brcm-sata " ) ;