2016-09-23 00:13:42 +03:00
/*
* Copyright ( c ) 2016 Mellanox Technologies . All rights reserved .
* Copyright ( c ) 2016 Vadim Pasternak < vadimp @ mellanox . com >
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions are met :
*
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission .
*
* Alternatively , this software may be distributed under the terms of the
* GNU General Public License ( " GPL " ) version 2 as published by the Free
* Software Foundation .
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS " AS IS "
* AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR
* CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS
* INTERRUPTION ) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN
* CONTRACT , STRICT LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE )
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE .
*/
# include <linux/device.h>
# include <linux/dmi.h>
# include <linux/i2c.h>
# include <linux/i2c-mux.h>
2018-01-23 06:55:11 +03:00
# include <linux/io.h>
2016-09-23 00:13:42 +03:00
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/platform_data/i2c-mux-reg.h>
2018-01-17 21:21:53 +03:00
# include <linux/platform_data/mlxreg.h>
2018-01-23 06:55:11 +03:00
# include <linux/regmap.h>
2016-09-23 00:13:42 +03:00
# define MLX_PLAT_DEVICE_NAME "mlxplat"
/* LPC bus IO offsets */
# define MLXPLAT_CPLD_LPC_I2C_BASE_ADRR 0x2000
# define MLXPLAT_CPLD_LPC_REG_BASE_ADRR 0x2500
2018-01-23 06:55:11 +03:00
# define MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET 0x3a
2018-01-26 22:03:44 +03:00
# define MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET 0x3b
# define MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET 0x40
# define MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET 0x41
2018-01-23 06:55:11 +03:00
# define MLXPLAT_CPLD_LPC_REG_PSU_OFFSET 0x58
2018-01-26 22:03:44 +03:00
# define MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET 0x59
# define MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET 0x5a
2018-01-23 06:55:11 +03:00
# define MLXPLAT_CPLD_LPC_REG_PWR_OFFSET 0x64
2018-01-26 22:03:44 +03:00
# define MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET 0x65
# define MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET 0x66
2018-01-23 06:55:11 +03:00
# define MLXPLAT_CPLD_LPC_REG_FAN_OFFSET 0x88
2018-01-26 22:03:44 +03:00
# define MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET 0x89
# define MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET 0x8a
2016-09-23 00:13:42 +03:00
# define MLXPLAT_CPLD_LPC_IO_RANGE 0x100
# define MLXPLAT_CPLD_LPC_I2C_CH1_OFF 0xdb
# define MLXPLAT_CPLD_LPC_I2C_CH2_OFF 0xda
# define MLXPLAT_CPLD_LPC_PIO_OFFSET 0x10000UL
# define MLXPLAT_CPLD_LPC_REG1 ((MLXPLAT_CPLD_LPC_REG_BASE_ADRR + \
MLXPLAT_CPLD_LPC_I2C_CH1_OFF ) | \
MLXPLAT_CPLD_LPC_PIO_OFFSET )
# define MLXPLAT_CPLD_LPC_REG2 ((MLXPLAT_CPLD_LPC_REG_BASE_ADRR + \
MLXPLAT_CPLD_LPC_I2C_CH2_OFF ) | \
MLXPLAT_CPLD_LPC_PIO_OFFSET )
2016-12-14 15:05:15 +03:00
/* Masks for aggregation, psu, pwr and fan event in CPLD related registers. */
# define MLXPLAT_CPLD_AGGR_PSU_MASK_DEF 0x08
# define MLXPLAT_CPLD_AGGR_PWR_MASK_DEF 0x08
# define MLXPLAT_CPLD_AGGR_FAN_MASK_DEF 0x40
# define MLXPLAT_CPLD_AGGR_MASK_DEF (MLXPLAT_CPLD_AGGR_PSU_MASK_DEF | \
MLXPLAT_CPLD_AGGR_FAN_MASK_DEF )
2018-02-02 11:45:47 +03:00
# define MLXPLAT_CPLD_AGGR_MASK_NG_DEF 0x04
# define MLXPLAT_CPLD_LOW_AGGR_MASK_LOW 0xc0
2016-12-14 15:05:15 +03:00
# define MLXPLAT_CPLD_AGGR_MASK_MSN21XX 0x04
# define MLXPLAT_CPLD_PSU_MASK GENMASK(1, 0)
# define MLXPLAT_CPLD_PWR_MASK GENMASK(1, 0)
# define MLXPLAT_CPLD_FAN_MASK GENMASK(3, 0)
2016-09-23 00:13:42 +03:00
/* Start channel numbers */
# define MLXPLAT_CPLD_CH1 2
# define MLXPLAT_CPLD_CH2 10
/* Number of LPC attached MUX platform devices */
# define MLXPLAT_CPLD_LPC_MUX_DEVS 2
2018-02-02 11:45:45 +03:00
/* Hotplug devices adapter numbers */
2018-02-02 11:45:46 +03:00
# define MLXPLAT_CPLD_NR_NONE -1
2018-02-02 11:45:45 +03:00
# define MLXPLAT_CPLD_PSU_DEFAULT_NR 10
2018-02-10 02:59:30 +03:00
# define MLXPLAT_CPLD_PSU_MSNXXXX_NR 4
2018-02-02 11:45:45 +03:00
# define MLXPLAT_CPLD_FAN1_DEFAULT_NR 11
# define MLXPLAT_CPLD_FAN2_DEFAULT_NR 12
# define MLXPLAT_CPLD_FAN3_DEFAULT_NR 13
# define MLXPLAT_CPLD_FAN4_DEFAULT_NR 14
2016-09-23 00:13:42 +03:00
/* mlxplat_priv - platform private data
* @ pdev_i2c - i2c controller platform device
* @ pdev_mux - array of mux platform devices
2018-01-26 01:02:53 +03:00
* @ pdev_hotplug - hotplug platform devices
2016-09-23 00:13:42 +03:00
*/
struct mlxplat_priv {
struct platform_device * pdev_i2c ;
struct platform_device * pdev_mux [ MLXPLAT_CPLD_LPC_MUX_DEVS ] ;
2016-10-27 22:55:54 +03:00
struct platform_device * pdev_hotplug ;
2016-09-23 00:13:42 +03:00
} ;
/* Regions for LPC I2C controller and LPC base register space */
static const struct resource mlxplat_lpc_resources [ ] = {
[ 0 ] = DEFINE_RES_NAMED ( MLXPLAT_CPLD_LPC_I2C_BASE_ADRR ,
MLXPLAT_CPLD_LPC_IO_RANGE ,
" mlxplat_cpld_lpc_i2c_ctrl " , IORESOURCE_IO ) ,
[ 1 ] = DEFINE_RES_NAMED ( MLXPLAT_CPLD_LPC_REG_BASE_ADRR ,
MLXPLAT_CPLD_LPC_IO_RANGE ,
" mlxplat_cpld_lpc_regs " ,
IORESOURCE_IO ) ,
} ;
/* Platform default channels */
static const int mlxplat_default_channels [ ] [ 8 ] = {
{
MLXPLAT_CPLD_CH1 , MLXPLAT_CPLD_CH1 + 1 , MLXPLAT_CPLD_CH1 + 2 ,
MLXPLAT_CPLD_CH1 + 3 , MLXPLAT_CPLD_CH1 + 4 , MLXPLAT_CPLD_CH1 +
5 , MLXPLAT_CPLD_CH1 + 6 , MLXPLAT_CPLD_CH1 + 7
} ,
{
MLXPLAT_CPLD_CH2 , MLXPLAT_CPLD_CH2 + 1 , MLXPLAT_CPLD_CH2 + 2 ,
MLXPLAT_CPLD_CH2 + 3 , MLXPLAT_CPLD_CH2 + 4 , MLXPLAT_CPLD_CH2 +
5 , MLXPLAT_CPLD_CH2 + 6 , MLXPLAT_CPLD_CH2 + 7
} ,
} ;
/* Platform channels for MSN21xx system family */
static const int mlxplat_msn21xx_channels [ ] = { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 } ;
/* Platform mux data */
static struct i2c_mux_reg_platform_data mlxplat_mux_data [ ] = {
{
. parent = 1 ,
. base_nr = MLXPLAT_CPLD_CH1 ,
. write_only = 1 ,
. reg = ( void __iomem * ) MLXPLAT_CPLD_LPC_REG1 ,
. reg_size = 1 ,
. idle_in_use = 1 ,
} ,
{
. parent = 1 ,
. base_nr = MLXPLAT_CPLD_CH2 ,
. write_only = 1 ,
. reg = ( void __iomem * ) MLXPLAT_CPLD_LPC_REG2 ,
. reg_size = 1 ,
. idle_in_use = 1 ,
} ,
} ;
2016-10-27 22:55:54 +03:00
/* Platform hotplug devices */
2018-01-23 06:55:11 +03:00
static struct i2c_board_info mlxplat_mlxcpld_psu [ ] = {
2016-10-27 22:55:54 +03:00
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " 24c02 " , 0x51 ) ,
2016-10-27 22:55:54 +03:00
} ,
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " 24c02 " , 0x50 ) ,
2016-10-27 22:55:54 +03:00
} ,
} ;
2018-01-23 06:55:11 +03:00
static struct i2c_board_info mlxplat_mlxcpld_pwr [ ] = {
2016-10-27 22:55:54 +03:00
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " dps460 " , 0x59 ) ,
2016-10-27 22:55:54 +03:00
} ,
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " dps460 " , 0x58 ) ,
2016-10-27 22:55:54 +03:00
} ,
} ;
2018-01-23 06:55:11 +03:00
static struct i2c_board_info mlxplat_mlxcpld_fan [ ] = {
2016-10-27 22:55:54 +03:00
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " 24c32 " , 0x50 ) ,
2016-10-27 22:55:54 +03:00
} ,
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " 24c32 " , 0x50 ) ,
2016-10-27 22:55:54 +03:00
} ,
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " 24c32 " , 0x50 ) ,
2016-10-27 22:55:54 +03:00
} ,
{
2018-01-23 06:55:11 +03:00
I2C_BOARD_INFO ( " 24c32 " , 0x50 ) ,
2016-10-27 22:55:54 +03:00
} ,
} ;
/* Platform hotplug default data */
2018-01-23 06:55:11 +03:00
static struct mlxreg_core_data mlxplat_mlxcpld_default_psu_items_data [ ] = {
{
. label = " psu1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_psu [ 0 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_PSU_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
{
. label = " psu2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_psu [ 1 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_PSU_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
} ;
static struct mlxreg_core_data mlxplat_mlxcpld_default_pwr_items_data [ ] = {
{
. label = " pwr1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_pwr [ 0 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_PSU_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
{
. label = " pwr2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_pwr [ 1 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_PSU_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
} ;
static struct mlxreg_core_data mlxplat_mlxcpld_default_fan_items_data [ ] = {
{
. label = " fan1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_fan [ 0 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_FAN1_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
{
. label = " fan2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_fan [ 1 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_FAN2_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
{
. label = " fan3 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 2 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_fan [ 2 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_FAN3_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
{
. label = " fan4 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 3 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_fan [ 3 ] ,
2018-02-02 11:45:45 +03:00
. hpdev . nr = MLXPLAT_CPLD_FAN4_DEFAULT_NR ,
2018-01-23 06:55:11 +03:00
} ,
} ;
static struct mlxreg_core_item mlxplat_mlxcpld_default_items [ ] = {
{
. data = mlxplat_mlxcpld_default_psu_items_data ,
. aggr_mask = MLXPLAT_CPLD_AGGR_PSU_MASK_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET ,
. mask = MLXPLAT_CPLD_PSU_MASK ,
. count = ARRAY_SIZE ( mlxplat_mlxcpld_psu ) ,
. inversed = 1 ,
. health = false ,
} ,
{
. data = mlxplat_mlxcpld_default_pwr_items_data ,
. aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = MLXPLAT_CPLD_PWR_MASK ,
. count = ARRAY_SIZE ( mlxplat_mlxcpld_pwr ) ,
. inversed = 0 ,
. health = false ,
} ,
{
. data = mlxplat_mlxcpld_default_fan_items_data ,
. aggr_mask = MLXPLAT_CPLD_AGGR_FAN_MASK_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = MLXPLAT_CPLD_FAN_MASK ,
. count = ARRAY_SIZE ( mlxplat_mlxcpld_fan ) ,
. inversed = 1 ,
. health = false ,
} ,
} ;
2016-10-27 22:55:54 +03:00
static
2018-01-23 06:55:11 +03:00
struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_default_data = {
. items = mlxplat_mlxcpld_default_items ,
. counter = ARRAY_SIZE ( mlxplat_mlxcpld_default_items ) ,
. cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET ,
. mask = MLXPLAT_CPLD_AGGR_MASK_DEF ,
2016-10-27 22:55:54 +03:00
} ;
2018-02-02 11:45:47 +03:00
static struct mlxreg_core_data mlxplat_mlxcpld_msn21xx_pwr_items_data [ ] = {
{
. label = " pwr1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
{
. label = " pwr2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
} ;
2016-10-27 22:55:54 +03:00
/* Platform hotplug MSN21xx system family data */
2018-01-23 06:55:11 +03:00
static struct mlxreg_core_item mlxplat_mlxcpld_msn21xx_items [ ] = {
{
2018-02-02 11:45:47 +03:00
. data = mlxplat_mlxcpld_msn21xx_pwr_items_data ,
2018-01-23 06:55:11 +03:00
. aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = MLXPLAT_CPLD_PWR_MASK ,
2018-02-02 11:45:47 +03:00
. count = ARRAY_SIZE ( mlxplat_mlxcpld_msn21xx_pwr_items_data ) ,
2018-01-23 06:55:11 +03:00
. inversed = 0 ,
. health = false ,
} ,
} ;
2016-10-27 22:55:54 +03:00
static
2018-01-23 06:55:11 +03:00
struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn21xx_data = {
. items = mlxplat_mlxcpld_msn21xx_items ,
. counter = ARRAY_SIZE ( mlxplat_mlxcpld_msn21xx_items ) ,
. cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET ,
. mask = MLXPLAT_CPLD_AGGR_MASK_DEF ,
2018-02-02 11:45:47 +03:00
. cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET ,
. mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW ,
2018-01-23 06:55:11 +03:00
} ;
2018-02-10 02:59:30 +03:00
/* Platform hotplug msn274x system family data */
static struct mlxreg_core_data mlxplat_mlxcpld_msn274x_psu_items_data [ ] = {
{
. label = " psu1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_psu [ 0 ] ,
. hpdev . nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR ,
} ,
{
. label = " psu2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_psu [ 1 ] ,
. hpdev . nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR ,
} ,
} ;
static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_pwr_items_data [ ] = {
{
. label = " pwr1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_pwr [ 0 ] ,
. hpdev . nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR ,
} ,
{
. label = " pwr2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . brdinfo = & mlxplat_mlxcpld_pwr [ 1 ] ,
. hpdev . nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR ,
} ,
} ;
static struct mlxreg_core_data mlxplat_mlxcpld_msn274x_fan_items_data [ ] = {
{
. label = " fan1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
{
. label = " fan2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
{
. label = " fan3 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 2 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
{
. label = " fan4 " ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = BIT ( 3 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
} ;
static struct mlxreg_core_item mlxplat_mlxcpld_msn274x_items [ ] = {
{
. data = mlxplat_mlxcpld_msn274x_psu_items_data ,
. aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET ,
. mask = MLXPLAT_CPLD_PSU_MASK ,
. count = ARRAY_SIZE ( mlxplat_mlxcpld_msn274x_psu_items_data ) ,
. inversed = 1 ,
. health = false ,
} ,
{
. data = mlxplat_mlxcpld_default_ng_pwr_items_data ,
. aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = MLXPLAT_CPLD_PWR_MASK ,
. count = ARRAY_SIZE ( mlxplat_mlxcpld_default_ng_pwr_items_data ) ,
. inversed = 0 ,
. health = false ,
} ,
{
. data = mlxplat_mlxcpld_msn274x_fan_items_data ,
. aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET ,
. mask = MLXPLAT_CPLD_FAN_MASK ,
. count = ARRAY_SIZE ( mlxplat_mlxcpld_msn274x_fan_items_data ) ,
. inversed = 1 ,
. health = false ,
} ,
} ;
static
struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn274x_data = {
. items = mlxplat_mlxcpld_msn274x_items ,
. counter = ARRAY_SIZE ( mlxplat_mlxcpld_msn274x_items ) ,
. cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET ,
. mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF ,
. cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET ,
. mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW ,
} ;
2018-02-10 02:59:31 +03:00
/* Platform hotplug MSN201x system family data */
static struct mlxreg_core_data mlxplat_mlxcpld_msn201x_pwr_items_data [ ] = {
{
. label = " pwr1 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 0 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
{
. label = " pwr2 " ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = BIT ( 1 ) ,
. hpdev . nr = MLXPLAT_CPLD_NR_NONE ,
} ,
} ;
static struct mlxreg_core_item mlxplat_mlxcpld_msn201x_items [ ] = {
{
. data = mlxplat_mlxcpld_msn201x_pwr_items_data ,
. aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF ,
. reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET ,
. mask = MLXPLAT_CPLD_PWR_MASK ,
. count = ARRAY_SIZE ( mlxplat_mlxcpld_msn201x_pwr_items_data ) ,
. inversed = 0 ,
. health = false ,
} ,
} ;
static
struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn201x_data = {
. items = mlxplat_mlxcpld_msn21xx_items ,
. counter = ARRAY_SIZE ( mlxplat_mlxcpld_msn201x_items ) ,
. cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET ,
. mask = MLXPLAT_CPLD_AGGR_MASK_DEF ,
. cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET ,
. mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW ,
} ;
2018-01-26 22:03:44 +03:00
static bool mlxplat_mlxcpld_writeable_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET :
return true ;
}
return false ;
}
static bool mlxplat_mlxcpld_readable_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET :
case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET :
case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET :
return true ;
}
return false ;
}
static bool mlxplat_mlxcpld_volatile_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET :
case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET :
case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET :
case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET :
return true ;
}
return false ;
}
2018-01-23 06:55:11 +03:00
struct mlxplat_mlxcpld_regmap_context {
void __iomem * base ;
} ;
static struct mlxplat_mlxcpld_regmap_context mlxplat_mlxcpld_regmap_ctx ;
static int
mlxplat_mlxcpld_reg_read ( void * context , unsigned int reg , unsigned int * val )
{
struct mlxplat_mlxcpld_regmap_context * ctx = context ;
* val = ioread8 ( ctx - > base + reg ) ;
return 0 ;
}
static int
mlxplat_mlxcpld_reg_write ( void * context , unsigned int reg , unsigned int val )
{
struct mlxplat_mlxcpld_regmap_context * ctx = context ;
iowrite8 ( val , ctx - > base + reg ) ;
return 0 ;
}
static const struct regmap_config mlxplat_mlxcpld_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = 255 ,
2018-01-26 22:03:44 +03:00
. cache_type = REGCACHE_FLAT ,
. writeable_reg = mlxplat_mlxcpld_writeable_reg ,
. readable_reg = mlxplat_mlxcpld_readable_reg ,
. volatile_reg = mlxplat_mlxcpld_volatile_reg ,
2018-01-23 06:55:11 +03:00
. reg_read = mlxplat_mlxcpld_reg_read ,
. reg_write = mlxplat_mlxcpld_reg_write ,
2016-10-27 22:55:54 +03:00
} ;
2016-12-14 15:05:15 +03:00
static struct resource mlxplat_mlxcpld_resources [ ] = {
2018-01-17 21:21:53 +03:00
[ 0 ] = DEFINE_RES_IRQ_NAMED ( 17 , " mlxreg-hotplug " ) ,
2016-10-27 22:55:54 +03:00
} ;
2017-10-05 13:42:11 +03:00
static struct platform_device * mlxplat_dev ;
2018-01-23 06:55:11 +03:00
static struct mlxreg_core_hotplug_platform_data * mlxplat_hotplug ;
2016-09-23 00:13:42 +03:00
static int __init mlxplat_dmi_default_matched ( const struct dmi_system_id * dmi )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( mlxplat_mux_data ) ; i + + ) {
mlxplat_mux_data [ i ] . values = mlxplat_default_channels [ i ] ;
mlxplat_mux_data [ i ] . n_values =
ARRAY_SIZE ( mlxplat_default_channels [ i ] ) ;
}
2016-12-14 15:05:15 +03:00
mlxplat_hotplug = & mlxplat_mlxcpld_default_data ;
2016-09-23 00:13:42 +03:00
return 1 ;
} ;
static int __init mlxplat_dmi_msn21xx_matched ( const struct dmi_system_id * dmi )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( mlxplat_mux_data ) ; i + + ) {
mlxplat_mux_data [ i ] . values = mlxplat_msn21xx_channels ;
mlxplat_mux_data [ i ] . n_values =
ARRAY_SIZE ( mlxplat_msn21xx_channels ) ;
}
2016-12-14 15:05:15 +03:00
mlxplat_hotplug = & mlxplat_mlxcpld_msn21xx_data ;
2016-09-23 00:13:42 +03:00
return 1 ;
} ;
2018-02-10 02:59:30 +03:00
static int __init mlxplat_dmi_msn274x_matched ( const struct dmi_system_id * dmi )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( mlxplat_mux_data ) ; i + + ) {
mlxplat_mux_data [ i ] . values = mlxplat_msn21xx_channels ;
mlxplat_mux_data [ i ] . n_values =
ARRAY_SIZE ( mlxplat_msn21xx_channels ) ;
}
mlxplat_hotplug = & mlxplat_mlxcpld_msn274x_data ;
return 1 ;
} ;
2018-02-10 02:59:31 +03:00
static int __init mlxplat_dmi_msn201x_matched ( const struct dmi_system_id * dmi )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( mlxplat_mux_data ) ; i + + ) {
mlxplat_mux_data [ i ] . values = mlxplat_msn21xx_channels ;
mlxplat_mux_data [ i ] . n_values =
ARRAY_SIZE ( mlxplat_msn21xx_channels ) ;
}
mlxplat_hotplug = & mlxplat_mlxcpld_msn201x_data ;
return 1 ;
} ;
2017-09-14 12:59:30 +03:00
static const struct dmi_system_id mlxplat_dmi_table [ ] __initconst = {
2018-02-10 02:59:30 +03:00
{
. callback = mlxplat_dmi_msn274x_matched ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " Mellanox Technologies " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MSN274 " ) ,
} ,
} ,
2016-09-23 00:13:42 +03:00
{
. callback = mlxplat_dmi_default_matched ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " Mellanox Technologies " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MSN24 " ) ,
} ,
} ,
{
. callback = mlxplat_dmi_default_matched ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " Mellanox Technologies " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MSN27 " ) ,
} ,
} ,
{
. callback = mlxplat_dmi_default_matched ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " Mellanox Technologies " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MSB " ) ,
} ,
} ,
{
. callback = mlxplat_dmi_default_matched ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " Mellanox Technologies " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MSX " ) ,
} ,
} ,
{
. callback = mlxplat_dmi_msn21xx_matched ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " Mellanox Technologies " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MSN21 " ) ,
} ,
} ,
2018-02-10 02:59:31 +03:00
{
. callback = mlxplat_dmi_msn201x_matched ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " Mellanox Technologies " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MSN201 " ) ,
} ,
} ,
2016-09-23 00:13:42 +03:00
{ }
} ;
2018-01-22 17:20:43 +03:00
MODULE_DEVICE_TABLE ( dmi , mlxplat_dmi_table ) ;
2016-09-23 00:13:42 +03:00
static int __init mlxplat_init ( void )
{
struct mlxplat_priv * priv ;
int i , err ;
if ( ! dmi_check_system ( mlxplat_dmi_table ) )
return - ENODEV ;
mlxplat_dev = platform_device_register_simple ( MLX_PLAT_DEVICE_NAME , - 1 ,
mlxplat_lpc_resources ,
ARRAY_SIZE ( mlxplat_lpc_resources ) ) ;
2016-09-24 14:48:13 +03:00
if ( IS_ERR ( mlxplat_dev ) )
return PTR_ERR ( mlxplat_dev ) ;
2016-09-23 00:13:42 +03:00
priv = devm_kzalloc ( & mlxplat_dev - > dev , sizeof ( struct mlxplat_priv ) ,
GFP_KERNEL ) ;
if ( ! priv ) {
err = - ENOMEM ;
goto fail_alloc ;
}
platform_set_drvdata ( mlxplat_dev , priv ) ;
priv - > pdev_i2c = platform_device_register_simple ( " i2c_mlxcpld " , - 1 ,
NULL , 0 ) ;
if ( IS_ERR ( priv - > pdev_i2c ) ) {
err = PTR_ERR ( priv - > pdev_i2c ) ;
goto fail_alloc ;
2016-10-27 20:26:50 +03:00
}
2016-09-23 00:13:42 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( mlxplat_mux_data ) ; i + + ) {
priv - > pdev_mux [ i ] = platform_device_register_resndata (
& mlxplat_dev - > dev ,
" i2c-mux-reg " , i , NULL ,
0 , & mlxplat_mux_data [ i ] ,
sizeof ( mlxplat_mux_data [ i ] ) ) ;
if ( IS_ERR ( priv - > pdev_mux [ i ] ) ) {
err = PTR_ERR ( priv - > pdev_mux [ i ] ) ;
goto fail_platform_mux_register ;
}
}
2018-01-23 06:55:11 +03:00
mlxplat_mlxcpld_regmap_ctx . base = devm_ioport_map ( & mlxplat_dev - > dev ,
mlxplat_lpc_resources [ 1 ] . start , 1 ) ;
2018-02-06 15:45:36 +03:00
if ( ! mlxplat_mlxcpld_regmap_ctx . base ) {
err = - ENOMEM ;
2018-01-23 06:55:11 +03:00
goto fail_platform_mux_register ;
}
mlxplat_hotplug - > regmap = devm_regmap_init ( & mlxplat_dev - > dev , NULL ,
& mlxplat_mlxcpld_regmap_ctx ,
& mlxplat_mlxcpld_regmap_config ) ;
if ( IS_ERR ( mlxplat_hotplug - > regmap ) ) {
err = PTR_ERR ( mlxplat_hotplug - > regmap ) ;
goto fail_platform_mux_register ;
}
2016-10-27 22:55:54 +03:00
priv - > pdev_hotplug = platform_device_register_resndata (
2018-01-17 21:21:53 +03:00
& mlxplat_dev - > dev , " mlxreg-hotplug " ,
2016-12-14 15:05:15 +03:00
PLATFORM_DEVID_NONE ,
mlxplat_mlxcpld_resources ,
ARRAY_SIZE ( mlxplat_mlxcpld_resources ) ,
2016-10-27 22:55:54 +03:00
mlxplat_hotplug , sizeof ( * mlxplat_hotplug ) ) ;
if ( IS_ERR ( priv - > pdev_hotplug ) ) {
err = PTR_ERR ( priv - > pdev_hotplug ) ;
goto fail_platform_mux_register ;
}
2018-01-26 22:03:44 +03:00
/* Sync registers with hardware. */
regcache_mark_dirty ( mlxplat_hotplug - > regmap ) ;
err = regcache_sync ( mlxplat_hotplug - > regmap ) ;
if ( err )
2018-02-01 00:55:13 +03:00
goto fail_platform_hotplug_register ;
2018-01-26 22:03:44 +03:00
2016-09-23 00:13:42 +03:00
return 0 ;
2018-02-01 00:55:13 +03:00
fail_platform_hotplug_register :
platform_device_unregister ( priv - > pdev_hotplug ) ;
2016-09-23 00:13:42 +03:00
fail_platform_mux_register :
2017-01-07 09:33:34 +03:00
while ( - - i > = 0 )
2016-09-23 00:13:42 +03:00
platform_device_unregister ( priv - > pdev_mux [ i ] ) ;
platform_device_unregister ( priv - > pdev_i2c ) ;
fail_alloc :
platform_device_unregister ( mlxplat_dev ) ;
return err ;
}
module_init ( mlxplat_init ) ;
static void __exit mlxplat_exit ( void )
{
struct mlxplat_priv * priv = platform_get_drvdata ( mlxplat_dev ) ;
int i ;
2016-10-27 22:55:54 +03:00
platform_device_unregister ( priv - > pdev_hotplug ) ;
2016-09-23 00:13:42 +03:00
for ( i = ARRAY_SIZE ( mlxplat_mux_data ) - 1 ; i > = 0 ; i - - )
platform_device_unregister ( priv - > pdev_mux [ i ] ) ;
platform_device_unregister ( priv - > pdev_i2c ) ;
platform_device_unregister ( mlxplat_dev ) ;
}
module_exit ( mlxplat_exit ) ;
MODULE_AUTHOR ( " Vadim Pasternak (vadimp@mellanox.com) " ) ;
MODULE_DESCRIPTION ( " Mellanox platform driver " ) ;
MODULE_LICENSE ( " Dual BSD/GPL " ) ;