Commit ef0df4fa authored by Vadim Pasternak's avatar Vadim Pasternak Committed by David S. Miller

mlxsw: core_thermal: Extend internal structures to support multi thermal areas

Introduce intermediate level for thermal zones areas.
Currently all thermal zones are associated with thermal objects located
within the main board. Such objects are created during driver
initialization and removed during driver de-initialization.

For line cards in modular system the thermal zones are to be associated
with the specific line card. They should be created whenever new line
card is available (inserted, validated, powered and enabled) and
removed, when line card is getting unavailable.
The thermal objects found on the line card #n are accessed by setting
slot index to #n, while for access to objects found on the main board
slot index should be set to default value zero.

Each thermal area contains the set of thermal zones associated with
particular slot index.
Thus introduction of thermal zone areas allows to use the same APIs for
the main board and line cards, by adding slot index argument.
Signed-off-by: default avatarVadim Pasternak <vadimp@nvidia.com>
Signed-off-by: default avatarIdo Schimmel <idosch@nvidia.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent fd27849d
...@@ -82,6 +82,15 @@ struct mlxsw_thermal_module { ...@@ -82,6 +82,15 @@ struct mlxsw_thermal_module {
struct thermal_zone_device *tzdev; struct thermal_zone_device *tzdev;
struct mlxsw_thermal_trip trips[MLXSW_THERMAL_NUM_TRIPS]; struct mlxsw_thermal_trip trips[MLXSW_THERMAL_NUM_TRIPS];
int module; /* Module or gearbox number */ int module; /* Module or gearbox number */
u8 slot_index;
};
struct mlxsw_thermal_area {
struct mlxsw_thermal_module *tz_module_arr;
u8 tz_module_num;
struct mlxsw_thermal_module *tz_gearbox_arr;
u8 tz_gearbox_num;
u8 slot_index;
}; };
struct mlxsw_thermal { struct mlxsw_thermal {
...@@ -92,12 +101,9 @@ struct mlxsw_thermal { ...@@ -92,12 +101,9 @@ struct mlxsw_thermal {
struct thermal_cooling_device *cdevs[MLXSW_MFCR_PWMS_MAX]; struct thermal_cooling_device *cdevs[MLXSW_MFCR_PWMS_MAX];
u8 cooling_levels[MLXSW_THERMAL_MAX_STATE + 1]; u8 cooling_levels[MLXSW_THERMAL_MAX_STATE + 1];
struct mlxsw_thermal_trip trips[MLXSW_THERMAL_NUM_TRIPS]; struct mlxsw_thermal_trip trips[MLXSW_THERMAL_NUM_TRIPS];
struct mlxsw_thermal_module *tz_module_arr;
u8 tz_module_num;
struct mlxsw_thermal_module *tz_gearbox_arr;
u8 tz_gearbox_num;
unsigned int tz_highest_score; unsigned int tz_highest_score;
struct thermal_zone_device *tz_highest_dev; struct thermal_zone_device *tz_highest_dev;
struct mlxsw_thermal_area line_cards[];
}; };
static inline u8 mlxsw_state_to_duty(int state) static inline u8 mlxsw_state_to_duty(int state)
...@@ -150,13 +156,15 @@ mlxsw_thermal_module_trips_update(struct device *dev, struct mlxsw_core *core, ...@@ -150,13 +156,15 @@ mlxsw_thermal_module_trips_update(struct device *dev, struct mlxsw_core *core,
* EEPROM if we got valid thresholds from MTMP. * EEPROM if we got valid thresholds from MTMP.
*/ */
if (!emerg_temp || !crit_temp) { if (!emerg_temp || !crit_temp) {
err = mlxsw_env_module_temp_thresholds_get(core, 0, tz->module, err = mlxsw_env_module_temp_thresholds_get(core, tz->slot_index,
tz->module,
SFP_TEMP_HIGH_WARN, SFP_TEMP_HIGH_WARN,
&crit_temp); &crit_temp);
if (err) if (err)
return err; return err;
err = mlxsw_env_module_temp_thresholds_get(core, 0, tz->module, err = mlxsw_env_module_temp_thresholds_get(core, tz->slot_index,
tz->module,
SFP_TEMP_HIGH_ALARM, SFP_TEMP_HIGH_ALARM,
&emerg_temp); &emerg_temp);
if (err) if (err)
...@@ -423,15 +431,16 @@ static int mlxsw_thermal_module_unbind(struct thermal_zone_device *tzdev, ...@@ -423,15 +431,16 @@ static int mlxsw_thermal_module_unbind(struct thermal_zone_device *tzdev,
static void static void
mlxsw_thermal_module_temp_and_thresholds_get(struct mlxsw_core *core, mlxsw_thermal_module_temp_and_thresholds_get(struct mlxsw_core *core,
u16 sensor_index, int *p_temp, u8 slot_index, u16 sensor_index,
int *p_crit_temp, int *p_temp, int *p_crit_temp,
int *p_emerg_temp) int *p_emerg_temp)
{ {
char mtmp_pl[MLXSW_REG_MTMP_LEN]; char mtmp_pl[MLXSW_REG_MTMP_LEN];
int err; int err;
/* Read module temperature and thresholds. */ /* Read module temperature and thresholds. */
mlxsw_reg_mtmp_pack(mtmp_pl, 0, sensor_index, false, false); mlxsw_reg_mtmp_pack(mtmp_pl, slot_index, sensor_index,
false, false);
err = mlxsw_reg_query(core, MLXSW_REG(mtmp), mtmp_pl); err = mlxsw_reg_query(core, MLXSW_REG(mtmp), mtmp_pl);
if (err) { if (err) {
/* Set temperature and thresholds to zero to avoid passing /* Set temperature and thresholds to zero to avoid passing
...@@ -462,6 +471,7 @@ static int mlxsw_thermal_module_temp_get(struct thermal_zone_device *tzdev, ...@@ -462,6 +471,7 @@ static int mlxsw_thermal_module_temp_get(struct thermal_zone_device *tzdev,
/* Read module temperature and thresholds. */ /* Read module temperature and thresholds. */
mlxsw_thermal_module_temp_and_thresholds_get(thermal->core, mlxsw_thermal_module_temp_and_thresholds_get(thermal->core,
tz->slot_index,
sensor_index, &temp, sensor_index, &temp,
&crit_temp, &emerg_temp); &crit_temp, &emerg_temp);
*p_temp = temp; *p_temp = temp;
...@@ -576,7 +586,7 @@ static int mlxsw_thermal_gearbox_temp_get(struct thermal_zone_device *tzdev, ...@@ -576,7 +586,7 @@ static int mlxsw_thermal_gearbox_temp_get(struct thermal_zone_device *tzdev,
int err; int err;
index = MLXSW_REG_MTMP_GBOX_INDEX_MIN + tz->module; index = MLXSW_REG_MTMP_GBOX_INDEX_MIN + tz->module;
mlxsw_reg_mtmp_pack(mtmp_pl, 0, index, false, false); mlxsw_reg_mtmp_pack(mtmp_pl, tz->slot_index, index, false, false);
err = mlxsw_reg_query(thermal->core, MLXSW_REG(mtmp), mtmp_pl); err = mlxsw_reg_query(thermal->core, MLXSW_REG(mtmp), mtmp_pl);
if (err) if (err)
...@@ -704,25 +714,28 @@ static void mlxsw_thermal_module_tz_fini(struct thermal_zone_device *tzdev) ...@@ -704,25 +714,28 @@ static void mlxsw_thermal_module_tz_fini(struct thermal_zone_device *tzdev)
static int static int
mlxsw_thermal_module_init(struct device *dev, struct mlxsw_core *core, mlxsw_thermal_module_init(struct device *dev, struct mlxsw_core *core,
struct mlxsw_thermal *thermal, u8 module) struct mlxsw_thermal *thermal,
struct mlxsw_thermal_area *area, u8 module)
{ {
struct mlxsw_thermal_module *module_tz; struct mlxsw_thermal_module *module_tz;
int dummy_temp, crit_temp, emerg_temp; int dummy_temp, crit_temp, emerg_temp;
u16 sensor_index; u16 sensor_index;
sensor_index = MLXSW_REG_MTMP_MODULE_INDEX_MIN + module; sensor_index = MLXSW_REG_MTMP_MODULE_INDEX_MIN + module;
module_tz = &thermal->tz_module_arr[module]; module_tz = &area->tz_module_arr[module];
/* Skip if parent is already set (case of port split). */ /* Skip if parent is already set (case of port split). */
if (module_tz->parent) if (module_tz->parent)
return 0; return 0;
module_tz->module = module; module_tz->module = module;
module_tz->slot_index = area->slot_index;
module_tz->parent = thermal; module_tz->parent = thermal;
memcpy(module_tz->trips, default_thermal_trips, memcpy(module_tz->trips, default_thermal_trips,
sizeof(thermal->trips)); sizeof(thermal->trips));
/* Initialize all trip point. */ /* Initialize all trip point. */
mlxsw_thermal_module_trips_reset(module_tz); mlxsw_thermal_module_trips_reset(module_tz);
/* Read module temperature and thresholds. */ /* Read module temperature and thresholds. */
mlxsw_thermal_module_temp_and_thresholds_get(core, sensor_index, &dummy_temp, mlxsw_thermal_module_temp_and_thresholds_get(core, area->slot_index,
sensor_index, &dummy_temp,
&crit_temp, &emerg_temp); &crit_temp, &emerg_temp);
/* Update trip point according to the module data. */ /* Update trip point according to the module data. */
return mlxsw_thermal_module_trips_update(dev, core, module_tz, return mlxsw_thermal_module_trips_update(dev, core, module_tz,
...@@ -740,34 +753,39 @@ static void mlxsw_thermal_module_fini(struct mlxsw_thermal_module *module_tz) ...@@ -740,34 +753,39 @@ static void mlxsw_thermal_module_fini(struct mlxsw_thermal_module *module_tz)
static int static int
mlxsw_thermal_modules_init(struct device *dev, struct mlxsw_core *core, mlxsw_thermal_modules_init(struct device *dev, struct mlxsw_core *core,
struct mlxsw_thermal *thermal) struct mlxsw_thermal *thermal,
struct mlxsw_thermal_area *area)
{ {
struct mlxsw_thermal_module *module_tz; struct mlxsw_thermal_module *module_tz;
char mgpir_pl[MLXSW_REG_MGPIR_LEN]; char mgpir_pl[MLXSW_REG_MGPIR_LEN];
int i, err; int i, err;
mlxsw_reg_mgpir_pack(mgpir_pl, 0); mlxsw_reg_mgpir_pack(mgpir_pl, area->slot_index);
err = mlxsw_reg_query(core, MLXSW_REG(mgpir), mgpir_pl); err = mlxsw_reg_query(core, MLXSW_REG(mgpir), mgpir_pl);
if (err) if (err)
return err; return err;
mlxsw_reg_mgpir_unpack(mgpir_pl, NULL, NULL, NULL, mlxsw_reg_mgpir_unpack(mgpir_pl, NULL, NULL, NULL,
&thermal->tz_module_num, NULL); &area->tz_module_num, NULL);
thermal->tz_module_arr = kcalloc(thermal->tz_module_num, /* For modular system module counter could be zero. */
sizeof(*thermal->tz_module_arr), if (!area->tz_module_num)
GFP_KERNEL); return 0;
if (!thermal->tz_module_arr)
area->tz_module_arr = kcalloc(area->tz_module_num,
sizeof(*area->tz_module_arr),
GFP_KERNEL);
if (!area->tz_module_arr)
return -ENOMEM; return -ENOMEM;
for (i = 0; i < thermal->tz_module_num; i++) { for (i = 0; i < area->tz_module_num; i++) {
err = mlxsw_thermal_module_init(dev, core, thermal, i); err = mlxsw_thermal_module_init(dev, core, thermal, area, i);
if (err) if (err)
goto err_thermal_module_init; goto err_thermal_module_init;
} }
for (i = 0; i < thermal->tz_module_num; i++) { for (i = 0; i < area->tz_module_num; i++) {
module_tz = &thermal->tz_module_arr[i]; module_tz = &area->tz_module_arr[i];
if (!module_tz->parent) if (!module_tz->parent)
continue; continue;
err = mlxsw_thermal_module_tz_init(module_tz); err = mlxsw_thermal_module_tz_init(module_tz);
...@@ -779,20 +797,21 @@ mlxsw_thermal_modules_init(struct device *dev, struct mlxsw_core *core, ...@@ -779,20 +797,21 @@ mlxsw_thermal_modules_init(struct device *dev, struct mlxsw_core *core,
err_thermal_module_tz_init: err_thermal_module_tz_init:
err_thermal_module_init: err_thermal_module_init:
for (i = thermal->tz_module_num - 1; i >= 0; i--) for (i = area->tz_module_num - 1; i >= 0; i--)
mlxsw_thermal_module_fini(&thermal->tz_module_arr[i]); mlxsw_thermal_module_fini(&area->tz_module_arr[i]);
kfree(thermal->tz_module_arr); kfree(area->tz_module_arr);
return err; return err;
} }
static void static void
mlxsw_thermal_modules_fini(struct mlxsw_thermal *thermal) mlxsw_thermal_modules_fini(struct mlxsw_thermal *thermal,
struct mlxsw_thermal_area *area)
{ {
int i; int i;
for (i = thermal->tz_module_num - 1; i >= 0; i--) for (i = area->tz_module_num - 1; i >= 0; i--)
mlxsw_thermal_module_fini(&thermal->tz_module_arr[i]); mlxsw_thermal_module_fini(&area->tz_module_arr[i]);
kfree(thermal->tz_module_arr); kfree(area->tz_module_arr);
} }
static int static int
...@@ -828,7 +847,8 @@ mlxsw_thermal_gearbox_tz_fini(struct mlxsw_thermal_module *gearbox_tz) ...@@ -828,7 +847,8 @@ mlxsw_thermal_gearbox_tz_fini(struct mlxsw_thermal_module *gearbox_tz)
static int static int
mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core, mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core,
struct mlxsw_thermal *thermal) struct mlxsw_thermal *thermal,
struct mlxsw_thermal_area *area)
{ {
enum mlxsw_reg_mgpir_device_type device_type; enum mlxsw_reg_mgpir_device_type device_type;
struct mlxsw_thermal_module *gearbox_tz; struct mlxsw_thermal_module *gearbox_tz;
...@@ -837,7 +857,7 @@ mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core, ...@@ -837,7 +857,7 @@ mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core,
int i; int i;
int err; int err;
mlxsw_reg_mgpir_pack(mgpir_pl, 0); mlxsw_reg_mgpir_pack(mgpir_pl, area->slot_index);
err = mlxsw_reg_query(core, MLXSW_REG(mgpir), mgpir_pl); err = mlxsw_reg_query(core, MLXSW_REG(mgpir), mgpir_pl);
if (err) if (err)
return err; return err;
...@@ -848,19 +868,20 @@ mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core, ...@@ -848,19 +868,20 @@ mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core,
!gbox_num) !gbox_num)
return 0; return 0;
thermal->tz_gearbox_num = gbox_num; area->tz_gearbox_num = gbox_num;
thermal->tz_gearbox_arr = kcalloc(thermal->tz_gearbox_num, area->tz_gearbox_arr = kcalloc(area->tz_gearbox_num,
sizeof(*thermal->tz_gearbox_arr), sizeof(*area->tz_gearbox_arr),
GFP_KERNEL); GFP_KERNEL);
if (!thermal->tz_gearbox_arr) if (!area->tz_gearbox_arr)
return -ENOMEM; return -ENOMEM;
for (i = 0; i < thermal->tz_gearbox_num; i++) { for (i = 0; i < area->tz_gearbox_num; i++) {
gearbox_tz = &thermal->tz_gearbox_arr[i]; gearbox_tz = &area->tz_gearbox_arr[i];
memcpy(gearbox_tz->trips, default_thermal_trips, memcpy(gearbox_tz->trips, default_thermal_trips,
sizeof(thermal->trips)); sizeof(thermal->trips));
gearbox_tz->module = i; gearbox_tz->module = i;
gearbox_tz->parent = thermal; gearbox_tz->parent = thermal;
gearbox_tz->slot_index = area->slot_index;
err = mlxsw_thermal_gearbox_tz_init(gearbox_tz); err = mlxsw_thermal_gearbox_tz_init(gearbox_tz);
if (err) if (err)
goto err_thermal_gearbox_tz_init; goto err_thermal_gearbox_tz_init;
...@@ -870,19 +891,20 @@ mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core, ...@@ -870,19 +891,20 @@ mlxsw_thermal_gearboxes_init(struct device *dev, struct mlxsw_core *core,
err_thermal_gearbox_tz_init: err_thermal_gearbox_tz_init:
for (i--; i >= 0; i--) for (i--; i >= 0; i--)
mlxsw_thermal_gearbox_tz_fini(&thermal->tz_gearbox_arr[i]); mlxsw_thermal_gearbox_tz_fini(&area->tz_gearbox_arr[i]);
kfree(thermal->tz_gearbox_arr); kfree(area->tz_gearbox_arr);
return err; return err;
} }
static void static void
mlxsw_thermal_gearboxes_fini(struct mlxsw_thermal *thermal) mlxsw_thermal_gearboxes_fini(struct mlxsw_thermal *thermal,
struct mlxsw_thermal_area *area)
{ {
int i; int i;
for (i = thermal->tz_gearbox_num - 1; i >= 0; i--) for (i = area->tz_gearbox_num - 1; i >= 0; i--)
mlxsw_thermal_gearbox_tz_fini(&thermal->tz_gearbox_arr[i]); mlxsw_thermal_gearbox_tz_fini(&area->tz_gearbox_arr[i]);
kfree(thermal->tz_gearbox_arr); kfree(area->tz_gearbox_arr);
} }
int mlxsw_thermal_init(struct mlxsw_core *core, int mlxsw_thermal_init(struct mlxsw_core *core,
...@@ -892,19 +914,29 @@ int mlxsw_thermal_init(struct mlxsw_core *core, ...@@ -892,19 +914,29 @@ int mlxsw_thermal_init(struct mlxsw_core *core,
char mfcr_pl[MLXSW_REG_MFCR_LEN] = { 0 }; char mfcr_pl[MLXSW_REG_MFCR_LEN] = { 0 };
enum mlxsw_reg_mfcr_pwm_frequency freq; enum mlxsw_reg_mfcr_pwm_frequency freq;
struct device *dev = bus_info->dev; struct device *dev = bus_info->dev;
char mgpir_pl[MLXSW_REG_MGPIR_LEN];
struct mlxsw_thermal *thermal; struct mlxsw_thermal *thermal;
u8 pwm_active, num_of_slots;
u16 tacho_active; u16 tacho_active;
u8 pwm_active;
int err, i; int err, i;
thermal = devm_kzalloc(dev, sizeof(*thermal), mlxsw_reg_mgpir_pack(mgpir_pl, 0);
GFP_KERNEL); err = mlxsw_reg_query(core, MLXSW_REG(mgpir), mgpir_pl);
if (err)
return err;
mlxsw_reg_mgpir_unpack(mgpir_pl, NULL, NULL, NULL, NULL,
&num_of_slots);
thermal = kzalloc(struct_size(thermal, line_cards, num_of_slots + 1),
GFP_KERNEL);
if (!thermal) if (!thermal)
return -ENOMEM; return -ENOMEM;
thermal->core = core; thermal->core = core;
thermal->bus_info = bus_info; thermal->bus_info = bus_info;
memcpy(thermal->trips, default_thermal_trips, sizeof(thermal->trips)); memcpy(thermal->trips, default_thermal_trips, sizeof(thermal->trips));
thermal->line_cards[0].slot_index = 0;
err = mlxsw_reg_query(thermal->core, MLXSW_REG(mfcr), mfcr_pl); err = mlxsw_reg_query(thermal->core, MLXSW_REG(mfcr), mfcr_pl);
if (err) { if (err) {
...@@ -970,11 +1002,13 @@ int mlxsw_thermal_init(struct mlxsw_core *core, ...@@ -970,11 +1002,13 @@ int mlxsw_thermal_init(struct mlxsw_core *core,
goto err_thermal_zone_device_register; goto err_thermal_zone_device_register;
} }
err = mlxsw_thermal_modules_init(dev, core, thermal); err = mlxsw_thermal_modules_init(dev, core, thermal,
&thermal->line_cards[0]);
if (err) if (err)
goto err_thermal_modules_init; goto err_thermal_modules_init;
err = mlxsw_thermal_gearboxes_init(dev, core, thermal); err = mlxsw_thermal_gearboxes_init(dev, core, thermal,
&thermal->line_cards[0]);
if (err) if (err)
goto err_thermal_gearboxes_init; goto err_thermal_gearboxes_init;
...@@ -986,9 +1020,9 @@ int mlxsw_thermal_init(struct mlxsw_core *core, ...@@ -986,9 +1020,9 @@ int mlxsw_thermal_init(struct mlxsw_core *core,
return 0; return 0;
err_thermal_zone_device_enable: err_thermal_zone_device_enable:
mlxsw_thermal_gearboxes_fini(thermal); mlxsw_thermal_gearboxes_fini(thermal, &thermal->line_cards[0]);
err_thermal_gearboxes_init: err_thermal_gearboxes_init:
mlxsw_thermal_modules_fini(thermal); mlxsw_thermal_modules_fini(thermal, &thermal->line_cards[0]);
err_thermal_modules_init: err_thermal_modules_init:
if (thermal->tzdev) { if (thermal->tzdev) {
thermal_zone_device_unregister(thermal->tzdev); thermal_zone_device_unregister(thermal->tzdev);
...@@ -1001,7 +1035,7 @@ int mlxsw_thermal_init(struct mlxsw_core *core, ...@@ -1001,7 +1035,7 @@ int mlxsw_thermal_init(struct mlxsw_core *core,
thermal_cooling_device_unregister(thermal->cdevs[i]); thermal_cooling_device_unregister(thermal->cdevs[i]);
err_reg_write: err_reg_write:
err_reg_query: err_reg_query:
devm_kfree(dev, thermal); kfree(thermal);
return err; return err;
} }
...@@ -1009,8 +1043,8 @@ void mlxsw_thermal_fini(struct mlxsw_thermal *thermal) ...@@ -1009,8 +1043,8 @@ void mlxsw_thermal_fini(struct mlxsw_thermal *thermal)
{ {
int i; int i;
mlxsw_thermal_gearboxes_fini(thermal); mlxsw_thermal_gearboxes_fini(thermal, &thermal->line_cards[0]);
mlxsw_thermal_modules_fini(thermal); mlxsw_thermal_modules_fini(thermal, &thermal->line_cards[0]);
if (thermal->tzdev) { if (thermal->tzdev) {
thermal_zone_device_unregister(thermal->tzdev); thermal_zone_device_unregister(thermal->tzdev);
thermal->tzdev = NULL; thermal->tzdev = NULL;
...@@ -1023,5 +1057,5 @@ void mlxsw_thermal_fini(struct mlxsw_thermal *thermal) ...@@ -1023,5 +1057,5 @@ void mlxsw_thermal_fini(struct mlxsw_thermal *thermal)
} }
} }
devm_kfree(thermal->bus_info->dev, thermal); kfree(thermal);
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment