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