18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * Alienware AlienFX control
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Copyright (C) 2014 Dell Inc <mario_limonciello@dell.com>
68c2ecf20Sopenharmony_ci */
78c2ecf20Sopenharmony_ci
88c2ecf20Sopenharmony_ci#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
98c2ecf20Sopenharmony_ci
108c2ecf20Sopenharmony_ci#include <linux/acpi.h>
118c2ecf20Sopenharmony_ci#include <linux/module.h>
128c2ecf20Sopenharmony_ci#include <linux/platform_device.h>
138c2ecf20Sopenharmony_ci#include <linux/dmi.h>
148c2ecf20Sopenharmony_ci#include <linux/leds.h>
158c2ecf20Sopenharmony_ci
168c2ecf20Sopenharmony_ci#define LEGACY_CONTROL_GUID		"A90597CE-A997-11DA-B012-B622A1EF5492"
178c2ecf20Sopenharmony_ci#define LEGACY_POWER_CONTROL_GUID	"A80593CE-A997-11DA-B012-B622A1EF5492"
188c2ecf20Sopenharmony_ci#define WMAX_CONTROL_GUID		"A70591CE-A997-11DA-B012-B622A1EF5492"
198c2ecf20Sopenharmony_ci
208c2ecf20Sopenharmony_ci#define WMAX_METHOD_HDMI_SOURCE		0x1
218c2ecf20Sopenharmony_ci#define WMAX_METHOD_HDMI_STATUS		0x2
228c2ecf20Sopenharmony_ci#define WMAX_METHOD_BRIGHTNESS		0x3
238c2ecf20Sopenharmony_ci#define WMAX_METHOD_ZONE_CONTROL	0x4
248c2ecf20Sopenharmony_ci#define WMAX_METHOD_HDMI_CABLE		0x5
258c2ecf20Sopenharmony_ci#define WMAX_METHOD_AMPLIFIER_CABLE	0x6
268c2ecf20Sopenharmony_ci#define WMAX_METHOD_DEEP_SLEEP_CONTROL	0x0B
278c2ecf20Sopenharmony_ci#define WMAX_METHOD_DEEP_SLEEP_STATUS	0x0C
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_ciMODULE_AUTHOR("Mario Limonciello <mario_limonciello@dell.com>");
308c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Alienware special feature control");
318c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL");
328c2ecf20Sopenharmony_ciMODULE_ALIAS("wmi:" LEGACY_CONTROL_GUID);
338c2ecf20Sopenharmony_ciMODULE_ALIAS("wmi:" WMAX_CONTROL_GUID);
348c2ecf20Sopenharmony_ci
358c2ecf20Sopenharmony_cienum INTERFACE_FLAGS {
368c2ecf20Sopenharmony_ci	LEGACY,
378c2ecf20Sopenharmony_ci	WMAX,
388c2ecf20Sopenharmony_ci};
398c2ecf20Sopenharmony_ci
408c2ecf20Sopenharmony_cienum LEGACY_CONTROL_STATES {
418c2ecf20Sopenharmony_ci	LEGACY_RUNNING = 1,
428c2ecf20Sopenharmony_ci	LEGACY_BOOTING = 0,
438c2ecf20Sopenharmony_ci	LEGACY_SUSPEND = 3,
448c2ecf20Sopenharmony_ci};
458c2ecf20Sopenharmony_ci
468c2ecf20Sopenharmony_cienum WMAX_CONTROL_STATES {
478c2ecf20Sopenharmony_ci	WMAX_RUNNING = 0xFF,
488c2ecf20Sopenharmony_ci	WMAX_BOOTING = 0,
498c2ecf20Sopenharmony_ci	WMAX_SUSPEND = 3,
508c2ecf20Sopenharmony_ci};
518c2ecf20Sopenharmony_ci
528c2ecf20Sopenharmony_cistruct quirk_entry {
538c2ecf20Sopenharmony_ci	u8 num_zones;
548c2ecf20Sopenharmony_ci	u8 hdmi_mux;
558c2ecf20Sopenharmony_ci	u8 amplifier;
568c2ecf20Sopenharmony_ci	u8 deepslp;
578c2ecf20Sopenharmony_ci};
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_cistatic struct quirk_entry *quirks;
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_ci
628c2ecf20Sopenharmony_cistatic struct quirk_entry quirk_inspiron5675 = {
638c2ecf20Sopenharmony_ci	.num_zones = 2,
648c2ecf20Sopenharmony_ci	.hdmi_mux = 0,
658c2ecf20Sopenharmony_ci	.amplifier = 0,
668c2ecf20Sopenharmony_ci	.deepslp = 0,
678c2ecf20Sopenharmony_ci};
688c2ecf20Sopenharmony_ci
698c2ecf20Sopenharmony_cistatic struct quirk_entry quirk_unknown = {
708c2ecf20Sopenharmony_ci	.num_zones = 2,
718c2ecf20Sopenharmony_ci	.hdmi_mux = 0,
728c2ecf20Sopenharmony_ci	.amplifier = 0,
738c2ecf20Sopenharmony_ci	.deepslp = 0,
748c2ecf20Sopenharmony_ci};
758c2ecf20Sopenharmony_ci
768c2ecf20Sopenharmony_cistatic struct quirk_entry quirk_x51_r1_r2 = {
778c2ecf20Sopenharmony_ci	.num_zones = 3,
788c2ecf20Sopenharmony_ci	.hdmi_mux = 0,
798c2ecf20Sopenharmony_ci	.amplifier = 0,
808c2ecf20Sopenharmony_ci	.deepslp = 0,
818c2ecf20Sopenharmony_ci};
828c2ecf20Sopenharmony_ci
838c2ecf20Sopenharmony_cistatic struct quirk_entry quirk_x51_r3 = {
848c2ecf20Sopenharmony_ci	.num_zones = 4,
858c2ecf20Sopenharmony_ci	.hdmi_mux = 0,
868c2ecf20Sopenharmony_ci	.amplifier = 1,
878c2ecf20Sopenharmony_ci	.deepslp = 0,
888c2ecf20Sopenharmony_ci};
898c2ecf20Sopenharmony_ci
908c2ecf20Sopenharmony_cistatic struct quirk_entry quirk_asm100 = {
918c2ecf20Sopenharmony_ci	.num_zones = 2,
928c2ecf20Sopenharmony_ci	.hdmi_mux = 1,
938c2ecf20Sopenharmony_ci	.amplifier = 0,
948c2ecf20Sopenharmony_ci	.deepslp = 0,
958c2ecf20Sopenharmony_ci};
968c2ecf20Sopenharmony_ci
978c2ecf20Sopenharmony_cistatic struct quirk_entry quirk_asm200 = {
988c2ecf20Sopenharmony_ci	.num_zones = 2,
998c2ecf20Sopenharmony_ci	.hdmi_mux = 1,
1008c2ecf20Sopenharmony_ci	.amplifier = 0,
1018c2ecf20Sopenharmony_ci	.deepslp = 1,
1028c2ecf20Sopenharmony_ci};
1038c2ecf20Sopenharmony_ci
1048c2ecf20Sopenharmony_cistatic struct quirk_entry quirk_asm201 = {
1058c2ecf20Sopenharmony_ci	.num_zones = 2,
1068c2ecf20Sopenharmony_ci	.hdmi_mux = 1,
1078c2ecf20Sopenharmony_ci	.amplifier = 1,
1088c2ecf20Sopenharmony_ci	.deepslp = 1,
1098c2ecf20Sopenharmony_ci};
1108c2ecf20Sopenharmony_ci
1118c2ecf20Sopenharmony_cistatic int __init dmi_matched(const struct dmi_system_id *dmi)
1128c2ecf20Sopenharmony_ci{
1138c2ecf20Sopenharmony_ci	quirks = dmi->driver_data;
1148c2ecf20Sopenharmony_ci	return 1;
1158c2ecf20Sopenharmony_ci}
1168c2ecf20Sopenharmony_ci
1178c2ecf20Sopenharmony_cistatic const struct dmi_system_id alienware_quirks[] __initconst = {
1188c2ecf20Sopenharmony_ci	{
1198c2ecf20Sopenharmony_ci	 .callback = dmi_matched,
1208c2ecf20Sopenharmony_ci	 .ident = "Alienware X51 R3",
1218c2ecf20Sopenharmony_ci	 .matches = {
1228c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_SYS_VENDOR, "Alienware"),
1238c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_PRODUCT_NAME, "Alienware X51 R3"),
1248c2ecf20Sopenharmony_ci		     },
1258c2ecf20Sopenharmony_ci	 .driver_data = &quirk_x51_r3,
1268c2ecf20Sopenharmony_ci	 },
1278c2ecf20Sopenharmony_ci	{
1288c2ecf20Sopenharmony_ci	 .callback = dmi_matched,
1298c2ecf20Sopenharmony_ci	 .ident = "Alienware X51 R2",
1308c2ecf20Sopenharmony_ci	 .matches = {
1318c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_SYS_VENDOR, "Alienware"),
1328c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_PRODUCT_NAME, "Alienware X51 R2"),
1338c2ecf20Sopenharmony_ci		     },
1348c2ecf20Sopenharmony_ci	 .driver_data = &quirk_x51_r1_r2,
1358c2ecf20Sopenharmony_ci	 },
1368c2ecf20Sopenharmony_ci	{
1378c2ecf20Sopenharmony_ci	 .callback = dmi_matched,
1388c2ecf20Sopenharmony_ci	 .ident = "Alienware X51 R1",
1398c2ecf20Sopenharmony_ci	 .matches = {
1408c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_SYS_VENDOR, "Alienware"),
1418c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_PRODUCT_NAME, "Alienware X51"),
1428c2ecf20Sopenharmony_ci		     },
1438c2ecf20Sopenharmony_ci	 .driver_data = &quirk_x51_r1_r2,
1448c2ecf20Sopenharmony_ci	 },
1458c2ecf20Sopenharmony_ci	{
1468c2ecf20Sopenharmony_ci	 .callback = dmi_matched,
1478c2ecf20Sopenharmony_ci	 .ident = "Alienware ASM100",
1488c2ecf20Sopenharmony_ci	 .matches = {
1498c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_SYS_VENDOR, "Alienware"),
1508c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_PRODUCT_NAME, "ASM100"),
1518c2ecf20Sopenharmony_ci		     },
1528c2ecf20Sopenharmony_ci	 .driver_data = &quirk_asm100,
1538c2ecf20Sopenharmony_ci	 },
1548c2ecf20Sopenharmony_ci	{
1558c2ecf20Sopenharmony_ci	 .callback = dmi_matched,
1568c2ecf20Sopenharmony_ci	 .ident = "Alienware ASM200",
1578c2ecf20Sopenharmony_ci	 .matches = {
1588c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_SYS_VENDOR, "Alienware"),
1598c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_PRODUCT_NAME, "ASM200"),
1608c2ecf20Sopenharmony_ci		     },
1618c2ecf20Sopenharmony_ci	 .driver_data = &quirk_asm200,
1628c2ecf20Sopenharmony_ci	 },
1638c2ecf20Sopenharmony_ci	{
1648c2ecf20Sopenharmony_ci	 .callback = dmi_matched,
1658c2ecf20Sopenharmony_ci	 .ident = "Alienware ASM201",
1668c2ecf20Sopenharmony_ci	 .matches = {
1678c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_SYS_VENDOR, "Alienware"),
1688c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_PRODUCT_NAME, "ASM201"),
1698c2ecf20Sopenharmony_ci		     },
1708c2ecf20Sopenharmony_ci	 .driver_data = &quirk_asm201,
1718c2ecf20Sopenharmony_ci	 },
1728c2ecf20Sopenharmony_ci	 {
1738c2ecf20Sopenharmony_ci	 .callback = dmi_matched,
1748c2ecf20Sopenharmony_ci	 .ident = "Dell Inc. Inspiron 5675",
1758c2ecf20Sopenharmony_ci	 .matches = {
1768c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
1778c2ecf20Sopenharmony_ci		     DMI_MATCH(DMI_PRODUCT_NAME, "Inspiron 5675"),
1788c2ecf20Sopenharmony_ci		     },
1798c2ecf20Sopenharmony_ci	 .driver_data = &quirk_inspiron5675,
1808c2ecf20Sopenharmony_ci	 },
1818c2ecf20Sopenharmony_ci	{}
1828c2ecf20Sopenharmony_ci};
1838c2ecf20Sopenharmony_ci
1848c2ecf20Sopenharmony_cistruct color_platform {
1858c2ecf20Sopenharmony_ci	u8 blue;
1868c2ecf20Sopenharmony_ci	u8 green;
1878c2ecf20Sopenharmony_ci	u8 red;
1888c2ecf20Sopenharmony_ci} __packed;
1898c2ecf20Sopenharmony_ci
1908c2ecf20Sopenharmony_cistruct platform_zone {
1918c2ecf20Sopenharmony_ci	u8 location;
1928c2ecf20Sopenharmony_ci	struct device_attribute *attr;
1938c2ecf20Sopenharmony_ci	struct color_platform colors;
1948c2ecf20Sopenharmony_ci};
1958c2ecf20Sopenharmony_ci
1968c2ecf20Sopenharmony_cistruct wmax_brightness_args {
1978c2ecf20Sopenharmony_ci	u32 led_mask;
1988c2ecf20Sopenharmony_ci	u32 percentage;
1998c2ecf20Sopenharmony_ci};
2008c2ecf20Sopenharmony_ci
2018c2ecf20Sopenharmony_cistruct wmax_basic_args {
2028c2ecf20Sopenharmony_ci	u8 arg;
2038c2ecf20Sopenharmony_ci};
2048c2ecf20Sopenharmony_ci
2058c2ecf20Sopenharmony_cistruct legacy_led_args {
2068c2ecf20Sopenharmony_ci	struct color_platform colors;
2078c2ecf20Sopenharmony_ci	u8 brightness;
2088c2ecf20Sopenharmony_ci	u8 state;
2098c2ecf20Sopenharmony_ci} __packed;
2108c2ecf20Sopenharmony_ci
2118c2ecf20Sopenharmony_cistruct wmax_led_args {
2128c2ecf20Sopenharmony_ci	u32 led_mask;
2138c2ecf20Sopenharmony_ci	struct color_platform colors;
2148c2ecf20Sopenharmony_ci	u8 state;
2158c2ecf20Sopenharmony_ci} __packed;
2168c2ecf20Sopenharmony_ci
2178c2ecf20Sopenharmony_cistatic struct platform_device *platform_device;
2188c2ecf20Sopenharmony_cistatic struct device_attribute *zone_dev_attrs;
2198c2ecf20Sopenharmony_cistatic struct attribute **zone_attrs;
2208c2ecf20Sopenharmony_cistatic struct platform_zone *zone_data;
2218c2ecf20Sopenharmony_ci
2228c2ecf20Sopenharmony_cistatic struct platform_driver platform_driver = {
2238c2ecf20Sopenharmony_ci	.driver = {
2248c2ecf20Sopenharmony_ci		   .name = "alienware-wmi",
2258c2ecf20Sopenharmony_ci		   }
2268c2ecf20Sopenharmony_ci};
2278c2ecf20Sopenharmony_ci
2288c2ecf20Sopenharmony_cistatic struct attribute_group zone_attribute_group = {
2298c2ecf20Sopenharmony_ci	.name = "rgb_zones",
2308c2ecf20Sopenharmony_ci};
2318c2ecf20Sopenharmony_ci
2328c2ecf20Sopenharmony_cistatic u8 interface;
2338c2ecf20Sopenharmony_cistatic u8 lighting_control_state;
2348c2ecf20Sopenharmony_cistatic u8 global_brightness;
2358c2ecf20Sopenharmony_ci
2368c2ecf20Sopenharmony_ci/*
2378c2ecf20Sopenharmony_ci * Helpers used for zone control
2388c2ecf20Sopenharmony_ci */
2398c2ecf20Sopenharmony_cistatic int parse_rgb(const char *buf, struct platform_zone *zone)
2408c2ecf20Sopenharmony_ci{
2418c2ecf20Sopenharmony_ci	long unsigned int rgb;
2428c2ecf20Sopenharmony_ci	int ret;
2438c2ecf20Sopenharmony_ci	union color_union {
2448c2ecf20Sopenharmony_ci		struct color_platform cp;
2458c2ecf20Sopenharmony_ci		int package;
2468c2ecf20Sopenharmony_ci	} repackager;
2478c2ecf20Sopenharmony_ci
2488c2ecf20Sopenharmony_ci	ret = kstrtoul(buf, 16, &rgb);
2498c2ecf20Sopenharmony_ci	if (ret)
2508c2ecf20Sopenharmony_ci		return ret;
2518c2ecf20Sopenharmony_ci
2528c2ecf20Sopenharmony_ci	/* RGB triplet notation is 24-bit hexadecimal */
2538c2ecf20Sopenharmony_ci	if (rgb > 0xFFFFFF)
2548c2ecf20Sopenharmony_ci		return -EINVAL;
2558c2ecf20Sopenharmony_ci
2568c2ecf20Sopenharmony_ci	repackager.package = rgb & 0x0f0f0f0f;
2578c2ecf20Sopenharmony_ci	pr_debug("alienware-wmi: r: %d g:%d b: %d\n",
2588c2ecf20Sopenharmony_ci		 repackager.cp.red, repackager.cp.green, repackager.cp.blue);
2598c2ecf20Sopenharmony_ci	zone->colors = repackager.cp;
2608c2ecf20Sopenharmony_ci	return 0;
2618c2ecf20Sopenharmony_ci}
2628c2ecf20Sopenharmony_ci
2638c2ecf20Sopenharmony_cistatic struct platform_zone *match_zone(struct device_attribute *attr)
2648c2ecf20Sopenharmony_ci{
2658c2ecf20Sopenharmony_ci	u8 zone;
2668c2ecf20Sopenharmony_ci
2678c2ecf20Sopenharmony_ci	for (zone = 0; zone < quirks->num_zones; zone++) {
2688c2ecf20Sopenharmony_ci		if ((struct device_attribute *)zone_data[zone].attr == attr) {
2698c2ecf20Sopenharmony_ci			pr_debug("alienware-wmi: matched zone location: %d\n",
2708c2ecf20Sopenharmony_ci				 zone_data[zone].location);
2718c2ecf20Sopenharmony_ci			return &zone_data[zone];
2728c2ecf20Sopenharmony_ci		}
2738c2ecf20Sopenharmony_ci	}
2748c2ecf20Sopenharmony_ci	return NULL;
2758c2ecf20Sopenharmony_ci}
2768c2ecf20Sopenharmony_ci
2778c2ecf20Sopenharmony_ci/*
2788c2ecf20Sopenharmony_ci * Individual RGB zone control
2798c2ecf20Sopenharmony_ci */
2808c2ecf20Sopenharmony_cistatic int alienware_update_led(struct platform_zone *zone)
2818c2ecf20Sopenharmony_ci{
2828c2ecf20Sopenharmony_ci	int method_id;
2838c2ecf20Sopenharmony_ci	acpi_status status;
2848c2ecf20Sopenharmony_ci	char *guid;
2858c2ecf20Sopenharmony_ci	struct acpi_buffer input;
2868c2ecf20Sopenharmony_ci	struct legacy_led_args legacy_args;
2878c2ecf20Sopenharmony_ci	struct wmax_led_args wmax_basic_args;
2888c2ecf20Sopenharmony_ci	if (interface == WMAX) {
2898c2ecf20Sopenharmony_ci		wmax_basic_args.led_mask = 1 << zone->location;
2908c2ecf20Sopenharmony_ci		wmax_basic_args.colors = zone->colors;
2918c2ecf20Sopenharmony_ci		wmax_basic_args.state = lighting_control_state;
2928c2ecf20Sopenharmony_ci		guid = WMAX_CONTROL_GUID;
2938c2ecf20Sopenharmony_ci		method_id = WMAX_METHOD_ZONE_CONTROL;
2948c2ecf20Sopenharmony_ci
2958c2ecf20Sopenharmony_ci		input.length = (acpi_size) sizeof(wmax_basic_args);
2968c2ecf20Sopenharmony_ci		input.pointer = &wmax_basic_args;
2978c2ecf20Sopenharmony_ci	} else {
2988c2ecf20Sopenharmony_ci		legacy_args.colors = zone->colors;
2998c2ecf20Sopenharmony_ci		legacy_args.brightness = global_brightness;
3008c2ecf20Sopenharmony_ci		legacy_args.state = 0;
3018c2ecf20Sopenharmony_ci		if (lighting_control_state == LEGACY_BOOTING ||
3028c2ecf20Sopenharmony_ci		    lighting_control_state == LEGACY_SUSPEND) {
3038c2ecf20Sopenharmony_ci			guid = LEGACY_POWER_CONTROL_GUID;
3048c2ecf20Sopenharmony_ci			legacy_args.state = lighting_control_state;
3058c2ecf20Sopenharmony_ci		} else
3068c2ecf20Sopenharmony_ci			guid = LEGACY_CONTROL_GUID;
3078c2ecf20Sopenharmony_ci		method_id = zone->location + 1;
3088c2ecf20Sopenharmony_ci
3098c2ecf20Sopenharmony_ci		input.length = (acpi_size) sizeof(legacy_args);
3108c2ecf20Sopenharmony_ci		input.pointer = &legacy_args;
3118c2ecf20Sopenharmony_ci	}
3128c2ecf20Sopenharmony_ci	pr_debug("alienware-wmi: guid %s method %d\n", guid, method_id);
3138c2ecf20Sopenharmony_ci
3148c2ecf20Sopenharmony_ci	status = wmi_evaluate_method(guid, 0, method_id, &input, NULL);
3158c2ecf20Sopenharmony_ci	if (ACPI_FAILURE(status))
3168c2ecf20Sopenharmony_ci		pr_err("alienware-wmi: zone set failure: %u\n", status);
3178c2ecf20Sopenharmony_ci	return ACPI_FAILURE(status);
3188c2ecf20Sopenharmony_ci}
3198c2ecf20Sopenharmony_ci
3208c2ecf20Sopenharmony_cistatic ssize_t zone_show(struct device *dev, struct device_attribute *attr,
3218c2ecf20Sopenharmony_ci			 char *buf)
3228c2ecf20Sopenharmony_ci{
3238c2ecf20Sopenharmony_ci	struct platform_zone *target_zone;
3248c2ecf20Sopenharmony_ci	target_zone = match_zone(attr);
3258c2ecf20Sopenharmony_ci	if (target_zone == NULL)
3268c2ecf20Sopenharmony_ci		return sprintf(buf, "red: -1, green: -1, blue: -1\n");
3278c2ecf20Sopenharmony_ci	return sprintf(buf, "red: %d, green: %d, blue: %d\n",
3288c2ecf20Sopenharmony_ci		       target_zone->colors.red,
3298c2ecf20Sopenharmony_ci		       target_zone->colors.green, target_zone->colors.blue);
3308c2ecf20Sopenharmony_ci
3318c2ecf20Sopenharmony_ci}
3328c2ecf20Sopenharmony_ci
3338c2ecf20Sopenharmony_cistatic ssize_t zone_set(struct device *dev, struct device_attribute *attr,
3348c2ecf20Sopenharmony_ci			const char *buf, size_t count)
3358c2ecf20Sopenharmony_ci{
3368c2ecf20Sopenharmony_ci	struct platform_zone *target_zone;
3378c2ecf20Sopenharmony_ci	int ret;
3388c2ecf20Sopenharmony_ci	target_zone = match_zone(attr);
3398c2ecf20Sopenharmony_ci	if (target_zone == NULL) {
3408c2ecf20Sopenharmony_ci		pr_err("alienware-wmi: invalid target zone\n");
3418c2ecf20Sopenharmony_ci		return 1;
3428c2ecf20Sopenharmony_ci	}
3438c2ecf20Sopenharmony_ci	ret = parse_rgb(buf, target_zone);
3448c2ecf20Sopenharmony_ci	if (ret)
3458c2ecf20Sopenharmony_ci		return ret;
3468c2ecf20Sopenharmony_ci	ret = alienware_update_led(target_zone);
3478c2ecf20Sopenharmony_ci	return ret ? ret : count;
3488c2ecf20Sopenharmony_ci}
3498c2ecf20Sopenharmony_ci
3508c2ecf20Sopenharmony_ci/*
3518c2ecf20Sopenharmony_ci * LED Brightness (Global)
3528c2ecf20Sopenharmony_ci */
3538c2ecf20Sopenharmony_cistatic int wmax_brightness(int brightness)
3548c2ecf20Sopenharmony_ci{
3558c2ecf20Sopenharmony_ci	acpi_status status;
3568c2ecf20Sopenharmony_ci	struct acpi_buffer input;
3578c2ecf20Sopenharmony_ci	struct wmax_brightness_args args = {
3588c2ecf20Sopenharmony_ci		.led_mask = 0xFF,
3598c2ecf20Sopenharmony_ci		.percentage = brightness,
3608c2ecf20Sopenharmony_ci	};
3618c2ecf20Sopenharmony_ci	input.length = (acpi_size) sizeof(args);
3628c2ecf20Sopenharmony_ci	input.pointer = &args;
3638c2ecf20Sopenharmony_ci	status = wmi_evaluate_method(WMAX_CONTROL_GUID, 0,
3648c2ecf20Sopenharmony_ci				     WMAX_METHOD_BRIGHTNESS, &input, NULL);
3658c2ecf20Sopenharmony_ci	if (ACPI_FAILURE(status))
3668c2ecf20Sopenharmony_ci		pr_err("alienware-wmi: brightness set failure: %u\n", status);
3678c2ecf20Sopenharmony_ci	return ACPI_FAILURE(status);
3688c2ecf20Sopenharmony_ci}
3698c2ecf20Sopenharmony_ci
3708c2ecf20Sopenharmony_cistatic void global_led_set(struct led_classdev *led_cdev,
3718c2ecf20Sopenharmony_ci			   enum led_brightness brightness)
3728c2ecf20Sopenharmony_ci{
3738c2ecf20Sopenharmony_ci	int ret;
3748c2ecf20Sopenharmony_ci	global_brightness = brightness;
3758c2ecf20Sopenharmony_ci	if (interface == WMAX)
3768c2ecf20Sopenharmony_ci		ret = wmax_brightness(brightness);
3778c2ecf20Sopenharmony_ci	else
3788c2ecf20Sopenharmony_ci		ret = alienware_update_led(&zone_data[0]);
3798c2ecf20Sopenharmony_ci	if (ret)
3808c2ecf20Sopenharmony_ci		pr_err("LED brightness update failed\n");
3818c2ecf20Sopenharmony_ci}
3828c2ecf20Sopenharmony_ci
3838c2ecf20Sopenharmony_cistatic enum led_brightness global_led_get(struct led_classdev *led_cdev)
3848c2ecf20Sopenharmony_ci{
3858c2ecf20Sopenharmony_ci	return global_brightness;
3868c2ecf20Sopenharmony_ci}
3878c2ecf20Sopenharmony_ci
3888c2ecf20Sopenharmony_cistatic struct led_classdev global_led = {
3898c2ecf20Sopenharmony_ci	.brightness_set = global_led_set,
3908c2ecf20Sopenharmony_ci	.brightness_get = global_led_get,
3918c2ecf20Sopenharmony_ci	.name = "alienware::global_brightness",
3928c2ecf20Sopenharmony_ci};
3938c2ecf20Sopenharmony_ci
3948c2ecf20Sopenharmony_ci/*
3958c2ecf20Sopenharmony_ci * Lighting control state device attribute (Global)
3968c2ecf20Sopenharmony_ci */
3978c2ecf20Sopenharmony_cistatic ssize_t show_control_state(struct device *dev,
3988c2ecf20Sopenharmony_ci				  struct device_attribute *attr, char *buf)
3998c2ecf20Sopenharmony_ci{
4008c2ecf20Sopenharmony_ci	if (lighting_control_state == LEGACY_BOOTING)
4018c2ecf20Sopenharmony_ci		return scnprintf(buf, PAGE_SIZE, "[booting] running suspend\n");
4028c2ecf20Sopenharmony_ci	else if (lighting_control_state == LEGACY_SUSPEND)
4038c2ecf20Sopenharmony_ci		return scnprintf(buf, PAGE_SIZE, "booting running [suspend]\n");
4048c2ecf20Sopenharmony_ci	return scnprintf(buf, PAGE_SIZE, "booting [running] suspend\n");
4058c2ecf20Sopenharmony_ci}
4068c2ecf20Sopenharmony_ci
4078c2ecf20Sopenharmony_cistatic ssize_t store_control_state(struct device *dev,
4088c2ecf20Sopenharmony_ci				   struct device_attribute *attr,
4098c2ecf20Sopenharmony_ci				   const char *buf, size_t count)
4108c2ecf20Sopenharmony_ci{
4118c2ecf20Sopenharmony_ci	long unsigned int val;
4128c2ecf20Sopenharmony_ci	if (strcmp(buf, "booting\n") == 0)
4138c2ecf20Sopenharmony_ci		val = LEGACY_BOOTING;
4148c2ecf20Sopenharmony_ci	else if (strcmp(buf, "suspend\n") == 0)
4158c2ecf20Sopenharmony_ci		val = LEGACY_SUSPEND;
4168c2ecf20Sopenharmony_ci	else if (interface == LEGACY)
4178c2ecf20Sopenharmony_ci		val = LEGACY_RUNNING;
4188c2ecf20Sopenharmony_ci	else
4198c2ecf20Sopenharmony_ci		val = WMAX_RUNNING;
4208c2ecf20Sopenharmony_ci	lighting_control_state = val;
4218c2ecf20Sopenharmony_ci	pr_debug("alienware-wmi: updated control state to %d\n",
4228c2ecf20Sopenharmony_ci		 lighting_control_state);
4238c2ecf20Sopenharmony_ci	return count;
4248c2ecf20Sopenharmony_ci}
4258c2ecf20Sopenharmony_ci
4268c2ecf20Sopenharmony_cistatic DEVICE_ATTR(lighting_control_state, 0644, show_control_state,
4278c2ecf20Sopenharmony_ci		   store_control_state);
4288c2ecf20Sopenharmony_ci
4298c2ecf20Sopenharmony_cistatic int alienware_zone_init(struct platform_device *dev)
4308c2ecf20Sopenharmony_ci{
4318c2ecf20Sopenharmony_ci	u8 zone;
4328c2ecf20Sopenharmony_ci	char buffer[10];
4338c2ecf20Sopenharmony_ci	char *name;
4348c2ecf20Sopenharmony_ci
4358c2ecf20Sopenharmony_ci	if (interface == WMAX) {
4368c2ecf20Sopenharmony_ci		lighting_control_state = WMAX_RUNNING;
4378c2ecf20Sopenharmony_ci	} else if (interface == LEGACY) {
4388c2ecf20Sopenharmony_ci		lighting_control_state = LEGACY_RUNNING;
4398c2ecf20Sopenharmony_ci	}
4408c2ecf20Sopenharmony_ci	global_led.max_brightness = 0x0F;
4418c2ecf20Sopenharmony_ci	global_brightness = global_led.max_brightness;
4428c2ecf20Sopenharmony_ci
4438c2ecf20Sopenharmony_ci	/*
4448c2ecf20Sopenharmony_ci	 *      - zone_dev_attrs num_zones + 1 is for individual zones and then
4458c2ecf20Sopenharmony_ci	 *        null terminated
4468c2ecf20Sopenharmony_ci	 *      - zone_attrs num_zones + 2 is for all attrs in zone_dev_attrs +
4478c2ecf20Sopenharmony_ci	 *        the lighting control + null terminated
4488c2ecf20Sopenharmony_ci	 *      - zone_data num_zones is for the distinct zones
4498c2ecf20Sopenharmony_ci	 */
4508c2ecf20Sopenharmony_ci	zone_dev_attrs =
4518c2ecf20Sopenharmony_ci	    kcalloc(quirks->num_zones + 1, sizeof(struct device_attribute),
4528c2ecf20Sopenharmony_ci		    GFP_KERNEL);
4538c2ecf20Sopenharmony_ci	if (!zone_dev_attrs)
4548c2ecf20Sopenharmony_ci		return -ENOMEM;
4558c2ecf20Sopenharmony_ci
4568c2ecf20Sopenharmony_ci	zone_attrs =
4578c2ecf20Sopenharmony_ci	    kcalloc(quirks->num_zones + 2, sizeof(struct attribute *),
4588c2ecf20Sopenharmony_ci		    GFP_KERNEL);
4598c2ecf20Sopenharmony_ci	if (!zone_attrs)
4608c2ecf20Sopenharmony_ci		return -ENOMEM;
4618c2ecf20Sopenharmony_ci
4628c2ecf20Sopenharmony_ci	zone_data =
4638c2ecf20Sopenharmony_ci	    kcalloc(quirks->num_zones, sizeof(struct platform_zone),
4648c2ecf20Sopenharmony_ci		    GFP_KERNEL);
4658c2ecf20Sopenharmony_ci	if (!zone_data)
4668c2ecf20Sopenharmony_ci		return -ENOMEM;
4678c2ecf20Sopenharmony_ci
4688c2ecf20Sopenharmony_ci	for (zone = 0; zone < quirks->num_zones; zone++) {
4698c2ecf20Sopenharmony_ci		sprintf(buffer, "zone%02hhX", zone);
4708c2ecf20Sopenharmony_ci		name = kstrdup(buffer, GFP_KERNEL);
4718c2ecf20Sopenharmony_ci		if (name == NULL)
4728c2ecf20Sopenharmony_ci			return 1;
4738c2ecf20Sopenharmony_ci		sysfs_attr_init(&zone_dev_attrs[zone].attr);
4748c2ecf20Sopenharmony_ci		zone_dev_attrs[zone].attr.name = name;
4758c2ecf20Sopenharmony_ci		zone_dev_attrs[zone].attr.mode = 0644;
4768c2ecf20Sopenharmony_ci		zone_dev_attrs[zone].show = zone_show;
4778c2ecf20Sopenharmony_ci		zone_dev_attrs[zone].store = zone_set;
4788c2ecf20Sopenharmony_ci		zone_data[zone].location = zone;
4798c2ecf20Sopenharmony_ci		zone_attrs[zone] = &zone_dev_attrs[zone].attr;
4808c2ecf20Sopenharmony_ci		zone_data[zone].attr = &zone_dev_attrs[zone];
4818c2ecf20Sopenharmony_ci	}
4828c2ecf20Sopenharmony_ci	zone_attrs[quirks->num_zones] = &dev_attr_lighting_control_state.attr;
4838c2ecf20Sopenharmony_ci	zone_attribute_group.attrs = zone_attrs;
4848c2ecf20Sopenharmony_ci
4858c2ecf20Sopenharmony_ci	led_classdev_register(&dev->dev, &global_led);
4868c2ecf20Sopenharmony_ci
4878c2ecf20Sopenharmony_ci	return sysfs_create_group(&dev->dev.kobj, &zone_attribute_group);
4888c2ecf20Sopenharmony_ci}
4898c2ecf20Sopenharmony_ci
4908c2ecf20Sopenharmony_cistatic void alienware_zone_exit(struct platform_device *dev)
4918c2ecf20Sopenharmony_ci{
4928c2ecf20Sopenharmony_ci	u8 zone;
4938c2ecf20Sopenharmony_ci
4948c2ecf20Sopenharmony_ci	sysfs_remove_group(&dev->dev.kobj, &zone_attribute_group);
4958c2ecf20Sopenharmony_ci	led_classdev_unregister(&global_led);
4968c2ecf20Sopenharmony_ci	if (zone_dev_attrs) {
4978c2ecf20Sopenharmony_ci		for (zone = 0; zone < quirks->num_zones; zone++)
4988c2ecf20Sopenharmony_ci			kfree(zone_dev_attrs[zone].attr.name);
4998c2ecf20Sopenharmony_ci	}
5008c2ecf20Sopenharmony_ci	kfree(zone_dev_attrs);
5018c2ecf20Sopenharmony_ci	kfree(zone_data);
5028c2ecf20Sopenharmony_ci	kfree(zone_attrs);
5038c2ecf20Sopenharmony_ci}
5048c2ecf20Sopenharmony_ci
5058c2ecf20Sopenharmony_cistatic acpi_status alienware_wmax_command(struct wmax_basic_args *in_args,
5068c2ecf20Sopenharmony_ci					  u32 command, int *out_data)
5078c2ecf20Sopenharmony_ci{
5088c2ecf20Sopenharmony_ci	acpi_status status;
5098c2ecf20Sopenharmony_ci	union acpi_object *obj;
5108c2ecf20Sopenharmony_ci	struct acpi_buffer input;
5118c2ecf20Sopenharmony_ci	struct acpi_buffer output;
5128c2ecf20Sopenharmony_ci
5138c2ecf20Sopenharmony_ci	input.length = (acpi_size) sizeof(*in_args);
5148c2ecf20Sopenharmony_ci	input.pointer = in_args;
5158c2ecf20Sopenharmony_ci	if (out_data) {
5168c2ecf20Sopenharmony_ci		output.length = ACPI_ALLOCATE_BUFFER;
5178c2ecf20Sopenharmony_ci		output.pointer = NULL;
5188c2ecf20Sopenharmony_ci		status = wmi_evaluate_method(WMAX_CONTROL_GUID, 0,
5198c2ecf20Sopenharmony_ci					     command, &input, &output);
5208c2ecf20Sopenharmony_ci		if (ACPI_SUCCESS(status)) {
5218c2ecf20Sopenharmony_ci			obj = (union acpi_object *)output.pointer;
5228c2ecf20Sopenharmony_ci			if (obj && obj->type == ACPI_TYPE_INTEGER)
5238c2ecf20Sopenharmony_ci				*out_data = (u32)obj->integer.value;
5248c2ecf20Sopenharmony_ci		}
5258c2ecf20Sopenharmony_ci		kfree(output.pointer);
5268c2ecf20Sopenharmony_ci	} else {
5278c2ecf20Sopenharmony_ci		status = wmi_evaluate_method(WMAX_CONTROL_GUID, 0,
5288c2ecf20Sopenharmony_ci					     command, &input, NULL);
5298c2ecf20Sopenharmony_ci	}
5308c2ecf20Sopenharmony_ci	return status;
5318c2ecf20Sopenharmony_ci}
5328c2ecf20Sopenharmony_ci
5338c2ecf20Sopenharmony_ci/*
5348c2ecf20Sopenharmony_ci *	The HDMI mux sysfs node indicates the status of the HDMI input mux.
5358c2ecf20Sopenharmony_ci *	It can toggle between standard system GPU output and HDMI input.
5368c2ecf20Sopenharmony_ci */
5378c2ecf20Sopenharmony_cistatic ssize_t show_hdmi_cable(struct device *dev,
5388c2ecf20Sopenharmony_ci			       struct device_attribute *attr, char *buf)
5398c2ecf20Sopenharmony_ci{
5408c2ecf20Sopenharmony_ci	acpi_status status;
5418c2ecf20Sopenharmony_ci	u32 out_data;
5428c2ecf20Sopenharmony_ci	struct wmax_basic_args in_args = {
5438c2ecf20Sopenharmony_ci		.arg = 0,
5448c2ecf20Sopenharmony_ci	};
5458c2ecf20Sopenharmony_ci	status =
5468c2ecf20Sopenharmony_ci	    alienware_wmax_command(&in_args, WMAX_METHOD_HDMI_CABLE,
5478c2ecf20Sopenharmony_ci				   (u32 *) &out_data);
5488c2ecf20Sopenharmony_ci	if (ACPI_SUCCESS(status)) {
5498c2ecf20Sopenharmony_ci		if (out_data == 0)
5508c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
5518c2ecf20Sopenharmony_ci					 "[unconnected] connected unknown\n");
5528c2ecf20Sopenharmony_ci		else if (out_data == 1)
5538c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
5548c2ecf20Sopenharmony_ci					 "unconnected [connected] unknown\n");
5558c2ecf20Sopenharmony_ci	}
5568c2ecf20Sopenharmony_ci	pr_err("alienware-wmi: unknown HDMI cable status: %d\n", status);
5578c2ecf20Sopenharmony_ci	return scnprintf(buf, PAGE_SIZE, "unconnected connected [unknown]\n");
5588c2ecf20Sopenharmony_ci}
5598c2ecf20Sopenharmony_ci
5608c2ecf20Sopenharmony_cistatic ssize_t show_hdmi_source(struct device *dev,
5618c2ecf20Sopenharmony_ci				struct device_attribute *attr, char *buf)
5628c2ecf20Sopenharmony_ci{
5638c2ecf20Sopenharmony_ci	acpi_status status;
5648c2ecf20Sopenharmony_ci	u32 out_data;
5658c2ecf20Sopenharmony_ci	struct wmax_basic_args in_args = {
5668c2ecf20Sopenharmony_ci		.arg = 0,
5678c2ecf20Sopenharmony_ci	};
5688c2ecf20Sopenharmony_ci	status =
5698c2ecf20Sopenharmony_ci	    alienware_wmax_command(&in_args, WMAX_METHOD_HDMI_STATUS,
5708c2ecf20Sopenharmony_ci				   (u32 *) &out_data);
5718c2ecf20Sopenharmony_ci
5728c2ecf20Sopenharmony_ci	if (ACPI_SUCCESS(status)) {
5738c2ecf20Sopenharmony_ci		if (out_data == 1)
5748c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
5758c2ecf20Sopenharmony_ci					 "[input] gpu unknown\n");
5768c2ecf20Sopenharmony_ci		else if (out_data == 2)
5778c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
5788c2ecf20Sopenharmony_ci					 "input [gpu] unknown\n");
5798c2ecf20Sopenharmony_ci	}
5808c2ecf20Sopenharmony_ci	pr_err("alienware-wmi: unknown HDMI source status: %u\n", status);
5818c2ecf20Sopenharmony_ci	return scnprintf(buf, PAGE_SIZE, "input gpu [unknown]\n");
5828c2ecf20Sopenharmony_ci}
5838c2ecf20Sopenharmony_ci
5848c2ecf20Sopenharmony_cistatic ssize_t toggle_hdmi_source(struct device *dev,
5858c2ecf20Sopenharmony_ci				  struct device_attribute *attr,
5868c2ecf20Sopenharmony_ci				  const char *buf, size_t count)
5878c2ecf20Sopenharmony_ci{
5888c2ecf20Sopenharmony_ci	acpi_status status;
5898c2ecf20Sopenharmony_ci	struct wmax_basic_args args;
5908c2ecf20Sopenharmony_ci	if (strcmp(buf, "gpu\n") == 0)
5918c2ecf20Sopenharmony_ci		args.arg = 1;
5928c2ecf20Sopenharmony_ci	else if (strcmp(buf, "input\n") == 0)
5938c2ecf20Sopenharmony_ci		args.arg = 2;
5948c2ecf20Sopenharmony_ci	else
5958c2ecf20Sopenharmony_ci		args.arg = 3;
5968c2ecf20Sopenharmony_ci	pr_debug("alienware-wmi: setting hdmi to %d : %s", args.arg, buf);
5978c2ecf20Sopenharmony_ci
5988c2ecf20Sopenharmony_ci	status = alienware_wmax_command(&args, WMAX_METHOD_HDMI_SOURCE, NULL);
5998c2ecf20Sopenharmony_ci
6008c2ecf20Sopenharmony_ci	if (ACPI_FAILURE(status))
6018c2ecf20Sopenharmony_ci		pr_err("alienware-wmi: HDMI toggle failed: results: %u\n",
6028c2ecf20Sopenharmony_ci		       status);
6038c2ecf20Sopenharmony_ci	return count;
6048c2ecf20Sopenharmony_ci}
6058c2ecf20Sopenharmony_ci
6068c2ecf20Sopenharmony_cistatic DEVICE_ATTR(cable, S_IRUGO, show_hdmi_cable, NULL);
6078c2ecf20Sopenharmony_cistatic DEVICE_ATTR(source, S_IRUGO | S_IWUSR, show_hdmi_source,
6088c2ecf20Sopenharmony_ci		   toggle_hdmi_source);
6098c2ecf20Sopenharmony_ci
6108c2ecf20Sopenharmony_cistatic struct attribute *hdmi_attrs[] = {
6118c2ecf20Sopenharmony_ci	&dev_attr_cable.attr,
6128c2ecf20Sopenharmony_ci	&dev_attr_source.attr,
6138c2ecf20Sopenharmony_ci	NULL,
6148c2ecf20Sopenharmony_ci};
6158c2ecf20Sopenharmony_ci
6168c2ecf20Sopenharmony_cistatic const struct attribute_group hdmi_attribute_group = {
6178c2ecf20Sopenharmony_ci	.name = "hdmi",
6188c2ecf20Sopenharmony_ci	.attrs = hdmi_attrs,
6198c2ecf20Sopenharmony_ci};
6208c2ecf20Sopenharmony_ci
6218c2ecf20Sopenharmony_cistatic void remove_hdmi(struct platform_device *dev)
6228c2ecf20Sopenharmony_ci{
6238c2ecf20Sopenharmony_ci	if (quirks->hdmi_mux > 0)
6248c2ecf20Sopenharmony_ci		sysfs_remove_group(&dev->dev.kobj, &hdmi_attribute_group);
6258c2ecf20Sopenharmony_ci}
6268c2ecf20Sopenharmony_ci
6278c2ecf20Sopenharmony_cistatic int create_hdmi(struct platform_device *dev)
6288c2ecf20Sopenharmony_ci{
6298c2ecf20Sopenharmony_ci	int ret;
6308c2ecf20Sopenharmony_ci
6318c2ecf20Sopenharmony_ci	ret = sysfs_create_group(&dev->dev.kobj, &hdmi_attribute_group);
6328c2ecf20Sopenharmony_ci	if (ret)
6338c2ecf20Sopenharmony_ci		remove_hdmi(dev);
6348c2ecf20Sopenharmony_ci	return ret;
6358c2ecf20Sopenharmony_ci}
6368c2ecf20Sopenharmony_ci
6378c2ecf20Sopenharmony_ci/*
6388c2ecf20Sopenharmony_ci * Alienware GFX amplifier support
6398c2ecf20Sopenharmony_ci * - Currently supports reading cable status
6408c2ecf20Sopenharmony_ci * - Leaving expansion room to possibly support dock/undock events later
6418c2ecf20Sopenharmony_ci */
6428c2ecf20Sopenharmony_cistatic ssize_t show_amplifier_status(struct device *dev,
6438c2ecf20Sopenharmony_ci				     struct device_attribute *attr, char *buf)
6448c2ecf20Sopenharmony_ci{
6458c2ecf20Sopenharmony_ci	acpi_status status;
6468c2ecf20Sopenharmony_ci	u32 out_data;
6478c2ecf20Sopenharmony_ci	struct wmax_basic_args in_args = {
6488c2ecf20Sopenharmony_ci		.arg = 0,
6498c2ecf20Sopenharmony_ci	};
6508c2ecf20Sopenharmony_ci	status =
6518c2ecf20Sopenharmony_ci	    alienware_wmax_command(&in_args, WMAX_METHOD_AMPLIFIER_CABLE,
6528c2ecf20Sopenharmony_ci				   (u32 *) &out_data);
6538c2ecf20Sopenharmony_ci	if (ACPI_SUCCESS(status)) {
6548c2ecf20Sopenharmony_ci		if (out_data == 0)
6558c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
6568c2ecf20Sopenharmony_ci					 "[unconnected] connected unknown\n");
6578c2ecf20Sopenharmony_ci		else if (out_data == 1)
6588c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
6598c2ecf20Sopenharmony_ci					 "unconnected [connected] unknown\n");
6608c2ecf20Sopenharmony_ci	}
6618c2ecf20Sopenharmony_ci	pr_err("alienware-wmi: unknown amplifier cable status: %d\n", status);
6628c2ecf20Sopenharmony_ci	return scnprintf(buf, PAGE_SIZE, "unconnected connected [unknown]\n");
6638c2ecf20Sopenharmony_ci}
6648c2ecf20Sopenharmony_ci
6658c2ecf20Sopenharmony_cistatic DEVICE_ATTR(status, S_IRUGO, show_amplifier_status, NULL);
6668c2ecf20Sopenharmony_ci
6678c2ecf20Sopenharmony_cistatic struct attribute *amplifier_attrs[] = {
6688c2ecf20Sopenharmony_ci	&dev_attr_status.attr,
6698c2ecf20Sopenharmony_ci	NULL,
6708c2ecf20Sopenharmony_ci};
6718c2ecf20Sopenharmony_ci
6728c2ecf20Sopenharmony_cistatic const struct attribute_group amplifier_attribute_group = {
6738c2ecf20Sopenharmony_ci	.name = "amplifier",
6748c2ecf20Sopenharmony_ci	.attrs = amplifier_attrs,
6758c2ecf20Sopenharmony_ci};
6768c2ecf20Sopenharmony_ci
6778c2ecf20Sopenharmony_cistatic void remove_amplifier(struct platform_device *dev)
6788c2ecf20Sopenharmony_ci{
6798c2ecf20Sopenharmony_ci	if (quirks->amplifier > 0)
6808c2ecf20Sopenharmony_ci		sysfs_remove_group(&dev->dev.kobj, &amplifier_attribute_group);
6818c2ecf20Sopenharmony_ci}
6828c2ecf20Sopenharmony_ci
6838c2ecf20Sopenharmony_cistatic int create_amplifier(struct platform_device *dev)
6848c2ecf20Sopenharmony_ci{
6858c2ecf20Sopenharmony_ci	int ret;
6868c2ecf20Sopenharmony_ci
6878c2ecf20Sopenharmony_ci	ret = sysfs_create_group(&dev->dev.kobj, &amplifier_attribute_group);
6888c2ecf20Sopenharmony_ci	if (ret)
6898c2ecf20Sopenharmony_ci		remove_amplifier(dev);
6908c2ecf20Sopenharmony_ci	return ret;
6918c2ecf20Sopenharmony_ci}
6928c2ecf20Sopenharmony_ci
6938c2ecf20Sopenharmony_ci/*
6948c2ecf20Sopenharmony_ci * Deep Sleep Control support
6958c2ecf20Sopenharmony_ci * - Modifies BIOS setting for deep sleep control allowing extra wakeup events
6968c2ecf20Sopenharmony_ci */
6978c2ecf20Sopenharmony_cistatic ssize_t show_deepsleep_status(struct device *dev,
6988c2ecf20Sopenharmony_ci				     struct device_attribute *attr, char *buf)
6998c2ecf20Sopenharmony_ci{
7008c2ecf20Sopenharmony_ci	acpi_status status;
7018c2ecf20Sopenharmony_ci	u32 out_data;
7028c2ecf20Sopenharmony_ci	struct wmax_basic_args in_args = {
7038c2ecf20Sopenharmony_ci		.arg = 0,
7048c2ecf20Sopenharmony_ci	};
7058c2ecf20Sopenharmony_ci	status = alienware_wmax_command(&in_args, WMAX_METHOD_DEEP_SLEEP_STATUS,
7068c2ecf20Sopenharmony_ci					(u32 *) &out_data);
7078c2ecf20Sopenharmony_ci	if (ACPI_SUCCESS(status)) {
7088c2ecf20Sopenharmony_ci		if (out_data == 0)
7098c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
7108c2ecf20Sopenharmony_ci					 "[disabled] s5 s5_s4\n");
7118c2ecf20Sopenharmony_ci		else if (out_data == 1)
7128c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
7138c2ecf20Sopenharmony_ci					 "disabled [s5] s5_s4\n");
7148c2ecf20Sopenharmony_ci		else if (out_data == 2)
7158c2ecf20Sopenharmony_ci			return scnprintf(buf, PAGE_SIZE,
7168c2ecf20Sopenharmony_ci					 "disabled s5 [s5_s4]\n");
7178c2ecf20Sopenharmony_ci	}
7188c2ecf20Sopenharmony_ci	pr_err("alienware-wmi: unknown deep sleep status: %d\n", status);
7198c2ecf20Sopenharmony_ci	return scnprintf(buf, PAGE_SIZE, "disabled s5 s5_s4 [unknown]\n");
7208c2ecf20Sopenharmony_ci}
7218c2ecf20Sopenharmony_ci
7228c2ecf20Sopenharmony_cistatic ssize_t toggle_deepsleep(struct device *dev,
7238c2ecf20Sopenharmony_ci				struct device_attribute *attr,
7248c2ecf20Sopenharmony_ci				const char *buf, size_t count)
7258c2ecf20Sopenharmony_ci{
7268c2ecf20Sopenharmony_ci	acpi_status status;
7278c2ecf20Sopenharmony_ci	struct wmax_basic_args args;
7288c2ecf20Sopenharmony_ci
7298c2ecf20Sopenharmony_ci	if (strcmp(buf, "disabled\n") == 0)
7308c2ecf20Sopenharmony_ci		args.arg = 0;
7318c2ecf20Sopenharmony_ci	else if (strcmp(buf, "s5\n") == 0)
7328c2ecf20Sopenharmony_ci		args.arg = 1;
7338c2ecf20Sopenharmony_ci	else
7348c2ecf20Sopenharmony_ci		args.arg = 2;
7358c2ecf20Sopenharmony_ci	pr_debug("alienware-wmi: setting deep sleep to %d : %s", args.arg, buf);
7368c2ecf20Sopenharmony_ci
7378c2ecf20Sopenharmony_ci	status = alienware_wmax_command(&args, WMAX_METHOD_DEEP_SLEEP_CONTROL,
7388c2ecf20Sopenharmony_ci					NULL);
7398c2ecf20Sopenharmony_ci
7408c2ecf20Sopenharmony_ci	if (ACPI_FAILURE(status))
7418c2ecf20Sopenharmony_ci		pr_err("alienware-wmi: deep sleep control failed: results: %u\n",
7428c2ecf20Sopenharmony_ci			status);
7438c2ecf20Sopenharmony_ci	return count;
7448c2ecf20Sopenharmony_ci}
7458c2ecf20Sopenharmony_ci
7468c2ecf20Sopenharmony_cistatic DEVICE_ATTR(deepsleep, S_IRUGO | S_IWUSR, show_deepsleep_status, toggle_deepsleep);
7478c2ecf20Sopenharmony_ci
7488c2ecf20Sopenharmony_cistatic struct attribute *deepsleep_attrs[] = {
7498c2ecf20Sopenharmony_ci	&dev_attr_deepsleep.attr,
7508c2ecf20Sopenharmony_ci	NULL,
7518c2ecf20Sopenharmony_ci};
7528c2ecf20Sopenharmony_ci
7538c2ecf20Sopenharmony_cistatic const struct attribute_group deepsleep_attribute_group = {
7548c2ecf20Sopenharmony_ci	.name = "deepsleep",
7558c2ecf20Sopenharmony_ci	.attrs = deepsleep_attrs,
7568c2ecf20Sopenharmony_ci};
7578c2ecf20Sopenharmony_ci
7588c2ecf20Sopenharmony_cistatic void remove_deepsleep(struct platform_device *dev)
7598c2ecf20Sopenharmony_ci{
7608c2ecf20Sopenharmony_ci	if (quirks->deepslp > 0)
7618c2ecf20Sopenharmony_ci		sysfs_remove_group(&dev->dev.kobj, &deepsleep_attribute_group);
7628c2ecf20Sopenharmony_ci}
7638c2ecf20Sopenharmony_ci
7648c2ecf20Sopenharmony_cistatic int create_deepsleep(struct platform_device *dev)
7658c2ecf20Sopenharmony_ci{
7668c2ecf20Sopenharmony_ci	int ret;
7678c2ecf20Sopenharmony_ci
7688c2ecf20Sopenharmony_ci	ret = sysfs_create_group(&dev->dev.kobj, &deepsleep_attribute_group);
7698c2ecf20Sopenharmony_ci	if (ret)
7708c2ecf20Sopenharmony_ci		remove_deepsleep(dev);
7718c2ecf20Sopenharmony_ci	return ret;
7728c2ecf20Sopenharmony_ci}
7738c2ecf20Sopenharmony_ci
7748c2ecf20Sopenharmony_cistatic int __init alienware_wmi_init(void)
7758c2ecf20Sopenharmony_ci{
7768c2ecf20Sopenharmony_ci	int ret;
7778c2ecf20Sopenharmony_ci
7788c2ecf20Sopenharmony_ci	if (wmi_has_guid(LEGACY_CONTROL_GUID))
7798c2ecf20Sopenharmony_ci		interface = LEGACY;
7808c2ecf20Sopenharmony_ci	else if (wmi_has_guid(WMAX_CONTROL_GUID))
7818c2ecf20Sopenharmony_ci		interface = WMAX;
7828c2ecf20Sopenharmony_ci	else {
7838c2ecf20Sopenharmony_ci		pr_warn("alienware-wmi: No known WMI GUID found\n");
7848c2ecf20Sopenharmony_ci		return -ENODEV;
7858c2ecf20Sopenharmony_ci	}
7868c2ecf20Sopenharmony_ci
7878c2ecf20Sopenharmony_ci	dmi_check_system(alienware_quirks);
7888c2ecf20Sopenharmony_ci	if (quirks == NULL)
7898c2ecf20Sopenharmony_ci		quirks = &quirk_unknown;
7908c2ecf20Sopenharmony_ci
7918c2ecf20Sopenharmony_ci	ret = platform_driver_register(&platform_driver);
7928c2ecf20Sopenharmony_ci	if (ret)
7938c2ecf20Sopenharmony_ci		goto fail_platform_driver;
7948c2ecf20Sopenharmony_ci	platform_device = platform_device_alloc("alienware-wmi", -1);
7958c2ecf20Sopenharmony_ci	if (!platform_device) {
7968c2ecf20Sopenharmony_ci		ret = -ENOMEM;
7978c2ecf20Sopenharmony_ci		goto fail_platform_device1;
7988c2ecf20Sopenharmony_ci	}
7998c2ecf20Sopenharmony_ci	ret = platform_device_add(platform_device);
8008c2ecf20Sopenharmony_ci	if (ret)
8018c2ecf20Sopenharmony_ci		goto fail_platform_device2;
8028c2ecf20Sopenharmony_ci
8038c2ecf20Sopenharmony_ci	if (quirks->hdmi_mux > 0) {
8048c2ecf20Sopenharmony_ci		ret = create_hdmi(platform_device);
8058c2ecf20Sopenharmony_ci		if (ret)
8068c2ecf20Sopenharmony_ci			goto fail_prep_hdmi;
8078c2ecf20Sopenharmony_ci	}
8088c2ecf20Sopenharmony_ci
8098c2ecf20Sopenharmony_ci	if (quirks->amplifier > 0) {
8108c2ecf20Sopenharmony_ci		ret = create_amplifier(platform_device);
8118c2ecf20Sopenharmony_ci		if (ret)
8128c2ecf20Sopenharmony_ci			goto fail_prep_amplifier;
8138c2ecf20Sopenharmony_ci	}
8148c2ecf20Sopenharmony_ci
8158c2ecf20Sopenharmony_ci	if (quirks->deepslp > 0) {
8168c2ecf20Sopenharmony_ci		ret = create_deepsleep(platform_device);
8178c2ecf20Sopenharmony_ci		if (ret)
8188c2ecf20Sopenharmony_ci			goto fail_prep_deepsleep;
8198c2ecf20Sopenharmony_ci	}
8208c2ecf20Sopenharmony_ci
8218c2ecf20Sopenharmony_ci	ret = alienware_zone_init(platform_device);
8228c2ecf20Sopenharmony_ci	if (ret)
8238c2ecf20Sopenharmony_ci		goto fail_prep_zones;
8248c2ecf20Sopenharmony_ci
8258c2ecf20Sopenharmony_ci	return 0;
8268c2ecf20Sopenharmony_ci
8278c2ecf20Sopenharmony_cifail_prep_zones:
8288c2ecf20Sopenharmony_ci	alienware_zone_exit(platform_device);
8298c2ecf20Sopenharmony_cifail_prep_deepsleep:
8308c2ecf20Sopenharmony_cifail_prep_amplifier:
8318c2ecf20Sopenharmony_cifail_prep_hdmi:
8328c2ecf20Sopenharmony_ci	platform_device_del(platform_device);
8338c2ecf20Sopenharmony_cifail_platform_device2:
8348c2ecf20Sopenharmony_ci	platform_device_put(platform_device);
8358c2ecf20Sopenharmony_cifail_platform_device1:
8368c2ecf20Sopenharmony_ci	platform_driver_unregister(&platform_driver);
8378c2ecf20Sopenharmony_cifail_platform_driver:
8388c2ecf20Sopenharmony_ci	return ret;
8398c2ecf20Sopenharmony_ci}
8408c2ecf20Sopenharmony_ci
8418c2ecf20Sopenharmony_cimodule_init(alienware_wmi_init);
8428c2ecf20Sopenharmony_ci
8438c2ecf20Sopenharmony_cistatic void __exit alienware_wmi_exit(void)
8448c2ecf20Sopenharmony_ci{
8458c2ecf20Sopenharmony_ci	if (platform_device) {
8468c2ecf20Sopenharmony_ci		alienware_zone_exit(platform_device);
8478c2ecf20Sopenharmony_ci		remove_hdmi(platform_device);
8488c2ecf20Sopenharmony_ci		platform_device_unregister(platform_device);
8498c2ecf20Sopenharmony_ci		platform_driver_unregister(&platform_driver);
8508c2ecf20Sopenharmony_ci	}
8518c2ecf20Sopenharmony_ci}
8528c2ecf20Sopenharmony_ci
8538c2ecf20Sopenharmony_cimodule_exit(alienware_wmi_exit);
854