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 08:47:45AM -0700, Guenter Roeck wrote:
> 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.
> 

Thanks for the feedback here and above! I'll send a v2 soon.

Aleksa

> > +
> > +	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]     [Kernel Newbies]     [Security]     [Netfilter]     [Bugtraq]     [Linux FS]     [Yosemite Forum]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Video 4 Linux]     [Device Mapper]     [Linux Resources]

  Powered by Linux