2024-01-29 04:19:28 -07:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
/*
|
2024-04-28 03:48:11 -07:00
|
|
|
* hwmon driver for NZXT Kraken X53/X63/X73, Z53/Z63/Z73 and 2023/2023 Elite all in one coolers.
|
|
|
|
* X53 and Z53 in code refer to all models in their respective series (shortened for brevity).
|
|
|
|
* 2023 models use the Z53 code paths.
|
2024-01-29 04:19:28 -07:00
|
|
|
*
|
|
|
|
* Copyright 2021 Jonas Malaco <jonas@protocubo.io>
|
|
|
|
* Copyright 2022 Aleksa Savic <savicaleksa83@gmail.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/debugfs.h>
|
|
|
|
#include <linux/hid.h>
|
|
|
|
#include <linux/hwmon.h>
|
|
|
|
#include <linux/hwmon-sysfs.h>
|
|
|
|
#include <linux/jiffies.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/mutex.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/wait.h>
|
2024-10-01 12:35:57 -07:00
|
|
|
#include <linux/unaligned.h>
|
2024-01-29 04:19:28 -07:00
|
|
|
|
|
|
|
#define USB_VENDOR_ID_NZXT 0x1e71
|
|
|
|
#define USB_PRODUCT_ID_X53 0x2007
|
|
|
|
#define USB_PRODUCT_ID_X53_SECOND 0x2014
|
|
|
|
#define USB_PRODUCT_ID_Z53 0x3008
|
2024-04-28 03:48:11 -07:00
|
|
|
#define USB_PRODUCT_ID_KRAKEN2023 0x300E
|
|
|
|
#define USB_PRODUCT_ID_KRAKEN2023_ELITE 0x300C
|
2024-01-29 04:19:28 -07:00
|
|
|
|
2024-04-28 03:48:11 -07:00
|
|
|
enum kinds { X53, Z53, KRAKEN2023 } __packed;
|
2024-01-29 04:19:28 -07:00
|
|
|
enum pwm_enable { off, manual, curve } __packed;
|
|
|
|
|
|
|
|
#define DRIVER_NAME "nzxt_kraken3"
|
|
|
|
#define STATUS_REPORT_ID 0x75
|
|
|
|
#define FIRMWARE_REPORT_ID 0x11
|
|
|
|
#define STATUS_VALIDITY 2000 /* In ms, equivalent to period of four status reports */
|
|
|
|
#define CUSTOM_CURVE_POINTS 40 /* For temps from 20C to 59C (critical temp) */
|
|
|
|
#define PUMP_DUTY_MIN 20 /* In percent */
|
|
|
|
|
|
|
|
/* Sensor report offsets for Kraken X53 and Z53 */
|
|
|
|
#define TEMP_SENSOR_START_OFFSET 15
|
|
|
|
#define TEMP_SENSOR_END_OFFSET 16
|
|
|
|
#define PUMP_SPEED_OFFSET 17
|
|
|
|
#define PUMP_DUTY_OFFSET 19
|
|
|
|
|
|
|
|
/* Firmware version report offset for Kraken X53 and Z53 */
|
|
|
|
#define FIRMWARE_VERSION_OFFSET 17
|
|
|
|
|
|
|
|
/* Sensor report offsets for Kraken Z53 */
|
|
|
|
#define Z53_FAN_SPEED_OFFSET 23
|
|
|
|
#define Z53_FAN_DUTY_OFFSET 25
|
|
|
|
|
|
|
|
/* Report offsets for control commands for Kraken X53 and Z53 */
|
|
|
|
#define SET_DUTY_ID_OFFSET 1
|
|
|
|
|
|
|
|
/* Control commands and their lengths for Kraken X53 and Z53 */
|
|
|
|
|
|
|
|
/* Last byte sets the report interval at 0.5s */
|
|
|
|
static const u8 set_interval_cmd[] = { 0x70, 0x02, 0x01, 0xB8, 1 };
|
|
|
|
static const u8 finish_init_cmd[] = { 0x70, 0x01 };
|
|
|
|
static const u8 __maybe_unused get_fw_version_cmd[] = { 0x10, 0x01 };
|
|
|
|
static const u8 set_pump_duty_cmd_header[] = { 0x72, 0x00, 0x00, 0x00 };
|
|
|
|
static const u8 z53_get_status_cmd[] = { 0x74, 0x01 };
|
|
|
|
|
|
|
|
#define SET_INTERVAL_CMD_LENGTH 5
|
|
|
|
#define FINISH_INIT_CMD_LENGTH 2
|
|
|
|
#define GET_FW_VERSION_CMD_LENGTH 2
|
|
|
|
#define MAX_REPORT_LENGTH 64
|
|
|
|
#define MIN_REPORT_LENGTH 20
|
|
|
|
#define SET_CURVE_DUTY_CMD_HEADER_LENGTH 4
|
|
|
|
/* 4 byte header and 40 duty offsets */
|
|
|
|
#define SET_CURVE_DUTY_CMD_LENGTH (4 + 40)
|
|
|
|
#define Z53_GET_STATUS_CMD_LENGTH 2
|
|
|
|
|
|
|
|
static const char *const kraken3_temp_label[] = {
|
|
|
|
"Coolant temp",
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *const kraken3_fan_label[] = {
|
|
|
|
"Pump speed",
|
|
|
|
"Fan speed"
|
|
|
|
};
|
|
|
|
|
|
|
|
struct kraken3_channel_info {
|
|
|
|
enum pwm_enable mode;
|
|
|
|
|
|
|
|
/* Both values are PWM */
|
|
|
|
u16 reported_duty;
|
|
|
|
u16 fixed_duty; /* Manually set fixed duty */
|
|
|
|
|
|
|
|
u8 pwm_points[CUSTOM_CURVE_POINTS];
|
|
|
|
};
|
|
|
|
|
|
|
|
struct kraken3_data {
|
|
|
|
struct hid_device *hdev;
|
|
|
|
struct device *hwmon_dev;
|
|
|
|
struct dentry *debugfs;
|
|
|
|
struct mutex buffer_lock; /* For locking access to buffer */
|
|
|
|
struct mutex z53_status_request_lock;
|
|
|
|
struct completion fw_version_processed;
|
|
|
|
/*
|
|
|
|
* For X53 devices, tracks whether an initial (one) sensor report was received to
|
|
|
|
* make fancontrol not bail outright. For Z53 devices, whether a status report
|
|
|
|
* was processed after requesting one.
|
|
|
|
*/
|
|
|
|
struct completion status_report_processed;
|
|
|
|
/* For locking the above completion */
|
|
|
|
spinlock_t status_completion_lock;
|
|
|
|
|
|
|
|
u8 *buffer;
|
|
|
|
struct kraken3_channel_info channel_info[2]; /* Pump and fan */
|
|
|
|
bool is_device_faulty;
|
|
|
|
|
|
|
|
/* Sensor values */
|
|
|
|
s32 temp_input[1];
|
|
|
|
u16 fan_input[2];
|
|
|
|
|
|
|
|
enum kinds kind;
|
|
|
|
u8 firmware_version[3];
|
|
|
|
|
|
|
|
unsigned long updated; /* jiffies */
|
|
|
|
};
|
|
|
|
|
|
|
|
static umode_t kraken3_is_visible(const void *data, enum hwmon_sensor_types type, u32 attr,
|
|
|
|
int channel)
|
|
|
|
{
|
|
|
|
const struct kraken3_data *priv = data;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case hwmon_temp:
|
|
|
|
if (channel < 1)
|
|
|
|
return 0444;
|
|
|
|
break;
|
|
|
|
case hwmon_fan:
|
|
|
|
switch (priv->kind) {
|
|
|
|
case X53:
|
|
|
|
/* Just the pump */
|
|
|
|
if (channel < 1)
|
|
|
|
return 0444;
|
|
|
|
break;
|
|
|
|
case Z53:
|
2024-04-28 03:48:11 -07:00
|
|
|
case KRAKEN2023:
|
2024-01-29 04:19:28 -07:00
|
|
|
/* Pump and fan */
|
|
|
|
if (channel < 2)
|
|
|
|
return 0444;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case hwmon_pwm:
|
|
|
|
switch (attr) {
|
|
|
|
case hwmon_pwm_enable:
|
|
|
|
case hwmon_pwm_input:
|
|
|
|
switch (priv->kind) {
|
|
|
|
case X53:
|
|
|
|
/* Just the pump */
|
|
|
|
if (channel < 1)
|
|
|
|
return 0644;
|
|
|
|
break;
|
|
|
|
case Z53:
|
2024-04-28 03:48:11 -07:00
|
|
|
case KRAKEN2023:
|
2024-01-29 04:19:28 -07:00
|
|
|
/* Pump and fan */
|
|
|
|
if (channel < 2)
|
|
|
|
return 0644;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Writes the command to the device with the rest of the report (up to 64 bytes) filled
|
|
|
|
* with zeroes.
|
|
|
|
*/
|
|
|
|
static int kraken3_write_expanded(struct kraken3_data *priv, const u8 *cmd, int cmd_length)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&priv->buffer_lock);
|
|
|
|
|
|
|
|
memcpy_and_pad(priv->buffer, MAX_REPORT_LENGTH, cmd, cmd_length, 0x00);
|
|
|
|
ret = hid_hw_output_report(priv->hdev, priv->buffer, MAX_REPORT_LENGTH);
|
|
|
|
|
|
|
|
mutex_unlock(&priv->buffer_lock);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_percent_to_pwm(long val)
|
|
|
|
{
|
|
|
|
return DIV_ROUND_CLOSEST(val * 255, 100);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_pwm_to_percent(long val, int channel)
|
|
|
|
{
|
|
|
|
int percent_value;
|
|
|
|
|
|
|
|
if (val < 0 || val > 255)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
percent_value = DIV_ROUND_CLOSEST(val * 100, 255);
|
|
|
|
|
|
|
|
/* Bring up pump duty to min value if needed */
|
|
|
|
if (channel == 0 && percent_value < PUMP_DUTY_MIN)
|
|
|
|
percent_value = PUMP_DUTY_MIN;
|
|
|
|
|
|
|
|
return percent_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_read_x53(struct kraken3_data *priv)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (completion_done(&priv->status_report_processed))
|
|
|
|
/*
|
|
|
|
* We're here because data is stale. This means that sensor reports haven't
|
|
|
|
* been received for some time in kraken3_raw_event(). On X-series sensor data
|
|
|
|
* can't be manually requested, so return an error.
|
|
|
|
*/
|
|
|
|
return -ENODATA;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Data needs to be read, but a sensor report wasn't yet received. It's usually
|
|
|
|
* fancontrol that requests data this early and it exits if it reads an error code.
|
|
|
|
* So, wait for the first report to be parsed (but up to STATUS_VALIDITY).
|
|
|
|
* This does not concern the Z series devices, because they send a sensor report
|
|
|
|
* only when requested.
|
|
|
|
*/
|
|
|
|
ret = wait_for_completion_interruptible_timeout(&priv->status_report_processed,
|
|
|
|
msecs_to_jiffies(STATUS_VALIDITY));
|
|
|
|
if (ret == 0)
|
|
|
|
return -ETIMEDOUT;
|
|
|
|
else if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* The first sensor report was parsed on time and reading can continue */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-04-28 03:48:11 -07:00
|
|
|
/* Covers Z53 and KRAKEN2023 device kinds */
|
2024-01-29 04:19:28 -07:00
|
|
|
static int kraken3_read_z53(struct kraken3_data *priv)
|
|
|
|
{
|
|
|
|
int ret = mutex_lock_interruptible(&priv->z53_status_request_lock);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (!time_after(jiffies, priv->updated + msecs_to_jiffies(STATUS_VALIDITY))) {
|
|
|
|
/* Data is up to date */
|
|
|
|
goto unlock_and_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Disable interrupts for a moment to safely reinit the completion,
|
|
|
|
* as hidraw calls could have allowed one or more readers to complete.
|
|
|
|
*/
|
|
|
|
spin_lock_bh(&priv->status_completion_lock);
|
|
|
|
reinit_completion(&priv->status_report_processed);
|
|
|
|
spin_unlock_bh(&priv->status_completion_lock);
|
|
|
|
|
|
|
|
/* Send command for getting status */
|
|
|
|
ret = kraken3_write_expanded(priv, z53_get_status_cmd, Z53_GET_STATUS_CMD_LENGTH);
|
|
|
|
if (ret < 0)
|
|
|
|
goto unlock_and_return;
|
|
|
|
|
|
|
|
/* Wait for completion from kraken3_raw_event() */
|
|
|
|
ret = wait_for_completion_interruptible_timeout(&priv->status_report_processed,
|
|
|
|
msecs_to_jiffies(STATUS_VALIDITY));
|
|
|
|
if (ret == 0)
|
|
|
|
ret = -ETIMEDOUT;
|
|
|
|
|
|
|
|
unlock_and_return:
|
|
|
|
mutex_unlock(&priv->z53_status_request_lock);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_read(struct device *dev, enum hwmon_sensor_types type, u32 attr, int channel,
|
|
|
|
long *val)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv = dev_get_drvdata(dev);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (time_after(jiffies, priv->updated + msecs_to_jiffies(STATUS_VALIDITY))) {
|
|
|
|
if (priv->kind == X53)
|
|
|
|
ret = kraken3_read_x53(priv);
|
|
|
|
else
|
|
|
|
ret = kraken3_read_z53(priv);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (priv->is_device_faulty)
|
|
|
|
return -ENODATA;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case hwmon_temp:
|
|
|
|
*val = priv->temp_input[channel];
|
|
|
|
break;
|
|
|
|
case hwmon_fan:
|
|
|
|
*val = priv->fan_input[channel];
|
|
|
|
break;
|
|
|
|
case hwmon_pwm:
|
|
|
|
switch (attr) {
|
|
|
|
case hwmon_pwm_enable:
|
|
|
|
*val = priv->channel_info[channel].mode;
|
|
|
|
break;
|
|
|
|
case hwmon_pwm_input:
|
|
|
|
*val = priv->channel_info[channel].reported_duty;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_read_string(struct device *dev, enum hwmon_sensor_types type, u32 attr,
|
|
|
|
int channel, const char **str)
|
|
|
|
{
|
|
|
|
switch (type) {
|
|
|
|
case hwmon_temp:
|
|
|
|
*str = kraken3_temp_label[channel];
|
|
|
|
break;
|
|
|
|
case hwmon_fan:
|
|
|
|
*str = kraken3_fan_label[channel];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Writes custom curve to device */
|
|
|
|
static int kraken3_write_curve(struct kraken3_data *priv, u8 *curve_array, int channel)
|
|
|
|
{
|
|
|
|
u8 fixed_duty_cmd[SET_CURVE_DUTY_CMD_LENGTH];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Copy command header */
|
|
|
|
memcpy(fixed_duty_cmd, set_pump_duty_cmd_header, SET_CURVE_DUTY_CMD_HEADER_LENGTH);
|
|
|
|
|
|
|
|
/* Set the correct ID for writing pump/fan duty (0x01 or 0x02, respectively) */
|
|
|
|
fixed_duty_cmd[SET_DUTY_ID_OFFSET] = channel + 1;
|
|
|
|
|
2024-04-28 03:48:11 -07:00
|
|
|
if (priv->kind == KRAKEN2023) {
|
|
|
|
/* These require 1s in the next one or two slots after SET_DUTY_ID_OFFSET */
|
|
|
|
fixed_duty_cmd[SET_DUTY_ID_OFFSET + 1] = 1;
|
|
|
|
if (channel == 1) /* Fan */
|
|
|
|
fixed_duty_cmd[SET_DUTY_ID_OFFSET + 2] = 1;
|
|
|
|
}
|
|
|
|
|
2024-01-29 04:19:28 -07:00
|
|
|
/* Copy curve to command */
|
|
|
|
memcpy(fixed_duty_cmd + SET_CURVE_DUTY_CMD_HEADER_LENGTH, curve_array, CUSTOM_CURVE_POINTS);
|
|
|
|
|
|
|
|
ret = kraken3_write_expanded(priv, fixed_duty_cmd, SET_CURVE_DUTY_CMD_LENGTH);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_write_fixed_duty(struct kraken3_data *priv, long val, int channel)
|
|
|
|
{
|
|
|
|
u8 fixed_curve_points[CUSTOM_CURVE_POINTS];
|
|
|
|
int ret, percent_val, i;
|
|
|
|
|
|
|
|
percent_val = kraken3_pwm_to_percent(val, channel);
|
|
|
|
if (percent_val < 0)
|
|
|
|
return percent_val;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The devices can only control the duty through a curve.
|
|
|
|
* Since we're setting a fixed duty here, fill the whole curve
|
|
|
|
* (ranging from 20C to 59C) with the same duty, except for
|
|
|
|
* the last point, the critical temperature, where it's maxed
|
|
|
|
* out for safety.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Fill the custom curve with the fixed value we're setting */
|
|
|
|
for (i = 0; i < CUSTOM_CURVE_POINTS - 1; i++)
|
|
|
|
fixed_curve_points[i] = percent_val;
|
|
|
|
|
|
|
|
/* Force duty to 100% at critical temp */
|
|
|
|
fixed_curve_points[CUSTOM_CURVE_POINTS - 1] = 100;
|
|
|
|
|
|
|
|
/* Write the fixed duty curve to the device */
|
|
|
|
ret = kraken3_write_curve(priv, fixed_curve_points, channel);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_write(struct device *dev, enum hwmon_sensor_types type, u32 attr, int channel,
|
|
|
|
long val)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv = dev_get_drvdata(dev);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case hwmon_pwm:
|
|
|
|
switch (attr) {
|
|
|
|
case hwmon_pwm_input:
|
|
|
|
/* Remember the last set fixed duty for channel */
|
|
|
|
priv->channel_info[channel].fixed_duty = val;
|
|
|
|
|
|
|
|
if (priv->channel_info[channel].mode == manual) {
|
|
|
|
ret = kraken3_write_fixed_duty(priv, val, channel);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Lock onto this value and report it until next interrupt status
|
|
|
|
* report is received, so userspace tools can continue to work.
|
|
|
|
*/
|
|
|
|
priv->channel_info[channel].reported_duty = val;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case hwmon_pwm_enable:
|
|
|
|
if (val < 0 || val > 2)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
switch (val) {
|
|
|
|
case 0:
|
|
|
|
/* Set channel to 100%, direct duty value */
|
|
|
|
ret = kraken3_write_fixed_duty(priv, 255, channel);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* We don't control anything anymore */
|
|
|
|
priv->channel_info[channel].mode = off;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
/* Apply the last known direct duty value */
|
|
|
|
ret =
|
|
|
|
kraken3_write_fixed_duty(priv,
|
|
|
|
priv->channel_info[channel].fixed_duty,
|
|
|
|
channel);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
priv->channel_info[channel].mode = manual;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* Apply the curve and note as enabled */
|
|
|
|
ret =
|
|
|
|
kraken3_write_curve(priv,
|
|
|
|
priv->channel_info[channel].pwm_points,
|
|
|
|
channel);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
priv->channel_info[channel].mode = curve;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t kraken3_fan_curve_pwm_store(struct device *dev, struct device_attribute *attr,
|
|
|
|
const char *buf, size_t count)
|
|
|
|
{
|
|
|
|
struct sensor_device_attribute_2 *dev_attr = to_sensor_dev_attr_2(attr);
|
|
|
|
struct kraken3_data *priv = dev_get_drvdata(dev);
|
|
|
|
long val;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (kstrtol(buf, 10, &val) < 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
val = kraken3_pwm_to_percent(val, dev_attr->nr);
|
|
|
|
if (val < 0)
|
|
|
|
return val;
|
|
|
|
|
|
|
|
priv->channel_info[dev_attr->nr].pwm_points[dev_attr->index] = val;
|
|
|
|
|
|
|
|
if (priv->channel_info[dev_attr->nr].mode == curve) {
|
|
|
|
/* Apply the curve */
|
|
|
|
ret =
|
|
|
|
kraken3_write_curve(priv,
|
|
|
|
priv->channel_info[dev_attr->nr].pwm_points, dev_attr->nr);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static umode_t kraken3_curve_props_are_visible(struct kobject *kobj, struct attribute *attr,
|
|
|
|
int index)
|
|
|
|
{
|
|
|
|
struct device *dev = kobj_to_dev(kobj);
|
|
|
|
struct kraken3_data *priv = dev_get_drvdata(dev);
|
|
|
|
|
2024-04-28 03:48:11 -07:00
|
|
|
/* X53 does not have a fan */
|
|
|
|
if (index >= CUSTOM_CURVE_POINTS && priv->kind == X53)
|
2024-01-29 04:19:28 -07:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
return attr->mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Custom pump curve from 20C to 59C (critical temp) */
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point1_pwm, kraken3_fan_curve_pwm, 0, 0);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point2_pwm, kraken3_fan_curve_pwm, 0, 1);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point3_pwm, kraken3_fan_curve_pwm, 0, 2);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point4_pwm, kraken3_fan_curve_pwm, 0, 3);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point5_pwm, kraken3_fan_curve_pwm, 0, 4);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point6_pwm, kraken3_fan_curve_pwm, 0, 5);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point7_pwm, kraken3_fan_curve_pwm, 0, 6);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point8_pwm, kraken3_fan_curve_pwm, 0, 7);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point9_pwm, kraken3_fan_curve_pwm, 0, 8);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point10_pwm, kraken3_fan_curve_pwm, 0, 9);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point11_pwm, kraken3_fan_curve_pwm, 0, 10);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point12_pwm, kraken3_fan_curve_pwm, 0, 11);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point13_pwm, kraken3_fan_curve_pwm, 0, 12);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point14_pwm, kraken3_fan_curve_pwm, 0, 13);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point15_pwm, kraken3_fan_curve_pwm, 0, 14);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point16_pwm, kraken3_fan_curve_pwm, 0, 15);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point17_pwm, kraken3_fan_curve_pwm, 0, 16);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point18_pwm, kraken3_fan_curve_pwm, 0, 17);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point19_pwm, kraken3_fan_curve_pwm, 0, 18);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point20_pwm, kraken3_fan_curve_pwm, 0, 19);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point21_pwm, kraken3_fan_curve_pwm, 0, 20);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point22_pwm, kraken3_fan_curve_pwm, 0, 21);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point23_pwm, kraken3_fan_curve_pwm, 0, 22);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point24_pwm, kraken3_fan_curve_pwm, 0, 23);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point25_pwm, kraken3_fan_curve_pwm, 0, 24);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point26_pwm, kraken3_fan_curve_pwm, 0, 25);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point27_pwm, kraken3_fan_curve_pwm, 0, 26);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point28_pwm, kraken3_fan_curve_pwm, 0, 27);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point29_pwm, kraken3_fan_curve_pwm, 0, 28);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point30_pwm, kraken3_fan_curve_pwm, 0, 29);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point31_pwm, kraken3_fan_curve_pwm, 0, 30);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point32_pwm, kraken3_fan_curve_pwm, 0, 31);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point33_pwm, kraken3_fan_curve_pwm, 0, 32);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point34_pwm, kraken3_fan_curve_pwm, 0, 33);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point35_pwm, kraken3_fan_curve_pwm, 0, 34);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point36_pwm, kraken3_fan_curve_pwm, 0, 35);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point37_pwm, kraken3_fan_curve_pwm, 0, 36);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point38_pwm, kraken3_fan_curve_pwm, 0, 37);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point39_pwm, kraken3_fan_curve_pwm, 0, 38);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point40_pwm, kraken3_fan_curve_pwm, 0, 39);
|
|
|
|
|
|
|
|
/* Custom fan curve from 20C to 59C (critical temp) */
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point1_pwm, kraken3_fan_curve_pwm, 1, 0);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point2_pwm, kraken3_fan_curve_pwm, 1, 1);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point3_pwm, kraken3_fan_curve_pwm, 1, 2);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point4_pwm, kraken3_fan_curve_pwm, 1, 3);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point5_pwm, kraken3_fan_curve_pwm, 1, 4);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point6_pwm, kraken3_fan_curve_pwm, 1, 5);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point7_pwm, kraken3_fan_curve_pwm, 1, 6);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point8_pwm, kraken3_fan_curve_pwm, 1, 7);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point9_pwm, kraken3_fan_curve_pwm, 1, 8);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point10_pwm, kraken3_fan_curve_pwm, 1, 9);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point11_pwm, kraken3_fan_curve_pwm, 1, 10);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point12_pwm, kraken3_fan_curve_pwm, 1, 11);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point13_pwm, kraken3_fan_curve_pwm, 1, 12);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point14_pwm, kraken3_fan_curve_pwm, 1, 13);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point15_pwm, kraken3_fan_curve_pwm, 1, 14);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point16_pwm, kraken3_fan_curve_pwm, 1, 15);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point17_pwm, kraken3_fan_curve_pwm, 1, 16);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point18_pwm, kraken3_fan_curve_pwm, 1, 17);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point19_pwm, kraken3_fan_curve_pwm, 1, 18);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point20_pwm, kraken3_fan_curve_pwm, 1, 19);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point21_pwm, kraken3_fan_curve_pwm, 1, 20);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point22_pwm, kraken3_fan_curve_pwm, 1, 21);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point23_pwm, kraken3_fan_curve_pwm, 1, 22);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point24_pwm, kraken3_fan_curve_pwm, 1, 23);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point25_pwm, kraken3_fan_curve_pwm, 1, 24);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point26_pwm, kraken3_fan_curve_pwm, 1, 25);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point27_pwm, kraken3_fan_curve_pwm, 1, 26);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point28_pwm, kraken3_fan_curve_pwm, 1, 27);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point29_pwm, kraken3_fan_curve_pwm, 1, 28);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point30_pwm, kraken3_fan_curve_pwm, 1, 29);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point31_pwm, kraken3_fan_curve_pwm, 1, 30);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point32_pwm, kraken3_fan_curve_pwm, 1, 31);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point33_pwm, kraken3_fan_curve_pwm, 1, 32);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point34_pwm, kraken3_fan_curve_pwm, 1, 33);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point35_pwm, kraken3_fan_curve_pwm, 1, 34);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point36_pwm, kraken3_fan_curve_pwm, 1, 35);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point37_pwm, kraken3_fan_curve_pwm, 1, 36);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point38_pwm, kraken3_fan_curve_pwm, 1, 37);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point39_pwm, kraken3_fan_curve_pwm, 1, 38);
|
|
|
|
static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point40_pwm, kraken3_fan_curve_pwm, 1, 39);
|
|
|
|
|
|
|
|
static struct attribute *kraken3_curve_attrs[] = {
|
|
|
|
/* Pump control curve */
|
|
|
|
&sensor_dev_attr_temp1_auto_point1_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point2_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point3_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point4_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point5_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point6_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point7_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point8_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point9_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point10_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point11_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point12_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point13_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point14_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point15_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point16_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point17_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point18_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point19_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point20_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point21_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point22_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point23_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point24_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point25_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point26_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point27_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point28_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point29_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point30_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point31_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point32_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point33_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point34_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point35_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point36_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point37_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point38_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point39_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp1_auto_point40_pwm.dev_attr.attr,
|
|
|
|
/* Fan control curve (Z53 only) */
|
|
|
|
&sensor_dev_attr_temp2_auto_point1_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point2_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point3_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point4_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point5_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point6_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point7_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point8_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point9_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point10_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point11_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point12_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point13_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point14_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point15_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point16_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point17_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point18_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point19_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point20_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point21_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point22_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point23_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point24_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point25_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point26_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point27_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point28_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point29_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point30_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point31_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point32_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point33_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point34_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point35_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point36_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point37_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point38_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point39_pwm.dev_attr.attr,
|
|
|
|
&sensor_dev_attr_temp2_auto_point40_pwm.dev_attr.attr,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct attribute_group kraken3_curves_group = {
|
|
|
|
.attrs = kraken3_curve_attrs,
|
|
|
|
.is_visible = kraken3_curve_props_are_visible
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct attribute_group *kraken3_groups[] = {
|
|
|
|
&kraken3_curves_group,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct hwmon_ops kraken3_hwmon_ops = {
|
|
|
|
.is_visible = kraken3_is_visible,
|
|
|
|
.read = kraken3_read,
|
|
|
|
.read_string = kraken3_read_string,
|
|
|
|
.write = kraken3_write
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct hwmon_channel_info *kraken3_info[] = {
|
|
|
|
HWMON_CHANNEL_INFO(temp,
|
|
|
|
HWMON_T_INPUT | HWMON_T_LABEL),
|
|
|
|
HWMON_CHANNEL_INFO(fan,
|
|
|
|
HWMON_F_INPUT | HWMON_F_LABEL,
|
|
|
|
HWMON_F_INPUT | HWMON_F_LABEL,
|
|
|
|
HWMON_F_INPUT | HWMON_F_LABEL,
|
|
|
|
HWMON_F_INPUT | HWMON_F_LABEL),
|
|
|
|
HWMON_CHANNEL_INFO(pwm,
|
|
|
|
HWMON_PWM_INPUT | HWMON_PWM_ENABLE,
|
|
|
|
HWMON_PWM_INPUT | HWMON_PWM_ENABLE),
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct hwmon_chip_info kraken3_chip_info = {
|
|
|
|
.ops = &kraken3_hwmon_ops,
|
|
|
|
.info = kraken3_info,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int kraken3_raw_event(struct hid_device *hdev, struct hid_report *report, u8 *data, int size)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv = hid_get_drvdata(hdev);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (size < MIN_REPORT_LENGTH)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (report->id == FIRMWARE_REPORT_ID) {
|
|
|
|
/* Read firmware version */
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
priv->firmware_version[i] = data[FIRMWARE_VERSION_OFFSET + i];
|
|
|
|
|
|
|
|
if (!completion_done(&priv->fw_version_processed))
|
|
|
|
complete_all(&priv->fw_version_processed);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (report->id != STATUS_REPORT_ID)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (data[TEMP_SENSOR_START_OFFSET] == 0xff && data[TEMP_SENSOR_END_OFFSET] == 0xff) {
|
|
|
|
hid_err_once(hdev,
|
|
|
|
"firmware or device is possibly damaged (is SATA power connected?), not parsing reports\n");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Mark first X-series device report as received,
|
|
|
|
* as well as all for Z-series, if faulty.
|
|
|
|
*/
|
|
|
|
spin_lock(&priv->status_completion_lock);
|
|
|
|
if (priv->kind != X53 || !completion_done(&priv->status_report_processed)) {
|
|
|
|
priv->is_device_faulty = true;
|
|
|
|
complete_all(&priv->status_report_processed);
|
|
|
|
}
|
|
|
|
spin_unlock(&priv->status_completion_lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Received normal data */
|
|
|
|
priv->is_device_faulty = false;
|
|
|
|
|
|
|
|
/* Temperature and fan sensor readings */
|
|
|
|
priv->temp_input[0] =
|
|
|
|
data[TEMP_SENSOR_START_OFFSET] * 1000 + data[TEMP_SENSOR_END_OFFSET] * 100;
|
|
|
|
|
|
|
|
priv->fan_input[0] = get_unaligned_le16(data + PUMP_SPEED_OFFSET);
|
|
|
|
priv->channel_info[0].reported_duty = kraken3_percent_to_pwm(data[PUMP_DUTY_OFFSET]);
|
|
|
|
|
|
|
|
spin_lock(&priv->status_completion_lock);
|
|
|
|
if (priv->kind == X53 && !completion_done(&priv->status_report_processed)) {
|
|
|
|
/* Mark first X-series device report as received */
|
|
|
|
complete_all(&priv->status_report_processed);
|
2024-04-28 03:48:11 -07:00
|
|
|
} else if (priv->kind == Z53 || priv->kind == KRAKEN2023) {
|
|
|
|
/* Additional readings for Z53 and KRAKEN2023 */
|
2024-01-29 04:19:28 -07:00
|
|
|
priv->fan_input[1] = get_unaligned_le16(data + Z53_FAN_SPEED_OFFSET);
|
|
|
|
priv->channel_info[1].reported_duty =
|
|
|
|
kraken3_percent_to_pwm(data[Z53_FAN_DUTY_OFFSET]);
|
|
|
|
|
|
|
|
if (!completion_done(&priv->status_report_processed))
|
|
|
|
complete_all(&priv->status_report_processed);
|
|
|
|
}
|
|
|
|
spin_unlock(&priv->status_completion_lock);
|
|
|
|
|
|
|
|
priv->updated = jiffies;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_init_device(struct hid_device *hdev)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv = hid_get_drvdata(hdev);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Set the polling interval */
|
|
|
|
ret = kraken3_write_expanded(priv, set_interval_cmd, SET_INTERVAL_CMD_LENGTH);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* Finalize the init process */
|
|
|
|
ret = kraken3_write_expanded(priv, finish_init_cmd, FINISH_INIT_CMD_LENGTH);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_get_fw_ver(struct hid_device *hdev)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv = hid_get_drvdata(hdev);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = kraken3_write_expanded(priv, get_fw_version_cmd, GET_FW_VERSION_CMD_LENGTH);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = wait_for_completion_interruptible_timeout(&priv->fw_version_processed,
|
|
|
|
msecs_to_jiffies(STATUS_VALIDITY));
|
|
|
|
if (ret == 0)
|
|
|
|
return -ETIMEDOUT;
|
|
|
|
else if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __maybe_unused kraken3_reset_resume(struct hid_device *hdev)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = kraken3_init_device(hdev);
|
|
|
|
if (ret)
|
|
|
|
hid_err(hdev, "req init (reset_resume) failed with %d\n", ret);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int firmware_version_show(struct seq_file *seqf, void *unused)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv = seqf->private;
|
|
|
|
|
|
|
|
seq_printf(seqf, "%u.%u.%u\n", priv->firmware_version[0], priv->firmware_version[1],
|
|
|
|
priv->firmware_version[2]);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
DEFINE_SHOW_ATTRIBUTE(firmware_version);
|
|
|
|
|
2024-04-28 03:48:10 -07:00
|
|
|
static void kraken3_debugfs_init(struct kraken3_data *priv, const char *device_name)
|
2024-01-29 04:19:28 -07:00
|
|
|
{
|
|
|
|
char name[64];
|
|
|
|
|
|
|
|
if (!priv->firmware_version[0])
|
|
|
|
return; /* Nothing to display in debugfs */
|
|
|
|
|
2024-04-28 03:48:10 -07:00
|
|
|
scnprintf(name, sizeof(name), "%s_%s-%s", DRIVER_NAME, device_name,
|
2024-01-29 04:19:28 -07:00
|
|
|
dev_name(&priv->hdev->dev));
|
|
|
|
|
|
|
|
priv->debugfs = debugfs_create_dir(name, NULL);
|
|
|
|
debugfs_create_file("firmware_version", 0444, priv->debugfs, priv, &firmware_version_fops);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int kraken3_probe(struct hid_device *hdev, const struct hid_device_id *id)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv;
|
2024-04-28 03:48:10 -07:00
|
|
|
const char *device_name;
|
2024-01-29 04:19:28 -07:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
priv = devm_kzalloc(&hdev->dev, sizeof(*priv), GFP_KERNEL);
|
|
|
|
if (!priv)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
priv->hdev = hdev;
|
|
|
|
hid_set_drvdata(hdev, priv);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize ->updated to STATUS_VALIDITY seconds in the past, making
|
|
|
|
* the initial empty data invalid for kraken3_read without the need for
|
|
|
|
* a special case there.
|
|
|
|
*/
|
|
|
|
priv->updated = jiffies - msecs_to_jiffies(STATUS_VALIDITY);
|
|
|
|
|
|
|
|
ret = hid_parse(hdev);
|
|
|
|
if (ret) {
|
|
|
|
hid_err(hdev, "hid parse failed with %d\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable hidraw so existing user-space tools can continue to work */
|
|
|
|
ret = hid_hw_start(hdev, HID_CONNECT_HIDRAW);
|
|
|
|
if (ret) {
|
|
|
|
hid_err(hdev, "hid hw start failed with %d\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = hid_hw_open(hdev);
|
|
|
|
if (ret) {
|
|
|
|
hid_err(hdev, "hid hw open failed with %d\n", ret);
|
|
|
|
goto fail_and_stop;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (hdev->product) {
|
|
|
|
case USB_PRODUCT_ID_X53:
|
|
|
|
case USB_PRODUCT_ID_X53_SECOND:
|
|
|
|
priv->kind = X53;
|
2024-04-28 03:48:10 -07:00
|
|
|
device_name = "x53";
|
2024-01-29 04:19:28 -07:00
|
|
|
break;
|
|
|
|
case USB_PRODUCT_ID_Z53:
|
|
|
|
priv->kind = Z53;
|
2024-04-28 03:48:10 -07:00
|
|
|
device_name = "z53";
|
2024-01-29 04:19:28 -07:00
|
|
|
break;
|
2024-04-28 03:48:11 -07:00
|
|
|
case USB_PRODUCT_ID_KRAKEN2023:
|
|
|
|
priv->kind = KRAKEN2023;
|
|
|
|
device_name = "kraken2023";
|
|
|
|
break;
|
|
|
|
case USB_PRODUCT_ID_KRAKEN2023_ELITE:
|
|
|
|
priv->kind = KRAKEN2023;
|
|
|
|
device_name = "kraken2023elite";
|
|
|
|
break;
|
2024-01-29 04:19:28 -07:00
|
|
|
default:
|
2024-05-12 09:07:01 -07:00
|
|
|
ret = -ENODEV;
|
|
|
|
goto fail_and_close;
|
2024-01-29 04:19:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
priv->buffer = devm_kzalloc(&hdev->dev, MAX_REPORT_LENGTH, GFP_KERNEL);
|
|
|
|
if (!priv->buffer) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto fail_and_close;
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_init(&priv->buffer_lock);
|
|
|
|
mutex_init(&priv->z53_status_request_lock);
|
|
|
|
init_completion(&priv->fw_version_processed);
|
|
|
|
init_completion(&priv->status_report_processed);
|
|
|
|
spin_lock_init(&priv->status_completion_lock);
|
|
|
|
|
|
|
|
hid_device_io_start(hdev);
|
|
|
|
ret = kraken3_init_device(hdev);
|
|
|
|
if (ret < 0) {
|
|
|
|
hid_err(hdev, "device init failed with %d\n", ret);
|
|
|
|
goto fail_and_close;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = kraken3_get_fw_ver(hdev);
|
|
|
|
if (ret < 0)
|
|
|
|
hid_warn(hdev, "fw version request failed with %d\n", ret);
|
|
|
|
|
2024-04-28 03:48:10 -07:00
|
|
|
priv->hwmon_dev = hwmon_device_register_with_info(&hdev->dev, device_name, priv,
|
2024-01-29 04:19:28 -07:00
|
|
|
&kraken3_chip_info, kraken3_groups);
|
|
|
|
if (IS_ERR(priv->hwmon_dev)) {
|
|
|
|
ret = PTR_ERR(priv->hwmon_dev);
|
|
|
|
hid_err(hdev, "hwmon registration failed with %d\n", ret);
|
|
|
|
goto fail_and_close;
|
|
|
|
}
|
|
|
|
|
2024-04-28 03:48:10 -07:00
|
|
|
kraken3_debugfs_init(priv, device_name);
|
2024-01-29 04:19:28 -07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail_and_close:
|
|
|
|
hid_hw_close(hdev);
|
|
|
|
fail_and_stop:
|
|
|
|
hid_hw_stop(hdev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void kraken3_remove(struct hid_device *hdev)
|
|
|
|
{
|
|
|
|
struct kraken3_data *priv = hid_get_drvdata(hdev);
|
|
|
|
|
|
|
|
debugfs_remove_recursive(priv->debugfs);
|
|
|
|
hwmon_device_unregister(priv->hwmon_dev);
|
|
|
|
|
|
|
|
hid_hw_close(hdev);
|
|
|
|
hid_hw_stop(hdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct hid_device_id kraken3_table[] = {
|
|
|
|
/* NZXT Kraken X53/X63/X73 have two possible product IDs */
|
|
|
|
{ HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_X53) },
|
|
|
|
{ HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_X53_SECOND) },
|
|
|
|
{ HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_Z53) },
|
2024-04-28 03:48:11 -07:00
|
|
|
{ HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_KRAKEN2023) },
|
|
|
|
{ HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_KRAKEN2023_ELITE) },
|
2024-01-29 04:19:28 -07:00
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
MODULE_DEVICE_TABLE(hid, kraken3_table);
|
|
|
|
|
|
|
|
static struct hid_driver kraken3_driver = {
|
|
|
|
.name = DRIVER_NAME,
|
|
|
|
.id_table = kraken3_table,
|
|
|
|
.probe = kraken3_probe,
|
|
|
|
.remove = kraken3_remove,
|
|
|
|
.raw_event = kraken3_raw_event,
|
|
|
|
#ifdef CONFIG_PM
|
|
|
|
.reset_resume = kraken3_reset_resume,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init kraken3_init(void)
|
|
|
|
{
|
|
|
|
return hid_register_driver(&kraken3_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit kraken3_exit(void)
|
|
|
|
{
|
|
|
|
hid_unregister_driver(&kraken3_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* When compiled into the kernel, initialize after the HID bus */
|
|
|
|
late_initcall(kraken3_init);
|
|
|
|
module_exit(kraken3_exit);
|
|
|
|
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_AUTHOR("Jonas Malaco <jonas@protocubo.io>");
|
|
|
|
MODULE_AUTHOR("Aleksa Savic <savicaleksa83@gmail.com>");
|
|
|
|
MODULE_DESCRIPTION("Hwmon driver for NZXT Kraken X53/X63/X73, Z53/Z63/Z73 coolers");
|