Re: [PATCH] hwmon: add driver for Aquacomputer D5 Next

From: Guenter Roeck
Date: Fri Aug 27 2021 - 11:47:51 EST


On Fri, Aug 27, 2021 at 10:25:05AM +0200, Aleksa Savic wrote:
> This driver exposes hardware sensors of the Aquacomputer D5 Next
> watercooling pump, which communicates through a proprietary USB HID
> protocol.
>
> Available sensors are pump and fan speed, power, voltage and current, as
> well as coolant temperature. Also available through debugfs are the serial
> number, firmware version and power-on count.
>
> Attaching a fan is optional and allows it to be controlled using
> temperature curves directly from the pump. If it's not connected,
> the fan-related sensors will report zeroes.
>
> The pump can be configured either through software or via its physical
> interface. Configuring the pump through this driver is not implemented,
> as it seems to require sending it a complete configuration. That
> includes addressable RGB LEDs, for which there is no standard sysfs
> interface. Thus, that task is better suited for userspace tools.
>
> This driver has been tested on x86_64, both in-kernel and as a module.
>
> Signed-off-by: Aleksa Savic <savicaleksa83@xxxxxxxxx>
> ---
> Documentation/hwmon/aquacomputer_d5next.rst | 61 ++++
> Documentation/hwmon/index.rst | 1 +
> MAINTAINERS | 7 +
> drivers/hwmon/Kconfig | 10 +
> drivers/hwmon/Makefile | 1 +
> drivers/hwmon/aquacomputer_d5next.c | 366 ++++++++++++++++++++
> 6 files changed, 446 insertions(+)
> create mode 100644 Documentation/hwmon/aquacomputer_d5next.rst
> create mode 100644 drivers/hwmon/aquacomputer_d5next.c
>
> diff --git a/Documentation/hwmon/aquacomputer_d5next.rst b/Documentation/hwmon/aquacomputer_d5next.rst
> new file mode 100644
> index 000000000000..1f4bb4ba2e4b
> --- /dev/null
> +++ b/Documentation/hwmon/aquacomputer_d5next.rst
> @@ -0,0 +1,61 @@
> +.. SPDX-License-Identifier: GPL-2.0-or-later
> +
> +Kernel driver aquacomputer-d5next
> +=================================
> +
> +Supported devices:
> +
> +* Aquacomputer D5 Next watercooling pump
> +
> +Author: Aleksa Savic
> +
> +Description
> +-----------
> +
> +This driver exposes hardware sensors of the Aquacomputer D5 Next watercooling
> +pump, which communicates through a proprietary USB HID protocol.
> +
> +Available sensors are pump and fan speed, power, voltage and current, as
> +well as coolant temperature. Also available through debugfs are the serial
> +number, firmware version and power-on count.
> +
> +Attaching a fan is optional and allows it to be controlled using temperature
> +curves directly from the pump. If it's not connected, the fan-related sensors
> +will report zeroes.
> +
> +The pump can be configured either through software or via its physical
> +interface. Configuring the pump through this driver is not implemented, as it
> +seems to require sending it a complete configuration. That includes addressable
> +RGB LEDs, for which there is no standard sysfs interface. Thus, that task is
> +better suited for userspace tools.
> +
> +Usage notes
> +-----------
> +
> +The pump communicates via HID reports. The driver is loaded automatically by
> +the kernel and supports hotswapping.
> +
> +Sysfs entries
> +-------------
> +
> +============ =============================================
> +temp1_input Coolant temperature (in millidegrees Celsius)
> +fan1_input Pump speed (in RPM)
> +fan2_input Fan speed (in RPM)
> +power1_input Pump power (in micro Watts)
> +power2_input Fan power (in micro Watts)
> +in0_input Pump voltage (in milli Volts)
> +in1_input Fan voltage (in milli Volts)
> +in2_input +5V rail voltage (in milli Volts)
> +curr1_input Pump current (in milli Amperes)
> +curr2_input Fan current (in milli Amperes)
> +============ =============================================
> +
> +Debugfs entries
> +---------------
> +
> +================ ===============================================
> +serial_number Serial number of the pump
> +firmware_version Version of installed firmware
> +power_cycles Count of how many times the pump was powered on
> +================ ===============================================
> diff --git a/Documentation/hwmon/index.rst b/Documentation/hwmon/index.rst
> index bc01601ea81a..77bfb2e2e8a9 100644
> --- a/Documentation/hwmon/index.rst
> +++ b/Documentation/hwmon/index.rst
> @@ -39,6 +39,7 @@ Hardware Monitoring Kernel Drivers
> adt7475
> aht10
> amc6821
> + aquacomputer_d5next
> asb100
> asc7621
> aspeed-pwm-tacho
> diff --git a/MAINTAINERS b/MAINTAINERS
> index d7b4f32875a9..ec0aa0dcf635 100644
> --- a/MAINTAINERS
> +++ b/MAINTAINERS
> @@ -1316,6 +1316,13 @@ L: linux-media@xxxxxxxxxxxxxxx
> S: Maintained
> F: drivers/media/i2c/aptina-pll.*
>
> +AQUACOMPUTER D5 NEXT PUMP SENSOR DRIVER
> +M: Aleksa Savic <savicaleksa83@xxxxxxxxx>
> +L: linux-hwmon@xxxxxxxxxxxxxxx
> +S: Maintained
> +F: Documentation/hwmon/aquacomputer_d5next.rst
> +F: drivers/hwmon/aquacomputer_d5next.c
> +
> AQUANTIA ETHERNET DRIVER (atlantic)
> M: Igor Russkikh <irusskikh@xxxxxxxxxxx>
> L: netdev@xxxxxxxxxxxxxxx
> diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig
> index e3675377bc5d..2bd563850f87 100644
> --- a/drivers/hwmon/Kconfig
> +++ b/drivers/hwmon/Kconfig
> @@ -254,6 +254,16 @@ config SENSORS_AHT10
> This driver can also be built as a module. If so, the module
> will be called aht10.
>
> +config SENSORS_AQUACOMPUTER_D5NEXT
> + tristate "Aquacomputer D5 Next watercooling pump"
> + depends on USB_HID
> + help
> + If you say yes here you get support for the Aquacomputer D5 Next
> + watercooling pump sensors.
> +
> + This driver can also be built as a module. If so, the module
> + will be called aquacomputer_d5next.
> +
> config SENSORS_AS370
> tristate "Synaptics AS370 SoC hardware monitoring driver"
> help
> diff --git a/drivers/hwmon/Makefile b/drivers/hwmon/Makefile
> index d712c61c1f5e..790a611a3188 100644
> --- a/drivers/hwmon/Makefile
> +++ b/drivers/hwmon/Makefile
> @@ -47,6 +47,7 @@ obj-$(CONFIG_SENSORS_ADT7475) += adt7475.o
> obj-$(CONFIG_SENSORS_AHT10) += aht10.o
> obj-$(CONFIG_SENSORS_AMD_ENERGY) += amd_energy.o
> obj-$(CONFIG_SENSORS_APPLESMC) += applesmc.o
> +obj-$(CONFIG_SENSORS_AQUACOMPUTER_D5NEXT) += aquacomputer_d5next.o
> obj-$(CONFIG_SENSORS_ARM_SCMI) += scmi-hwmon.o
> obj-$(CONFIG_SENSORS_ARM_SCPI) += scpi-hwmon.o
> obj-$(CONFIG_SENSORS_AS370) += as370-hwmon.o
> diff --git a/drivers/hwmon/aquacomputer_d5next.c b/drivers/hwmon/aquacomputer_d5next.c
> new file mode 100644
> index 000000000000..0f831b0eb94c
> --- /dev/null
> +++ b/drivers/hwmon/aquacomputer_d5next.c
> @@ -0,0 +1,366 @@
> +// SPDX-License-Identifier: GPL-2.0+
> +/*
> + * hwmon driver for Aquacomputer D5 Next watercooling pump
> + *
> + * The D5 Next sends HID reports (with ID 0x01) every second to report sensor values
> + * (coolant temperature, pump and fan speed, voltage, current and power). It responds to
> + * Get_Report requests, but returns a dummy value of no use.
> + *
> + * Copyright 2021 Aleksa Savic <savicaleksa83@xxxxxxxxx>
> + */
> +
> +#include <asm/unaligned.h>
> +#include <linux/debugfs.h>
> +#include <linux/hid.h>
> +#include <linux/hwmon.h>
> +#include <linux/jiffies.h>
> +#include <linux/module.h>

