2008-04-27 15:55:59 +04:00
/****************************************************************************
* Driver for Solarflare Solarstorm network controllers and boards
2008-12-13 09:00:49 +03:00
* Copyright 2007 - 2008 Solarflare Communications Inc .
2008-04-27 15:55:59 +04:00
*
* 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 , incorporated herein by reference .
*/
/*****************************************************************************
2008-12-13 09:00:49 +03:00
* Support for the SFE4001 and SFN4111T NICs .
*
* The SFE4001 does not power - up fully at reset due to its high power
* consumption . We control its power via a PCA9539 I / O expander .
* Both boards have a MAX6647 temperature monitor which we expose to
* the lm90 driver .
*
* This also provides minimal support for reflashing the PHY , which is
* initiated by resetting it with the FLASH_CFG_1 pin pulled down .
* On SFE4001 rev A2 and later this is connected to the 3 V3X output of
* the IO - expander ; on the SFN4111T it is connected to Falcon ' s GPIO3 .
* We represent reflash mode as PHY_MODE_SPECIAL and make it mutually
* exclusive with the network device being open .
2008-04-27 15:55:59 +04:00
*/
2008-12-13 09:00:49 +03:00
2008-04-27 15:55:59 +04:00
# include <linux/delay.h>
2008-09-01 15:48:17 +04:00
# include "net_driver.h"
2008-04-27 15:55:59 +04:00
# include "efx.h"
# include "phy.h"
# include "boards.h"
# include "falcon.h"
# include "falcon_hwdefs.h"
2008-09-01 15:48:41 +04:00
# include "falcon_io.h"
2008-04-27 15:55:59 +04:00
# include "mac.h"
2008-11-04 23:34:56 +03:00
# include "workarounds.h"
2008-04-27 15:55:59 +04:00
/**************************************************************************
*
* I2C IO Expander device
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define PCA9539 0x74
# define P0_IN 0x00
# define P0_OUT 0x02
# define P0_INVERT 0x04
# define P0_CONFIG 0x06
# define P0_EN_1V0X_LBN 0
# define P0_EN_1V0X_WIDTH 1
# define P0_EN_1V2_LBN 1
# define P0_EN_1V2_WIDTH 1
# define P0_EN_2V5_LBN 2
# define P0_EN_2V5_WIDTH 1
# define P0_EN_3V3X_LBN 3
# define P0_EN_3V3X_WIDTH 1
# define P0_EN_5V_LBN 4
# define P0_EN_5V_WIDTH 1
# define P0_SHORTEN_JTAG_LBN 5
# define P0_SHORTEN_JTAG_WIDTH 1
# define P0_X_TRST_LBN 6
# define P0_X_TRST_WIDTH 1
# define P0_DSP_RESET_LBN 7
# define P0_DSP_RESET_WIDTH 1
# define P1_IN 0x01
# define P1_OUT 0x03
# define P1_INVERT 0x05
# define P1_CONFIG 0x07
# define P1_AFE_PWD_LBN 0
# define P1_AFE_PWD_WIDTH 1
# define P1_DSP_PWD25_LBN 1
# define P1_DSP_PWD25_WIDTH 1
# define P1_RESERVED_LBN 2
# define P1_RESERVED_WIDTH 2
# define P1_SPARE_LBN 4
# define P1_SPARE_WIDTH 4
2008-11-04 23:34:56 +03:00
/* Temperature Sensor */
# define MAX664X_REG_RSL 0x02
# define MAX664X_REG_WLHO 0x0B
2008-04-27 15:55:59 +04:00
2008-05-31 01:27:04 +04:00
static void sfe4001_poweroff ( struct efx_nic * efx )
2008-04-27 15:55:59 +04:00
{
2008-05-31 01:27:04 +04:00
struct i2c_client * ioexp_client = efx - > board_info . ioexp_client ;
struct i2c_client * hwmon_client = efx - > board_info . hwmon_client ;
2008-04-27 15:55:59 +04:00
2008-05-31 01:27:04 +04:00
/* Turn off all power rails and disable outputs */
i2c_smbus_write_byte_data ( ioexp_client , P0_OUT , 0xff ) ;
i2c_smbus_write_byte_data ( ioexp_client , P1_CONFIG , 0xff ) ;
i2c_smbus_write_byte_data ( ioexp_client , P0_CONFIG , 0xff ) ;
2008-04-27 15:55:59 +04:00
2008-05-31 01:27:04 +04:00
/* Clear any over-temperature alert */
2008-11-04 23:34:56 +03:00
i2c_smbus_read_byte_data ( hwmon_client , MAX664X_REG_RSL ) ;
2008-05-31 01:27:04 +04:00
}
2008-04-27 15:55:59 +04:00
2008-09-01 15:48:17 +04:00
static int sfe4001_poweron ( struct efx_nic * efx )
2008-05-31 01:27:04 +04:00
{
2008-09-01 15:48:17 +04:00
struct i2c_client * hwmon_client = efx - > board_info . hwmon_client ;
struct i2c_client * ioexp_client = efx - > board_info . ioexp_client ;
2008-09-01 15:45:48 +04:00
unsigned int i , j ;
2008-04-27 15:55:59 +04:00
int rc ;
2008-05-31 01:27:04 +04:00
u8 out ;
2008-04-27 15:55:59 +04:00
/* Clear any previous over-temperature alert */
2008-11-04 23:34:56 +03:00
rc = i2c_smbus_read_byte_data ( hwmon_client , MAX664X_REG_RSL ) ;
2008-05-31 01:27:04 +04:00
if ( rc < 0 )
2008-09-01 15:48:17 +04:00
return rc ;
2008-04-27 15:55:59 +04:00
/* Enable port 0 and port 1 outputs on IO expander */
2008-05-31 01:27:04 +04:00
rc = i2c_smbus_write_byte_data ( ioexp_client , P0_CONFIG , 0x00 ) ;
2008-04-27 15:55:59 +04:00
if ( rc )
2008-09-01 15:48:17 +04:00
return rc ;
2008-05-31 01:27:04 +04:00
rc = i2c_smbus_write_byte_data ( ioexp_client , P1_CONFIG ,
0xff & ~ ( 1 < < P1_SPARE_LBN ) ) ;
2008-04-27 15:55:59 +04:00
if ( rc )
2008-05-31 01:27:04 +04:00
goto fail_on ;
2008-04-27 15:55:59 +04:00
2008-09-01 15:45:48 +04:00
/* If PHY power is on, turn it all off and wait 1 second to
* ensure a full reset .
*/
rc = i2c_smbus_read_byte_data ( ioexp_client , P0_OUT ) ;
if ( rc < 0 )
goto fail_on ;
2008-04-27 15:55:59 +04:00
out = 0xff & ~ ( ( 0 < < P0_EN_1V2_LBN ) | ( 0 < < P0_EN_2V5_LBN ) |
( 0 < < P0_EN_3V3X_LBN ) | ( 0 < < P0_EN_5V_LBN ) |
( 0 < < P0_EN_1V0X_LBN ) ) ;
2008-09-01 15:45:48 +04:00
if ( rc ! = out ) {
EFX_INFO ( efx , " power-cycling PHY \n " ) ;
rc = i2c_smbus_write_byte_data ( ioexp_client , P0_OUT , out ) ;
if ( rc )
goto fail_on ;
schedule_timeout_uninterruptible ( HZ ) ;
}
2008-04-27 15:55:59 +04:00
2008-09-01 15:45:48 +04:00
for ( i = 0 ; i < 20 ; + + i ) {
2008-04-27 15:55:59 +04:00
/* Turn on 1.2V, 2.5V, 3.3V and 5V power rails */
out = 0xff & ~ ( ( 1 < < P0_EN_1V2_LBN ) | ( 1 < < P0_EN_2V5_LBN ) |
( 1 < < P0_EN_3V3X_LBN ) | ( 1 < < P0_EN_5V_LBN ) |
( 1 < < P0_X_TRST_LBN ) ) ;
2008-09-01 15:48:17 +04:00
if ( efx - > phy_mode & PHY_MODE_SPECIAL )
2008-05-07 15:55:13 +04:00
out | = 1 < < P0_EN_3V3X_LBN ;
2008-04-27 15:55:59 +04:00
2008-05-31 01:27:04 +04:00
rc = i2c_smbus_write_byte_data ( ioexp_client , P0_OUT , out ) ;
2008-04-27 15:55:59 +04:00
if ( rc )
2008-05-31 01:27:04 +04:00
goto fail_on ;
2008-04-27 15:55:59 +04:00
msleep ( 10 ) ;
/* Turn on 1V power rail */
out & = ~ ( 1 < < P0_EN_1V0X_LBN ) ;
2008-05-31 01:27:04 +04:00
rc = i2c_smbus_write_byte_data ( ioexp_client , P0_OUT , out ) ;
2008-04-27 15:55:59 +04:00
if ( rc )
2008-05-31 01:27:04 +04:00
goto fail_on ;
2008-04-27 15:55:59 +04:00
2008-09-01 15:45:48 +04:00
EFX_INFO ( efx , " waiting for DSP boot (attempt %d)... \n " , i ) ;
2008-04-27 15:55:59 +04:00
2008-09-01 15:45:48 +04:00
/* In flash config mode, DSP does not turn on AFE, so
* just wait 1 second .
*/
2008-09-01 15:48:17 +04:00
if ( efx - > phy_mode & PHY_MODE_SPECIAL ) {
2008-09-01 15:45:48 +04:00
schedule_timeout_uninterruptible ( HZ ) ;
2008-09-01 15:48:17 +04:00
return 0 ;
2008-09-01 15:45:48 +04:00
}
for ( j = 0 ; j < 10 ; + + j ) {
msleep ( 100 ) ;
/* Check DSP has asserted AFE power line */
rc = i2c_smbus_read_byte_data ( ioexp_client , P1_IN ) ;
if ( rc < 0 )
goto fail_on ;
if ( rc & ( 1 < < P1_AFE_PWD_LBN ) )
2008-09-01 15:48:17 +04:00
return 0 ;
2008-09-01 15:45:48 +04:00
}
}
2008-04-27 15:55:59 +04:00
2008-09-01 15:45:48 +04:00
EFX_INFO ( efx , " timed out waiting for DSP boot \n " ) ;
2008-04-27 15:55:59 +04:00
rc = - ETIMEDOUT ;
2008-09-01 15:48:17 +04:00
fail_on :
sfe4001_poweroff ( efx ) ;
return rc ;
}
2008-12-13 09:00:49 +03:00
static int sfn4111t_reset ( struct efx_nic * efx )
2008-11-04 23:34:56 +03:00
{
2008-12-13 09:00:49 +03:00
efx_oword_t reg ;
2008-11-04 23:34:56 +03:00
2008-12-13 09:00:49 +03:00
/* GPIO pins are also used for I2C, so block that temporarily */
mutex_lock ( & efx - > i2c_adap . bus_lock ) ;
2008-11-04 23:34:56 +03:00
2008-12-13 09:00:49 +03:00
falcon_read ( efx , & reg , GPIO_CTL_REG_KER ) ;
EFX_SET_OWORD_FIELD ( reg , GPIO2_OEN , true ) ;
EFX_SET_OWORD_FIELD ( reg , GPIO2_OUT , false ) ;
falcon_write ( efx , & reg , GPIO_CTL_REG_KER ) ;
msleep ( 1000 ) ;
EFX_SET_OWORD_FIELD ( reg , GPIO2_OUT , true ) ;
EFX_SET_OWORD_FIELD ( reg , GPIO3_OEN , true ) ;
EFX_SET_OWORD_FIELD ( reg , GPIO3_OUT ,
! ( efx - > phy_mode & PHY_MODE_SPECIAL ) ) ;
falcon_write ( efx , & reg , GPIO_CTL_REG_KER ) ;
2008-11-04 23:34:56 +03:00
2008-12-13 09:00:49 +03:00
mutex_unlock ( & efx - > i2c_adap . bus_lock ) ;
2008-11-04 23:34:56 +03:00
2008-12-13 09:00:49 +03:00
ssleep ( 1 ) ;
return 0 ;
2008-11-04 23:34:56 +03:00
}
2008-09-01 15:48:17 +04:00
static ssize_t show_phy_flash_cfg ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct efx_nic * efx = pci_get_drvdata ( to_pci_dev ( dev ) ) ;
return sprintf ( buf , " %d \n " , ! ! ( efx - > phy_mode & PHY_MODE_SPECIAL ) ) ;
}
static ssize_t set_phy_flash_cfg ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
{
struct efx_nic * efx = pci_get_drvdata ( to_pci_dev ( dev ) ) ;
enum efx_phy_mode old_mode , new_mode ;
int err ;
rtnl_lock ( ) ;
old_mode = efx - > phy_mode ;
if ( count = = 0 | | * buf = = ' 0 ' )
new_mode = old_mode & ~ PHY_MODE_SPECIAL ;
else
new_mode = PHY_MODE_SPECIAL ;
if ( old_mode = = new_mode ) {
err = 0 ;
} else if ( efx - > state ! = STATE_RUNNING | | netif_running ( efx - > net_dev ) ) {
err = - EBUSY ;
} else {
efx - > phy_mode = new_mode ;
2008-12-13 09:00:49 +03:00
if ( efx - > board_info . type = = EFX_BOARD_SFE4001 )
err = sfe4001_poweron ( efx ) ;
else
err = sfn4111t_reset ( efx ) ;
2008-09-01 15:48:17 +04:00
efx_reconfigure_port ( efx ) ;
}
rtnl_unlock ( ) ;
return err ? err : count ;
}
static DEVICE_ATTR ( phy_flash_cfg , 0644 , show_phy_flash_cfg , set_phy_flash_cfg ) ;
static void sfe4001_fini ( struct efx_nic * efx )
{
EFX_INFO ( efx , " %s \n " , __func__ ) ;
device_remove_file ( & efx - > pci_dev - > dev , & dev_attr_phy_flash_cfg ) ;
sfe4001_poweroff ( efx ) ;
i2c_unregister_device ( efx - > board_info . ioexp_client ) ;
i2c_unregister_device ( efx - > board_info . hwmon_client ) ;
}
2008-12-13 09:00:49 +03:00
static int sfe4001_check_hw ( struct efx_nic * efx )
{
s32 status ;
/* If XAUI link is up then do not monitor */
if ( EFX_WORKAROUND_7884 ( efx ) & & efx - > mac_up )
return 0 ;
/* Check the powered status of the PHY. Lack of power implies that
* the MAX6647 has shut down power to it , probably due to a temp .
* alarm . Reading the power status rather than the MAX6647 status
* directly because the later is read - to - clear and would thus
* start to power up the PHY again when polled , causing us to blip
* the power undesirably .
* We know we can read from the IO expander because we did
* it during power - on . Assume failure now is bad news . */
status = i2c_smbus_read_byte_data ( efx - > board_info . ioexp_client , P1_IN ) ;
if ( status > = 0 & &
( status & ( ( 1 < < P1_AFE_PWD_LBN ) | ( 1 < < P1_DSP_PWD25_LBN ) ) ) ! = 0 )
return 0 ;
/* Use board power control, not PHY power control */
sfe4001_poweroff ( efx ) ;
efx - > phy_mode = PHY_MODE_OFF ;
return ( status < 0 ) ? - EIO : - ERANGE ;
}
2008-11-04 23:34:56 +03:00
static struct i2c_board_info sfe4001_hwmon_info = {
I2C_BOARD_INFO ( " max6647 " , 0x4e ) ,
. irq = - 1 ,
} ;
2008-09-01 15:48:17 +04:00
/* This board uses an I2C expander to provider power to the PHY, which needs to
* be turned on before the PHY can be used .
* Context : Process context , rtnl lock held
*/
int sfe4001_init ( struct efx_nic * efx )
{
int rc ;
2008-11-04 23:34:56 +03:00
# if defined(CONFIG_SENSORS_LM90) || defined(CONFIG_SENSORS_LM90_MODULE)
efx - > board_info . hwmon_client =
i2c_new_device ( & efx - > i2c_adap , & sfe4001_hwmon_info ) ;
# else
efx - > board_info . hwmon_client =
i2c_new_dummy ( & efx - > i2c_adap , sfe4001_hwmon_info . addr ) ;
# endif
if ( ! efx - > board_info . hwmon_client )
2008-09-01 15:48:17 +04:00
return - EIO ;
2008-11-04 23:34:56 +03:00
/* Raise board/PHY high limit from 85 to 90 degrees Celsius */
rc = i2c_smbus_write_byte_data ( efx - > board_info . hwmon_client ,
MAX664X_REG_WLHO , 90 ) ;
2008-09-01 15:48:17 +04:00
if ( rc )
2008-11-04 23:34:56 +03:00
goto fail_hwmon ;
2008-09-01 15:48:17 +04:00
efx - > board_info . ioexp_client = i2c_new_dummy ( & efx - > i2c_adap , PCA9539 ) ;
if ( ! efx - > board_info . ioexp_client ) {
rc = - EIO ;
goto fail_hwmon ;
}
/* 10Xpress has fixed-function LED pins, so there is no board-specific
* blink code . */
efx - > board_info . blink = tenxpress_phy_blink ;
2008-11-04 23:34:56 +03:00
efx - > board_info . monitor = sfe4001_check_hw ;
2008-09-01 15:48:17 +04:00
efx - > board_info . fini = sfe4001_fini ;
rc = sfe4001_poweron ( efx ) ;
if ( rc )
goto fail_ioexp ;
rc = device_create_file ( & efx - > pci_dev - > dev , & dev_attr_phy_flash_cfg ) ;
if ( rc )
goto fail_on ;
2008-04-27 15:55:59 +04:00
EFX_INFO ( efx , " PHY is powered on \n " ) ;
return 0 ;
2008-05-31 01:27:04 +04:00
fail_on :
sfe4001_poweroff ( efx ) ;
fail_ioexp :
2008-09-01 15:48:17 +04:00
i2c_unregister_device ( efx - > board_info . ioexp_client ) ;
2008-05-31 01:27:04 +04:00
fail_hwmon :
2008-11-04 23:34:56 +03:00
i2c_unregister_device ( efx - > board_info . hwmon_client ) ;
2008-04-27 15:55:59 +04:00
return rc ;
}
2008-12-13 09:00:49 +03:00
static int sfn4111t_check_hw ( struct efx_nic * efx )
{
s32 status ;
/* If XAUI link is up then do not monitor */
if ( EFX_WORKAROUND_7884 ( efx ) & & efx - > mac_up )
return 0 ;
/* Test LHIGH, RHIGH, FAULT, EOT and IOT alarms */
status = i2c_smbus_read_byte_data ( efx - > board_info . hwmon_client ,
MAX664X_REG_RSL ) ;
if ( status < 0 )
return - EIO ;
if ( status & 0x57 )
return - ERANGE ;
return 0 ;
}
static void sfn4111t_fini ( struct efx_nic * efx )
{
EFX_INFO ( efx , " %s \n " , __func__ ) ;
device_remove_file ( & efx - > pci_dev - > dev , & dev_attr_phy_flash_cfg ) ;
i2c_unregister_device ( efx - > board_info . hwmon_client ) ;
}
static struct i2c_board_info sfn4111t_hwmon_info = {
I2C_BOARD_INFO ( " max6647 " , 0x4e ) ,
. irq = - 1 ,
} ;
int sfn4111t_init ( struct efx_nic * efx )
{
int rc ;
efx - > board_info . hwmon_client =
i2c_new_device ( & efx - > i2c_adap , & sfn4111t_hwmon_info ) ;
if ( ! efx - > board_info . hwmon_client )
return - EIO ;
efx - > board_info . blink = tenxpress_phy_blink ;
efx - > board_info . monitor = sfn4111t_check_hw ;
efx - > board_info . fini = sfn4111t_fini ;
rc = device_create_file ( & efx - > pci_dev - > dev , & dev_attr_phy_flash_cfg ) ;
if ( rc )
goto fail_hwmon ;
if ( efx - > phy_mode & PHY_MODE_SPECIAL )
sfn4111t_reset ( efx ) ;
return 0 ;
fail_hwmon :
i2c_unregister_device ( efx - > board_info . hwmon_client ) ;
return rc ;
}