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

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



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
> 



[Index of Archives]     [LM Sensors]     [Linux Sound]     [ALSA Users]     [ALSA Devel]     [Linux Audio Users]     [Linux Media]     [Kernel]     [Gimp]     [Yosemite News]     [Linux Media]

  Powered by Linux