xref: /dflybsd-src/sys/dev/acpica/acpi_thinkpad/acpi_thinkpad.c (revision c91db74236e5aa8b26bad43072e4e40a1fe4f5ff)
15db2f26eSSascha Wildner /*
25db2f26eSSascha Wildner  * Copyright (c) 2004 Takanori Watanabe
35db2f26eSSascha Wildner  * Copyright (c) 2005 Markus Brueffer <markus@FreeBSD.org>
45db2f26eSSascha Wildner  * All rights reserved.
55db2f26eSSascha Wildner  *
65db2f26eSSascha Wildner  * Redistribution and use in source and binary forms, with or without
75db2f26eSSascha Wildner  * modification, are permitted provided that the following conditions
85db2f26eSSascha Wildner  * are met:
95db2f26eSSascha Wildner  * 1. Redistributions of source code must retain the above copyright
105db2f26eSSascha Wildner  *    notice, this list of conditions and the following disclaimer.
115db2f26eSSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
125db2f26eSSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
135db2f26eSSascha Wildner  *    documentation and/or other materials provided with the distribution.
145db2f26eSSascha Wildner  *
155db2f26eSSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
165db2f26eSSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
175db2f26eSSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
185db2f26eSSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
195db2f26eSSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
205db2f26eSSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
215db2f26eSSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
225db2f26eSSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
235db2f26eSSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
245db2f26eSSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
255db2f26eSSascha Wildner  * SUCH DAMAGE.
265db2f26eSSascha Wildner  *
2720befb6eSPierre-Alain TORET  * $FreeBSD: head/sys/dev/acpi_support/acpi_ibm.c 358333 2020-02-26 14:26:36Z kaktus $
285db2f26eSSascha Wildner  */
295db2f26eSSascha Wildner 
305db2f26eSSascha Wildner /*
3120befb6eSPierre-Alain TORET  * Driver for extra ACPI-controlled gadgets found on ThinkPad laptops.
3220befb6eSPierre-Alain TORET  * Inspired by the ibm-acpi and tpb projects which implement these features
3320befb6eSPierre-Alain TORET  * on Linux.
345db2f26eSSascha Wildner  *
355db2f26eSSascha Wildner  *   acpi-ibm: <http://ibm-acpi.sourceforge.net/>
365db2f26eSSascha Wildner  *        tpb: <http://www.nongnu.org/tpb/>
375db2f26eSSascha Wildner  */
385db2f26eSSascha Wildner 
395db2f26eSSascha Wildner #include "opt_acpi.h"
405db2f26eSSascha Wildner #include <sys/param.h>
415db2f26eSSascha Wildner #include <sys/kernel.h>
425db2f26eSSascha Wildner #include <sys/bus.h>
435db2f26eSSascha Wildner #include <machine/cpufunc.h>
445db2f26eSSascha Wildner #include <sys/module.h>
455db2f26eSSascha Wildner #include <sys/sensors.h>
46a31756d0SMarkus Pfeiffer #include <sys/sbuf.h>
475db2f26eSSascha Wildner #include <sys/sysctl.h>
485db2f26eSSascha Wildner #include <sys/lock.h>
4920befb6eSPierre-Alain TORET #include <sys/power.h>
505db2f26eSSascha Wildner #include <machine/clock.h>
51e9283513SSascha Wildner #include <dev/misc/led/led.h>
525db2f26eSSascha Wildner 
535db2f26eSSascha Wildner #include "acpi.h"
545db2f26eSSascha Wildner #include "accommon.h"
555db2f26eSSascha Wildner #include "acpivar.h"
565db2f26eSSascha Wildner 
575db2f26eSSascha Wildner #define _COMPONENT	ACPI_OEM
585db2f26eSSascha Wildner ACPI_MODULE_NAME("THINKPAD")
595db2f26eSSascha Wildner 
605db2f26eSSascha Wildner /* Internal methods */
615db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_EVENTS		1
625db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_EVENTMASK		2
635db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_HOTKEY		3
645db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_BRIGHTNESS		4
655db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_VOLUME		5
665db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_MUTE		6
675db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_THINKLIGHT		7
685db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_BLUETOOTH		8
695db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_WLAN		9
705db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_FANSPEED		10
715db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_FANLEVEL		11
725db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_FANSTATUS		12
735db2f26eSSascha Wildner #define ACPI_THINKPAD_METHOD_THERMAL		13
74a31756d0SMarkus Pfeiffer #define ACPI_THINKPAD_METHOD_HANDLEREVENTS	14
7520befb6eSPierre-Alain TORET #define ACPI_THINKPAD_METHOD_MIC_LED		15
7620befb6eSPierre-Alain TORET #define ACPI_THINKPAD_METHOD_PRIVACYGUARD	16
775db2f26eSSascha Wildner 
785db2f26eSSascha Wildner /* Hotkeys/Buttons */
795db2f26eSSascha Wildner #define THINKPAD_RTC_HOTKEY1			0x64
805db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_HOME		(1 << 0)
815db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_SEARCH		(1 << 1)
825db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_MAIL		(1 << 2)
835db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_WLAN		(1 << 5)
845db2f26eSSascha Wildner #define THINKPAD_RTC_HOTKEY2			0x65
855db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_THINKPAD		(1 << 3)
865db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_ZOOM		(1 << 5)
875db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_VIDEO		(1 << 6)
885db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_HIBERNATE		(1 << 7)
895db2f26eSSascha Wildner #define THINKPAD_RTC_THINKLIGHT			0x66
905db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_THINKLIGHT		(1 << 4)
915db2f26eSSascha Wildner #define THINKPAD_RTC_SCREENEXPAND		0x67
925db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_SCREENEXPAND	(1 << 5)
935db2f26eSSascha Wildner #define THINKPAD_RTC_BRIGHTNESS			0x6c
945db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_BRIGHTNESS		(1 << 5)
955db2f26eSSascha Wildner #define THINKPAD_RTC_VOLUME			0x6e
965db2f26eSSascha Wildner #define   THINKPAD_RTC_MASK_VOLUME		(1 << 7)
975db2f26eSSascha Wildner 
985db2f26eSSascha Wildner /* Embedded Controller registers */
995db2f26eSSascha Wildner #define THINKPAD_EC_BRIGHTNESS			0x31
1005db2f26eSSascha Wildner #define   THINKPAD_EC_MASK_BRI			0x7
1015db2f26eSSascha Wildner #define THINKPAD_EC_VOLUME			0x30
1025db2f26eSSascha Wildner #define   THINKPAD_EC_MASK_VOL			0xf
1035db2f26eSSascha Wildner #define   THINKPAD_EC_MASK_MUTE			(1 << 6)
1045db2f26eSSascha Wildner #define THINKPAD_EC_FANSTATUS			0x2F
1055db2f26eSSascha Wildner #define   THINKPAD_EC_MASK_FANLEVEL		0x3f
1065db2f26eSSascha Wildner #define   THINKPAD_EC_MASK_FANDISENGAGED	(1 << 6)
1075db2f26eSSascha Wildner #define   THINKPAD_EC_MASK_FANSTATUS		(1 << 7)
1085db2f26eSSascha Wildner #define THINKPAD_EC_FANSPEED			0x84
1095db2f26eSSascha Wildner 
1105db2f26eSSascha Wildner /* CMOS Commands */
1115db2f26eSSascha Wildner #define THINKPAD_CMOS_VOLUME_DOWN		0
1125db2f26eSSascha Wildner #define THINKPAD_CMOS_VOLUME_UP			1
1135db2f26eSSascha Wildner #define THINKPAD_CMOS_VOLUME_MUTE		2
1145db2f26eSSascha Wildner #define THINKPAD_CMOS_BRIGHTNESS_UP		4
1155db2f26eSSascha Wildner #define THINKPAD_CMOS_BRIGHTNESS_DOWN		5
1165db2f26eSSascha Wildner 
1175db2f26eSSascha Wildner /* ACPI methods */
1185db2f26eSSascha Wildner #define THINKPAD_NAME_KEYLIGHT			"KBLT"
1195db2f26eSSascha Wildner #define THINKPAD_NAME_WLAN_BT_GET		"GBDC"
1205db2f26eSSascha Wildner #define THINKPAD_NAME_WLAN_BT_SET		"SBDC"
1215db2f26eSSascha Wildner #define   THINKPAD_NAME_MASK_BT			(1 << 1)
1225db2f26eSSascha Wildner #define   THINKPAD_NAME_MASK_WLAN		(1 << 2)
1235db2f26eSSascha Wildner #define THINKPAD_NAME_THERMAL_GET		"TMP7"
1245db2f26eSSascha Wildner #define THINKPAD_NAME_THERMAL_UPDT		"UPDT"
12520befb6eSPierre-Alain TORET #define THINKPAD_NAME_PRIVACYGUARD_GET	"GSSS"
12620befb6eSPierre-Alain TORET #define THINKPAD_NAME_PRIVACYGUARD_SET	"SSSS"
1275db2f26eSSascha Wildner 
1285db2f26eSSascha Wildner #define THINKPAD_NAME_EVENTS_STATUS_GET		"DHKC"
1295db2f26eSSascha Wildner #define THINKPAD_NAME_EVENTS_MASK_GET		"DHKN"
1305db2f26eSSascha Wildner #define THINKPAD_NAME_EVENTS_STATUS_SET		"MHKC"
1315db2f26eSSascha Wildner #define THINKPAD_NAME_EVENTS_MASK_SET		"MHKM"
1325db2f26eSSascha Wildner #define THINKPAD_NAME_EVENTS_GET		"MHKP"
1335db2f26eSSascha Wildner #define THINKPAD_NAME_EVENTS_AVAILMASK		"MHKA"
1345db2f26eSSascha Wildner 
1355db2f26eSSascha Wildner #define	THINKPAD_NUM_SENSORS			9
1365db2f26eSSascha Wildner #define	THINKPAD_TEMP_SENSORS			8
1375db2f26eSSascha Wildner 
138a31756d0SMarkus Pfeiffer /* Event Code */
139a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_LCD_BACKLIGHT		0x03
140a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_SUSPEND_TO_RAM		0x04
141a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_BLUETOOTH		0x05
142a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_SCREEN_EXPAND		0x07
143a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_SUSPEND_TO_DISK		0x0c
144a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_BRIGHTNESS_UP		0x10
145a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_BRIGHTNESS_DOWN		0x11
146a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_THINKLIGHT		0x12
147a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_ZOOM			0x14
148a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_VOLUME_UP		0x15
149a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_VOLUME_DOWN		0x16
150a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_MUTE			0x17
151a31756d0SMarkus Pfeiffer #define THINKPAD_EVENT_ACCESS_THINKPAD_BUTTON	0x18
152a31756d0SMarkus Pfeiffer 
15320befb6eSPierre-Alain TORET /* Device-specific register flags */
15420befb6eSPierre-Alain TORET #define THINKPAD_FLAG_PRIVACYGUARD_DEVICE_PRESENT	0x10000
15520befb6eSPierre-Alain TORET #define THINKPAD_FLAG_PRIVACYGUARD_ON	0x1
15620befb6eSPierre-Alain TORET 
1575db2f26eSSascha Wildner #define ABS(x) (((x) < 0)? -(x) : (x))
1585db2f26eSSascha Wildner 
1595db2f26eSSascha Wildner struct acpi_thinkpad_softc {
1605db2f26eSSascha Wildner 	device_t	dev;
1615db2f26eSSascha Wildner 	ACPI_HANDLE	handle;
1625db2f26eSSascha Wildner 
1635db2f26eSSascha Wildner 	/* Embedded controller */
1645db2f26eSSascha Wildner 	device_t	ec_dev;
1655db2f26eSSascha Wildner 	ACPI_HANDLE	ec_handle;
1665db2f26eSSascha Wildner 
1675db2f26eSSascha Wildner 	/* CMOS */
1685db2f26eSSascha Wildner 	ACPI_HANDLE	cmos_handle;
1695db2f26eSSascha Wildner 
1705db2f26eSSascha Wildner 	/* Fan status */
1715db2f26eSSascha Wildner 	ACPI_HANDLE	fan_handle;
1725db2f26eSSascha Wildner 	int		fan_levels;
1735db2f26eSSascha Wildner 
1745db2f26eSSascha Wildner 	/* Keylight commands and states */
1755db2f26eSSascha Wildner 	ACPI_HANDLE	light_handle;
1765db2f26eSSascha Wildner 	int		light_cmd_on;
1775db2f26eSSascha Wildner 	int		light_cmd_off;
1785db2f26eSSascha Wildner 	int		light_val;
1795db2f26eSSascha Wildner 	int		light_get_supported;
1805db2f26eSSascha Wildner 	int		light_set_supported;
1815db2f26eSSascha Wildner 
1825db2f26eSSascha Wildner 	/* led(4) interface */
1835db2f26eSSascha Wildner 	struct cdev	*led_dev;
1845db2f26eSSascha Wildner 	int		led_busy;
1855db2f26eSSascha Wildner 	int		led_state;
1865db2f26eSSascha Wildner 
18720befb6eSPierre-Alain TORET 	/* Mic led handle */
18820befb6eSPierre-Alain TORET 	ACPI_HANDLE	mic_led_handle;
18920befb6eSPierre-Alain TORET 	int		mic_led_state;
19020befb6eSPierre-Alain TORET 
1915db2f26eSSascha Wildner 	int		wlan_bt_flags;
1925db2f26eSSascha Wildner 	int		thermal_updt_supported;
1935db2f26eSSascha Wildner 
1945db2f26eSSascha Wildner 	unsigned int	events_availmask;
1955db2f26eSSascha Wildner 	unsigned int	events_initialmask;
1965db2f26eSSascha Wildner 	int		events_mask_supported;
1975db2f26eSSascha Wildner 	int		events_enable;
1985db2f26eSSascha Wildner 
1995db2f26eSSascha Wildner 	/* sensors(9) related */
2005db2f26eSSascha Wildner 	struct ksensordev sensordev;
2015db2f26eSSascha Wildner 	struct ksensor sensors[THINKPAD_NUM_SENSORS];
2025db2f26eSSascha Wildner 
203a31756d0SMarkus Pfeiffer 	unsigned int	handler_events;
204a31756d0SMarkus Pfeiffer 
20520befb6eSPierre-Alain TORET 	struct sysctl_ctx_list	*sysctl_ctx;
2065db2f26eSSascha Wildner 	struct sysctl_oid	*sysctl_tree;
2075db2f26eSSascha Wildner };
2085db2f26eSSascha Wildner 
2095db2f26eSSascha Wildner static struct {
2105db2f26eSSascha Wildner 	char	*name;
2115db2f26eSSascha Wildner 	int	method;
2125db2f26eSSascha Wildner 	char	*description;
21320befb6eSPierre-Alain TORET 	int	flag_rdonly;
2145db2f26eSSascha Wildner } acpi_thinkpad_sysctls[] = {
2155db2f26eSSascha Wildner 	{
2165db2f26eSSascha Wildner 		.name		= "events",
2175db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_EVENTS,
2185db2f26eSSascha Wildner 		.description	= "ACPI events enable",
2195db2f26eSSascha Wildner 	},
2205db2f26eSSascha Wildner 	{
2215db2f26eSSascha Wildner 		.name		= "eventmask",
2225db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_EVENTMASK,
2235db2f26eSSascha Wildner 		.description	= "ACPI eventmask",
2245db2f26eSSascha Wildner 	},
2255db2f26eSSascha Wildner 	{
2265db2f26eSSascha Wildner 		.name		= "hotkey",
2275db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_HOTKEY,
2285db2f26eSSascha Wildner 		.description	= "Key Status",
22920befb6eSPierre-Alain TORET 		.flag_rdonly	= 1
2305db2f26eSSascha Wildner 	},
2315db2f26eSSascha Wildner 	{
2325db2f26eSSascha Wildner 		.name		= "lcd_brightness",
2335db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_BRIGHTNESS,
2345db2f26eSSascha Wildner 		.description	= "LCD Brightness",
2355db2f26eSSascha Wildner 	},
2365db2f26eSSascha Wildner 	{
2375db2f26eSSascha Wildner 		.name		= "volume",
2385db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_VOLUME,
2395db2f26eSSascha Wildner 		.description	= "Volume",
2405db2f26eSSascha Wildner 	},
2415db2f26eSSascha Wildner 	{
2425db2f26eSSascha Wildner 		.name		= "mute",
2435db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_MUTE,
2445db2f26eSSascha Wildner 		.description	= "Mute",
2455db2f26eSSascha Wildner 	},
2465db2f26eSSascha Wildner 	{
2475db2f26eSSascha Wildner 		.name		= "thinklight",
2485db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_THINKLIGHT,
2495db2f26eSSascha Wildner 		.description	= "Thinklight enable",
2505db2f26eSSascha Wildner 	},
2515db2f26eSSascha Wildner 	{
2525db2f26eSSascha Wildner 		.name		= "bluetooth",
2535db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_BLUETOOTH,
2545db2f26eSSascha Wildner 		.description	= "Bluetooth enable",
2555db2f26eSSascha Wildner 	},
2565db2f26eSSascha Wildner 	{
2575db2f26eSSascha Wildner 		.name		= "wlan",
2585db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_WLAN,
2595db2f26eSSascha Wildner 		.description	= "WLAN enable",
26020befb6eSPierre-Alain TORET 		.flag_rdonly	= 1
2615db2f26eSSascha Wildner 	},
2625db2f26eSSascha Wildner 	{
263a31756d0SMarkus Pfeiffer 		.name		= "fan_speed",
264a31756d0SMarkus Pfeiffer 		.method		= ACPI_THINKPAD_METHOD_FANSPEED,
265a31756d0SMarkus Pfeiffer 		.description	= "Fan speed",
26620befb6eSPierre-Alain TORET 		.flag_rdonly	= 1
267a31756d0SMarkus Pfeiffer 	},
268a31756d0SMarkus Pfeiffer 	{
2695db2f26eSSascha Wildner 		.name		= "fan_level",
2705db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_FANLEVEL,
2715db2f26eSSascha Wildner 		.description	= "Fan level",
2725db2f26eSSascha Wildner 	},
2735db2f26eSSascha Wildner 	{
2745db2f26eSSascha Wildner 		.name		= "fan",
2755db2f26eSSascha Wildner 		.method		= ACPI_THINKPAD_METHOD_FANSTATUS,
2765db2f26eSSascha Wildner 		.description	= "Fan enable",
2775db2f26eSSascha Wildner 	},
27820befb6eSPierre-Alain TORET 	{
27920befb6eSPierre-Alain TORET 		.name		= "mic_led",
28020befb6eSPierre-Alain TORET 		.method		= ACPI_THINKPAD_METHOD_MIC_LED,
28120befb6eSPierre-Alain TORET 		.description	= "Mic led",
28220befb6eSPierre-Alain TORET 	},
28320befb6eSPierre-Alain TORET 	{
28420befb6eSPierre-Alain TORET 		.name		= "privacyguard",
28520befb6eSPierre-Alain TORET 		.method		= ACPI_THINKPAD_METHOD_PRIVACYGUARD,
28620befb6eSPierre-Alain TORET 		.description	= "PrivacyGuard enable",
28720befb6eSPierre-Alain TORET 	},
2885db2f26eSSascha Wildner 	{ NULL, 0, NULL, 0 }
2895db2f26eSSascha Wildner };
2905db2f26eSSascha Wildner 
29120befb6eSPierre-Alain TORET /*
29220befb6eSPierre-Alain TORET  * Per-model default list of event mask.
29320befb6eSPierre-Alain TORET  */
294*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_RFKILL_MASK		(1U << 4)
295*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_DSWITCH_MASK		(1U << 6)
296*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_BRIGHTNESS_UP_MASK	(1U << 15)
297*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_BRIGHTNESS_DOWN_MASK	(1U << 16)
298*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_SEARCH_MASK		(1U << 18)
299*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_MICMUTE_MASK		(1U << 26)
300*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_SETTINGS_MASK	(1U << 28)
301*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_VIEWOPEN_MASK	(1U << 30)
302*c91db742SSascha Wildner #define	ACPI_THINKPAD_HKEY_VIEWALL_MASK		(1U << 31)
30320befb6eSPierre-Alain TORET 
30420befb6eSPierre-Alain TORET struct acpi_thinkpad_models {
30520befb6eSPierre-Alain TORET 	const char *maker;
30620befb6eSPierre-Alain TORET 	const char *product;
30720befb6eSPierre-Alain TORET 	uint32_t eventmask;
30820befb6eSPierre-Alain TORET } acpi_thinkpad_models[] = {
30920befb6eSPierre-Alain TORET 	{ "LENOVO", "20BSCTO1WW",
31020befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_RFKILL_MASK |
31120befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_DSWITCH_MASK |
31220befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_BRIGHTNESS_UP_MASK |
31320befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_BRIGHTNESS_DOWN_MASK |
31420befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_SEARCH_MASK |
31520befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_MICMUTE_MASK |
31620befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_SETTINGS_MASK |
31720befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_VIEWOPEN_MASK |
31820befb6eSPierre-Alain TORET 	  ACPI_THINKPAD_HKEY_VIEWALL_MASK
31920befb6eSPierre-Alain TORET 	}
32020befb6eSPierre-Alain TORET };
32120befb6eSPierre-Alain TORET 
32220befb6eSPierre-Alain TORET ACPI_SERIAL_DECL(thinkpad, "ThinkPad ACPI Extras");
3235db2f26eSSascha Wildner 
3245db2f26eSSascha Wildner static int	acpi_thinkpad_probe(device_t dev);
3255db2f26eSSascha Wildner static int	acpi_thinkpad_attach(device_t dev);
3265db2f26eSSascha Wildner static int	acpi_thinkpad_detach(device_t dev);
327a31756d0SMarkus Pfeiffer static int	acpi_thinkpad_resume(device_t dev);
328a31756d0SMarkus Pfeiffer 
329a31756d0SMarkus Pfeiffer static void	thinkpad_led(void *softc, int onoff);
330a31756d0SMarkus Pfeiffer static void	thinkpad_led_task(struct acpi_thinkpad_softc *sc, int pending __unused);
3315db2f26eSSascha Wildner 
3325db2f26eSSascha Wildner static int	acpi_thinkpad_sysctl(SYSCTL_HANDLER_ARGS);
33320befb6eSPierre-Alain TORET static int	acpi_thinkpad_sysctl_init(struct acpi_thinkpad_softc *sc, int method);
33420befb6eSPierre-Alain TORET static int	acpi_thinkpad_sysctl_get(struct acpi_thinkpad_softc *sc, int method);
33520befb6eSPierre-Alain TORET static int	acpi_thinkpad_sysctl_set(struct acpi_thinkpad_softc *sc, int method, int val);
3365db2f26eSSascha Wildner 
33720befb6eSPierre-Alain TORET static int	acpi_thinkpad_eventmask_set(struct acpi_thinkpad_softc *sc, int val);
33820befb6eSPierre-Alain TORET static int	acpi_thinkpad_thermal_sysctl(SYSCTL_HANDLER_ARGS);
339a31756d0SMarkus Pfeiffer static int	acpi_thinkpad_handlerevents_sysctl(SYSCTL_HANDLER_ARGS);
34020befb6eSPierre-Alain TORET static void	acpi_thinkpad_notify(ACPI_HANDLE h, UINT32 notify, void *context);
3415db2f26eSSascha Wildner static void	acpi_thinkpad_refresh(void *);
3425db2f26eSSascha Wildner 
343a31756d0SMarkus Pfeiffer static int	acpi_thinkpad_brightness_set(struct acpi_thinkpad_softc *sc, int arg);
344a31756d0SMarkus Pfeiffer static int	acpi_thinkpad_bluetooth_set(struct acpi_thinkpad_softc *sc, int arg);
345a31756d0SMarkus Pfeiffer static int	acpi_thinkpad_thinklight_set(struct acpi_thinkpad_softc *sc, int arg);
346a31756d0SMarkus Pfeiffer static int	acpi_thinkpad_volume_set(struct acpi_thinkpad_softc *sc, int arg);
347a31756d0SMarkus Pfeiffer static int	acpi_thinkpad_mute_set(struct acpi_thinkpad_softc *sc, int arg);
34820befb6eSPierre-Alain TORET static int	acpi_thinkpad_privacyguard_get(struct acpi_thinkpad_softc *sc);
34920befb6eSPierre-Alain TORET static ACPI_STATUS	acpi_thinkpad_privacyguard_set(struct acpi_thinkpad_softc *sc, int arg);
35020befb6eSPierre-Alain TORET static ACPI_STATUS	acpi_thinkpad_privacyguard_acpi_call(struct acpi_thinkpad_softc *sc, bool write, int *arg);
35120befb6eSPierre-Alain TORET 
35220befb6eSPierre-Alain TORET static int	acpi_status_to_errno(ACPI_STATUS status);
353a31756d0SMarkus Pfeiffer 
3545db2f26eSSascha Wildner static device_method_t acpi_thinkpad_methods[] = {
3555db2f26eSSascha Wildner 	/* Device interface */
3565db2f26eSSascha Wildner 	DEVMETHOD(device_probe, acpi_thinkpad_probe),
3575db2f26eSSascha Wildner 	DEVMETHOD(device_attach, acpi_thinkpad_attach),
3585db2f26eSSascha Wildner 	DEVMETHOD(device_detach, acpi_thinkpad_detach),
359a31756d0SMarkus Pfeiffer 	DEVMETHOD(device_resume, acpi_thinkpad_resume),
360a31756d0SMarkus Pfeiffer 
361d3c9c58eSSascha Wildner 	DEVMETHOD_END
3625db2f26eSSascha Wildner };
3635db2f26eSSascha Wildner 
3645db2f26eSSascha Wildner static driver_t	acpi_thinkpad_driver = {
3655db2f26eSSascha Wildner 	"acpi_thinkpad",
3665db2f26eSSascha Wildner 	acpi_thinkpad_methods,
3675db2f26eSSascha Wildner 	sizeof(struct acpi_thinkpad_softc),
3685db2f26eSSascha Wildner };
3695db2f26eSSascha Wildner 
3705db2f26eSSascha Wildner static devclass_t acpi_thinkpad_devclass;
3715db2f26eSSascha Wildner 
37220befb6eSPierre-Alain TORET DRIVER_MODULE(acpi_thinkpad, acpi, acpi_thinkpad_driver, acpi_thinkpad_devclass, NULL, NULL);
3735db2f26eSSascha Wildner MODULE_DEPEND(acpi_thinkpad, acpi, 1, 1, 1);
374f2c400e8SImre Vadász MODULE_DEPEND(acpi_thinkpad, led, 1, 1, 1);
37520befb6eSPierre-Alain TORET static char    *thinkpad_ids[] = {"IBM0068", "LEN0068", "LEN0268", NULL};
37620befb6eSPierre-Alain TORET 
37720befb6eSPierre-Alain TORET static int
acpi_status_to_errno(ACPI_STATUS status)37820befb6eSPierre-Alain TORET acpi_status_to_errno(ACPI_STATUS status)
37920befb6eSPierre-Alain TORET {
38020befb6eSPierre-Alain TORET 	switch (status) {
38120befb6eSPierre-Alain TORET 	case AE_OK:
38220befb6eSPierre-Alain TORET 		return (0);
38320befb6eSPierre-Alain TORET 	case AE_BAD_PARAMETER:
38420befb6eSPierre-Alain TORET 		return (EINVAL);
38520befb6eSPierre-Alain TORET 	default:
38620befb6eSPierre-Alain TORET 		return (ENODEV);
38720befb6eSPierre-Alain TORET 	}
38820befb6eSPierre-Alain TORET }
3895db2f26eSSascha Wildner 
390a31756d0SMarkus Pfeiffer static void
thinkpad_led(void * softc,int onoff)391a31756d0SMarkus Pfeiffer thinkpad_led(void *softc, int onoff)
392a31756d0SMarkus Pfeiffer {
39320befb6eSPierre-Alain TORET 	struct acpi_thinkpad_softc *sc = softc;
394a31756d0SMarkus Pfeiffer 
395a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
396a31756d0SMarkus Pfeiffer 
397a31756d0SMarkus Pfeiffer 	if (sc->led_busy)
398a31756d0SMarkus Pfeiffer 		return;
399a31756d0SMarkus Pfeiffer 
400a31756d0SMarkus Pfeiffer 	sc->led_busy = 1;
401a31756d0SMarkus Pfeiffer 	sc->led_state = onoff;
402a31756d0SMarkus Pfeiffer 
403a31756d0SMarkus Pfeiffer 	AcpiOsExecute(OSL_NOTIFY_HANDLER, (void *)thinkpad_led_task, sc);
404a31756d0SMarkus Pfeiffer }
405a31756d0SMarkus Pfeiffer 
406a31756d0SMarkus Pfeiffer static void
thinkpad_led_task(struct acpi_thinkpad_softc * sc,int pending __unused)407a31756d0SMarkus Pfeiffer thinkpad_led_task(struct acpi_thinkpad_softc *sc, int pending __unused)
408a31756d0SMarkus Pfeiffer {
409a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
410a31756d0SMarkus Pfeiffer 
411a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_BEGIN(thinkpad);
412a31756d0SMarkus Pfeiffer 	acpi_thinkpad_sysctl_set(sc, ACPI_THINKPAD_METHOD_THINKLIGHT, sc->led_state);
413a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_END(thinkpad);
414a31756d0SMarkus Pfeiffer 
415a31756d0SMarkus Pfeiffer 	sc->led_busy = 0;
416a31756d0SMarkus Pfeiffer }
417a31756d0SMarkus Pfeiffer 
4185db2f26eSSascha Wildner static int
acpi_thinkpad_mic_led_set(struct acpi_thinkpad_softc * sc,int arg)41920befb6eSPierre-Alain TORET acpi_thinkpad_mic_led_set(struct acpi_thinkpad_softc *sc, int arg)
42020befb6eSPierre-Alain TORET {
42120befb6eSPierre-Alain TORET 	ACPI_OBJECT_LIST input;
42220befb6eSPierre-Alain TORET 	ACPI_OBJECT params[1];
42320befb6eSPierre-Alain TORET 	ACPI_STATUS status;
42420befb6eSPierre-Alain TORET 
42520befb6eSPierre-Alain TORET 	if (arg < 0 || arg > 1)
42620befb6eSPierre-Alain TORET 		return (EINVAL);
42720befb6eSPierre-Alain TORET 
42820befb6eSPierre-Alain TORET 	if (sc->mic_led_handle) {
42920befb6eSPierre-Alain TORET 		params[0].Type = ACPI_TYPE_INTEGER;
43020befb6eSPierre-Alain TORET 		params[0].Integer.Value = 0;
43120befb6eSPierre-Alain TORET 		/* mic led: 0 off, 2 on */
43220befb6eSPierre-Alain TORET 		if (arg == 1)
43320befb6eSPierre-Alain TORET 			params[0].Integer.Value = 2;
43420befb6eSPierre-Alain TORET 
43520befb6eSPierre-Alain TORET 		input.Pointer = params;
43620befb6eSPierre-Alain TORET 		input.Count = 1;
43720befb6eSPierre-Alain TORET 
43820befb6eSPierre-Alain TORET 		status = AcpiEvaluateObject(sc->handle, "MMTS", &input, NULL);
43920befb6eSPierre-Alain TORET 		if (ACPI_SUCCESS(status))
44020befb6eSPierre-Alain TORET 			sc->mic_led_state = arg;
44120befb6eSPierre-Alain TORET 		return (status);
44220befb6eSPierre-Alain TORET 	}
44320befb6eSPierre-Alain TORET 
44420befb6eSPierre-Alain TORET 	return (0);
44520befb6eSPierre-Alain TORET }
44620befb6eSPierre-Alain TORET 
44720befb6eSPierre-Alain TORET static int
acpi_thinkpad_probe(device_t dev)4485db2f26eSSascha Wildner acpi_thinkpad_probe(device_t dev)
4495db2f26eSSascha Wildner {
45020befb6eSPierre-Alain TORET 	if (acpi_disabled("thinkpad") || device_get_unit(dev) != 0 || ACPI_ID_PROBE(device_get_parent(dev), dev, thinkpad_ids) == NULL)
4515db2f26eSSascha Wildner 		return (ENXIO);
4525db2f26eSSascha Wildner 
45320befb6eSPierre-Alain TORET 	device_set_desc(dev, "ThinkPad ACPI Extras");
45420befb6eSPierre-Alain TORET 
4555db2f26eSSascha Wildner 	return (0);
4565db2f26eSSascha Wildner }
4575db2f26eSSascha Wildner 
4585db2f26eSSascha Wildner static int
acpi_thinkpad_attach(device_t dev)4595db2f26eSSascha Wildner acpi_thinkpad_attach(device_t dev)
4605db2f26eSSascha Wildner {
4615db2f26eSSascha Wildner 	int			i;
46220befb6eSPierre-Alain TORET 	int hkey;
46320befb6eSPierre-Alain TORET 	struct acpi_thinkpad_softc *sc;
46420befb6eSPierre-Alain TORET 	char *maker, *product;
46520befb6eSPierre-Alain TORET 	ACPI_OBJECT_LIST input;
46620befb6eSPierre-Alain TORET 	ACPI_OBJECT params[1];
46720befb6eSPierre-Alain TORET 	ACPI_OBJECT out_obj;
46820befb6eSPierre-Alain TORET 	ACPI_BUFFER result;
46920befb6eSPierre-Alain TORET 	devclass_t ec_devclass;
4705db2f26eSSascha Wildner 
4710742f40dSImre Vadasz 	ACPI_SERIAL_INIT(thinkpad);
4725db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t) __func__);
4735db2f26eSSascha Wildner 
4745db2f26eSSascha Wildner 	sc = device_get_softc(dev);
4755db2f26eSSascha Wildner 	sc->dev = dev;
4765db2f26eSSascha Wildner 	sc->handle = acpi_get_handle(dev);
4775db2f26eSSascha Wildner 
4785db2f26eSSascha Wildner 	/* Look for the first embedded controller */
4795db2f26eSSascha Wildner 	if (!(ec_devclass = devclass_find ("acpi_ec"))) {
4805db2f26eSSascha Wildner 		if (bootverbose)
4815db2f26eSSascha Wildner 			device_printf(dev, "Couldn't find acpi_ec devclass\n");
4825db2f26eSSascha Wildner 		return (EINVAL);
4835db2f26eSSascha Wildner 	}
4845db2f26eSSascha Wildner 	if (!(sc->ec_dev = devclass_get_device(ec_devclass, 0))) {
4855db2f26eSSascha Wildner 		if (bootverbose)
4865db2f26eSSascha Wildner 			device_printf(dev, "Couldn't find acpi_ec device\n");
4875db2f26eSSascha Wildner 		return (EINVAL);
4885db2f26eSSascha Wildner 	}
4895db2f26eSSascha Wildner 	sc->ec_handle = acpi_get_handle(sc->ec_dev);
4905db2f26eSSascha Wildner 
49120befb6eSPierre-Alain TORET 	/* Get the sysctl tree */
49220befb6eSPierre-Alain TORET 	sc->sysctl_ctx = device_get_sysctl_ctx(dev);
49320befb6eSPierre-Alain TORET 	sc->sysctl_tree = device_get_sysctl_tree(dev);
4945db2f26eSSascha Wildner 
4955db2f26eSSascha Wildner 	/* Look for event mask and hook up the nodes */
4965db2f26eSSascha Wildner 	sc->events_mask_supported = ACPI_SUCCESS(acpi_GetInteger(sc->handle,
4975db2f26eSSascha Wildner 	    THINKPAD_NAME_EVENTS_MASK_GET, &sc->events_initialmask));
4985db2f26eSSascha Wildner 
4995db2f26eSSascha Wildner 	if (sc->events_mask_supported) {
50020befb6eSPierre-Alain TORET 		SYSCTL_ADD_UINT(sc->sysctl_ctx,
50120befb6eSPierre-Alain TORET 		    SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO, "initialmask",
50220befb6eSPierre-Alain TORET 		    CTLFLAG_RD, &sc->events_initialmask, 0,
50320befb6eSPierre-Alain TORET 		    "Initial eventmask");
5045db2f26eSSascha Wildner 
50520befb6eSPierre-Alain TORET 		if (ACPI_SUCCESS (acpi_GetInteger(sc->handle, "MHKV", &hkey))) {
50620befb6eSPierre-Alain TORET 			device_printf(dev, "Firmware version is 0x%X\n", hkey);
50720befb6eSPierre-Alain TORET 			switch (hkey >> 8) {
50820befb6eSPierre-Alain TORET 			case 1:
5095db2f26eSSascha Wildner 				/* The availmask is the bitmask of supported events */
5105db2f26eSSascha Wildner 				if (ACPI_FAILURE(acpi_GetInteger(sc->handle,
5115db2f26eSSascha Wildner 				    THINKPAD_NAME_EVENTS_AVAILMASK, &sc->events_availmask)))
5125db2f26eSSascha Wildner 					sc->events_availmask = 0xffffffff;
51320befb6eSPierre-Alain TORET 				break;
5145db2f26eSSascha Wildner 
51520befb6eSPierre-Alain TORET 			case 2:
51620befb6eSPierre-Alain TORET 				result.Length = sizeof(out_obj);
51720befb6eSPierre-Alain TORET 				result.Pointer = &out_obj;
51820befb6eSPierre-Alain TORET 				params[0].Type = ACPI_TYPE_INTEGER;
51920befb6eSPierre-Alain TORET 				params[0].Integer.Value = 1;
52020befb6eSPierre-Alain TORET 				input.Pointer = params;
52120befb6eSPierre-Alain TORET 				input.Count = 1;
52220befb6eSPierre-Alain TORET 
52320befb6eSPierre-Alain TORET 				sc->events_availmask = 0xffffffff;
52420befb6eSPierre-Alain TORET 
52520befb6eSPierre-Alain TORET 				if (ACPI_SUCCESS(AcpiEvaluateObject (sc->handle,
52620befb6eSPierre-Alain TORET 				    THINKPAD_NAME_EVENTS_AVAILMASK, &input, &result)))
52720befb6eSPierre-Alain TORET 					sc->events_availmask = out_obj.Integer.Value;
52820befb6eSPierre-Alain TORET 				break;
52920befb6eSPierre-Alain TORET 			default:
53020befb6eSPierre-Alain TORET 				device_printf(dev, "Unknown firmware version 0x%x\n", hkey);
53120befb6eSPierre-Alain TORET 				break;
53220befb6eSPierre-Alain TORET 			}
53320befb6eSPierre-Alain TORET 		} else
53420befb6eSPierre-Alain TORET 			sc->events_availmask = 0xffffffff;
53520befb6eSPierre-Alain TORET 
53620befb6eSPierre-Alain TORET 		SYSCTL_ADD_UINT(sc->sysctl_ctx,
5375db2f26eSSascha Wildner 				SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO,
5385db2f26eSSascha Wildner 				"availmask", CTLFLAG_RD,
5395db2f26eSSascha Wildner 				&sc->events_availmask, 0, "Mask of supported events");
5405db2f26eSSascha Wildner 	}
5415db2f26eSSascha Wildner 
5425db2f26eSSascha Wildner 	/* Hook up proc nodes */
5435db2f26eSSascha Wildner 	for (i = 0; acpi_thinkpad_sysctls[i].name != NULL; i++) {
54420befb6eSPierre-Alain TORET 		if (!acpi_thinkpad_sysctl_init(sc, acpi_thinkpad_sysctls[i].method))
5455db2f26eSSascha Wildner 			continue;
5465db2f26eSSascha Wildner 
54720befb6eSPierre-Alain TORET 		if (acpi_thinkpad_sysctls[i].flag_rdonly != 0) {
54820befb6eSPierre-Alain TORET 			SYSCTL_ADD_PROC(sc->sysctl_ctx,
5495db2f26eSSascha Wildner 			    SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO,
5505db2f26eSSascha Wildner 			    acpi_thinkpad_sysctls[i].name,
551323237e6SDaniel Fojt 			    CTLTYPE_UINT | CTLFLAG_RD,
552323237e6SDaniel Fojt 			    sc, i, acpi_thinkpad_sysctl, "IU",
5535db2f26eSSascha Wildner 			    acpi_thinkpad_sysctls[i].description);
55420befb6eSPierre-Alain TORET 		} else {
55520befb6eSPierre-Alain TORET 			SYSCTL_ADD_PROC(sc->sysctl_ctx,
55620befb6eSPierre-Alain TORET 			    SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO,
55720befb6eSPierre-Alain TORET 			    acpi_thinkpad_sysctls[i].name,
558323237e6SDaniel Fojt 			    CTLTYPE_UINT | CTLFLAG_RW,
559323237e6SDaniel Fojt 			    sc, i, acpi_thinkpad_sysctl, "IU",
56020befb6eSPierre-Alain TORET 			    acpi_thinkpad_sysctls[i].description);
56120befb6eSPierre-Alain TORET 		}
56220befb6eSPierre-Alain TORET 	}
56320befb6eSPierre-Alain TORET 
56420befb6eSPierre-Alain TORET 	/* Hook up thermal node */
56520befb6eSPierre-Alain TORET 	if (acpi_thinkpad_sysctl_init(sc, ACPI_THINKPAD_METHOD_THERMAL)) {
56620befb6eSPierre-Alain TORET 		SYSCTL_ADD_PROC(sc->sysctl_ctx,
56720befb6eSPierre-Alain TORET 		    SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO, "thermal",
56820befb6eSPierre-Alain TORET 		    CTLTYPE_INT | CTLFLAG_RD, sc, 0,
56920befb6eSPierre-Alain TORET 		    acpi_thinkpad_thermal_sysctl, "I", "Thermal zones");
5705db2f26eSSascha Wildner 	}
5715db2f26eSSascha Wildner 
572a31756d0SMarkus Pfeiffer 	/* Hook up handlerevents node */
573a31756d0SMarkus Pfeiffer 	if (acpi_thinkpad_sysctl_init(sc, ACPI_THINKPAD_METHOD_HANDLEREVENTS)) {
57420befb6eSPierre-Alain TORET 		SYSCTL_ADD_PROC(sc->sysctl_ctx,
57520befb6eSPierre-Alain TORET 		    SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO, "handlerevents",
57620befb6eSPierre-Alain TORET 		    CTLTYPE_STRING | CTLFLAG_RW, sc, 0,
57720befb6eSPierre-Alain TORET 		    acpi_thinkpad_handlerevents_sysctl, "I",
578a31756d0SMarkus Pfeiffer 		    "devd(8) events handled by acpi_thinkpad");
579a31756d0SMarkus Pfeiffer 	}
5805db2f26eSSascha Wildner 
5815db2f26eSSascha Wildner 	/* Handle notifies */
5825db2f26eSSascha Wildner 	AcpiInstallNotifyHandler(sc->handle, ACPI_DEVICE_NOTIFY,
5835db2f26eSSascha Wildner 	    acpi_thinkpad_notify, dev);
5845db2f26eSSascha Wildner 
5855db2f26eSSascha Wildner 	/* Attach sensors(9). */
5861bedd63aSSepherosa Ziehau 	sensor_task_register(sc, acpi_thinkpad_refresh, 5);
5875db2f26eSSascha Wildner 
5885db2f26eSSascha Wildner 	strlcpy(sc->sensordev.xname, device_get_nameunit(sc->dev),
5895db2f26eSSascha Wildner 	    sizeof(sc->sensordev.xname));
5905db2f26eSSascha Wildner 
5915db2f26eSSascha Wildner 	for (i = 0; i < THINKPAD_TEMP_SENSORS; i++) {
5925db2f26eSSascha Wildner 		sc->sensors[i].type = SENSOR_TEMP;
5935db2f26eSSascha Wildner 		sensor_attach(&sc->sensordev, &sc->sensors[i]);
5945db2f26eSSascha Wildner 	}
5955db2f26eSSascha Wildner 
5965db2f26eSSascha Wildner 	sc->sensors[i].type = SENSOR_FANRPM;
5975db2f26eSSascha Wildner 	sensor_attach(&sc->sensordev, &sc->sensors[i]);
5985db2f26eSSascha Wildner 
5995db2f26eSSascha Wildner 	sensordev_install(&sc->sensordev);
6005db2f26eSSascha Wildner 
601a31756d0SMarkus Pfeiffer 	/* Hook up light to led(4) */
602a31756d0SMarkus Pfeiffer 	if (sc->light_set_supported)
60320befb6eSPierre-Alain TORET 		sc->led_dev = led_create_state(thinkpad_led, sc, "thinklight",
60420befb6eSPierre-Alain TORET 		    (sc->light_val ? 1 : 0));
60520befb6eSPierre-Alain TORET 
60620befb6eSPierre-Alain TORET 	/* Enable per-model events. */
60720befb6eSPierre-Alain TORET 	maker = kgetenv("smbios.system.maker");
60820befb6eSPierre-Alain TORET 	product = kgetenv("smbios.system.product");
60920befb6eSPierre-Alain TORET 	if (maker == NULL || product == NULL)
61020befb6eSPierre-Alain TORET 		goto nosmbios;
61120befb6eSPierre-Alain TORET 
61220befb6eSPierre-Alain TORET 	for (i = 0; i < nitems(acpi_thinkpad_models); i++) {
61320befb6eSPierre-Alain TORET 		if (strcmp(maker, acpi_thinkpad_models[i].maker) == 0 &&
61420befb6eSPierre-Alain TORET 		    strcmp(product, acpi_thinkpad_models[i].product) == 0) {
61520befb6eSPierre-Alain TORET 			ACPI_SERIAL_BEGIN(thinkpad);
61620befb6eSPierre-Alain TORET 			acpi_thinkpad_sysctl_set(sc, ACPI_THINKPAD_METHOD_EVENTMASK,
61720befb6eSPierre-Alain TORET 			    acpi_thinkpad_models[i].eventmask);
61820befb6eSPierre-Alain TORET 			ACPI_SERIAL_END(thinkpad);
61920befb6eSPierre-Alain TORET 		}
62020befb6eSPierre-Alain TORET 	}
62120befb6eSPierre-Alain TORET 
62220befb6eSPierre-Alain TORET nosmbios:
62320befb6eSPierre-Alain TORET 	kfreeenv(maker);
62420befb6eSPierre-Alain TORET 	kfreeenv(product);
62520befb6eSPierre-Alain TORET 
62620befb6eSPierre-Alain TORET 	/* Enable events by default. */
62720befb6eSPierre-Alain TORET 	ACPI_SERIAL_BEGIN(thinkpad);
62820befb6eSPierre-Alain TORET 	acpi_thinkpad_sysctl_set(sc, ACPI_THINKPAD_METHOD_EVENTS, 1);
62920befb6eSPierre-Alain TORET 	ACPI_SERIAL_END(thinkpad);
630a31756d0SMarkus Pfeiffer 
6315db2f26eSSascha Wildner 	return (0);
6325db2f26eSSascha Wildner }
6335db2f26eSSascha Wildner 
6345db2f26eSSascha Wildner static int
acpi_thinkpad_detach(device_t dev)6355db2f26eSSascha Wildner acpi_thinkpad_detach(device_t dev)
6365db2f26eSSascha Wildner {
6375db2f26eSSascha Wildner 	int i;
6385db2f26eSSascha Wildner 
6395db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t) __func__);
6405db2f26eSSascha Wildner 
6415db2f26eSSascha Wildner 	struct acpi_thinkpad_softc *sc = device_get_softc(dev);
6425db2f26eSSascha Wildner 
6435db2f26eSSascha Wildner 	/* Disable events and restore eventmask */
644a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_BEGIN(thinkpad);
6455db2f26eSSascha Wildner 	acpi_thinkpad_sysctl_set(sc, ACPI_THINKPAD_METHOD_EVENTS, 0);
64620befb6eSPierre-Alain TORET 	acpi_thinkpad_sysctl_set(sc, ACPI_THINKPAD_METHOD_EVENTMASK, sc->events_initialmask);
647a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_END(thinkpad);
6485db2f26eSSascha Wildner 
64920befb6eSPierre-Alain TORET 	AcpiRemoveNotifyHandler(sc->handle, ACPI_DEVICE_NOTIFY, acpi_thinkpad_notify);
6505db2f26eSSascha Wildner 
6515db2f26eSSascha Wildner 	if (sc->sysctl_tree != NULL)
65220befb6eSPierre-Alain TORET 		sysctl_ctx_free(sc->sysctl_ctx);
6535db2f26eSSascha Wildner 
6545db2f26eSSascha Wildner 	sensordev_deinstall(&sc->sensordev);
6555db2f26eSSascha Wildner 	for (i = 0; i < THINKPAD_NUM_SENSORS; i++)
6565db2f26eSSascha Wildner 		sensor_detach(&sc->sensordev, &sc->sensors[i]);
6575db2f26eSSascha Wildner 	sensor_task_unregister(sc);
6585db2f26eSSascha Wildner 
659a31756d0SMarkus Pfeiffer 	if (sc->led_dev != NULL)
660a31756d0SMarkus Pfeiffer 		led_destroy(sc->led_dev);
661a31756d0SMarkus Pfeiffer 
662a31756d0SMarkus Pfeiffer 	return (0);
663a31756d0SMarkus Pfeiffer }
664a31756d0SMarkus Pfeiffer 
665a31756d0SMarkus Pfeiffer static int
acpi_thinkpad_resume(device_t dev)666a31756d0SMarkus Pfeiffer acpi_thinkpad_resume(device_t dev)
667a31756d0SMarkus Pfeiffer {
668a31756d0SMarkus Pfeiffer 	struct acpi_thinkpad_softc *sc = device_get_softc(dev);
669a31756d0SMarkus Pfeiffer 
670a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t) __func__);
671a31756d0SMarkus Pfeiffer 
672a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_BEGIN(thinkpad);
673a31756d0SMarkus Pfeiffer 	for (int i = 0; acpi_thinkpad_sysctls[i].name != NULL; i++) {
674a31756d0SMarkus Pfeiffer 		int val;
675a31756d0SMarkus Pfeiffer 
676a31756d0SMarkus Pfeiffer 		val = acpi_thinkpad_sysctl_get(sc, i);
677a31756d0SMarkus Pfeiffer 
67820befb6eSPierre-Alain TORET 		if (acpi_thinkpad_sysctls[i].flag_rdonly != 0)
679a31756d0SMarkus Pfeiffer 			continue;
680a31756d0SMarkus Pfeiffer 
681a31756d0SMarkus Pfeiffer 		acpi_thinkpad_sysctl_set(sc, i, val);
682a31756d0SMarkus Pfeiffer 	}
683a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_END(thinkpad);
684a31756d0SMarkus Pfeiffer 
68520befb6eSPierre-Alain TORET 	/* The mic led does not turn back on when sysctl_set is called in the above loop */
68620befb6eSPierre-Alain TORET 	acpi_thinkpad_mic_led_set(sc, sc->mic_led_state);
68720befb6eSPierre-Alain TORET 
6885db2f26eSSascha Wildner 	return (0);
6895db2f26eSSascha Wildner }
6905db2f26eSSascha Wildner 
6915db2f26eSSascha Wildner static int
acpi_thinkpad_eventmask_set(struct acpi_thinkpad_softc * sc,int val)6925db2f26eSSascha Wildner acpi_thinkpad_eventmask_set(struct acpi_thinkpad_softc *sc, int val)
6935db2f26eSSascha Wildner {
6945db2f26eSSascha Wildner 	int i;
6955db2f26eSSascha Wildner 	ACPI_OBJECT		arg[2];
6965db2f26eSSascha Wildner 	ACPI_OBJECT_LIST	args;
6975db2f26eSSascha Wildner 	ACPI_STATUS		status;
6985db2f26eSSascha Wildner 
6995db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
700a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
7015db2f26eSSascha Wildner 
7025db2f26eSSascha Wildner 	args.Count = 2;
7035db2f26eSSascha Wildner 	args.Pointer = arg;
7045db2f26eSSascha Wildner 	arg[0].Type = ACPI_TYPE_INTEGER;
7055db2f26eSSascha Wildner 	arg[1].Type = ACPI_TYPE_INTEGER;
7065db2f26eSSascha Wildner 
7075db2f26eSSascha Wildner 	for (i = 0; i < 32; ++i) {
7085db2f26eSSascha Wildner 		arg[0].Integer.Value = i + 1;
709*c91db742SSascha Wildner 		arg[1].Integer.Value = (((1U << i) & val) != 0);
7105db2f26eSSascha Wildner 		status = AcpiEvaluateObject(sc->handle,
7115db2f26eSSascha Wildner 		    THINKPAD_NAME_EVENTS_MASK_SET, &args, NULL);
7125db2f26eSSascha Wildner 
7135db2f26eSSascha Wildner 		if (ACPI_FAILURE(status))
7145db2f26eSSascha Wildner 			return (status);
7155db2f26eSSascha Wildner 	}
7165db2f26eSSascha Wildner 
7175db2f26eSSascha Wildner 	return (0);
7185db2f26eSSascha Wildner }
7195db2f26eSSascha Wildner 
7205db2f26eSSascha Wildner static int
acpi_thinkpad_sysctl(SYSCTL_HANDLER_ARGS)7215db2f26eSSascha Wildner acpi_thinkpad_sysctl(SYSCTL_HANDLER_ARGS)
7225db2f26eSSascha Wildner {
7235db2f26eSSascha Wildner 	struct acpi_thinkpad_softc	*sc;
7245db2f26eSSascha Wildner 	int			arg;
7255db2f26eSSascha Wildner 	int			error = 0;
7265db2f26eSSascha Wildner 	int			function;
7275db2f26eSSascha Wildner 	int			method;
7285db2f26eSSascha Wildner 
7295db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
7305db2f26eSSascha Wildner 
7315db2f26eSSascha Wildner 	sc = (struct acpi_thinkpad_softc *)oidp->oid_arg1;
7325db2f26eSSascha Wildner 	function = oidp->oid_arg2;
7335db2f26eSSascha Wildner 	method = acpi_thinkpad_sysctls[function].method;
7345db2f26eSSascha Wildner 
735a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_BEGIN(thinkpad);
7365db2f26eSSascha Wildner 	arg = acpi_thinkpad_sysctl_get(sc, method);
7375db2f26eSSascha Wildner 	error = sysctl_handle_int(oidp, &arg, 0, req);
7385db2f26eSSascha Wildner 
7395db2f26eSSascha Wildner 	/* Sanity check */
7405db2f26eSSascha Wildner 	if (error != 0 || req->newptr == NULL)
7415db2f26eSSascha Wildner 		goto out;
7425db2f26eSSascha Wildner 
7435db2f26eSSascha Wildner 	/* Update */
7445db2f26eSSascha Wildner 	error = acpi_thinkpad_sysctl_set(sc, method, arg);
7455db2f26eSSascha Wildner 
7465db2f26eSSascha Wildner out:
747a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_END(thinkpad);
7485db2f26eSSascha Wildner 	return (error);
7495db2f26eSSascha Wildner }
7505db2f26eSSascha Wildner 
7515db2f26eSSascha Wildner static int
acpi_thinkpad_sysctl_get(struct acpi_thinkpad_softc * sc,int method)7525db2f26eSSascha Wildner acpi_thinkpad_sysctl_get(struct acpi_thinkpad_softc *sc, int method)
7535db2f26eSSascha Wildner {
754a31756d0SMarkus Pfeiffer 	UINT64		val_ec;
7555db2f26eSSascha Wildner 	int 		val = 0, key;
7565db2f26eSSascha Wildner 
7575db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
758a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
7595db2f26eSSascha Wildner 
7605db2f26eSSascha Wildner 	switch (method) {
7615db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_EVENTS:
76220befb6eSPierre-Alain TORET 		acpi_GetInteger(sc->handle, THINKPAD_NAME_EVENTS_STATUS_GET, &val);
7635db2f26eSSascha Wildner 		break;
7645db2f26eSSascha Wildner 
7655db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_EVENTMASK:
7665db2f26eSSascha Wildner 		if (sc->events_mask_supported)
76720befb6eSPierre-Alain TORET 			acpi_GetInteger(sc->handle, THINKPAD_NAME_EVENTS_MASK_GET, &val);
7685db2f26eSSascha Wildner 		break;
7695db2f26eSSascha Wildner 
7705db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_HOTKEY:
7715db2f26eSSascha Wildner 		/*
7725db2f26eSSascha Wildner 		 * Construct the hotkey as a bitmask as illustrated below.
7735db2f26eSSascha Wildner 		 * Note that whenever a key was pressed, the respecting bit
7745db2f26eSSascha Wildner 		 * toggles and nothing else changes.
7755db2f26eSSascha Wildner 		 * +--+--+-+-+-+-+-+-+-+-+-+-+
7765db2f26eSSascha Wildner 		 * |11|10|9|8|7|6|5|4|3|2|1|0|
7775db2f26eSSascha Wildner 		 * +--+--+-+-+-+-+-+-+-+-+-+-+
7785db2f26eSSascha Wildner 		 *   |  | | | | | | | | | | |
7795db2f26eSSascha Wildner 		 *   |  | | | | | | | | | | +- Home Button
7805db2f26eSSascha Wildner 		 *   |  | | | | | | | | | +--- Search Button
7815db2f26eSSascha Wildner 		 *   |  | | | | | | | | +----- Mail Button
7825db2f26eSSascha Wildner 		 *   |  | | | | | | | +------- Thinkpad Button
7835db2f26eSSascha Wildner 		 *   |  | | | | | | +--------- Zoom (Fn + Space)
7845db2f26eSSascha Wildner 		 *   |  | | | | | +----------- WLAN Button
7855db2f26eSSascha Wildner 		 *   |  | | | | +------------- Video Button
7865db2f26eSSascha Wildner 		 *   |  | | | +--------------- Hibernate Button
7875db2f26eSSascha Wildner 		 *   |  | | +----------------- Thinklight Button
7885db2f26eSSascha Wildner 		 *   |  | +------------------- Screen expand (Fn + F8)
7895db2f26eSSascha Wildner 		 *   |  +--------------------- Brightness
7905db2f26eSSascha Wildner 		 *   +------------------------ Volume/Mute
7915db2f26eSSascha Wildner 		 */
7925db2f26eSSascha Wildner 		key = rtcin(THINKPAD_RTC_HOTKEY1);
79320befb6eSPierre-Alain TORET 		val = (THINKPAD_RTC_MASK_HOME | THINKPAD_RTC_MASK_SEARCH | THINKPAD_RTC_MASK_MAIL | THINKPAD_RTC_MASK_WLAN) & key;
7945db2f26eSSascha Wildner 		key = rtcin(THINKPAD_RTC_HOTKEY2);
79520befb6eSPierre-Alain TORET 		val |= (THINKPAD_RTC_MASK_THINKPAD | THINKPAD_RTC_MASK_VIDEO | THINKPAD_RTC_MASK_HIBERNATE) & key;
7965db2f26eSSascha Wildner 		val |= (THINKPAD_RTC_MASK_ZOOM & key) >> 1;
7975db2f26eSSascha Wildner 		key = rtcin(THINKPAD_RTC_THINKLIGHT);
7985db2f26eSSascha Wildner 		val |= (THINKPAD_RTC_MASK_THINKLIGHT & key) << 4;
7995db2f26eSSascha Wildner 		key = rtcin(THINKPAD_RTC_SCREENEXPAND);
8005db2f26eSSascha Wildner 		val |= (THINKPAD_RTC_MASK_THINKLIGHT & key) << 4;
8015db2f26eSSascha Wildner 		key = rtcin(THINKPAD_RTC_BRIGHTNESS);
8025db2f26eSSascha Wildner 		val |= (THINKPAD_RTC_MASK_BRIGHTNESS & key) << 5;
8035db2f26eSSascha Wildner 		key = rtcin(THINKPAD_RTC_VOLUME);
8045db2f26eSSascha Wildner 		val |= (THINKPAD_RTC_MASK_VOLUME & key) << 4;
8055db2f26eSSascha Wildner 		break;
8065db2f26eSSascha Wildner 
8075db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_BRIGHTNESS:
8085db2f26eSSascha Wildner 		ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_BRIGHTNESS, &val_ec, 1);
8095db2f26eSSascha Wildner 		val = val_ec & THINKPAD_EC_MASK_BRI;
8105db2f26eSSascha Wildner 		break;
8115db2f26eSSascha Wildner 
8125db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_VOLUME:
8135db2f26eSSascha Wildner 		ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_VOLUME, &val_ec, 1);
8145db2f26eSSascha Wildner 		val = val_ec & THINKPAD_EC_MASK_VOL;
8155db2f26eSSascha Wildner 		break;
8165db2f26eSSascha Wildner 
8175db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_MUTE:
8185db2f26eSSascha Wildner 		ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_VOLUME, &val_ec, 1);
81920befb6eSPierre-Alain TORET 		val = ((val_ec & THINKPAD_EC_MASK_MUTE) == THINKPAD_EC_MASK_MUTE);
8205db2f26eSSascha Wildner 		break;
8215db2f26eSSascha Wildner 
8225db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_THINKLIGHT:
8235db2f26eSSascha Wildner 		if (sc->light_get_supported)
82420befb6eSPierre-Alain TORET 			acpi_GetInteger(sc->ec_handle, THINKPAD_NAME_KEYLIGHT, &val);
8255db2f26eSSascha Wildner 		else
8265db2f26eSSascha Wildner 			val = sc->light_val;
8275db2f26eSSascha Wildner 		break;
8285db2f26eSSascha Wildner 
8295db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_BLUETOOTH:
8305db2f26eSSascha Wildner 		acpi_GetInteger(sc->handle, THINKPAD_NAME_WLAN_BT_GET, &val);
8315db2f26eSSascha Wildner 		sc->wlan_bt_flags = val;
8325db2f26eSSascha Wildner 		val = ((val & THINKPAD_NAME_MASK_BT) != 0);
8335db2f26eSSascha Wildner 		break;
8345db2f26eSSascha Wildner 
8355db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_WLAN:
8365db2f26eSSascha Wildner 		acpi_GetInteger(sc->handle, THINKPAD_NAME_WLAN_BT_GET, &val);
8375db2f26eSSascha Wildner 		sc->wlan_bt_flags = val;
8385db2f26eSSascha Wildner 		val = ((val & THINKPAD_NAME_MASK_WLAN) != 0);
8395db2f26eSSascha Wildner 		break;
8405db2f26eSSascha Wildner 
8415db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANSPEED:
8425db2f26eSSascha Wildner 		if (sc->fan_handle) {
84320befb6eSPierre-Alain TORET 			if (ACPI_FAILURE(acpi_GetInteger(sc->fan_handle, NULL, &val)))
8445db2f26eSSascha Wildner 				val = -1;
84520befb6eSPierre-Alain TORET 		} else {
84620befb6eSPierre-Alain TORET 			ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_FANSPEED, &val_ec, 2);
8475db2f26eSSascha Wildner 			val = val_ec;
8485db2f26eSSascha Wildner 		}
8495db2f26eSSascha Wildner 		break;
8505db2f26eSSascha Wildner 
8515db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANLEVEL:
8525db2f26eSSascha Wildner 		/*
8535db2f26eSSascha Wildner 		 * The THINKPAD_EC_FANSTATUS register works as follows:
8545db2f26eSSascha Wildner 		 * Bit 0-5 indicate the level at which the fan operates. Only
8555db2f26eSSascha Wildner 		 *       values between 0 and 7 have an effect. Everything
8565db2f26eSSascha Wildner 		 *       above 7 is treated the same as level 7
8575db2f26eSSascha Wildner 		 * Bit 6 overrides the fan speed limit if set to 1
8585db2f26eSSascha Wildner 		 * Bit 7 indicates at which mode the fan operates:
8595db2f26eSSascha Wildner 		 *       manual (0) or automatic (1)
8605db2f26eSSascha Wildner 		 */
8615db2f26eSSascha Wildner 		if (!sc->fan_handle) {
86220befb6eSPierre-Alain TORET 			ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_FANSTATUS, &val_ec, 1);
8635db2f26eSSascha Wildner 			val = val_ec & THINKPAD_EC_MASK_FANLEVEL;
8645db2f26eSSascha Wildner 		}
8655db2f26eSSascha Wildner 		break;
8665db2f26eSSascha Wildner 
8675db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANSTATUS:
8685db2f26eSSascha Wildner 		if (!sc->fan_handle) {
86920befb6eSPierre-Alain TORET 			ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_FANSTATUS, &val_ec, 1);
87020befb6eSPierre-Alain TORET 			val = (val_ec & THINKPAD_EC_MASK_FANSTATUS) == THINKPAD_EC_MASK_FANSTATUS;
87120befb6eSPierre-Alain TORET 		} else
87220befb6eSPierre-Alain TORET 			val = -1;
87320befb6eSPierre-Alain TORET 		break;
87420befb6eSPierre-Alain TORET 
87520befb6eSPierre-Alain TORET 	case ACPI_THINKPAD_METHOD_MIC_LED:
87620befb6eSPierre-Alain TORET 		if (sc->mic_led_handle)
87720befb6eSPierre-Alain TORET 			return sc->mic_led_state;
8785db2f26eSSascha Wildner 		else
8795db2f26eSSascha Wildner 			val = -1;
8805db2f26eSSascha Wildner 		break;
88120befb6eSPierre-Alain TORET 
88220befb6eSPierre-Alain TORET 	case ACPI_THINKPAD_METHOD_PRIVACYGUARD:
88320befb6eSPierre-Alain TORET 		val = acpi_thinkpad_privacyguard_get(sc);
88420befb6eSPierre-Alain TORET 		break;
8855db2f26eSSascha Wildner 	}
8865db2f26eSSascha Wildner 
8875db2f26eSSascha Wildner 	return (val);
8885db2f26eSSascha Wildner }
8895db2f26eSSascha Wildner 
8905db2f26eSSascha Wildner static int
acpi_thinkpad_sysctl_set(struct acpi_thinkpad_softc * sc,int method,int arg)8915db2f26eSSascha Wildner acpi_thinkpad_sysctl_set(struct acpi_thinkpad_softc *sc, int method, int arg)
8925db2f26eSSascha Wildner {
893a31756d0SMarkus Pfeiffer 	int			val;
894a31756d0SMarkus Pfeiffer 	UINT64			val_ec;
8955db2f26eSSascha Wildner 	ACPI_STATUS		status;
8965db2f26eSSascha Wildner 
8975db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
898a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
8995db2f26eSSascha Wildner 
9005db2f26eSSascha Wildner 	switch (method) {
9015db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_EVENTS:
9025db2f26eSSascha Wildner 		if (arg < 0 || arg > 1)
9035db2f26eSSascha Wildner 			return (EINVAL);
9045db2f26eSSascha Wildner 
90520befb6eSPierre-Alain TORET 		status = acpi_SetInteger(sc->handle, THINKPAD_NAME_EVENTS_STATUS_SET, arg);
9065db2f26eSSascha Wildner 		if (ACPI_FAILURE(status))
9075db2f26eSSascha Wildner 			return (status);
9085db2f26eSSascha Wildner 		if (sc->events_mask_supported)
90920befb6eSPierre-Alain TORET 			return acpi_thinkpad_eventmask_set(sc, sc->events_availmask);
9105db2f26eSSascha Wildner 		break;
9115db2f26eSSascha Wildner 
9125db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_EVENTMASK:
9135db2f26eSSascha Wildner 		if (sc->events_mask_supported)
9145db2f26eSSascha Wildner 			return acpi_thinkpad_eventmask_set(sc, arg);
9155db2f26eSSascha Wildner 		break;
9165db2f26eSSascha Wildner 
9175db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_BRIGHTNESS:
918a31756d0SMarkus Pfeiffer 		return acpi_thinkpad_brightness_set(sc, arg);
9195db2f26eSSascha Wildner 		break;
9205db2f26eSSascha Wildner 
9215db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_VOLUME:
922a31756d0SMarkus Pfeiffer 		return acpi_thinkpad_volume_set(sc, arg);
9235db2f26eSSascha Wildner 		break;
9245db2f26eSSascha Wildner 
9255db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_MUTE:
926a31756d0SMarkus Pfeiffer 		return acpi_thinkpad_mute_set(sc, arg);
9275db2f26eSSascha Wildner 		break;
9285db2f26eSSascha Wildner 
92920befb6eSPierre-Alain TORET 	case ACPI_THINKPAD_METHOD_MIC_LED:
93020befb6eSPierre-Alain TORET 		return acpi_thinkpad_mic_led_set(sc, arg);
93120befb6eSPierre-Alain TORET 		break;
93220befb6eSPierre-Alain TORET 
9335db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_THINKLIGHT:
934a31756d0SMarkus Pfeiffer 		return acpi_thinkpad_thinklight_set(sc, arg);
9355db2f26eSSascha Wildner 		break;
9365db2f26eSSascha Wildner 
9375db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_BLUETOOTH:
938a31756d0SMarkus Pfeiffer 		return acpi_thinkpad_bluetooth_set(sc, arg);
9395db2f26eSSascha Wildner 		break;
9405db2f26eSSascha Wildner 
94120befb6eSPierre-Alain TORET 	case ACPI_THINKPAD_METHOD_PRIVACYGUARD:
94220befb6eSPierre-Alain TORET 		return (acpi_status_to_errno(acpi_thinkpad_privacyguard_set(sc, arg)));
94320befb6eSPierre-Alain TORET 		break;
94420befb6eSPierre-Alain TORET 
9455db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANLEVEL:
9465db2f26eSSascha Wildner 		if (arg < 0 || arg > 7)
9475db2f26eSSascha Wildner 			return (EINVAL);
9485db2f26eSSascha Wildner 
9495db2f26eSSascha Wildner 		if (!sc->fan_handle) {
9505db2f26eSSascha Wildner 			/* Read the current fanstatus */
95120befb6eSPierre-Alain TORET 			ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_FANSTATUS, &val_ec, 1);
9525db2f26eSSascha Wildner 			val = val_ec & (~THINKPAD_EC_MASK_FANLEVEL);
9535db2f26eSSascha Wildner 
95420befb6eSPierre-Alain TORET 			return ACPI_EC_WRITE(sc->ec_dev, THINKPAD_EC_FANSTATUS, val | arg, 1);
9555db2f26eSSascha Wildner 		}
9565db2f26eSSascha Wildner 		break;
9575db2f26eSSascha Wildner 
9585db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANSTATUS:
9595db2f26eSSascha Wildner 		if (arg < 0 || arg > 1)
9605db2f26eSSascha Wildner 			return (EINVAL);
9615db2f26eSSascha Wildner 
9625db2f26eSSascha Wildner 		if (!sc->fan_handle) {
9635db2f26eSSascha Wildner 			/* Read the current fanstatus */
96420befb6eSPierre-Alain TORET 			ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_FANSTATUS, &val_ec, 1);
9655db2f26eSSascha Wildner 
9665db2f26eSSascha Wildner 			return ACPI_EC_WRITE(sc->ec_dev, THINKPAD_EC_FANSTATUS,
96720befb6eSPierre-Alain TORET 			    (arg == 1) ? (val_ec | THINKPAD_EC_MASK_FANSTATUS) : (val_ec & (~THINKPAD_EC_MASK_FANSTATUS)), 1);
9685db2f26eSSascha Wildner 		}
9695db2f26eSSascha Wildner 		break;
9705db2f26eSSascha Wildner 	}
9715db2f26eSSascha Wildner 
9725db2f26eSSascha Wildner 	return (0);
9735db2f26eSSascha Wildner }
9745db2f26eSSascha Wildner 
9755db2f26eSSascha Wildner static int
acpi_thinkpad_sysctl_init(struct acpi_thinkpad_softc * sc,int method)9765db2f26eSSascha Wildner acpi_thinkpad_sysctl_init(struct acpi_thinkpad_softc *sc, int method)
9775db2f26eSSascha Wildner {
9785db2f26eSSascha Wildner 	int 			dummy;
9795db2f26eSSascha Wildner 	ACPI_OBJECT_TYPE 	cmos_t;
9805db2f26eSSascha Wildner 	ACPI_HANDLE		ledb_handle;
9815db2f26eSSascha Wildner 
9825db2f26eSSascha Wildner 	switch (method) {
9835db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_EVENTS:
9845db2f26eSSascha Wildner 		return (TRUE);
9855db2f26eSSascha Wildner 
9865db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_EVENTMASK:
9875db2f26eSSascha Wildner 		return (sc->events_mask_supported);
9885db2f26eSSascha Wildner 
9895db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_HOTKEY:
9905db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_BRIGHTNESS:
9915db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_VOLUME:
9925db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_MUTE:
9935db2f26eSSascha Wildner 		/* EC is required here, which was already checked before */
9945db2f26eSSascha Wildner 		return (TRUE);
9955db2f26eSSascha Wildner 
99620befb6eSPierre-Alain TORET 	case ACPI_THINKPAD_METHOD_MIC_LED:
99720befb6eSPierre-Alain TORET 		if (ACPI_SUCCESS(AcpiGetHandle(sc->handle, "MMTS", &sc->mic_led_handle)))
99820befb6eSPierre-Alain TORET 		{
99920befb6eSPierre-Alain TORET 			/* Turn off mic led by default */
100020befb6eSPierre-Alain TORET 			acpi_thinkpad_mic_led_set(sc, 0);
100120befb6eSPierre-Alain TORET 			return (TRUE);
100220befb6eSPierre-Alain TORET 		} else
100320befb6eSPierre-Alain TORET 			sc->mic_led_handle = NULL;
100420befb6eSPierre-Alain TORET 		return (FALSE);
100520befb6eSPierre-Alain TORET 
10065db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_THINKLIGHT:
10075db2f26eSSascha Wildner 		sc->cmos_handle = NULL;
100820befb6eSPierre-Alain TORET 		sc->light_get_supported = ACPI_SUCCESS(acpi_GetInteger(sc->ec_handle, THINKPAD_NAME_KEYLIGHT, &sc->light_val));
10095db2f26eSSascha Wildner 
101020befb6eSPierre-Alain TORET 		if ((ACPI_SUCCESS(AcpiGetHandle(sc->handle, "\\UCMS", &sc->light_handle)) ||
101120befb6eSPierre-Alain TORET 		     ACPI_SUCCESS(AcpiGetHandle(sc->handle, "\\CMOS", &sc->light_handle)) ||
101220befb6eSPierre-Alain TORET 		     ACPI_SUCCESS(AcpiGetHandle(sc->handle, "\\CMS", &sc->light_handle))) &&
10135db2f26eSSascha Wildner 		     ACPI_SUCCESS(AcpiGetType(sc->light_handle, &cmos_t)) &&
10145db2f26eSSascha Wildner 		     cmos_t == ACPI_TYPE_METHOD) {
10155db2f26eSSascha Wildner 			sc->light_cmd_on = 0x0c;
10165db2f26eSSascha Wildner 			sc->light_cmd_off = 0x0d;
10175db2f26eSSascha Wildner 			sc->cmos_handle = sc->light_handle;
10185db2f26eSSascha Wildner 		}
101920befb6eSPierre-Alain TORET 		else if (ACPI_SUCCESS(AcpiGetHandle(sc->handle, "\\LGHT", &sc->light_handle))) {
10205db2f26eSSascha Wildner 			sc->light_cmd_on = 1;
10215db2f26eSSascha Wildner 			sc->light_cmd_off = 0;
102220befb6eSPierre-Alain TORET 		} else
10235db2f26eSSascha Wildner 			sc->light_handle = NULL;
10245db2f26eSSascha Wildner 
10255db2f26eSSascha Wildner 		sc->light_set_supported = (sc->light_handle &&
10265db2f26eSSascha Wildner 		    ACPI_FAILURE(AcpiGetHandle(sc->ec_handle, "LEDB",
10275db2f26eSSascha Wildner 		    &ledb_handle)));
10285db2f26eSSascha Wildner 
10295db2f26eSSascha Wildner 		if (sc->light_get_supported)
10305db2f26eSSascha Wildner 			return (TRUE);
10315db2f26eSSascha Wildner 
10325db2f26eSSascha Wildner 		if (sc->light_set_supported) {
10335db2f26eSSascha Wildner 			sc->light_val = 0;
10345db2f26eSSascha Wildner 			return (TRUE);
10355db2f26eSSascha Wildner 		}
10365db2f26eSSascha Wildner 
10375db2f26eSSascha Wildner 		return (FALSE);
10385db2f26eSSascha Wildner 
10395db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_BLUETOOTH:
10405db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_WLAN:
104120befb6eSPierre-Alain TORET 		if (ACPI_SUCCESS(acpi_GetInteger(sc->handle, THINKPAD_NAME_WLAN_BT_GET, &dummy)))
10425db2f26eSSascha Wildner 			return (TRUE);
10435db2f26eSSascha Wildner 		return (FALSE);
10445db2f26eSSascha Wildner 
10455db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANSPEED:
10465db2f26eSSascha Wildner 		/*
10475db2f26eSSascha Wildner 		 * Some models report the fan speed in levels from 0-7
10485db2f26eSSascha Wildner 		 * Newer models report it contiguously
10495db2f26eSSascha Wildner 		 */
105020befb6eSPierre-Alain TORET 		sc->fan_levels =
105120befb6eSPierre-Alain TORET 		    (ACPI_SUCCESS(AcpiGetHandle(sc->handle, "GFAN", &sc->fan_handle)) ||
105220befb6eSPierre-Alain TORET 		     ACPI_SUCCESS(AcpiGetHandle(sc->handle, "\\FSPD", &sc->fan_handle)));
10535db2f26eSSascha Wildner 		return (TRUE);
10545db2f26eSSascha Wildner 
10555db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANLEVEL:
10565db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_FANSTATUS:
10575db2f26eSSascha Wildner 		/*
10585db2f26eSSascha Wildner 		 * Fan status is only supported on those models,
10595db2f26eSSascha Wildner 		 * which report fan RPM contiguously, not in levels
10605db2f26eSSascha Wildner 		 */
10615db2f26eSSascha Wildner 		if (sc->fan_levels)
10625db2f26eSSascha Wildner 			return (FALSE);
10635db2f26eSSascha Wildner 		return (TRUE);
10645db2f26eSSascha Wildner 
10655db2f26eSSascha Wildner 	case ACPI_THINKPAD_METHOD_THERMAL:
106620befb6eSPierre-Alain TORET 		if (ACPI_SUCCESS(acpi_GetInteger(sc->ec_handle, THINKPAD_NAME_THERMAL_GET, &dummy))) {
106720befb6eSPierre-Alain TORET 			sc->thermal_updt_supported = ACPI_SUCCESS(acpi_GetInteger(sc->ec_handle, THINKPAD_NAME_THERMAL_UPDT, &dummy));
10685db2f26eSSascha Wildner 			return (TRUE);
10695db2f26eSSascha Wildner 		}
10705db2f26eSSascha Wildner 		return (FALSE);
1071a31756d0SMarkus Pfeiffer 
1072a31756d0SMarkus Pfeiffer 	case ACPI_THINKPAD_METHOD_HANDLEREVENTS:
1073a31756d0SMarkus Pfeiffer 		return (TRUE);
107420befb6eSPierre-Alain TORET 
107520befb6eSPierre-Alain TORET 	case ACPI_THINKPAD_METHOD_PRIVACYGUARD:
107620befb6eSPierre-Alain TORET 		return (acpi_thinkpad_privacyguard_get(sc) != -1);
10775db2f26eSSascha Wildner 	}
10785db2f26eSSascha Wildner 	return (FALSE);
10795db2f26eSSascha Wildner }
10805db2f26eSSascha Wildner 
1081a31756d0SMarkus Pfeiffer static int
acpi_thinkpad_thermal_sysctl(SYSCTL_HANDLER_ARGS)108220befb6eSPierre-Alain TORET acpi_thinkpad_thermal_sysctl(SYSCTL_HANDLER_ARGS)
108320befb6eSPierre-Alain TORET {
108420befb6eSPierre-Alain TORET         struct acpi_thinkpad_softc   *sc;
108520befb6eSPierre-Alain TORET         int                     error = 0;
108620befb6eSPierre-Alain TORET         char                    temp_cmd[] = "TMP0";
108720befb6eSPierre-Alain TORET         int                     temp[8];
108820befb6eSPierre-Alain TORET 
108920befb6eSPierre-Alain TORET         ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
109020befb6eSPierre-Alain TORET 
109120befb6eSPierre-Alain TORET         sc = (struct acpi_thinkpad_softc *)oidp->oid_arg1;
109220befb6eSPierre-Alain TORET 
109320befb6eSPierre-Alain TORET         ACPI_SERIAL_BEGIN(thinkpad);
109420befb6eSPierre-Alain TORET 
109520befb6eSPierre-Alain TORET         for (int i = 0; i < 8; ++i) {
109620befb6eSPierre-Alain TORET                 temp_cmd[3] = '0' + i;
109720befb6eSPierre-Alain TORET 
109820befb6eSPierre-Alain TORET                 /*
109920befb6eSPierre-Alain TORET                  * The TMPx methods seem to return +/- 128 or 0
110020befb6eSPierre-Alain TORET                  * when the respecting sensor is not available
110120befb6eSPierre-Alain TORET                  */
110220befb6eSPierre-Alain TORET                 if (ACPI_FAILURE(acpi_GetInteger(sc->ec_handle, temp_cmd,
110320befb6eSPierre-Alain TORET                     &temp[i])) || ABS(temp[i]) == 128 || temp[i] == 0)
110420befb6eSPierre-Alain TORET                         temp[i] = -1;
110520befb6eSPierre-Alain TORET                 else if (sc->thermal_updt_supported)
110620befb6eSPierre-Alain TORET                         /* Temperature is reported in tenth of Kelvin */
110720befb6eSPierre-Alain TORET                         temp[i] = (temp[i] - 2731 + 5) / 10;
110820befb6eSPierre-Alain TORET         }
110920befb6eSPierre-Alain TORET 
111020befb6eSPierre-Alain TORET         error = sysctl_handle_opaque(oidp, &temp, 8*sizeof(int), req);
111120befb6eSPierre-Alain TORET 
111220befb6eSPierre-Alain TORET         ACPI_SERIAL_END(thinkpad);
111320befb6eSPierre-Alain TORET         return (error);
111420befb6eSPierre-Alain TORET }
111520befb6eSPierre-Alain TORET 
111620befb6eSPierre-Alain TORET 
111720befb6eSPierre-Alain TORET static int
acpi_thinkpad_handlerevents_sysctl(SYSCTL_HANDLER_ARGS)1118a31756d0SMarkus Pfeiffer acpi_thinkpad_handlerevents_sysctl(SYSCTL_HANDLER_ARGS)
1119a31756d0SMarkus Pfeiffer {
1120a31756d0SMarkus Pfeiffer 	struct acpi_thinkpad_softc	*sc;
1121a31756d0SMarkus Pfeiffer 	int			error = 0;
1122a31756d0SMarkus Pfeiffer 	struct sbuf		sb;
1123a31756d0SMarkus Pfeiffer 	char			*cp, *ep;
1124a31756d0SMarkus Pfeiffer 	int			l, val;
1125a31756d0SMarkus Pfeiffer 	unsigned int		handler_events;
112620befb6eSPierre-Alain TORET 	char			temp[128];
1127a31756d0SMarkus Pfeiffer 
1128a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1129a31756d0SMarkus Pfeiffer 
1130a31756d0SMarkus Pfeiffer 	sc = (struct acpi_thinkpad_softc *)oidp->oid_arg1;
1131a31756d0SMarkus Pfeiffer 
1132a31756d0SMarkus Pfeiffer 	if (sbuf_new(&sb, NULL, 128, SBUF_AUTOEXTEND) == NULL)
1133a31756d0SMarkus Pfeiffer 		return (ENOMEM);
1134a31756d0SMarkus Pfeiffer 
1135a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_BEGIN(thinkpad);
1136a31756d0SMarkus Pfeiffer 
1137a31756d0SMarkus Pfeiffer 	/* Get old values if this is a get request. */
1138a31756d0SMarkus Pfeiffer 	if (req->newptr == NULL) {
1139a31756d0SMarkus Pfeiffer 		for (int i = 0; i < 8 * sizeof(sc->handler_events); i++)
1140*c91db742SSascha Wildner 			if (sc->handler_events & (1U << i))
1141a31756d0SMarkus Pfeiffer 				sbuf_printf(&sb, "0x%02x ", i + 1);
1142a31756d0SMarkus Pfeiffer 		if (sbuf_len(&sb) == 0)
1143a31756d0SMarkus Pfeiffer 			sbuf_printf(&sb, "NONE");
1144a31756d0SMarkus Pfeiffer 	}
1145a31756d0SMarkus Pfeiffer 
1146a31756d0SMarkus Pfeiffer 	sbuf_trim(&sb);
1147a31756d0SMarkus Pfeiffer 	sbuf_finish(&sb);
114820befb6eSPierre-Alain TORET 	strlcpy(temp, sbuf_data(&sb), sizeof(temp));
1149a31756d0SMarkus Pfeiffer 	sbuf_delete(&sb);
1150a31756d0SMarkus Pfeiffer 
115120befb6eSPierre-Alain TORET 	error = sysctl_handle_string(oidp, temp, sizeof(temp), req);
115220befb6eSPierre-Alain TORET 
115320befb6eSPierre-Alain TORET 	/* Check for error or no change */
1154a31756d0SMarkus Pfeiffer 	if (error != 0 || req->newptr == NULL)
1155a31756d0SMarkus Pfeiffer 		goto out;
1156a31756d0SMarkus Pfeiffer 
1157a31756d0SMarkus Pfeiffer 	/* If the user is setting a string, parse it. */
1158a31756d0SMarkus Pfeiffer 	handler_events = 0;
115920befb6eSPierre-Alain TORET 	cp = temp;
1160a31756d0SMarkus Pfeiffer 	while (*cp) {
1161a31756d0SMarkus Pfeiffer 		if (isspace(*cp)) {
1162a31756d0SMarkus Pfeiffer 			cp++;
1163a31756d0SMarkus Pfeiffer 			continue;
1164a31756d0SMarkus Pfeiffer 		}
1165a31756d0SMarkus Pfeiffer 
1166a31756d0SMarkus Pfeiffer 		ep = cp;
1167a31756d0SMarkus Pfeiffer 
1168a31756d0SMarkus Pfeiffer 		while (*ep && !isspace(*ep))
1169a31756d0SMarkus Pfeiffer 			ep++;
1170a31756d0SMarkus Pfeiffer 
1171a31756d0SMarkus Pfeiffer 		l = ep - cp;
1172a31756d0SMarkus Pfeiffer 		if (l == 0)
1173a31756d0SMarkus Pfeiffer 			break;
1174a31756d0SMarkus Pfeiffer 
1175a31756d0SMarkus Pfeiffer 		if (strncmp(cp, "NONE", 4) == 0) {
1176a31756d0SMarkus Pfeiffer 			cp = ep;
1177a31756d0SMarkus Pfeiffer 			continue;
1178a31756d0SMarkus Pfeiffer 		}
1179a31756d0SMarkus Pfeiffer 
1180a31756d0SMarkus Pfeiffer 		if (l >= 3 && cp[0] == '0' && (cp[1] == 'X' || cp[1] == 'x'))
1181a31756d0SMarkus Pfeiffer 			val = strtoul(cp, &ep, 16);
1182a31756d0SMarkus Pfeiffer 		else
1183a31756d0SMarkus Pfeiffer 			val = strtoul(cp, &ep, 10);
1184a31756d0SMarkus Pfeiffer 
1185a31756d0SMarkus Pfeiffer 		if (val == 0 || ep == cp || val >= 8 * sizeof(handler_events)) {
1186a31756d0SMarkus Pfeiffer 			cp[l] = '\0';
1187a31756d0SMarkus Pfeiffer 			device_printf(sc->dev, "invalid event code: %s\n", cp);
1188a31756d0SMarkus Pfeiffer 			error = EINVAL;
1189a31756d0SMarkus Pfeiffer 			goto out;
1190a31756d0SMarkus Pfeiffer 		}
1191a31756d0SMarkus Pfeiffer 
1192a31756d0SMarkus Pfeiffer 		handler_events |= 1 << (val - 1);
1193a31756d0SMarkus Pfeiffer 
1194a31756d0SMarkus Pfeiffer 		cp = ep;
1195a31756d0SMarkus Pfeiffer 	}
1196a31756d0SMarkus Pfeiffer 
1197a31756d0SMarkus Pfeiffer 	sc->handler_events = handler_events;
1198a31756d0SMarkus Pfeiffer out:
1199a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_END(thinkpad);
1200a31756d0SMarkus Pfeiffer 	return (error);
1201a31756d0SMarkus Pfeiffer }
1202a31756d0SMarkus Pfeiffer 
1203a31756d0SMarkus Pfeiffer static int
acpi_thinkpad_brightness_set(struct acpi_thinkpad_softc * sc,int arg)1204a31756d0SMarkus Pfeiffer acpi_thinkpad_brightness_set(struct acpi_thinkpad_softc *sc, int arg)
1205a31756d0SMarkus Pfeiffer {
1206a31756d0SMarkus Pfeiffer 	int			val, step;
1207a31756d0SMarkus Pfeiffer 	UINT64			val_ec;
1208a31756d0SMarkus Pfeiffer 	ACPI_OBJECT		Arg;
1209a31756d0SMarkus Pfeiffer 	ACPI_OBJECT_LIST	Args;
1210a31756d0SMarkus Pfeiffer 	ACPI_STATUS		status;
1211a31756d0SMarkus Pfeiffer 
1212a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1213a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
1214a31756d0SMarkus Pfeiffer 
1215a31756d0SMarkus Pfeiffer 	if (arg < 0 || arg > 7)
1216a31756d0SMarkus Pfeiffer 		return (EINVAL);
1217a31756d0SMarkus Pfeiffer 
1218a31756d0SMarkus Pfeiffer 	/* Read the current brightness */
1219a31756d0SMarkus Pfeiffer 	status = ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_BRIGHTNESS, &val_ec, 1);
1220a31756d0SMarkus Pfeiffer 	if (ACPI_FAILURE(status))
1221a31756d0SMarkus Pfeiffer 		return (status);
1222a31756d0SMarkus Pfeiffer 
1223a31756d0SMarkus Pfeiffer 	if (sc->cmos_handle) {
1224a31756d0SMarkus Pfeiffer 		val = val_ec & THINKPAD_EC_MASK_BRI;
1225a31756d0SMarkus Pfeiffer 
1226a31756d0SMarkus Pfeiffer 		Args.Count = 1;
1227a31756d0SMarkus Pfeiffer 		Args.Pointer = &Arg;
1228a31756d0SMarkus Pfeiffer 		Arg.Type = ACPI_TYPE_INTEGER;
1229a31756d0SMarkus Pfeiffer 		Arg.Integer.Value = (arg > val) ? THINKPAD_CMOS_BRIGHTNESS_UP :
1230a31756d0SMarkus Pfeiffer 		                                  THINKPAD_CMOS_BRIGHTNESS_DOWN;
1231a31756d0SMarkus Pfeiffer 
1232a31756d0SMarkus Pfeiffer 		step = (arg > val) ? 1 : -1;
1233a31756d0SMarkus Pfeiffer 		for (int i = val; i != arg; i += step) {
123420befb6eSPierre-Alain TORET 			status = AcpiEvaluateObject(sc->cmos_handle, NULL, &Args, NULL);
1235a31756d0SMarkus Pfeiffer 			if (ACPI_FAILURE(status)) {
1236a31756d0SMarkus Pfeiffer 				/* Record the last value */
1237a31756d0SMarkus Pfeiffer 				if (i != val) {
1238a31756d0SMarkus Pfeiffer 					ACPI_EC_WRITE(sc->ec_dev,
1239a31756d0SMarkus Pfeiffer 					    THINKPAD_EC_BRIGHTNESS, i - step, 1);
1240a31756d0SMarkus Pfeiffer 				}
1241a31756d0SMarkus Pfeiffer 				return (status);
1242a31756d0SMarkus Pfeiffer 			}
1243a31756d0SMarkus Pfeiffer 		}
1244a31756d0SMarkus Pfeiffer 	}
1245a31756d0SMarkus Pfeiffer 
1246a31756d0SMarkus Pfeiffer 	return ACPI_EC_WRITE(sc->ec_dev, THINKPAD_EC_BRIGHTNESS, arg, 1);
1247a31756d0SMarkus Pfeiffer }
1248a31756d0SMarkus Pfeiffer 
1249a31756d0SMarkus Pfeiffer static int
acpi_thinkpad_bluetooth_set(struct acpi_thinkpad_softc * sc,int arg)1250a31756d0SMarkus Pfeiffer acpi_thinkpad_bluetooth_set(struct acpi_thinkpad_softc *sc, int arg)
1251a31756d0SMarkus Pfeiffer {
1252a31756d0SMarkus Pfeiffer 	int			val;
1253a31756d0SMarkus Pfeiffer 
1254a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1255a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
1256a31756d0SMarkus Pfeiffer 
1257a31756d0SMarkus Pfeiffer 	if (arg < 0 || arg > 1)
1258a31756d0SMarkus Pfeiffer 		return (EINVAL);
1259a31756d0SMarkus Pfeiffer 
1260a31756d0SMarkus Pfeiffer 	val = (arg == 1) ? sc->wlan_bt_flags | THINKPAD_NAME_MASK_BT :
1261a31756d0SMarkus Pfeiffer 			   sc->wlan_bt_flags & (~THINKPAD_NAME_MASK_BT);
1262a31756d0SMarkus Pfeiffer 	return acpi_SetInteger(sc->handle, THINKPAD_NAME_WLAN_BT_SET, val);
1263a31756d0SMarkus Pfeiffer }
1264a31756d0SMarkus Pfeiffer 
1265a31756d0SMarkus Pfeiffer static int
acpi_thinkpad_thinklight_set(struct acpi_thinkpad_softc * sc,int arg)1266a31756d0SMarkus Pfeiffer acpi_thinkpad_thinklight_set(struct acpi_thinkpad_softc *sc, int arg)
1267a31756d0SMarkus Pfeiffer {
1268a31756d0SMarkus Pfeiffer 	ACPI_OBJECT		Arg;
1269a31756d0SMarkus Pfeiffer 	ACPI_OBJECT_LIST	Args;
1270a31756d0SMarkus Pfeiffer 	ACPI_STATUS		status;
1271a31756d0SMarkus Pfeiffer 
1272a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1273a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
1274a31756d0SMarkus Pfeiffer 
1275a31756d0SMarkus Pfeiffer 	if (arg < 0 || arg > 1)
1276a31756d0SMarkus Pfeiffer 		return (EINVAL);
1277a31756d0SMarkus Pfeiffer 
1278a31756d0SMarkus Pfeiffer 	if (sc->light_set_supported) {
1279a31756d0SMarkus Pfeiffer 		Args.Count = 1;
1280a31756d0SMarkus Pfeiffer 		Args.Pointer = &Arg;
1281a31756d0SMarkus Pfeiffer 		Arg.Type = ACPI_TYPE_INTEGER;
1282a31756d0SMarkus Pfeiffer 		Arg.Integer.Value = arg ? sc->light_cmd_on : sc->light_cmd_off;
1283a31756d0SMarkus Pfeiffer 
1284a31756d0SMarkus Pfeiffer 		status = AcpiEvaluateObject(sc->light_handle, NULL,
1285a31756d0SMarkus Pfeiffer 					    &Args, NULL);
1286a31756d0SMarkus Pfeiffer 		if (ACPI_SUCCESS(status))
1287a31756d0SMarkus Pfeiffer 			sc->light_val = arg;
1288a31756d0SMarkus Pfeiffer 		return (status);
1289a31756d0SMarkus Pfeiffer 	}
1290a31756d0SMarkus Pfeiffer 
1291a31756d0SMarkus Pfeiffer 	return (0);
1292a31756d0SMarkus Pfeiffer }
1293a31756d0SMarkus Pfeiffer 
129420befb6eSPierre-Alain TORET /*
129520befb6eSPierre-Alain TORET  * Helper function to make a get or set ACPI call to the PrivacyGuard handle.
129620befb6eSPierre-Alain TORET  * Only meant to be used internally by the get/set functions below.
129720befb6eSPierre-Alain TORET  */
129820befb6eSPierre-Alain TORET static ACPI_STATUS
acpi_thinkpad_privacyguard_acpi_call(struct acpi_thinkpad_softc * sc,bool write,int * arg)129920befb6eSPierre-Alain TORET acpi_thinkpad_privacyguard_acpi_call(struct acpi_thinkpad_softc *sc, bool write, int *arg)
130020befb6eSPierre-Alain TORET {
130120befb6eSPierre-Alain TORET 	ACPI_OBJECT		Arg;
130220befb6eSPierre-Alain TORET 	ACPI_OBJECT_LIST	Args;
130320befb6eSPierre-Alain TORET 	ACPI_STATUS		status;
130420befb6eSPierre-Alain TORET 	ACPI_OBJECT		out_obj;
130520befb6eSPierre-Alain TORET 	ACPI_BUFFER		result;
130620befb6eSPierre-Alain TORET 
130720befb6eSPierre-Alain TORET 	Arg.Type = ACPI_TYPE_INTEGER;
130820befb6eSPierre-Alain TORET 	Arg.Integer.Value = (write ? *arg : 0);
130920befb6eSPierre-Alain TORET 	Args.Count = 1;
131020befb6eSPierre-Alain TORET 	Args.Pointer = &Arg;
131120befb6eSPierre-Alain TORET 	result.Length = sizeof(out_obj);
131220befb6eSPierre-Alain TORET 	result.Pointer = &out_obj;
131320befb6eSPierre-Alain TORET 
131420befb6eSPierre-Alain TORET 	status = AcpiEvaluateObject(sc->handle,
131520befb6eSPierre-Alain TORET 	    (write ? THINKPAD_NAME_PRIVACYGUARD_SET : THINKPAD_NAME_PRIVACYGUARD_GET),
131620befb6eSPierre-Alain TORET 	    &Args, &result);
131720befb6eSPierre-Alain TORET 	if (ACPI_SUCCESS(status) && !write)
131820befb6eSPierre-Alain TORET 		*arg = out_obj.Integer.Value;
131920befb6eSPierre-Alain TORET 
132020befb6eSPierre-Alain TORET 	return (status);
132120befb6eSPierre-Alain TORET }
132220befb6eSPierre-Alain TORET 
132320befb6eSPierre-Alain TORET /*
132420befb6eSPierre-Alain TORET  * Returns -1 if the device is not present.
132520befb6eSPierre-Alain TORET  */
132620befb6eSPierre-Alain TORET static int
acpi_thinkpad_privacyguard_get(struct acpi_thinkpad_softc * sc)132720befb6eSPierre-Alain TORET acpi_thinkpad_privacyguard_get(struct acpi_thinkpad_softc *sc)
132820befb6eSPierre-Alain TORET {
132920befb6eSPierre-Alain TORET 	ACPI_STATUS status;
133020befb6eSPierre-Alain TORET 	int val;
133120befb6eSPierre-Alain TORET 
133220befb6eSPierre-Alain TORET 	status = acpi_thinkpad_privacyguard_acpi_call(sc, false, &val);
133320befb6eSPierre-Alain TORET 	if (ACPI_SUCCESS(status) &&
133420befb6eSPierre-Alain TORET 	    (val & THINKPAD_FLAG_PRIVACYGUARD_DEVICE_PRESENT))
133520befb6eSPierre-Alain TORET 		return (val & THINKPAD_FLAG_PRIVACYGUARD_ON);
133620befb6eSPierre-Alain TORET 
133720befb6eSPierre-Alain TORET 	return (-1);
133820befb6eSPierre-Alain TORET }
133920befb6eSPierre-Alain TORET 
134020befb6eSPierre-Alain TORET static ACPI_STATUS
acpi_thinkpad_privacyguard_set(struct acpi_thinkpad_softc * sc,int arg)134120befb6eSPierre-Alain TORET acpi_thinkpad_privacyguard_set(struct acpi_thinkpad_softc *sc, int arg)
134220befb6eSPierre-Alain TORET {
134320befb6eSPierre-Alain TORET 	if (arg < 0 || arg > 1)
134420befb6eSPierre-Alain TORET 		return (AE_BAD_PARAMETER);
134520befb6eSPierre-Alain TORET 
134620befb6eSPierre-Alain TORET 	return (acpi_thinkpad_privacyguard_acpi_call(sc, true, &arg));
134720befb6eSPierre-Alain TORET }
134820befb6eSPierre-Alain TORET 
1349a31756d0SMarkus Pfeiffer static int
acpi_thinkpad_volume_set(struct acpi_thinkpad_softc * sc,int arg)1350a31756d0SMarkus Pfeiffer acpi_thinkpad_volume_set(struct acpi_thinkpad_softc *sc, int arg)
1351a31756d0SMarkus Pfeiffer {
1352a31756d0SMarkus Pfeiffer 	int			val, step;
1353a31756d0SMarkus Pfeiffer 	UINT64			val_ec;
1354a31756d0SMarkus Pfeiffer 	ACPI_OBJECT		Arg;
1355a31756d0SMarkus Pfeiffer 	ACPI_OBJECT_LIST	Args;
1356a31756d0SMarkus Pfeiffer 	ACPI_STATUS		status;
1357a31756d0SMarkus Pfeiffer 
1358a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1359a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
1360a31756d0SMarkus Pfeiffer 
1361a31756d0SMarkus Pfeiffer 	if (arg < 0 || arg > 14)
1362a31756d0SMarkus Pfeiffer 		return (EINVAL);
1363a31756d0SMarkus Pfeiffer 
1364a31756d0SMarkus Pfeiffer 	/* Read the current volume */
1365a31756d0SMarkus Pfeiffer 	status = ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_VOLUME, &val_ec, 1);
1366a31756d0SMarkus Pfeiffer 	if (ACPI_FAILURE(status))
1367a31756d0SMarkus Pfeiffer 		return (status);
1368a31756d0SMarkus Pfeiffer 
1369a31756d0SMarkus Pfeiffer 	if (sc->cmos_handle) {
1370a31756d0SMarkus Pfeiffer 		val = val_ec & THINKPAD_EC_MASK_VOL;
1371a31756d0SMarkus Pfeiffer 
1372a31756d0SMarkus Pfeiffer 		Args.Count = 1;
1373a31756d0SMarkus Pfeiffer 		Args.Pointer = &Arg;
1374a31756d0SMarkus Pfeiffer 		Arg.Type = ACPI_TYPE_INTEGER;
1375a31756d0SMarkus Pfeiffer 		Arg.Integer.Value = (arg > val) ? THINKPAD_CMOS_VOLUME_UP :
1376a31756d0SMarkus Pfeiffer 						  THINKPAD_CMOS_VOLUME_DOWN;
1377a31756d0SMarkus Pfeiffer 
1378a31756d0SMarkus Pfeiffer 		step = (arg > val) ? 1 : -1;
1379a31756d0SMarkus Pfeiffer 		for (int i = val; i != arg; i += step) {
1380a31756d0SMarkus Pfeiffer 			status = AcpiEvaluateObject(sc->cmos_handle, NULL,
1381a31756d0SMarkus Pfeiffer 						    &Args, NULL);
1382a31756d0SMarkus Pfeiffer 			if (ACPI_FAILURE(status)) {
1383a31756d0SMarkus Pfeiffer 				/* Record the last value */
1384a31756d0SMarkus Pfeiffer 				if (i != val) {
1385a31756d0SMarkus Pfeiffer 					val_ec = i - step +
1386a31756d0SMarkus Pfeiffer 						 (val_ec & (~THINKPAD_EC_MASK_VOL));
1387a31756d0SMarkus Pfeiffer 					ACPI_EC_WRITE(sc->ec_dev, THINKPAD_EC_VOLUME,
1388a31756d0SMarkus Pfeiffer 						      val_ec, 1);
1389a31756d0SMarkus Pfeiffer 				}
1390a31756d0SMarkus Pfeiffer 				return (status);
1391a31756d0SMarkus Pfeiffer 			}
1392a31756d0SMarkus Pfeiffer 		}
1393a31756d0SMarkus Pfeiffer 	}
1394a31756d0SMarkus Pfeiffer 
1395a31756d0SMarkus Pfeiffer 	val_ec = arg + (val_ec & (~THINKPAD_EC_MASK_VOL));
1396a31756d0SMarkus Pfeiffer 	return ACPI_EC_WRITE(sc->ec_dev, THINKPAD_EC_VOLUME, val_ec, 1);
1397a31756d0SMarkus Pfeiffer }
1398a31756d0SMarkus Pfeiffer 
1399a31756d0SMarkus Pfeiffer static int
acpi_thinkpad_mute_set(struct acpi_thinkpad_softc * sc,int arg)1400a31756d0SMarkus Pfeiffer acpi_thinkpad_mute_set(struct acpi_thinkpad_softc *sc, int arg)
1401a31756d0SMarkus Pfeiffer {
1402a31756d0SMarkus Pfeiffer 	UINT64			val_ec;
1403a31756d0SMarkus Pfeiffer 	ACPI_OBJECT		Arg;
1404a31756d0SMarkus Pfeiffer 	ACPI_OBJECT_LIST	Args;
1405a31756d0SMarkus Pfeiffer 	ACPI_STATUS		status;
1406a31756d0SMarkus Pfeiffer 
1407a31756d0SMarkus Pfeiffer 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1408a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_ASSERT(thinkpad);
1409a31756d0SMarkus Pfeiffer 
1410a31756d0SMarkus Pfeiffer 	if (arg < 0 || arg > 1)
1411a31756d0SMarkus Pfeiffer 		return (EINVAL);
1412a31756d0SMarkus Pfeiffer 
1413a31756d0SMarkus Pfeiffer 	status = ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_VOLUME, &val_ec, 1);
1414a31756d0SMarkus Pfeiffer 	if (ACPI_FAILURE(status))
1415a31756d0SMarkus Pfeiffer 		return (status);
1416a31756d0SMarkus Pfeiffer 
1417a31756d0SMarkus Pfeiffer 	if (sc->cmos_handle) {
1418a31756d0SMarkus Pfeiffer 		Args.Count = 1;
1419a31756d0SMarkus Pfeiffer 		Args.Pointer = &Arg;
1420a31756d0SMarkus Pfeiffer 		Arg.Type = ACPI_TYPE_INTEGER;
1421a31756d0SMarkus Pfeiffer 		Arg.Integer.Value = THINKPAD_CMOS_VOLUME_MUTE;
1422a31756d0SMarkus Pfeiffer 
1423a31756d0SMarkus Pfeiffer 		status = AcpiEvaluateObject(sc->cmos_handle, NULL, &Args, NULL);
1424a31756d0SMarkus Pfeiffer 		if (ACPI_FAILURE(status))
1425a31756d0SMarkus Pfeiffer 			return (status);
1426a31756d0SMarkus Pfeiffer 	}
1427a31756d0SMarkus Pfeiffer 
1428a31756d0SMarkus Pfeiffer 	val_ec = (arg == 1) ? val_ec | THINKPAD_EC_MASK_MUTE :
1429a31756d0SMarkus Pfeiffer 			      val_ec & (~THINKPAD_EC_MASK_MUTE);
1430a31756d0SMarkus Pfeiffer 	return ACPI_EC_WRITE(sc->ec_dev, THINKPAD_EC_VOLUME, val_ec, 1);
1431a31756d0SMarkus Pfeiffer }
1432a31756d0SMarkus Pfeiffer 
1433a31756d0SMarkus Pfeiffer static void
acpi_thinkpad_eventhandler(struct acpi_thinkpad_softc * sc,int arg)1434a31756d0SMarkus Pfeiffer acpi_thinkpad_eventhandler(struct acpi_thinkpad_softc *sc, int arg)
1435a31756d0SMarkus Pfeiffer {
1436a31756d0SMarkus Pfeiffer 	int			val;
1437a31756d0SMarkus Pfeiffer 	UINT64			val_ec;
1438a31756d0SMarkus Pfeiffer 	ACPI_STATUS		status;
1439a31756d0SMarkus Pfeiffer 
1440a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_BEGIN(thinkpad);
1441a31756d0SMarkus Pfeiffer 	switch (arg) {
1442a31756d0SMarkus Pfeiffer #if 0 /* XXX */
1443a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_SUSPEND_TO_RAM:
1444a31756d0SMarkus Pfeiffer 		power_pm_suspend(POWER_SLEEP_STATE_SUSPEND);
1445a31756d0SMarkus Pfeiffer 		break;
1446a31756d0SMarkus Pfeiffer #endif
1447a31756d0SMarkus Pfeiffer 
1448a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_BLUETOOTH:
1449a31756d0SMarkus Pfeiffer 		acpi_thinkpad_bluetooth_set(sc, (sc->wlan_bt_flags == 0));
1450a31756d0SMarkus Pfeiffer 		break;
1451a31756d0SMarkus Pfeiffer 
1452a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_BRIGHTNESS_UP:
1453a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_BRIGHTNESS_DOWN:
1454a31756d0SMarkus Pfeiffer 		/* Read the current brightness */
1455a31756d0SMarkus Pfeiffer 		status = ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_BRIGHTNESS,
1456a31756d0SMarkus Pfeiffer 				      &val_ec, 1);
1457a31756d0SMarkus Pfeiffer 		if (ACPI_FAILURE(status))
14589c897caeSSascha Wildner 			goto done;
1459a31756d0SMarkus Pfeiffer 
1460a31756d0SMarkus Pfeiffer 		val = val_ec & THINKPAD_EC_MASK_BRI;
1461a31756d0SMarkus Pfeiffer 		val = (arg == THINKPAD_EVENT_BRIGHTNESS_UP) ? val + 1 : val - 1;
1462a31756d0SMarkus Pfeiffer 		acpi_thinkpad_brightness_set(sc, val);
1463a31756d0SMarkus Pfeiffer 		break;
1464a31756d0SMarkus Pfeiffer 
1465a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_THINKLIGHT:
1466a31756d0SMarkus Pfeiffer 		acpi_thinkpad_thinklight_set(sc, (sc->light_val == 0));
1467a31756d0SMarkus Pfeiffer 		break;
1468a31756d0SMarkus Pfeiffer 
1469a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_VOLUME_UP:
1470a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_VOLUME_DOWN:
1471a31756d0SMarkus Pfeiffer 		/* Read the current volume */
1472a31756d0SMarkus Pfeiffer 		status = ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_VOLUME, &val_ec, 1);
1473a31756d0SMarkus Pfeiffer 		if (ACPI_FAILURE(status))
14749c897caeSSascha Wildner 			goto done;
1475a31756d0SMarkus Pfeiffer 
1476a31756d0SMarkus Pfeiffer 		val = val_ec & THINKPAD_EC_MASK_VOL;
1477a31756d0SMarkus Pfeiffer 		val = (arg == THINKPAD_EVENT_VOLUME_UP) ? val + 1 : val - 1;
1478a31756d0SMarkus Pfeiffer 		acpi_thinkpad_volume_set(sc, val);
1479a31756d0SMarkus Pfeiffer 		break;
1480a31756d0SMarkus Pfeiffer 
1481a31756d0SMarkus Pfeiffer 	case THINKPAD_EVENT_MUTE:
1482a31756d0SMarkus Pfeiffer 		/* Read the current value */
1483a31756d0SMarkus Pfeiffer 		status = ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_VOLUME, &val_ec, 1);
1484a31756d0SMarkus Pfeiffer 		if (ACPI_FAILURE(status))
14859c897caeSSascha Wildner 			goto done;
1486a31756d0SMarkus Pfeiffer 
1487a31756d0SMarkus Pfeiffer 		val = ((val_ec & THINKPAD_EC_MASK_MUTE) == THINKPAD_EC_MASK_MUTE);
1488a31756d0SMarkus Pfeiffer 		acpi_thinkpad_mute_set(sc, (val == 0));
1489a31756d0SMarkus Pfeiffer 		break;
1490a31756d0SMarkus Pfeiffer 
1491a31756d0SMarkus Pfeiffer 	default:
1492a31756d0SMarkus Pfeiffer 		break;
1493a31756d0SMarkus Pfeiffer 	}
14949c897caeSSascha Wildner done:
1495a31756d0SMarkus Pfeiffer 	ACPI_SERIAL_END(thinkpad);
1496a31756d0SMarkus Pfeiffer }
1497a31756d0SMarkus Pfeiffer 
14985db2f26eSSascha Wildner static void
acpi_thinkpad_notify(ACPI_HANDLE h,UINT32 notify,void * context)14995db2f26eSSascha Wildner acpi_thinkpad_notify(ACPI_HANDLE h, UINT32 notify, void *context)
15005db2f26eSSascha Wildner {
15015db2f26eSSascha Wildner 	int		event, arg, type;
15025db2f26eSSascha Wildner 	device_t	dev = context;
15035db2f26eSSascha Wildner 	struct acpi_thinkpad_softc *sc = device_get_softc(dev);
15045db2f26eSSascha Wildner 
15055db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE_U32((char *)(uintptr_t)__func__, notify);
15065db2f26eSSascha Wildner 
15075db2f26eSSascha Wildner 	if (notify != 0x80)
1508d1fb95bdSSascha Wildner 		device_printf(sc->dev, "unknown notify: %#x\n", notify);
15095db2f26eSSascha Wildner 
15105db2f26eSSascha Wildner 	for (;;) {
151120befb6eSPierre-Alain TORET 		acpi_GetInteger(acpi_get_handle(dev), THINKPAD_NAME_EVENTS_GET, &event);
15125db2f26eSSascha Wildner 		if (event == 0)
15135db2f26eSSascha Wildner 			break;
15145db2f26eSSascha Wildner 
15155db2f26eSSascha Wildner 		type = (event >> 12) & 0xf;
15165db2f26eSSascha Wildner 		arg = event & 0xfff;
15175db2f26eSSascha Wildner 		switch (type) {
15185db2f26eSSascha Wildner 		case 1:
15195db2f26eSSascha Wildner 			if (!(sc->events_availmask & (1 << (arg - 1)))) {
15205db2f26eSSascha Wildner 				device_printf(dev, "Unknown key %d\n", arg);
15215db2f26eSSascha Wildner 				break;
15225db2f26eSSascha Wildner 			}
15235db2f26eSSascha Wildner 
1524a31756d0SMarkus Pfeiffer 			/* Execute event handler */
1525a31756d0SMarkus Pfeiffer 			if (sc->handler_events & (1 << (arg - 1)))
1526a31756d0SMarkus Pfeiffer 				acpi_thinkpad_eventhandler(sc, (arg & 0xff));
1527a31756d0SMarkus Pfeiffer 
15285db2f26eSSascha Wildner 			/* Notify devd(8) */
15295db2f26eSSascha Wildner 			acpi_UserNotify("THINKPAD", h, (arg & 0xff));
15305db2f26eSSascha Wildner 			break;
15315db2f26eSSascha Wildner 		default:
15325db2f26eSSascha Wildner 			break;
15335db2f26eSSascha Wildner 		}
15345db2f26eSSascha Wildner 	}
15355db2f26eSSascha Wildner }
15365db2f26eSSascha Wildner 
15375db2f26eSSascha Wildner static void
acpi_thinkpad_refresh(void * arg)15385db2f26eSSascha Wildner acpi_thinkpad_refresh(void *arg)
15395db2f26eSSascha Wildner {
15405db2f26eSSascha Wildner 	struct acpi_thinkpad_softc *sc = (struct acpi_thinkpad_softc *)arg;
15415db2f26eSSascha Wildner 	int i, data;
15425db2f26eSSascha Wildner 
15435db2f26eSSascha Wildner 	for (i = 0; i < THINKPAD_TEMP_SENSORS; i++) {
15445db2f26eSSascha Wildner 		char temp_cmd[] = "TMP0";
15455db2f26eSSascha Wildner 
15465db2f26eSSascha Wildner 		temp_cmd[3] = '0' + i;
15475db2f26eSSascha Wildner 		/*
15485db2f26eSSascha Wildner 		 * The TMPx methods seem to return +/- 128 or 0
15495db2f26eSSascha Wildner 		 * when the respecting sensor is not available
15505db2f26eSSascha Wildner 		 */
15515db2f26eSSascha Wildner 		if (ACPI_FAILURE(acpi_GetInteger(sc->ec_handle, temp_cmd,
15525db2f26eSSascha Wildner 		    &data)) || ABS(data) == 128 || data == 0) {
15535db2f26eSSascha Wildner 			sc->sensors[i].flags |= SENSOR_FINVALID;
15545db2f26eSSascha Wildner 			continue;
15555db2f26eSSascha Wildner 		}
15565db2f26eSSascha Wildner 		if (sc->thermal_updt_supported)
15575db2f26eSSascha Wildner 			/* Temperature is reported in tenth of Kelvin */
15585db2f26eSSascha Wildner 			sc->sensors[i].value = data * 100000 - 50000;
15595db2f26eSSascha Wildner 		else
15605db2f26eSSascha Wildner 			sc->sensors[i].value = data * 1000000 + 273150000;
15615db2f26eSSascha Wildner 		sc->sensors[i].flags &= ~SENSOR_FINVALID;
15625db2f26eSSascha Wildner 	}
15635db2f26eSSascha Wildner 
15645db2f26eSSascha Wildner 	if (sc->fan_handle) {
15655db2f26eSSascha Wildner 		if (ACPI_FAILURE(acpi_GetInteger(sc->fan_handle,
15665db2f26eSSascha Wildner 		    NULL, &data)))
15675db2f26eSSascha Wildner 			sc->sensors[i].flags |= SENSOR_FINVALID;
15685db2f26eSSascha Wildner 		sc->sensors[i].value = data;
15695db2f26eSSascha Wildner 		sc->sensors[i].flags &= ~SENSOR_FINVALID;
15705db2f26eSSascha Wildner 	} else {
1571417dc5a4SSascha Wildner 		UINT64 speed;
15725db2f26eSSascha Wildner 
15735db2f26eSSascha Wildner 		ACPI_EC_READ(sc->ec_dev, THINKPAD_EC_FANSPEED, &speed, 2);
15745db2f26eSSascha Wildner 		sc->sensors[i].value = speed;
15755db2f26eSSascha Wildner 		sc->sensors[i].flags &= ~SENSOR_FINVALID;
15765db2f26eSSascha Wildner 	}
15775db2f26eSSascha Wildner }
1578