1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3 * hwmon driver for NZXT Kraken X53/X63/X73, Z53/Z63/Z73 and 2023/2023 Elite all in one coolers.
4 * X53 and Z53 in code refer to all models in their respective series (shortened for brevity).
5 * 2023 models use the Z53 code paths.
6 *
7 * Copyright 2021 Jonas Malaco <jonas@protocubo.io>
8 * Copyright 2022 Aleksa Savic <savicaleksa83@gmail.com>
9 */
10
11 #include <linux/debugfs.h>
12 #include <linux/hid.h>
13 #include <linux/hwmon.h>
14 #include <linux/hwmon-sysfs.h>
15 #include <linux/jiffies.h>
16 #include <linux/module.h>
17 #include <linux/mutex.h>
18 #include <linux/spinlock.h>
19 #include <linux/wait.h>
20 #include <linux/unaligned.h>
21
22 #define USB_VENDOR_ID_NZXT 0x1e71
23 #define USB_PRODUCT_ID_X53 0x2007
24 #define USB_PRODUCT_ID_X53_SECOND 0x2014
25 #define USB_PRODUCT_ID_Z53 0x3008
26 #define USB_PRODUCT_ID_KRAKEN2023 0x300E
27 #define USB_PRODUCT_ID_KRAKEN2023_ELITE 0x300C
28
29 enum kinds { X53, Z53, KRAKEN2023 } __packed;
30 enum pwm_enable { off, manual, curve } __packed;
31
32 #define DRIVER_NAME "nzxt_kraken3"
33 #define STATUS_REPORT_ID 0x75
34 #define FIRMWARE_REPORT_ID 0x11
35 #define STATUS_VALIDITY 2000 /* In ms, equivalent to period of four status reports */
36 #define CUSTOM_CURVE_POINTS 40 /* For temps from 20C to 59C (critical temp) */
37 #define PUMP_DUTY_MIN 20 /* In percent */
38
39 /* Sensor report offsets for Kraken X53 and Z53 */
40 #define TEMP_SENSOR_START_OFFSET 15
41 #define TEMP_SENSOR_END_OFFSET 16
42 #define PUMP_SPEED_OFFSET 17
43 #define PUMP_DUTY_OFFSET 19
44
45 /* Firmware version report offset for Kraken X53 and Z53 */
46 #define FIRMWARE_VERSION_OFFSET 17
47
48 /* Sensor report offsets for Kraken Z53 */
49 #define Z53_FAN_SPEED_OFFSET 23
50 #define Z53_FAN_DUTY_OFFSET 25
51
52 /* Report offsets for control commands for Kraken X53 and Z53 */
53 #define SET_DUTY_ID_OFFSET 1
54
55 /* Control commands and their lengths for Kraken X53 and Z53 */
56
57 /* Last byte sets the report interval at 0.5s */
58 static const u8 set_interval_cmd[] = { 0x70, 0x02, 0x01, 0xB8, 1 };
59 static const u8 finish_init_cmd[] = { 0x70, 0x01 };
60 static const u8 __maybe_unused get_fw_version_cmd[] = { 0x10, 0x01 };
61 static const u8 set_pump_duty_cmd_header[] = { 0x72, 0x00, 0x00, 0x00 };
62 static const u8 z53_get_status_cmd[] = { 0x74, 0x01 };
63
64 #define SET_INTERVAL_CMD_LENGTH 5
65 #define FINISH_INIT_CMD_LENGTH 2
66 #define GET_FW_VERSION_CMD_LENGTH 2
67 #define MAX_REPORT_LENGTH 64
68 #define MIN_REPORT_LENGTH 20
69 #define SET_CURVE_DUTY_CMD_HEADER_LENGTH 4
70 /* 4 byte header and 40 duty offsets */
71 #define SET_CURVE_DUTY_CMD_LENGTH (4 + 40)
72 #define Z53_GET_STATUS_CMD_LENGTH 2
73
74 static const char *const kraken3_temp_label[] = {
75 "Coolant temp",
76 };
77
78 static const char *const kraken3_fan_label[] = {
79 "Pump speed",
80 "Fan speed"
81 };
82
83 struct kraken3_channel_info {
84 enum pwm_enable mode;
85
86 /* Both values are PWM */
87 u16 reported_duty;
88 u16 fixed_duty; /* Manually set fixed duty */
89
90 u8 pwm_points[CUSTOM_CURVE_POINTS];
91 };
92
93 struct kraken3_data {
94 struct hid_device *hdev;
95 struct device *hwmon_dev;
96 struct dentry *debugfs;
97 struct mutex buffer_lock; /* For locking access to buffer */
98 struct mutex z53_status_request_lock;
99 struct completion fw_version_processed;
100 /*
101 * For X53 devices, tracks whether an initial (one) sensor report was received to
102 * make fancontrol not bail outright. For Z53 devices, whether a status report
103 * was processed after requesting one.
104 */
105 struct completion status_report_processed;
106 /* For locking the above completion */
107 spinlock_t status_completion_lock;
108
109 u8 *buffer;
110 struct kraken3_channel_info channel_info[2]; /* Pump and fan */
111 bool is_device_faulty;
112
113 /* Sensor values */
114 s32 temp_input[1];
115 u16 fan_input[2];
116
117 enum kinds kind;
118 u8 firmware_version[3];
119
120 unsigned long updated; /* jiffies */
121 };
122
kraken3_is_visible(const void * data,enum hwmon_sensor_types type,u32 attr,int channel)123 static umode_t kraken3_is_visible(const void *data, enum hwmon_sensor_types type, u32 attr,
124 int channel)
125 {
126 const struct kraken3_data *priv = data;
127
128 switch (type) {
129 case hwmon_temp:
130 if (channel < 1)
131 return 0444;
132 break;
133 case hwmon_fan:
134 switch (priv->kind) {
135 case X53:
136 /* Just the pump */
137 if (channel < 1)
138 return 0444;
139 break;
140 case Z53:
141 case KRAKEN2023:
142 /* Pump and fan */
143 if (channel < 2)
144 return 0444;
145 break;
146 default:
147 break;
148 }
149 break;
150 case hwmon_pwm:
151 switch (attr) {
152 case hwmon_pwm_enable:
153 case hwmon_pwm_input:
154 switch (priv->kind) {
155 case X53:
156 /* Just the pump */
157 if (channel < 1)
158 return 0644;
159 break;
160 case Z53:
161 case KRAKEN2023:
162 /* Pump and fan */
163 if (channel < 2)
164 return 0644;
165 break;
166 default:
167 break;
168 }
169 break;
170 default:
171 break;
172 }
173 break;
174 default:
175 break;
176 }
177
178 return 0;
179 }
180
181 /*
182 * Writes the command to the device with the rest of the report (up to 64 bytes) filled
183 * with zeroes.
184 */
kraken3_write_expanded(struct kraken3_data * priv,const u8 * cmd,int cmd_length)185 static int kraken3_write_expanded(struct kraken3_data *priv, const u8 *cmd, int cmd_length)
186 {
187 int ret;
188
189 mutex_lock(&priv->buffer_lock);
190
191 memcpy_and_pad(priv->buffer, MAX_REPORT_LENGTH, cmd, cmd_length, 0x00);
192 ret = hid_hw_output_report(priv->hdev, priv->buffer, MAX_REPORT_LENGTH);
193
194 mutex_unlock(&priv->buffer_lock);
195 return ret;
196 }
197
kraken3_percent_to_pwm(long val)198 static int kraken3_percent_to_pwm(long val)
199 {
200 return DIV_ROUND_CLOSEST(val * 255, 100);
201 }
202
kraken3_pwm_to_percent(long val,int channel)203 static int kraken3_pwm_to_percent(long val, int channel)
204 {
205 int percent_value;
206
207 if (val < 0 || val > 255)
208 return -EINVAL;
209
210 percent_value = DIV_ROUND_CLOSEST(val * 100, 255);
211
212 /* Bring up pump duty to min value if needed */
213 if (channel == 0 && percent_value < PUMP_DUTY_MIN)
214 percent_value = PUMP_DUTY_MIN;
215
216 return percent_value;
217 }
218
kraken3_read_x53(struct kraken3_data * priv)219 static int kraken3_read_x53(struct kraken3_data *priv)
220 {
221 int ret;
222
223 if (completion_done(&priv->status_report_processed))
224 /*
225 * We're here because data is stale. This means that sensor reports haven't
226 * been received for some time in kraken3_raw_event(). On X-series sensor data
227 * can't be manually requested, so return an error.
228 */
229 return -ENODATA;
230
231 /*
232 * Data needs to be read, but a sensor report wasn't yet received. It's usually
233 * fancontrol that requests data this early and it exits if it reads an error code.
234 * So, wait for the first report to be parsed (but up to STATUS_VALIDITY).
235 * This does not concern the Z series devices, because they send a sensor report
236 * only when requested.
237 */
238 ret = wait_for_completion_interruptible_timeout(&priv->status_report_processed,
239 msecs_to_jiffies(STATUS_VALIDITY));
240 if (ret == 0)
241 return -ETIMEDOUT;
242 else if (ret < 0)
243 return ret;
244
245 /* The first sensor report was parsed on time and reading can continue */
246 return 0;
247 }
248
249 /* Covers Z53 and KRAKEN2023 device kinds */
kraken3_read_z53(struct kraken3_data * priv)250 static int kraken3_read_z53(struct kraken3_data *priv)
251 {
252 int ret = mutex_lock_interruptible(&priv->z53_status_request_lock);
253
254 if (ret < 0)
255 return ret;
256
257 if (!time_after(jiffies, priv->updated + msecs_to_jiffies(STATUS_VALIDITY))) {
258 /* Data is up to date */
259 goto unlock_and_return;
260 }
261
262 /*
263 * Disable interrupts for a moment to safely reinit the completion,
264 * as hidraw calls could have allowed one or more readers to complete.
265 */
266 spin_lock_bh(&priv->status_completion_lock);
267 reinit_completion(&priv->status_report_processed);
268 spin_unlock_bh(&priv->status_completion_lock);
269
270 /* Send command for getting status */
271 ret = kraken3_write_expanded(priv, z53_get_status_cmd, Z53_GET_STATUS_CMD_LENGTH);
272 if (ret < 0)
273 goto unlock_and_return;
274
275 /* Wait for completion from kraken3_raw_event() */
276 ret = wait_for_completion_interruptible_timeout(&priv->status_report_processed,
277 msecs_to_jiffies(STATUS_VALIDITY));
278 if (ret == 0)
279 ret = -ETIMEDOUT;
280
281 unlock_and_return:
282 mutex_unlock(&priv->z53_status_request_lock);
283 if (ret < 0)
284 return ret;
285
286 return 0;
287 }
288
kraken3_read(struct device * dev,enum hwmon_sensor_types type,u32 attr,int channel,long * val)289 static int kraken3_read(struct device *dev, enum hwmon_sensor_types type, u32 attr, int channel,
290 long *val)
291 {
292 struct kraken3_data *priv = dev_get_drvdata(dev);
293 int ret;
294
295 if (time_after(jiffies, priv->updated + msecs_to_jiffies(STATUS_VALIDITY))) {
296 if (priv->kind == X53)
297 ret = kraken3_read_x53(priv);
298 else
299 ret = kraken3_read_z53(priv);
300
301 if (ret < 0)
302 return ret;
303
304 if (priv->is_device_faulty)
305 return -ENODATA;
306 }
307
308 switch (type) {
309 case hwmon_temp:
310 *val = priv->temp_input[channel];
311 break;
312 case hwmon_fan:
313 *val = priv->fan_input[channel];
314 break;
315 case hwmon_pwm:
316 switch (attr) {
317 case hwmon_pwm_enable:
318 *val = priv->channel_info[channel].mode;
319 break;
320 case hwmon_pwm_input:
321 *val = priv->channel_info[channel].reported_duty;
322 break;
323 default:
324 return -EOPNOTSUPP;
325 }
326 break;
327 default:
328 return -EOPNOTSUPP;
329 }
330
331 return 0;
332 }
333
kraken3_read_string(struct device * dev,enum hwmon_sensor_types type,u32 attr,int channel,const char ** str)334 static int kraken3_read_string(struct device *dev, enum hwmon_sensor_types type, u32 attr,
335 int channel, const char **str)
336 {
337 switch (type) {
338 case hwmon_temp:
339 *str = kraken3_temp_label[channel];
340 break;
341 case hwmon_fan:
342 *str = kraken3_fan_label[channel];
343 break;
344 default:
345 return -EOPNOTSUPP;
346 }
347
348 return 0;
349 }
350
351 /* Writes custom curve to device */
kraken3_write_curve(struct kraken3_data * priv,u8 * curve_array,int channel)352 static int kraken3_write_curve(struct kraken3_data *priv, u8 *curve_array, int channel)
353 {
354 u8 fixed_duty_cmd[SET_CURVE_DUTY_CMD_LENGTH];
355 int ret;
356
357 /* Copy command header */
358 memcpy(fixed_duty_cmd, set_pump_duty_cmd_header, SET_CURVE_DUTY_CMD_HEADER_LENGTH);
359
360 /* Set the correct ID for writing pump/fan duty (0x01 or 0x02, respectively) */
361 fixed_duty_cmd[SET_DUTY_ID_OFFSET] = channel + 1;
362
363 if (priv->kind == KRAKEN2023) {
364 /* These require 1s in the next one or two slots after SET_DUTY_ID_OFFSET */
365 fixed_duty_cmd[SET_DUTY_ID_OFFSET + 1] = 1;
366 if (channel == 1) /* Fan */
367 fixed_duty_cmd[SET_DUTY_ID_OFFSET + 2] = 1;
368 }
369
370 /* Copy curve to command */
371 memcpy(fixed_duty_cmd + SET_CURVE_DUTY_CMD_HEADER_LENGTH, curve_array, CUSTOM_CURVE_POINTS);
372
373 ret = kraken3_write_expanded(priv, fixed_duty_cmd, SET_CURVE_DUTY_CMD_LENGTH);
374 return ret;
375 }
376
kraken3_write_fixed_duty(struct kraken3_data * priv,long val,int channel)377 static int kraken3_write_fixed_duty(struct kraken3_data *priv, long val, int channel)
378 {
379 u8 fixed_curve_points[CUSTOM_CURVE_POINTS];
380 int ret, percent_val, i;
381
382 percent_val = kraken3_pwm_to_percent(val, channel);
383 if (percent_val < 0)
384 return percent_val;
385
386 /*
387 * The devices can only control the duty through a curve.
388 * Since we're setting a fixed duty here, fill the whole curve
389 * (ranging from 20C to 59C) with the same duty, except for
390 * the last point, the critical temperature, where it's maxed
391 * out for safety.
392 */
393
394 /* Fill the custom curve with the fixed value we're setting */
395 for (i = 0; i < CUSTOM_CURVE_POINTS - 1; i++)
396 fixed_curve_points[i] = percent_val;
397
398 /* Force duty to 100% at critical temp */
399 fixed_curve_points[CUSTOM_CURVE_POINTS - 1] = 100;
400
401 /* Write the fixed duty curve to the device */
402 ret = kraken3_write_curve(priv, fixed_curve_points, channel);
403 return ret;
404 }
405
kraken3_write(struct device * dev,enum hwmon_sensor_types type,u32 attr,int channel,long val)406 static int kraken3_write(struct device *dev, enum hwmon_sensor_types type, u32 attr, int channel,
407 long val)
408 {
409 struct kraken3_data *priv = dev_get_drvdata(dev);
410 int ret;
411
412 switch (type) {
413 case hwmon_pwm:
414 switch (attr) {
415 case hwmon_pwm_input:
416 /* Remember the last set fixed duty for channel */
417 priv->channel_info[channel].fixed_duty = val;
418
419 if (priv->channel_info[channel].mode == manual) {
420 ret = kraken3_write_fixed_duty(priv, val, channel);
421 if (ret < 0)
422 return ret;
423
424 /*
425 * Lock onto this value and report it until next interrupt status
426 * report is received, so userspace tools can continue to work.
427 */
428 priv->channel_info[channel].reported_duty = val;
429 }
430 break;
431 case hwmon_pwm_enable:
432 if (val < 0 || val > 2)
433 return -EINVAL;
434
435 switch (val) {
436 case 0:
437 /* Set channel to 100%, direct duty value */
438 ret = kraken3_write_fixed_duty(priv, 255, channel);
439 if (ret < 0)
440 return ret;
441
442 /* We don't control anything anymore */
443 priv->channel_info[channel].mode = off;
444 break;
445 case 1:
446 /* Apply the last known direct duty value */
447 ret =
448 kraken3_write_fixed_duty(priv,
449 priv->channel_info[channel].fixed_duty,
450 channel);
451 if (ret < 0)
452 return ret;
453
454 priv->channel_info[channel].mode = manual;
455 break;
456 case 2:
457 /* Apply the curve and note as enabled */
458 ret =
459 kraken3_write_curve(priv,
460 priv->channel_info[channel].pwm_points,
461 channel);
462 if (ret < 0)
463 return ret;
464
465 priv->channel_info[channel].mode = curve;
466 break;
467 default:
468 break;
469 }
470 break;
471 default:
472 return -EOPNOTSUPP;
473 }
474 break;
475 default:
476 return -EOPNOTSUPP;
477 }
478
479 return 0;
480 }
481
kraken3_fan_curve_pwm_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)482 static ssize_t kraken3_fan_curve_pwm_store(struct device *dev, struct device_attribute *attr,
483 const char *buf, size_t count)
484 {
485 struct sensor_device_attribute_2 *dev_attr = to_sensor_dev_attr_2(attr);
486 struct kraken3_data *priv = dev_get_drvdata(dev);
487 long val;
488 int ret;
489
490 if (kstrtol(buf, 10, &val) < 0)
491 return -EINVAL;
492
493 val = kraken3_pwm_to_percent(val, dev_attr->nr);
494 if (val < 0)
495 return val;
496
497 priv->channel_info[dev_attr->nr].pwm_points[dev_attr->index] = val;
498
499 if (priv->channel_info[dev_attr->nr].mode == curve) {
500 /* Apply the curve */
501 ret =
502 kraken3_write_curve(priv,
503 priv->channel_info[dev_attr->nr].pwm_points, dev_attr->nr);
504 if (ret < 0)
505 return ret;
506 }
507
508 return count;
509 }
510
kraken3_curve_props_are_visible(struct kobject * kobj,struct attribute * attr,int index)511 static umode_t kraken3_curve_props_are_visible(struct kobject *kobj, struct attribute *attr,
512 int index)
513 {
514 struct device *dev = kobj_to_dev(kobj);
515 struct kraken3_data *priv = dev_get_drvdata(dev);
516
517 /* X53 does not have a fan */
518 if (index >= CUSTOM_CURVE_POINTS && priv->kind == X53)
519 return 0;
520
521 return attr->mode;
522 }
523
524 /* Custom pump curve from 20C to 59C (critical temp) */
525 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point1_pwm, kraken3_fan_curve_pwm, 0, 0);
526 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point2_pwm, kraken3_fan_curve_pwm, 0, 1);
527 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point3_pwm, kraken3_fan_curve_pwm, 0, 2);
528 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point4_pwm, kraken3_fan_curve_pwm, 0, 3);
529 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point5_pwm, kraken3_fan_curve_pwm, 0, 4);
530 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point6_pwm, kraken3_fan_curve_pwm, 0, 5);
531 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point7_pwm, kraken3_fan_curve_pwm, 0, 6);
532 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point8_pwm, kraken3_fan_curve_pwm, 0, 7);
533 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point9_pwm, kraken3_fan_curve_pwm, 0, 8);
534 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point10_pwm, kraken3_fan_curve_pwm, 0, 9);
535 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point11_pwm, kraken3_fan_curve_pwm, 0, 10);
536 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point12_pwm, kraken3_fan_curve_pwm, 0, 11);
537 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point13_pwm, kraken3_fan_curve_pwm, 0, 12);
538 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point14_pwm, kraken3_fan_curve_pwm, 0, 13);
539 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point15_pwm, kraken3_fan_curve_pwm, 0, 14);
540 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point16_pwm, kraken3_fan_curve_pwm, 0, 15);
541 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point17_pwm, kraken3_fan_curve_pwm, 0, 16);
542 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point18_pwm, kraken3_fan_curve_pwm, 0, 17);
543 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point19_pwm, kraken3_fan_curve_pwm, 0, 18);
544 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point20_pwm, kraken3_fan_curve_pwm, 0, 19);
545 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point21_pwm, kraken3_fan_curve_pwm, 0, 20);
546 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point22_pwm, kraken3_fan_curve_pwm, 0, 21);
547 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point23_pwm, kraken3_fan_curve_pwm, 0, 22);
548 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point24_pwm, kraken3_fan_curve_pwm, 0, 23);
549 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point25_pwm, kraken3_fan_curve_pwm, 0, 24);
550 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point26_pwm, kraken3_fan_curve_pwm, 0, 25);
551 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point27_pwm, kraken3_fan_curve_pwm, 0, 26);
552 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point28_pwm, kraken3_fan_curve_pwm, 0, 27);
553 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point29_pwm, kraken3_fan_curve_pwm, 0, 28);
554 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point30_pwm, kraken3_fan_curve_pwm, 0, 29);
555 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point31_pwm, kraken3_fan_curve_pwm, 0, 30);
556 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point32_pwm, kraken3_fan_curve_pwm, 0, 31);
557 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point33_pwm, kraken3_fan_curve_pwm, 0, 32);
558 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point34_pwm, kraken3_fan_curve_pwm, 0, 33);
559 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point35_pwm, kraken3_fan_curve_pwm, 0, 34);
560 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point36_pwm, kraken3_fan_curve_pwm, 0, 35);
561 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point37_pwm, kraken3_fan_curve_pwm, 0, 36);
562 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point38_pwm, kraken3_fan_curve_pwm, 0, 37);
563 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point39_pwm, kraken3_fan_curve_pwm, 0, 38);
564 static SENSOR_DEVICE_ATTR_2_WO(temp1_auto_point40_pwm, kraken3_fan_curve_pwm, 0, 39);
565
566 /* Custom fan curve from 20C to 59C (critical temp) */
567 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point1_pwm, kraken3_fan_curve_pwm, 1, 0);
568 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point2_pwm, kraken3_fan_curve_pwm, 1, 1);
569 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point3_pwm, kraken3_fan_curve_pwm, 1, 2);
570 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point4_pwm, kraken3_fan_curve_pwm, 1, 3);
571 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point5_pwm, kraken3_fan_curve_pwm, 1, 4);
572 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point6_pwm, kraken3_fan_curve_pwm, 1, 5);
573 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point7_pwm, kraken3_fan_curve_pwm, 1, 6);
574 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point8_pwm, kraken3_fan_curve_pwm, 1, 7);
575 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point9_pwm, kraken3_fan_curve_pwm, 1, 8);
576 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point10_pwm, kraken3_fan_curve_pwm, 1, 9);
577 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point11_pwm, kraken3_fan_curve_pwm, 1, 10);
578 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point12_pwm, kraken3_fan_curve_pwm, 1, 11);
579 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point13_pwm, kraken3_fan_curve_pwm, 1, 12);
580 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point14_pwm, kraken3_fan_curve_pwm, 1, 13);
581 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point15_pwm, kraken3_fan_curve_pwm, 1, 14);
582 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point16_pwm, kraken3_fan_curve_pwm, 1, 15);
583 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point17_pwm, kraken3_fan_curve_pwm, 1, 16);
584 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point18_pwm, kraken3_fan_curve_pwm, 1, 17);
585 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point19_pwm, kraken3_fan_curve_pwm, 1, 18);
586 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point20_pwm, kraken3_fan_curve_pwm, 1, 19);
587 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point21_pwm, kraken3_fan_curve_pwm, 1, 20);
588 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point22_pwm, kraken3_fan_curve_pwm, 1, 21);
589 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point23_pwm, kraken3_fan_curve_pwm, 1, 22);
590 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point24_pwm, kraken3_fan_curve_pwm, 1, 23);
591 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point25_pwm, kraken3_fan_curve_pwm, 1, 24);
592 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point26_pwm, kraken3_fan_curve_pwm, 1, 25);
593 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point27_pwm, kraken3_fan_curve_pwm, 1, 26);
594 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point28_pwm, kraken3_fan_curve_pwm, 1, 27);
595 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point29_pwm, kraken3_fan_curve_pwm, 1, 28);
596 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point30_pwm, kraken3_fan_curve_pwm, 1, 29);
597 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point31_pwm, kraken3_fan_curve_pwm, 1, 30);
598 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point32_pwm, kraken3_fan_curve_pwm, 1, 31);
599 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point33_pwm, kraken3_fan_curve_pwm, 1, 32);
600 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point34_pwm, kraken3_fan_curve_pwm, 1, 33);
601 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point35_pwm, kraken3_fan_curve_pwm, 1, 34);
602 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point36_pwm, kraken3_fan_curve_pwm, 1, 35);
603 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point37_pwm, kraken3_fan_curve_pwm, 1, 36);
604 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point38_pwm, kraken3_fan_curve_pwm, 1, 37);
605 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point39_pwm, kraken3_fan_curve_pwm, 1, 38);
606 static SENSOR_DEVICE_ATTR_2_WO(temp2_auto_point40_pwm, kraken3_fan_curve_pwm, 1, 39);
607
608 static struct attribute *kraken3_curve_attrs[] = {
609 /* Pump control curve */
610 &sensor_dev_attr_temp1_auto_point1_pwm.dev_attr.attr,
611 &sensor_dev_attr_temp1_auto_point2_pwm.dev_attr.attr,
612 &sensor_dev_attr_temp1_auto_point3_pwm.dev_attr.attr,
613 &sensor_dev_attr_temp1_auto_point4_pwm.dev_attr.attr,
614 &sensor_dev_attr_temp1_auto_point5_pwm.dev_attr.attr,
615 &sensor_dev_attr_temp1_auto_point6_pwm.dev_attr.attr,
616 &sensor_dev_attr_temp1_auto_point7_pwm.dev_attr.attr,
617 &sensor_dev_attr_temp1_auto_point8_pwm.dev_attr.attr,
618 &sensor_dev_attr_temp1_auto_point9_pwm.dev_attr.attr,
619 &sensor_dev_attr_temp1_auto_point10_pwm.dev_attr.attr,
620 &sensor_dev_attr_temp1_auto_point11_pwm.dev_attr.attr,
621 &sensor_dev_attr_temp1_auto_point12_pwm.dev_attr.attr,
622 &sensor_dev_attr_temp1_auto_point13_pwm.dev_attr.attr,
623 &sensor_dev_attr_temp1_auto_point14_pwm.dev_attr.attr,
624 &sensor_dev_attr_temp1_auto_point15_pwm.dev_attr.attr,
625 &sensor_dev_attr_temp1_auto_point16_pwm.dev_attr.attr,
626 &sensor_dev_attr_temp1_auto_point17_pwm.dev_attr.attr,
627 &sensor_dev_attr_temp1_auto_point18_pwm.dev_attr.attr,
628 &sensor_dev_attr_temp1_auto_point19_pwm.dev_attr.attr,
629 &sensor_dev_attr_temp1_auto_point20_pwm.dev_attr.attr,
630 &sensor_dev_attr_temp1_auto_point21_pwm.dev_attr.attr,
631 &sensor_dev_attr_temp1_auto_point22_pwm.dev_attr.attr,
632 &sensor_dev_attr_temp1_auto_point23_pwm.dev_attr.attr,
633 &sensor_dev_attr_temp1_auto_point24_pwm.dev_attr.attr,
634 &sensor_dev_attr_temp1_auto_point25_pwm.dev_attr.attr,
635 &sensor_dev_attr_temp1_auto_point26_pwm.dev_attr.attr,
636 &sensor_dev_attr_temp1_auto_point27_pwm.dev_attr.attr,
637 &sensor_dev_attr_temp1_auto_point28_pwm.dev_attr.attr,
638 &sensor_dev_attr_temp1_auto_point29_pwm.dev_attr.attr,
639 &sensor_dev_attr_temp1_auto_point30_pwm.dev_attr.attr,
640 &sensor_dev_attr_temp1_auto_point31_pwm.dev_attr.attr,
641 &sensor_dev_attr_temp1_auto_point32_pwm.dev_attr.attr,
642 &sensor_dev_attr_temp1_auto_point33_pwm.dev_attr.attr,
643 &sensor_dev_attr_temp1_auto_point34_pwm.dev_attr.attr,
644 &sensor_dev_attr_temp1_auto_point35_pwm.dev_attr.attr,
645 &sensor_dev_attr_temp1_auto_point36_pwm.dev_attr.attr,
646 &sensor_dev_attr_temp1_auto_point37_pwm.dev_attr.attr,
647 &sensor_dev_attr_temp1_auto_point38_pwm.dev_attr.attr,
648 &sensor_dev_attr_temp1_auto_point39_pwm.dev_attr.attr,
649 &sensor_dev_attr_temp1_auto_point40_pwm.dev_attr.attr,
650 /* Fan control curve (Z53 only) */
651 &sensor_dev_attr_temp2_auto_point1_pwm.dev_attr.attr,
652 &sensor_dev_attr_temp2_auto_point2_pwm.dev_attr.attr,
653 &sensor_dev_attr_temp2_auto_point3_pwm.dev_attr.attr,
654 &sensor_dev_attr_temp2_auto_point4_pwm.dev_attr.attr,
655 &sensor_dev_attr_temp2_auto_point5_pwm.dev_attr.attr,
656 &sensor_dev_attr_temp2_auto_point6_pwm.dev_attr.attr,
657 &sensor_dev_attr_temp2_auto_point7_pwm.dev_attr.attr,
658 &sensor_dev_attr_temp2_auto_point8_pwm.dev_attr.attr,
659 &sensor_dev_attr_temp2_auto_point9_pwm.dev_attr.attr,
660 &sensor_dev_attr_temp2_auto_point10_pwm.dev_attr.attr,
661 &sensor_dev_attr_temp2_auto_point11_pwm.dev_attr.attr,
662 &sensor_dev_attr_temp2_auto_point12_pwm.dev_attr.attr,
663 &sensor_dev_attr_temp2_auto_point13_pwm.dev_attr.attr,
664 &sensor_dev_attr_temp2_auto_point14_pwm.dev_attr.attr,
665 &sensor_dev_attr_temp2_auto_point15_pwm.dev_attr.attr,
666 &sensor_dev_attr_temp2_auto_point16_pwm.dev_attr.attr,
667 &sensor_dev_attr_temp2_auto_point17_pwm.dev_attr.attr,
668 &sensor_dev_attr_temp2_auto_point18_pwm.dev_attr.attr,
669 &sensor_dev_attr_temp2_auto_point19_pwm.dev_attr.attr,
670 &sensor_dev_attr_temp2_auto_point20_pwm.dev_attr.attr,
671 &sensor_dev_attr_temp2_auto_point21_pwm.dev_attr.attr,
672 &sensor_dev_attr_temp2_auto_point22_pwm.dev_attr.attr,
673 &sensor_dev_attr_temp2_auto_point23_pwm.dev_attr.attr,
674 &sensor_dev_attr_temp2_auto_point24_pwm.dev_attr.attr,
675 &sensor_dev_attr_temp2_auto_point25_pwm.dev_attr.attr,
676 &sensor_dev_attr_temp2_auto_point26_pwm.dev_attr.attr,
677 &sensor_dev_attr_temp2_auto_point27_pwm.dev_attr.attr,
678 &sensor_dev_attr_temp2_auto_point28_pwm.dev_attr.attr,
679 &sensor_dev_attr_temp2_auto_point29_pwm.dev_attr.attr,
680 &sensor_dev_attr_temp2_auto_point30_pwm.dev_attr.attr,
681 &sensor_dev_attr_temp2_auto_point31_pwm.dev_attr.attr,
682 &sensor_dev_attr_temp2_auto_point32_pwm.dev_attr.attr,
683 &sensor_dev_attr_temp2_auto_point33_pwm.dev_attr.attr,
684 &sensor_dev_attr_temp2_auto_point34_pwm.dev_attr.attr,
685 &sensor_dev_attr_temp2_auto_point35_pwm.dev_attr.attr,
686 &sensor_dev_attr_temp2_auto_point36_pwm.dev_attr.attr,
687 &sensor_dev_attr_temp2_auto_point37_pwm.dev_attr.attr,
688 &sensor_dev_attr_temp2_auto_point38_pwm.dev_attr.attr,
689 &sensor_dev_attr_temp2_auto_point39_pwm.dev_attr.attr,
690 &sensor_dev_attr_temp2_auto_point40_pwm.dev_attr.attr,
691 NULL
692 };
693
694 static const struct attribute_group kraken3_curves_group = {
695 .attrs = kraken3_curve_attrs,
696 .is_visible = kraken3_curve_props_are_visible
697 };
698
699 static const struct attribute_group *kraken3_groups[] = {
700 &kraken3_curves_group,
701 NULL
702 };
703
704 static const struct hwmon_ops kraken3_hwmon_ops = {
705 .is_visible = kraken3_is_visible,
706 .read = kraken3_read,
707 .read_string = kraken3_read_string,
708 .write = kraken3_write
709 };
710
711 static const struct hwmon_channel_info *kraken3_info[] = {
712 HWMON_CHANNEL_INFO(temp,
713 HWMON_T_INPUT | HWMON_T_LABEL),
714 HWMON_CHANNEL_INFO(fan,
715 HWMON_F_INPUT | HWMON_F_LABEL,
716 HWMON_F_INPUT | HWMON_F_LABEL,
717 HWMON_F_INPUT | HWMON_F_LABEL,
718 HWMON_F_INPUT | HWMON_F_LABEL),
719 HWMON_CHANNEL_INFO(pwm,
720 HWMON_PWM_INPUT | HWMON_PWM_ENABLE,
721 HWMON_PWM_INPUT | HWMON_PWM_ENABLE),
722 NULL
723 };
724
725 static const struct hwmon_chip_info kraken3_chip_info = {
726 .ops = &kraken3_hwmon_ops,
727 .info = kraken3_info,
728 };
729
kraken3_raw_event(struct hid_device * hdev,struct hid_report * report,u8 * data,int size)730 static int kraken3_raw_event(struct hid_device *hdev, struct hid_report *report, u8 *data, int size)
731 {
732 struct kraken3_data *priv = hid_get_drvdata(hdev);
733 int i;
734
735 if (size < MIN_REPORT_LENGTH)
736 return 0;
737
738 if (report->id == FIRMWARE_REPORT_ID) {
739 /* Read firmware version */
740 for (i = 0; i < 3; i++)
741 priv->firmware_version[i] = data[FIRMWARE_VERSION_OFFSET + i];
742
743 if (!completion_done(&priv->fw_version_processed))
744 complete_all(&priv->fw_version_processed);
745
746 return 0;
747 }
748
749 if (report->id != STATUS_REPORT_ID)
750 return 0;
751
752 if (data[TEMP_SENSOR_START_OFFSET] == 0xff && data[TEMP_SENSOR_END_OFFSET] == 0xff) {
753 hid_err_once(hdev,
754 "firmware or device is possibly damaged (is SATA power connected?), not parsing reports\n");
755
756 /*
757 * Mark first X-series device report as received,
758 * as well as all for Z-series, if faulty.
759 */
760 spin_lock(&priv->status_completion_lock);
761 if (priv->kind != X53 || !completion_done(&priv->status_report_processed)) {
762 priv->is_device_faulty = true;
763 complete_all(&priv->status_report_processed);
764 }
765 spin_unlock(&priv->status_completion_lock);
766
767 return 0;
768 }
769
770 /* Received normal data */
771 priv->is_device_faulty = false;
772
773 /* Temperature and fan sensor readings */
774 priv->temp_input[0] =
775 data[TEMP_SENSOR_START_OFFSET] * 1000 + data[TEMP_SENSOR_END_OFFSET] * 100;
776
777 priv->fan_input[0] = get_unaligned_le16(data + PUMP_SPEED_OFFSET);
778 priv->channel_info[0].reported_duty = kraken3_percent_to_pwm(data[PUMP_DUTY_OFFSET]);
779
780 spin_lock(&priv->status_completion_lock);
781 if (priv->kind == X53 && !completion_done(&priv->status_report_processed)) {
782 /* Mark first X-series device report as received */
783 complete_all(&priv->status_report_processed);
784 } else if (priv->kind == Z53 || priv->kind == KRAKEN2023) {
785 /* Additional readings for Z53 and KRAKEN2023 */
786 priv->fan_input[1] = get_unaligned_le16(data + Z53_FAN_SPEED_OFFSET);
787 priv->channel_info[1].reported_duty =
788 kraken3_percent_to_pwm(data[Z53_FAN_DUTY_OFFSET]);
789
790 if (!completion_done(&priv->status_report_processed))
791 complete_all(&priv->status_report_processed);
792 }
793 spin_unlock(&priv->status_completion_lock);
794
795 priv->updated = jiffies;
796
797 return 0;
798 }
799
kraken3_init_device(struct hid_device * hdev)800 static int kraken3_init_device(struct hid_device *hdev)
801 {
802 struct kraken3_data *priv = hid_get_drvdata(hdev);
803 int ret;
804
805 /* Set the polling interval */
806 ret = kraken3_write_expanded(priv, set_interval_cmd, SET_INTERVAL_CMD_LENGTH);
807 if (ret < 0)
808 return ret;
809
810 /* Finalize the init process */
811 ret = kraken3_write_expanded(priv, finish_init_cmd, FINISH_INIT_CMD_LENGTH);
812 if (ret < 0)
813 return ret;
814
815 return 0;
816 }
817
kraken3_get_fw_ver(struct hid_device * hdev)818 static int kraken3_get_fw_ver(struct hid_device *hdev)
819 {
820 struct kraken3_data *priv = hid_get_drvdata(hdev);
821 int ret;
822
823 ret = kraken3_write_expanded(priv, get_fw_version_cmd, GET_FW_VERSION_CMD_LENGTH);
824 if (ret < 0)
825 return ret;
826
827 ret = wait_for_completion_interruptible_timeout(&priv->fw_version_processed,
828 msecs_to_jiffies(STATUS_VALIDITY));
829 if (ret == 0)
830 return -ETIMEDOUT;
831 else if (ret < 0)
832 return ret;
833
834 return 0;
835 }
836
kraken3_reset_resume(struct hid_device * hdev)837 static int __maybe_unused kraken3_reset_resume(struct hid_device *hdev)
838 {
839 int ret;
840
841 ret = kraken3_init_device(hdev);
842 if (ret)
843 hid_err(hdev, "req init (reset_resume) failed with %d\n", ret);
844
845 return ret;
846 }
847
firmware_version_show(struct seq_file * seqf,void * unused)848 static int firmware_version_show(struct seq_file *seqf, void *unused)
849 {
850 struct kraken3_data *priv = seqf->private;
851
852 seq_printf(seqf, "%u.%u.%u\n", priv->firmware_version[0], priv->firmware_version[1],
853 priv->firmware_version[2]);
854
855 return 0;
856 }
857 DEFINE_SHOW_ATTRIBUTE(firmware_version);
858
kraken3_debugfs_init(struct kraken3_data * priv,const char * device_name)859 static void kraken3_debugfs_init(struct kraken3_data *priv, const char *device_name)
860 {
861 char name[64];
862
863 if (!priv->firmware_version[0])
864 return; /* Nothing to display in debugfs */
865
866 scnprintf(name, sizeof(name), "%s_%s-%s", DRIVER_NAME, device_name,
867 dev_name(&priv->hdev->dev));
868
869 priv->debugfs = debugfs_create_dir(name, NULL);
870 debugfs_create_file("firmware_version", 0444, priv->debugfs, priv, &firmware_version_fops);
871 }
872
kraken3_probe(struct hid_device * hdev,const struct hid_device_id * id)873 static int kraken3_probe(struct hid_device *hdev, const struct hid_device_id *id)
874 {
875 struct kraken3_data *priv;
876 const char *device_name;
877 int ret;
878
879 priv = devm_kzalloc(&hdev->dev, sizeof(*priv), GFP_KERNEL);
880 if (!priv)
881 return -ENOMEM;
882
883 priv->hdev = hdev;
884 hid_set_drvdata(hdev, priv);
885
886 /*
887 * Initialize ->updated to STATUS_VALIDITY seconds in the past, making
888 * the initial empty data invalid for kraken3_read without the need for
889 * a special case there.
890 */
891 priv->updated = jiffies - msecs_to_jiffies(STATUS_VALIDITY);
892
893 ret = hid_parse(hdev);
894 if (ret) {
895 hid_err(hdev, "hid parse failed with %d\n", ret);
896 return ret;
897 }
898
899 /* Enable hidraw so existing user-space tools can continue to work */
900 ret = hid_hw_start(hdev, HID_CONNECT_HIDRAW);
901 if (ret) {
902 hid_err(hdev, "hid hw start failed with %d\n", ret);
903 return ret;
904 }
905
906 ret = hid_hw_open(hdev);
907 if (ret) {
908 hid_err(hdev, "hid hw open failed with %d\n", ret);
909 goto fail_and_stop;
910 }
911
912 switch (hdev->product) {
913 case USB_PRODUCT_ID_X53:
914 case USB_PRODUCT_ID_X53_SECOND:
915 priv->kind = X53;
916 device_name = "x53";
917 break;
918 case USB_PRODUCT_ID_Z53:
919 priv->kind = Z53;
920 device_name = "z53";
921 break;
922 case USB_PRODUCT_ID_KRAKEN2023:
923 priv->kind = KRAKEN2023;
924 device_name = "kraken2023";
925 break;
926 case USB_PRODUCT_ID_KRAKEN2023_ELITE:
927 priv->kind = KRAKEN2023;
928 device_name = "kraken2023elite";
929 break;
930 default:
931 ret = -ENODEV;
932 goto fail_and_close;
933 }
934
935 priv->buffer = devm_kzalloc(&hdev->dev, MAX_REPORT_LENGTH, GFP_KERNEL);
936 if (!priv->buffer) {
937 ret = -ENOMEM;
938 goto fail_and_close;
939 }
940
941 mutex_init(&priv->buffer_lock);
942 mutex_init(&priv->z53_status_request_lock);
943 init_completion(&priv->fw_version_processed);
944 init_completion(&priv->status_report_processed);
945 spin_lock_init(&priv->status_completion_lock);
946
947 hid_device_io_start(hdev);
948 ret = kraken3_init_device(hdev);
949 if (ret < 0) {
950 hid_err(hdev, "device init failed with %d\n", ret);
951 goto fail_and_close;
952 }
953
954 ret = kraken3_get_fw_ver(hdev);
955 if (ret < 0)
956 hid_warn(hdev, "fw version request failed with %d\n", ret);
957
958 priv->hwmon_dev = hwmon_device_register_with_info(&hdev->dev, device_name, priv,
959 &kraken3_chip_info, kraken3_groups);
960 if (IS_ERR(priv->hwmon_dev)) {
961 ret = PTR_ERR(priv->hwmon_dev);
962 hid_err(hdev, "hwmon registration failed with %d\n", ret);
963 goto fail_and_close;
964 }
965
966 kraken3_debugfs_init(priv, device_name);
967
968 return 0;
969
970 fail_and_close:
971 hid_hw_close(hdev);
972 fail_and_stop:
973 hid_hw_stop(hdev);
974 return ret;
975 }
976
kraken3_remove(struct hid_device * hdev)977 static void kraken3_remove(struct hid_device *hdev)
978 {
979 struct kraken3_data *priv = hid_get_drvdata(hdev);
980
981 debugfs_remove_recursive(priv->debugfs);
982 hwmon_device_unregister(priv->hwmon_dev);
983
984 hid_hw_close(hdev);
985 hid_hw_stop(hdev);
986 }
987
988 static const struct hid_device_id kraken3_table[] = {
989 /* NZXT Kraken X53/X63/X73 have two possible product IDs */
990 { HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_X53) },
991 { HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_X53_SECOND) },
992 { HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_Z53) },
993 { HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_KRAKEN2023) },
994 { HID_USB_DEVICE(USB_VENDOR_ID_NZXT, USB_PRODUCT_ID_KRAKEN2023_ELITE) },
995 { }
996 };
997
998 MODULE_DEVICE_TABLE(hid, kraken3_table);
999
1000 static struct hid_driver kraken3_driver = {
1001 .name = DRIVER_NAME,
1002 .id_table = kraken3_table,
1003 .probe = kraken3_probe,
1004 .remove = kraken3_remove,
1005 .raw_event = kraken3_raw_event,
1006 #ifdef CONFIG_PM
1007 .reset_resume = kraken3_reset_resume,
1008 #endif
1009 };
1010
kraken3_init(void)1011 static int __init kraken3_init(void)
1012 {
1013 return hid_register_driver(&kraken3_driver);
1014 }
1015
kraken3_exit(void)1016 static void __exit kraken3_exit(void)
1017 {
1018 hid_unregister_driver(&kraken3_driver);
1019 }
1020
1021 /* When compiled into the kernel, initialize after the HID bus */
1022 late_initcall(kraken3_init);
1023 module_exit(kraken3_exit);
1024
1025 MODULE_LICENSE("GPL");
1026 MODULE_AUTHOR("Jonas Malaco <jonas@protocubo.io>");
1027 MODULE_AUTHOR("Aleksa Savic <savicaleksa83@gmail.com>");
1028 MODULE_DESCRIPTION("Hwmon driver for NZXT Kraken X53/X63/X73, Z53/Z63/Z73 coolers");
1029