Linux Audio

Check our new training course

Loading...
Note: File does not exist in v3.1.
  1// SPDX-License-Identifier: GPL-2.0-or-later
  2/*
  3 *  Copyright (C) 2021 Thomas Weißschuh <thomas@weissschuh.net>
  4 */
  5#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  6
  7#include <linux/acpi.h>
  8#include <linux/dmi.h>
  9#include <linux/hwmon.h>
 10#include <linux/module.h>
 11#include <linux/wmi.h>
 12
 13#define GIGABYTE_WMI_GUID	"DEADBEEF-2001-0000-00A0-C90629100000"
 14#define NUM_TEMPERATURE_SENSORS	6
 15
 16static bool force_load;
 17module_param(force_load, bool, 0444);
 18MODULE_PARM_DESC(force_load, "Force loading on unknown platform");
 19
 20static u8 usable_sensors_mask;
 21
 22enum gigabyte_wmi_commandtype {
 23	GIGABYTE_WMI_BUILD_DATE_QUERY       =   0x1,
 24	GIGABYTE_WMI_MAINBOARD_TYPE_QUERY   =   0x2,
 25	GIGABYTE_WMI_FIRMWARE_VERSION_QUERY =   0x4,
 26	GIGABYTE_WMI_MAINBOARD_NAME_QUERY   =   0x5,
 27	GIGABYTE_WMI_TEMPERATURE_QUERY      = 0x125,
 28};
 29
 30struct gigabyte_wmi_args {
 31	u32 arg1;
 32};
 33
 34static int gigabyte_wmi_perform_query(struct wmi_device *wdev,
 35				      enum gigabyte_wmi_commandtype command,
 36				      struct gigabyte_wmi_args *args, struct acpi_buffer *out)
 37{
 38	const struct acpi_buffer in = {
 39		.length = sizeof(*args),
 40		.pointer = args,
 41	};
 42
 43	acpi_status ret = wmidev_evaluate_method(wdev, 0x0, command, &in, out);
 44
 45	if (ACPI_FAILURE(ret))
 46		return -EIO;
 47
 48	return 0;
 49}
 50
 51static int gigabyte_wmi_query_integer(struct wmi_device *wdev,
 52				      enum gigabyte_wmi_commandtype command,
 53				      struct gigabyte_wmi_args *args, u64 *res)
 54{
 55	union acpi_object *obj;
 56	struct acpi_buffer result = { ACPI_ALLOCATE_BUFFER, NULL };
 57	int ret;
 58
 59	ret = gigabyte_wmi_perform_query(wdev, command, args, &result);
 60	if (ret)
 61		return ret;
 62	obj = result.pointer;
 63	if (obj && obj->type == ACPI_TYPE_INTEGER)
 64		*res = obj->integer.value;
 65	else
 66		ret = -EIO;
 67	kfree(result.pointer);
 68	return ret;
 69}
 70
 71static int gigabyte_wmi_temperature(struct wmi_device *wdev, u8 sensor, long *res)
 72{
 73	struct gigabyte_wmi_args args = {
 74		.arg1 = sensor,
 75	};
 76	u64 temp;
 77	acpi_status ret;
 78
 79	ret = gigabyte_wmi_query_integer(wdev, GIGABYTE_WMI_TEMPERATURE_QUERY, &args, &temp);
 80	if (ret == 0) {
 81		if (temp == 0)
 82			return -ENODEV;
 83		*res = (s8)temp * 1000; // value is a signed 8-bit integer
 84	}
 85	return ret;
 86}
 87
 88static int gigabyte_wmi_hwmon_read(struct device *dev, enum hwmon_sensor_types type,
 89				   u32 attr, int channel, long *val)
 90{
 91	struct wmi_device *wdev = dev_get_drvdata(dev);
 92
 93	return gigabyte_wmi_temperature(wdev, channel, val);
 94}
 95
 96static umode_t gigabyte_wmi_hwmon_is_visible(const void *data, enum hwmon_sensor_types type,
 97					     u32 attr, int channel)
 98{
 99	return usable_sensors_mask & BIT(channel) ? 0444  : 0;
100}
101
102static const struct hwmon_channel_info *gigabyte_wmi_hwmon_info[] = {
103	HWMON_CHANNEL_INFO(temp,
104			   HWMON_T_INPUT,
105			   HWMON_T_INPUT,
106			   HWMON_T_INPUT,
107			   HWMON_T_INPUT,
108			   HWMON_T_INPUT,
109			   HWMON_T_INPUT),
110	NULL
111};
112
113static const struct hwmon_ops gigabyte_wmi_hwmon_ops = {
114	.read = gigabyte_wmi_hwmon_read,
115	.is_visible = gigabyte_wmi_hwmon_is_visible,
116};
117
118static const struct hwmon_chip_info gigabyte_wmi_hwmon_chip_info = {
119	.ops = &gigabyte_wmi_hwmon_ops,
120	.info = gigabyte_wmi_hwmon_info,
121};
122
123static u8 gigabyte_wmi_detect_sensor_usability(struct wmi_device *wdev)
124{
125	int i;
126	long temp;
127	u8 r = 0;
128
129	for (i = 0; i < NUM_TEMPERATURE_SENSORS; i++) {
130		if (!gigabyte_wmi_temperature(wdev, i, &temp))
131			r |= BIT(i);
132	}
133	return r;
134}
135
136#define DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME(name) \
137	{ .matches = { \
138		DMI_EXACT_MATCH(DMI_BOARD_VENDOR, "Gigabyte Technology Co., Ltd."), \
139		DMI_EXACT_MATCH(DMI_BOARD_NAME, name), \
140	}}
141
142static const struct dmi_system_id gigabyte_wmi_known_working_platforms[] = {
143	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B450M DS3H-CF"),
144	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B450M DS3H WIFI-CF"),
145	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B450M S2H V2"),
146	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B550 AORUS ELITE AX V2"),
147	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B550 AORUS ELITE"),
148	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B550 AORUS ELITE V2"),
149	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B550 GAMING X V2"),
150	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B550I AORUS PRO AX"),
151	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B550M AORUS PRO-P"),
152	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B550M DS3H"),
153	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B660 GAMING X DDR4"),
154	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("B660I AORUS PRO DDR4"),
155	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("Z390 I AORUS PRO WIFI-CF"),
156	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("Z490 AORUS ELITE AC"),
157	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("X570 AORUS ELITE"),
158	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("X570 AORUS ELITE WIFI"),
159	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("X570 GAMING X"),
160	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("X570 I AORUS PRO WIFI"),
161	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("X570 UD"),
162	DMI_EXACT_MATCH_GIGABYTE_BOARD_NAME("Z690M AORUS ELITE AX DDR4"),
163	{ }
164};
165
166static int gigabyte_wmi_probe(struct wmi_device *wdev, const void *context)
167{
168	struct device *hwmon_dev;
169
170	if (!dmi_check_system(gigabyte_wmi_known_working_platforms)) {
171		if (!force_load)
172			return -ENODEV;
173		dev_warn(&wdev->dev, "Forcing load on unknown platform");
174	}
175
176	usable_sensors_mask = gigabyte_wmi_detect_sensor_usability(wdev);
177	if (!usable_sensors_mask) {
178		dev_info(&wdev->dev, "No temperature sensors usable");
179		return -ENODEV;
180	}
181
182	hwmon_dev = devm_hwmon_device_register_with_info(&wdev->dev, "gigabyte_wmi", wdev,
183							 &gigabyte_wmi_hwmon_chip_info, NULL);
184
185	return PTR_ERR_OR_ZERO(hwmon_dev);
186}
187
188static const struct wmi_device_id gigabyte_wmi_id_table[] = {
189	{ GIGABYTE_WMI_GUID, NULL },
190	{ }
191};
192
193static struct wmi_driver gigabyte_wmi_driver = {
194	.driver = {
195		.name = "gigabyte-wmi",
196	},
197	.id_table = gigabyte_wmi_id_table,
198	.probe = gigabyte_wmi_probe,
199};
200module_wmi_driver(gigabyte_wmi_driver);
201
202MODULE_DEVICE_TABLE(wmi, gigabyte_wmi_id_table);
203MODULE_AUTHOR("Thomas Weißschuh <thomas@weissschuh.net>");
204MODULE_DESCRIPTION("Gigabyte WMI temperature driver");
205MODULE_LICENSE("GPL");