162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2017 Chelsio Communications. All rights reserved. 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Written by: Ganesh Goudar (ganeshgr@chelsio.com) 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include "cxgb4.h" 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#define CXGB4_NUM_TRIPS 1 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_cistatic int cxgb4_thermal_get_temp(struct thermal_zone_device *tzdev, 1362306a36Sopenharmony_ci int *temp) 1462306a36Sopenharmony_ci{ 1562306a36Sopenharmony_ci struct adapter *adap = thermal_zone_device_priv(tzdev); 1662306a36Sopenharmony_ci u32 param, val; 1762306a36Sopenharmony_ci int ret; 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci param = (FW_PARAMS_MNEM_V(FW_PARAMS_MNEM_DEV) | 2062306a36Sopenharmony_ci FW_PARAMS_PARAM_X_V(FW_PARAMS_PARAM_DEV_DIAG) | 2162306a36Sopenharmony_ci FW_PARAMS_PARAM_Y_V(FW_PARAM_DEV_DIAG_TMP)); 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci ret = t4_query_params(adap, adap->mbox, adap->pf, 0, 1, 2462306a36Sopenharmony_ci ¶m, &val); 2562306a36Sopenharmony_ci if (ret < 0 || val == 0) 2662306a36Sopenharmony_ci return -1; 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_ci *temp = val * 1000; 2962306a36Sopenharmony_ci return 0; 3062306a36Sopenharmony_ci} 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_cistatic struct thermal_zone_device_ops cxgb4_thermal_ops = { 3362306a36Sopenharmony_ci .get_temp = cxgb4_thermal_get_temp, 3462306a36Sopenharmony_ci}; 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_cistatic struct thermal_trip trip = { .type = THERMAL_TRIP_CRITICAL } ; 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_ciint cxgb4_thermal_init(struct adapter *adap) 3962306a36Sopenharmony_ci{ 4062306a36Sopenharmony_ci struct ch_thermal *ch_thermal = &adap->ch_thermal; 4162306a36Sopenharmony_ci char ch_tz_name[THERMAL_NAME_LENGTH]; 4262306a36Sopenharmony_ci int num_trip = CXGB4_NUM_TRIPS; 4362306a36Sopenharmony_ci u32 param, val; 4462306a36Sopenharmony_ci int ret; 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_ci /* on older firmwares we may not get the trip temperature, 4762306a36Sopenharmony_ci * set the num of trips to 0. 4862306a36Sopenharmony_ci */ 4962306a36Sopenharmony_ci param = (FW_PARAMS_MNEM_V(FW_PARAMS_MNEM_DEV) | 5062306a36Sopenharmony_ci FW_PARAMS_PARAM_X_V(FW_PARAMS_PARAM_DEV_DIAG) | 5162306a36Sopenharmony_ci FW_PARAMS_PARAM_Y_V(FW_PARAM_DEV_DIAG_MAXTMPTHRESH)); 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci ret = t4_query_params(adap, adap->mbox, adap->pf, 0, 1, 5462306a36Sopenharmony_ci ¶m, &val); 5562306a36Sopenharmony_ci if (ret < 0) { 5662306a36Sopenharmony_ci num_trip = 0; /* could not get trip temperature */ 5762306a36Sopenharmony_ci } else { 5862306a36Sopenharmony_ci trip.temperature = val * 1000; 5962306a36Sopenharmony_ci } 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_ci snprintf(ch_tz_name, sizeof(ch_tz_name), "cxgb4_%s", adap->name); 6262306a36Sopenharmony_ci ch_thermal->tzdev = thermal_zone_device_register_with_trips(ch_tz_name, &trip, num_trip, 6362306a36Sopenharmony_ci 0, adap, 6462306a36Sopenharmony_ci &cxgb4_thermal_ops, 6562306a36Sopenharmony_ci NULL, 0, 0); 6662306a36Sopenharmony_ci if (IS_ERR(ch_thermal->tzdev)) { 6762306a36Sopenharmony_ci ret = PTR_ERR(ch_thermal->tzdev); 6862306a36Sopenharmony_ci dev_err(adap->pdev_dev, "Failed to register thermal zone\n"); 6962306a36Sopenharmony_ci ch_thermal->tzdev = NULL; 7062306a36Sopenharmony_ci return ret; 7162306a36Sopenharmony_ci } 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci ret = thermal_zone_device_enable(ch_thermal->tzdev); 7462306a36Sopenharmony_ci if (ret) { 7562306a36Sopenharmony_ci dev_err(adap->pdev_dev, "Failed to enable thermal zone\n"); 7662306a36Sopenharmony_ci thermal_zone_device_unregister(adap->ch_thermal.tzdev); 7762306a36Sopenharmony_ci return ret; 7862306a36Sopenharmony_ci } 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_ci return 0; 8162306a36Sopenharmony_ci} 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ciint cxgb4_thermal_remove(struct adapter *adap) 8462306a36Sopenharmony_ci{ 8562306a36Sopenharmony_ci if (adap->ch_thermal.tzdev) { 8662306a36Sopenharmony_ci thermal_zone_device_unregister(adap->ch_thermal.tzdev); 8762306a36Sopenharmony_ci adap->ch_thermal.tzdev = NULL; 8862306a36Sopenharmony_ci } 8962306a36Sopenharmony_ci return 0; 9062306a36Sopenharmony_ci} 91