#include <linux/seq_file.h>

> +
> +#define DRIVER_NAME "aquacomputer-d5next"
> +
> +#define D5NEXT_STATUS_REPORT_ID 0x01
> +#define D5NEXT_STATUS_UPDATE_INTERVAL 1 /* In seconds */
> +
> +/* Register offsets for the D5 Next pump */
> +
> +#define D5NEXT_SERIAL_FIRST_PART 3
> +#define D5NEXT_SERIAL_SECOND_PART 5
> +#define D5NEXT_FIRMWARE_VERSION 13

Please always use

#define<space>WHAT<tab>value

with aligned values.

> +#define D5NEXT_POWER_CYCLES 24
> +
> +#define D5NEXT_COOLANT_TEMP 87
> +
> +#define D5NEXT_PUMP_SPEED 116
> +#define D5NEXT_FAN_SPEED 103
> +
> +#define D5NEXT_PUMP_POWER 114
> +#define D5NEXT_FAN_POWER 101
> +
> +#define D5NEXT_PUMP_VOLTAGE 110
> +#define D5NEXT_FAN_VOLTAGE 97
> +#define D5NEXT_5V_VOLTAGE 57
> +
> +#define D5NEXT_PUMP_CURRENT 112
> +#define D5NEXT_FAN_CURRENT 99
> +
> +/* Labels for provided values */
> +
> +#define L_COOLANT_TEMP "Coolant temp"
> +
> +#define L_PUMP_SPEED "Pump speed"
> +#define L_FAN_SPEED "Fan speed"
> +
> +#define L_PUMP_POWER "Pump power"
> +#define L_FAN_POWER "Fan power"
> +
> +#define L_PUMP_VOLTAGE "Pump voltage"
> +#define L_FAN_VOLTAGE "Fan voltage"
> +#define L_5V_VOLTAGE "+5V voltage"
> +
> +#define L_PUMP_CURRENT "Pump current"
> +#define L_FAN_CURRENT "Fan current"
> +
> +static const char *const label_temp[] = {
> + L_COOLANT_TEMP,
> +};
> +
> +static const char *const label_speeds[] = {
> + L_PUMP_SPEED,
> + L_FAN_SPEED,
> +};
> +
> +static const char *const label_power[] = {
> + L_PUMP_POWER,
> + L_FAN_POWER,
> +};
> +
> +static const char *const label_voltages[] = {
> + L_PUMP_VOLTAGE,
> + L_FAN_VOLTAGE,
> + L_5V_VOLTAGE,
> +};
> +
> +static const char *const label_current[] = {
> + L_PUMP_CURRENT,
> + L_FAN_CURRENT,
> +};
> +
> +struct d5next_data {
> + struct hid_device *hdev;
> + struct device *hwmon_dev;
> + struct dentry *debugfs;
> + s32 temp_input[1];

This doesn't have to be an array.

> + u16 speed_input[2];
> + u32 power_input[2];
> + u16 voltage_input[3];
> + u16 current_input[2];
> + u32 serial_number[2];
> + u16 firmware_version;
> + u32 power_cycles; /* How many times the device was powered on */
> + unsigned long updated;
> +};
> +
> +static umode_t d5next_is_visible(const void *data, enum hwmon_sensor_types type, u32 attr,
> + int channel)
> +{
> + return 0444;
> +}
> +
> +static int d5next_read(struct device *dev, enum hwmon_sensor_types type, u32 attr, int channel,
> + long *val)
> +{
> + struct d5next_data *priv = dev_get_drvdata(dev);
> +
> + if (time_after(jiffies, priv->updated + D5NEXT_STATUS_UPDATE_INTERVAL * HZ))
> + return -ENODATA;

This seems a bit strict; it results in ENODATA if a single update
is missed or if it comes just a little late. I would suggest to relax
it a bit.

Also, D5NEXT_STATUS_UPDATE_INTERVAL is always used with "* HZ".
I would suggest to make that part of the define.

> +
> + switch (type) {
> + case hwmon_temp:
> + *val = priv->temp_input[channel];
> + break;
> + case hwmon_fan:
> + *val = priv->speed_input[channel];
> + break;
> + case hwmon_power:
> + *val = priv->power_input[channel];
> + break;
> + case hwmon_in:
> + *val = priv->voltage_input[channel];
> + break;
> + case hwmon_curr:
> + *val = priv->current_input[channel];
> + break;
> + default:
> + return -EOPNOTSUPP;
> + }
> +
> + return 0;
> +}
> +
> +static int d5next_read_string(struct device *dev, enum hwmon_sensor_types type, u32 attr,
> + int channel, const char **str)
> +{
> + switch (type) {
> + case hwmon_temp:
> + *str = label_temp[channel];
> + break;
> + case hwmon_fan:
> + *str = label_speeds[channel];
> + break;
> + case hwmon_power:
> + *str = label_power[channel];
> + break;
> + case hwmon_in:
> + *str = label_voltages[channel];
> + break;
> + case hwmon_curr:
> + *str = label_current[channel];
> + break;
> + default:
> + return -EOPNOTSUPP;
> + }
> +
> + return 0;
> +}
> +
> +static const struct hwmon_ops d5next_hwmon_ops = {
> + .is_visible = d5next_is_visible,
> + .read = d5next_read,
> + .read_string = d5next_read_string,
> +};
> +
> +static const struct hwmon_channel_info *d5next_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_CHANNEL_INFO(power, HWMON_P_INPUT | HWMON_P_LABEL, HWMON_P_INPUT | HWMON_P_LABEL),
> + HWMON_CHANNEL_INFO(in, HWMON_I_INPUT | HWMON_I_LABEL, HWMON_I_INPUT | HWMON_I_LABEL,
> + HWMON_I_INPUT | HWMON_I_LABEL),
> + HWMON_CHANNEL_INFO(curr, HWMON_C_INPUT | HWMON_C_LABEL, HWMON_C_INPUT | HWMON_C_LABEL),
> + NULL
> +};
> +
> +static const struct hwmon_chip_info d5next_chip_info = {
> + .ops = &d5next_hwmon_ops,
> + .info = d5next_info,
> +};
> +
> +static int d5next_raw_event(struct hid_device *hdev, struct hid_report *report, u8 *data, int size)
> +{
> + struct d5next_data *priv;
> +
> + if (report->id != D5NEXT_STATUS_REPORT_ID)
> + return 0;
> +
> + priv = hid_get_drvdata(hdev);
> +
> + /* Info provided with every report */
> +
> + priv->serial_number[0] = get_unaligned_be16(data + D5NEXT_SERIAL_FIRST_PART);
> + priv->serial_number[1] = get_unaligned_be16(data + D5NEXT_SERIAL_SECOND_PART);
> +
> + priv->firmware_version = get_unaligned_be16(data + D5NEXT_FIRMWARE_VERSION);
> + priv->power_cycles = get_unaligned_be32(data + D5NEXT_POWER_CYCLES);
> +
> + /* Sensor readings */
> +
> + priv->temp_input[0] = get_unaligned_be16(data + D5NEXT_COOLANT_TEMP) * 10;
> +
> + priv->speed_input[0] = get_unaligned_be16(data + D5NEXT_PUMP_SPEED);
> + priv->speed_input[1] = get_unaligned_be16(data + D5NEXT_FAN_SPEED);
> +
> + priv->power_input[0] = get_unaligned_be16(data + D5NEXT_PUMP_POWER) * 10000;
> + priv->power_input[1] = get_unaligned_be16(data + D5NEXT_FAN_POWER) * 10000;
> +
> + priv->voltage_input[0] = get_unaligned_be16(data + D5NEXT_PUMP_VOLTAGE) * 10;
> + priv->voltage_input[1] = get_unaligned_be16(data + D5NEXT_FAN_VOLTAGE) * 10;
> + priv->voltage_input[2] = get_unaligned_be16(data + D5NEXT_5V_VOLTAGE) * 10;
> +
> + priv->current_input[0] = get_unaligned_be16(data + D5NEXT_PUMP_CURRENT);
> + priv->current_input[1] = get_unaligned_be16(data + D5NEXT_FAN_CURRENT);
> +
> + priv->updated = jiffies;
> +
> + return 0;
> +}
> +
> +#ifdef CONFIG_DEBUG_FS
> +
> +static int serial_number_show(struct seq_file *seqf, void *unused)
> +{
> + struct d5next_data *priv = seqf->private;
> +
> + seq_printf(seqf, "%05u-%05u\n", priv->serial_number[0], priv->serial_number[1]);
> +
> + return 0;
> +}
> +DEFINE_SHOW_ATTRIBUTE(serial_number);
> +
> +static int firmware_version_show(struct seq_file *seqf, void *unused)
> +{
> + struct d5next_data *priv = seqf->private;
> +
> + seq_printf(seqf, "%u\n", priv->firmware_version);
> +
> + return 0;
> +}
> +DEFINE_SHOW_ATTRIBUTE(firmware_version);
> +
> +static int power_cycles_show(struct seq_file *seqf, void *unused)
> +{
> + struct d5next_data *priv = seqf->private;
> +
> + seq_printf(seqf, "%u\n", priv->power_cycles);
> +
> + return 0;
> +}
> +DEFINE_SHOW_ATTRIBUTE(power_cycles);
> +
> +static void d5next_debugfs_init(struct d5next_data *priv)
> +{
> + char name[32];
> +
> + scnprintf(name, sizeof(name), "%s-%s", DRIVER_NAME, dev_name(&priv->hdev->dev));
> +
> + priv->debugfs = debugfs_create_dir(name, NULL);
> + debugfs_create_file("serial_number", 0444, priv->debugfs, priv, &serial_number_fops);
> + debugfs_create_file("firmware_version", 0444, priv->debugfs, priv, &firmware_version_fops);
> + debugfs_create_file("power_cycles", 0444, priv->debugfs, priv, &power_cycles_fops);
> +}
> +
> +#else
> +
> +static void d5next_debugfs_init(struct d5next_data *priv)
> +{
> +}
> +
> +#endif
> +
> +static int d5next_probe(struct hid_device *hdev, const struct hid_device_id *id)
> +{
> + struct d5next_data *priv;
> + int ret;
> +
> + priv = devm_kzalloc(&hdev->dev, sizeof(*priv), GFP_KERNEL);
> + if (!priv)
> + return -ENOMEM;
> +
> + priv->hdev = hdev;
> + hid_set_drvdata(hdev, priv);
> +
> + priv->updated = jiffies - D5NEXT_STATUS_UPDATE_INTERVAL * HZ;
> +
> + ret = hid_parse(hdev);
> + if (ret)
> + return ret;
> +
> + ret = hid_hw_start(hdev, HID_CONNECT_HIDRAW);
> + if (ret)
> + return ret;
> +
> + ret = hid_hw_open(hdev);
> + if (ret)
> + goto fail_and_stop;
> +
> + priv->hwmon_dev = hwmon_device_register_with_info(&hdev->dev, "d5next", priv,
> + &d5next_chip_info, NULL);
> +
> + if (IS_ERR(priv->hwmon_dev)) {
> + ret = PTR_ERR(priv->hwmon_dev);
> + goto fail_and_close;
> + }
> +
> + d5next_debugfs_init(priv);
> +
> + return 0;
> +
> +fail_and_close:
> + hid_hw_close(hdev);
> +fail_and_stop:
> + hid_hw_stop(hdev);
> + return ret;
> +}
> +
> +static void d5next_remove(struct hid_device *hdev)
> +{
> + struct d5next_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 d5next_table[] = {
> + { HID_USB_DEVICE(0x0c70, 0xf00e) }, /* Aquacomputer D5 Next */
> + {},
> +};
> +
> +MODULE_DEVICE_TABLE(hid, d5next_table);
> +
> +static struct hid_driver d5next_driver = {
> + .name = DRIVER_NAME,
> + .id_table = d5next_table,
> + .probe = d5next_probe,
> + .remove = d5next_remove,
> + .raw_event = d5next_raw_event,
> +};
> +
> +static int __init d5next_init(void)
> +{
> + return hid_register_driver(&d5next_driver);
> +}
> +
> +static void __exit d5next_exit(void)
> +{
> + hid_unregister_driver(&d5next_driver);
> +}
> +
> +/* Request to initialize after the HID bus to ensure it's not being loaded before */
> +
> +late_initcall(d5next_init);
> +module_exit(d5next_exit);
> +
> +MODULE_LICENSE("GPL");
> +MODULE_AUTHOR("Aleksa Savic <savicaleksa83@xxxxxxxxx>");
> +MODULE_DESCRIPTION("Hwmon driver for Aquacomputer D5 Next pump");
> --
> 2.31.1
>