c94b7f9bab
Recent commit52fde2c07d
("nvme: set dma alignment to dword") has caused a regression on our platform. It turned out that the nvme_get_log() method invocation caused the nvme_hwmon_data structure instance corruption. In particular the nvme_hwmon_data.ctrl pointer was overwritten either with zeros or with garbage. After some research we discovered that the problem happened even before the actual NVME DMA execution, but during the buffer mapping. Since our platform is DMA-noncoherent, the mapping implied the cache-line invalidations or write-backs depending on the DMA-direction parameter. In case of the NVME SMART log getting the DMA was performed from-device-to-memory, thus the cache-invalidation was activated during the buffer mapping. Since the log-buffer isn't cache-line aligned, the cache-invalidation caused the neighbour data to be discarded. The neighbouring data turned to be the data surrounding the buffer in the framework of the nvme_hwmon_data structure. In order to fix that we need to make sure that the whole log-buffer is defined within the cache-line-aligned memory region so the cache-invalidation procedure wouldn't involve the adjacent data. One of the option to guarantee that is to kmalloc the DMA-buffer [1]. Seeing the rest of the NVME core driver prefer that method it has been chosen to fix this problem too. Note after a deeper researches we found out that the denoted commit wasn't a root cause of the problem. It just revealed the invalidity by activating the DMA-based NVME SMART log getting performed in the framework of the NVME hwmon driver. The problem was here since the initial commit of the driver. [1] Documentation/core-api/dma-api-howto.rst Fixes:400b6a7b13
("nvme: Add hardware monitoring support") Signed-off-by: Serge Semin <Sergey.Semin@baikalelectronics.ru> Signed-off-by: Christoph Hellwig <hch@lst.de>
280 lines
6.2 KiB
C
280 lines
6.2 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* NVM Express hardware monitoring support
|
|
* Copyright (c) 2019, Guenter Roeck
|
|
*/
|
|
|
|
#include <linux/hwmon.h>
|
|
#include <linux/units.h>
|
|
#include <asm/unaligned.h>
|
|
|
|
#include "nvme.h"
|
|
|
|
struct nvme_hwmon_data {
|
|
struct nvme_ctrl *ctrl;
|
|
struct nvme_smart_log *log;
|
|
struct mutex read_lock;
|
|
};
|
|
|
|
static int nvme_get_temp_thresh(struct nvme_ctrl *ctrl, int sensor, bool under,
|
|
long *temp)
|
|
{
|
|
unsigned int threshold = sensor << NVME_TEMP_THRESH_SELECT_SHIFT;
|
|
u32 status;
|
|
int ret;
|
|
|
|
if (under)
|
|
threshold |= NVME_TEMP_THRESH_TYPE_UNDER;
|
|
|
|
ret = nvme_get_features(ctrl, NVME_FEAT_TEMP_THRESH, threshold, NULL, 0,
|
|
&status);
|
|
if (ret > 0)
|
|
return -EIO;
|
|
if (ret < 0)
|
|
return ret;
|
|
*temp = kelvin_to_millicelsius(status & NVME_TEMP_THRESH_MASK);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int nvme_set_temp_thresh(struct nvme_ctrl *ctrl, int sensor, bool under,
|
|
long temp)
|
|
{
|
|
unsigned int threshold = sensor << NVME_TEMP_THRESH_SELECT_SHIFT;
|
|
int ret;
|
|
|
|
temp = millicelsius_to_kelvin(temp);
|
|
threshold |= clamp_val(temp, 0, NVME_TEMP_THRESH_MASK);
|
|
|
|
if (under)
|
|
threshold |= NVME_TEMP_THRESH_TYPE_UNDER;
|
|
|
|
ret = nvme_set_features(ctrl, NVME_FEAT_TEMP_THRESH, threshold, NULL, 0,
|
|
NULL);
|
|
if (ret > 0)
|
|
return -EIO;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int nvme_hwmon_get_smart_log(struct nvme_hwmon_data *data)
|
|
{
|
|
return nvme_get_log(data->ctrl, NVME_NSID_ALL, NVME_LOG_SMART, 0,
|
|
NVME_CSI_NVM, data->log, sizeof(*data->log), 0);
|
|
}
|
|
|
|
static int nvme_hwmon_read(struct device *dev, enum hwmon_sensor_types type,
|
|
u32 attr, int channel, long *val)
|
|
{
|
|
struct nvme_hwmon_data *data = dev_get_drvdata(dev);
|
|
struct nvme_smart_log *log = data->log;
|
|
int temp;
|
|
int err;
|
|
|
|
/*
|
|
* First handle attributes which don't require us to read
|
|
* the smart log.
|
|
*/
|
|
switch (attr) {
|
|
case hwmon_temp_max:
|
|
return nvme_get_temp_thresh(data->ctrl, channel, false, val);
|
|
case hwmon_temp_min:
|
|
return nvme_get_temp_thresh(data->ctrl, channel, true, val);
|
|
case hwmon_temp_crit:
|
|
*val = kelvin_to_millicelsius(data->ctrl->cctemp);
|
|
return 0;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
mutex_lock(&data->read_lock);
|
|
err = nvme_hwmon_get_smart_log(data);
|
|
if (err)
|
|
goto unlock;
|
|
|
|
switch (attr) {
|
|
case hwmon_temp_input:
|
|
if (!channel)
|
|
temp = get_unaligned_le16(log->temperature);
|
|
else
|
|
temp = le16_to_cpu(log->temp_sensor[channel - 1]);
|
|
*val = kelvin_to_millicelsius(temp);
|
|
break;
|
|
case hwmon_temp_alarm:
|
|
*val = !!(log->critical_warning & NVME_SMART_CRIT_TEMPERATURE);
|
|
break;
|
|
default:
|
|
err = -EOPNOTSUPP;
|
|
break;
|
|
}
|
|
unlock:
|
|
mutex_unlock(&data->read_lock);
|
|
return err;
|
|
}
|
|
|
|
static int nvme_hwmon_write(struct device *dev, enum hwmon_sensor_types type,
|
|
u32 attr, int channel, long val)
|
|
{
|
|
struct nvme_hwmon_data *data = dev_get_drvdata(dev);
|
|
|
|
switch (attr) {
|
|
case hwmon_temp_max:
|
|
return nvme_set_temp_thresh(data->ctrl, channel, false, val);
|
|
case hwmon_temp_min:
|
|
return nvme_set_temp_thresh(data->ctrl, channel, true, val);
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return -EOPNOTSUPP;
|
|
}
|
|
|
|
static const char * const nvme_hwmon_sensor_names[] = {
|
|
"Composite",
|
|
"Sensor 1",
|
|
"Sensor 2",
|
|
"Sensor 3",
|
|
"Sensor 4",
|
|
"Sensor 5",
|
|
"Sensor 6",
|
|
"Sensor 7",
|
|
"Sensor 8",
|
|
};
|
|
|
|
static int nvme_hwmon_read_string(struct device *dev,
|
|
enum hwmon_sensor_types type, u32 attr,
|
|
int channel, const char **str)
|
|
{
|
|
*str = nvme_hwmon_sensor_names[channel];
|
|
return 0;
|
|
}
|
|
|
|
static umode_t nvme_hwmon_is_visible(const void *_data,
|
|
enum hwmon_sensor_types type,
|
|
u32 attr, int channel)
|
|
{
|
|
const struct nvme_hwmon_data *data = _data;
|
|
|
|
switch (attr) {
|
|
case hwmon_temp_crit:
|
|
if (!channel && data->ctrl->cctemp)
|
|
return 0444;
|
|
break;
|
|
case hwmon_temp_max:
|
|
case hwmon_temp_min:
|
|
if ((!channel && data->ctrl->wctemp) ||
|
|
(channel && data->log->temp_sensor[channel - 1])) {
|
|
if (data->ctrl->quirks &
|
|
NVME_QUIRK_NO_TEMP_THRESH_CHANGE)
|
|
return 0444;
|
|
return 0644;
|
|
}
|
|
break;
|
|
case hwmon_temp_alarm:
|
|
if (!channel)
|
|
return 0444;
|
|
break;
|
|
case hwmon_temp_input:
|
|
case hwmon_temp_label:
|
|
if (!channel || data->log->temp_sensor[channel - 1])
|
|
return 0444;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static const struct hwmon_channel_info *nvme_hwmon_info[] = {
|
|
HWMON_CHANNEL_INFO(chip, HWMON_C_REGISTER_TZ),
|
|
HWMON_CHANNEL_INFO(temp,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_CRIT | HWMON_T_LABEL | HWMON_T_ALARM,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL,
|
|
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
|
|
HWMON_T_LABEL),
|
|
NULL
|
|
};
|
|
|
|
static const struct hwmon_ops nvme_hwmon_ops = {
|
|
.is_visible = nvme_hwmon_is_visible,
|
|
.read = nvme_hwmon_read,
|
|
.read_string = nvme_hwmon_read_string,
|
|
.write = nvme_hwmon_write,
|
|
};
|
|
|
|
static const struct hwmon_chip_info nvme_hwmon_chip_info = {
|
|
.ops = &nvme_hwmon_ops,
|
|
.info = nvme_hwmon_info,
|
|
};
|
|
|
|
int nvme_hwmon_init(struct nvme_ctrl *ctrl)
|
|
{
|
|
struct device *dev = ctrl->device;
|
|
struct nvme_hwmon_data *data;
|
|
struct device *hwmon;
|
|
int err;
|
|
|
|
data = kzalloc(sizeof(*data), GFP_KERNEL);
|
|
if (!data)
|
|
return -ENOMEM;
|
|
|
|
data->log = kzalloc(sizeof(*data->log), GFP_KERNEL);
|
|
if (!data->log) {
|
|
err = -ENOMEM;
|
|
goto err_free_data;
|
|
}
|
|
|
|
data->ctrl = ctrl;
|
|
mutex_init(&data->read_lock);
|
|
|
|
err = nvme_hwmon_get_smart_log(data);
|
|
if (err) {
|
|
dev_warn(dev, "Failed to read smart log (error %d)\n", err);
|
|
goto err_free_log;
|
|
}
|
|
|
|
hwmon = hwmon_device_register_with_info(dev, "nvme",
|
|
data, &nvme_hwmon_chip_info,
|
|
NULL);
|
|
if (IS_ERR(hwmon)) {
|
|
dev_warn(dev, "Failed to instantiate hwmon device\n");
|
|
err = PTR_ERR(hwmon);
|
|
goto err_free_log;
|
|
}
|
|
ctrl->hwmon_device = hwmon;
|
|
return 0;
|
|
|
|
err_free_log:
|
|
kfree(data->log);
|
|
err_free_data:
|
|
kfree(data);
|
|
return err;
|
|
}
|
|
|
|
void nvme_hwmon_exit(struct nvme_ctrl *ctrl)
|
|
{
|
|
if (ctrl->hwmon_device) {
|
|
struct nvme_hwmon_data *data =
|
|
dev_get_drvdata(ctrl->hwmon_device);
|
|
|
|
hwmon_device_unregister(ctrl->hwmon_device);
|
|
ctrl->hwmon_device = NULL;
|
|
kfree(data->log);
|
|
kfree(data);
|
|
}
|
|
}
|