1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Copyright (C) 2021 Thomas Weißschuh <thomas@weissschuh.net>
5 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
7 #include <linux/acpi.h>
8 #include <linux/hwmon.h>
9 #include <linux/module.h>
10 #include <linux/wmi.h>
12 #define GIGABYTE_WMI_GUID "DEADBEEF-2001-0000-00A0-C90629100000"
13 #define NUM_TEMPERATURE_SENSORS 6
15 static u8 usable_sensors_mask
;
17 enum gigabyte_wmi_commandtype
{
18 GIGABYTE_WMI_BUILD_DATE_QUERY
= 0x1,
19 GIGABYTE_WMI_MAINBOARD_TYPE_QUERY
= 0x2,
20 GIGABYTE_WMI_FIRMWARE_VERSION_QUERY
= 0x4,
21 GIGABYTE_WMI_MAINBOARD_NAME_QUERY
= 0x5,
22 GIGABYTE_WMI_TEMPERATURE_QUERY
= 0x125,
25 struct gigabyte_wmi_args
{
29 static int gigabyte_wmi_perform_query(struct wmi_device
*wdev
,
30 enum gigabyte_wmi_commandtype command
,
31 struct gigabyte_wmi_args
*args
, struct acpi_buffer
*out
)
33 const struct acpi_buffer in
= {
34 .length
= sizeof(*args
),
38 acpi_status ret
= wmidev_evaluate_method(wdev
, 0x0, command
, &in
, out
);
40 if (ACPI_FAILURE(ret
))
46 static int gigabyte_wmi_query_integer(struct wmi_device
*wdev
,
47 enum gigabyte_wmi_commandtype command
,
48 struct gigabyte_wmi_args
*args
, u64
*res
)
50 union acpi_object
*obj
;
51 struct acpi_buffer result
= { ACPI_ALLOCATE_BUFFER
, NULL
};
54 ret
= gigabyte_wmi_perform_query(wdev
, command
, args
, &result
);
58 if (obj
&& obj
->type
== ACPI_TYPE_INTEGER
)
59 *res
= obj
->integer
.value
;
62 kfree(result
.pointer
);
66 static int gigabyte_wmi_temperature(struct wmi_device
*wdev
, u8 sensor
, long *res
)
68 struct gigabyte_wmi_args args
= {
74 ret
= gigabyte_wmi_query_integer(wdev
, GIGABYTE_WMI_TEMPERATURE_QUERY
, &args
, &temp
);
78 *res
= (s8
)temp
* 1000; // value is a signed 8-bit integer
83 static int gigabyte_wmi_hwmon_read(struct device
*dev
, enum hwmon_sensor_types type
,
84 u32 attr
, int channel
, long *val
)
86 struct wmi_device
*wdev
= dev_get_drvdata(dev
);
88 return gigabyte_wmi_temperature(wdev
, channel
, val
);
91 static umode_t
gigabyte_wmi_hwmon_is_visible(const void *data
, enum hwmon_sensor_types type
,
92 u32 attr
, int channel
)
94 return usable_sensors_mask
& BIT(channel
) ? 0444 : 0;
97 static const struct hwmon_channel_info
* const gigabyte_wmi_hwmon_info
[] = {
98 HWMON_CHANNEL_INFO(temp
,
108 static const struct hwmon_ops gigabyte_wmi_hwmon_ops
= {
109 .read
= gigabyte_wmi_hwmon_read
,
110 .is_visible
= gigabyte_wmi_hwmon_is_visible
,
113 static const struct hwmon_chip_info gigabyte_wmi_hwmon_chip_info
= {
114 .ops
= &gigabyte_wmi_hwmon_ops
,
115 .info
= gigabyte_wmi_hwmon_info
,
118 static u8
gigabyte_wmi_detect_sensor_usability(struct wmi_device
*wdev
)
124 for (i
= 0; i
< NUM_TEMPERATURE_SENSORS
; i
++) {
125 if (!gigabyte_wmi_temperature(wdev
, i
, &temp
))
131 static int gigabyte_wmi_probe(struct wmi_device
*wdev
, const void *context
)
133 struct device
*hwmon_dev
;
135 usable_sensors_mask
= gigabyte_wmi_detect_sensor_usability(wdev
);
136 if (!usable_sensors_mask
) {
137 dev_info(&wdev
->dev
, "No temperature sensors usable");
141 hwmon_dev
= devm_hwmon_device_register_with_info(&wdev
->dev
, "gigabyte_wmi", wdev
,
142 &gigabyte_wmi_hwmon_chip_info
, NULL
);
144 return PTR_ERR_OR_ZERO(hwmon_dev
);
147 static const struct wmi_device_id gigabyte_wmi_id_table
[] = {
148 { GIGABYTE_WMI_GUID
, NULL
},
152 static struct wmi_driver gigabyte_wmi_driver
= {
154 .name
= "gigabyte-wmi",
156 .id_table
= gigabyte_wmi_id_table
,
157 .probe
= gigabyte_wmi_probe
,
159 module_wmi_driver(gigabyte_wmi_driver
);
161 MODULE_DEVICE_TABLE(wmi
, gigabyte_wmi_id_table
);
162 MODULE_AUTHOR("Thomas Weißschuh <thomas@weissschuh.net>");
163 MODULE_DESCRIPTION("Gigabyte WMI temperature driver");
164 MODULE_LICENSE("GPL");