xref: /dragonfly/sys/dev/acpica/acpi_asus/acpi_asus.c (revision f2c400e8)
15db2f26eSSascha Wildner /*-
25db2f26eSSascha Wildner  * Copyright (c) 2004, 2005 Philip Paeps <philip@FreeBSD.org>
35db2f26eSSascha Wildner  * All rights reserved.
45db2f26eSSascha Wildner  *
55db2f26eSSascha Wildner  * Redistribution and use in source and binary forms, with or without
65db2f26eSSascha Wildner  * modification, are permitted provided that the following conditions
75db2f26eSSascha Wildner  * are met:
85db2f26eSSascha Wildner  * 1. Redistributions of source code must retain the above copyright
95db2f26eSSascha Wildner  *    notice, this list of conditions and the following disclaimer.
105db2f26eSSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
115db2f26eSSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
125db2f26eSSascha Wildner  *    documentation and/or other materials provided with the distribution.
135db2f26eSSascha Wildner  *
145db2f26eSSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
155db2f26eSSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
165db2f26eSSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
175db2f26eSSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
185db2f26eSSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
195db2f26eSSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
205db2f26eSSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
215db2f26eSSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
225db2f26eSSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
235db2f26eSSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
245db2f26eSSascha Wildner  * SUCH DAMAGE.
255db2f26eSSascha Wildner  *
265db2f26eSSascha Wildner  * $FreeBSD: src/sys/dev/acpi_support/acpi_asus.c,v 1.44 2010/06/11 20:08:20 jkim Exp $
275db2f26eSSascha Wildner  */
285db2f26eSSascha Wildner 
295db2f26eSSascha Wildner /*
305db2f26eSSascha Wildner  * Driver for extra ACPI-controlled gadgets (hotkeys, leds, etc) found on
315db2f26eSSascha Wildner  * recent Asus (and Medion) laptops.  Inspired by the acpi4asus project which
325db2f26eSSascha Wildner  * implements these features in the Linux kernel.
335db2f26eSSascha Wildner  *
345db2f26eSSascha Wildner  *   <http://sourceforge.net/projects/acpi4asus/>
355db2f26eSSascha Wildner  *
365db2f26eSSascha Wildner  * Currently should support most features, but could use some more testing.
375db2f26eSSascha Wildner  * Particularly the display-switching stuff is a bit hairy.  If you have an
385db2f26eSSascha Wildner  * Asus laptop which doesn't appear to be supported, or strange things happen
395db2f26eSSascha Wildner  * when using this driver, please report to <acpi@FreeBSD.org>.
405db2f26eSSascha Wildner  */
415db2f26eSSascha Wildner 
425db2f26eSSascha Wildner #include "opt_acpi.h"
435db2f26eSSascha Wildner #include <sys/param.h>
445db2f26eSSascha Wildner #include <sys/kernel.h>
455db2f26eSSascha Wildner #include <sys/module.h>
465db2f26eSSascha Wildner #include <sys/bus.h>
475db2f26eSSascha Wildner #include <sys/sbuf.h>
485db2f26eSSascha Wildner 
495db2f26eSSascha Wildner #include "acpi.h"
505db2f26eSSascha Wildner #include "accommon.h"
515db2f26eSSascha Wildner #include "acpivar.h"
525db2f26eSSascha Wildner 
53e9283513SSascha Wildner #include <dev/misc/led/led.h>
545db2f26eSSascha Wildner 
555db2f26eSSascha Wildner /* Methods */
565db2f26eSSascha Wildner #define ACPI_ASUS_METHOD_BRN	1
575db2f26eSSascha Wildner #define ACPI_ASUS_METHOD_DISP	2
585db2f26eSSascha Wildner #define ACPI_ASUS_METHOD_LCD	3
595db2f26eSSascha Wildner #define ACPI_ASUS_METHOD_CAMERA	4
605db2f26eSSascha Wildner #define ACPI_ASUS_METHOD_CARDRD	5
615db2f26eSSascha Wildner #define ACPI_ASUS_METHOD_WLAN	6
625db2f26eSSascha Wildner 
635db2f26eSSascha Wildner #define _COMPONENT	ACPI_OEM
645db2f26eSSascha Wildner ACPI_MODULE_NAME("ASUS")
655db2f26eSSascha Wildner 
665db2f26eSSascha Wildner struct acpi_asus_model {
675db2f26eSSascha Wildner 	char	*name;
685db2f26eSSascha Wildner 	char	*bled_set;
695db2f26eSSascha Wildner 	char	*dled_set;
705db2f26eSSascha Wildner 	char	*gled_set;
715db2f26eSSascha Wildner 	char	*mled_set;
725db2f26eSSascha Wildner 	char	*tled_set;
735db2f26eSSascha Wildner 	char	*wled_set;
745db2f26eSSascha Wildner 
755db2f26eSSascha Wildner 	char	*brn_get;
765db2f26eSSascha Wildner 	char	*brn_set;
775db2f26eSSascha Wildner 	char	*brn_up;
785db2f26eSSascha Wildner 	char	*brn_dn;
795db2f26eSSascha Wildner 
805db2f26eSSascha Wildner 	char	*lcd_get;
815db2f26eSSascha Wildner 	char	*lcd_set;
825db2f26eSSascha Wildner 
835db2f26eSSascha Wildner 	char	*disp_get;
845db2f26eSSascha Wildner 	char	*disp_set;
855db2f26eSSascha Wildner 
865db2f26eSSascha Wildner 	char	*cam_get;
875db2f26eSSascha Wildner 	char	*cam_set;
885db2f26eSSascha Wildner 
895db2f26eSSascha Wildner 	char	*crd_get;
905db2f26eSSascha Wildner 	char	*crd_set;
915db2f26eSSascha Wildner 
925db2f26eSSascha Wildner 	char	*wlan_get;
935db2f26eSSascha Wildner 	char	*wlan_set;
945db2f26eSSascha Wildner 
955db2f26eSSascha Wildner 	void	(*n_func)(ACPI_HANDLE, UINT32, void *);
965db2f26eSSascha Wildner 
975db2f26eSSascha Wildner 	char	*lcdd;
985db2f26eSSascha Wildner 	void	(*lcdd_n_func)(ACPI_HANDLE, UINT32, void *);
995db2f26eSSascha Wildner };
1005db2f26eSSascha Wildner 
1015db2f26eSSascha Wildner struct acpi_asus_led {
1025db2f26eSSascha Wildner 	struct acpi_asus_softc *sc;
1035db2f26eSSascha Wildner 	struct cdev	*cdev;
1045db2f26eSSascha Wildner 	int		busy;
1055db2f26eSSascha Wildner 	int		state;
1065db2f26eSSascha Wildner 	enum {
1075db2f26eSSascha Wildner 		ACPI_ASUS_LED_BLED,
1085db2f26eSSascha Wildner 		ACPI_ASUS_LED_DLED,
1095db2f26eSSascha Wildner 		ACPI_ASUS_LED_GLED,
1105db2f26eSSascha Wildner 		ACPI_ASUS_LED_MLED,
1115db2f26eSSascha Wildner 		ACPI_ASUS_LED_TLED,
1125db2f26eSSascha Wildner 		ACPI_ASUS_LED_WLED,
1135db2f26eSSascha Wildner 	} type;
1145db2f26eSSascha Wildner };
1155db2f26eSSascha Wildner 
1165db2f26eSSascha Wildner struct acpi_asus_softc {
1175db2f26eSSascha Wildner 	device_t		dev;
1185db2f26eSSascha Wildner 	ACPI_HANDLE		handle;
1195db2f26eSSascha Wildner 	ACPI_HANDLE		lcdd_handle;
1205db2f26eSSascha Wildner 
1215db2f26eSSascha Wildner 	struct acpi_asus_model	*model;
1225db2f26eSSascha Wildner 	struct sysctl_ctx_list	sysctl_ctx;
1235db2f26eSSascha Wildner 	struct sysctl_oid	*sysctl_tree;
1245db2f26eSSascha Wildner 	struct acpi_asus_led	s_bled;
1255db2f26eSSascha Wildner 	struct acpi_asus_led	s_dled;
1265db2f26eSSascha Wildner 	struct acpi_asus_led	s_gled;
1275db2f26eSSascha Wildner 	struct acpi_asus_led	s_mled;
1285db2f26eSSascha Wildner 	struct acpi_asus_led	s_tled;
1295db2f26eSSascha Wildner 	struct acpi_asus_led	s_wled;
1305db2f26eSSascha Wildner 
1315db2f26eSSascha Wildner 	int			s_brn;
1325db2f26eSSascha Wildner 	int			s_disp;
1335db2f26eSSascha Wildner 	int			s_lcd;
1345db2f26eSSascha Wildner 	int			s_cam;
1355db2f26eSSascha Wildner 	int			s_crd;
1365db2f26eSSascha Wildner 	int			s_wlan;
1375db2f26eSSascha Wildner };
1385db2f26eSSascha Wildner 
1395db2f26eSSascha Wildner static void	acpi_asus_lcdd_notify(ACPI_HANDLE h, UINT32 notify,
1405db2f26eSSascha Wildner     void *context);
1415db2f26eSSascha Wildner 
1425db2f26eSSascha Wildner /*
1435db2f26eSSascha Wildner  * We can identify Asus laptops from the string they return
1445db2f26eSSascha Wildner  * as a result of calling the ATK0100 'INIT' method.
1455db2f26eSSascha Wildner  */
1465db2f26eSSascha Wildner static struct acpi_asus_model acpi_asus_models[] = {
1475db2f26eSSascha Wildner 	{
1485db2f26eSSascha Wildner 		.name		= "xxN",
1495db2f26eSSascha Wildner 		.mled_set	= "MLED",
1505db2f26eSSascha Wildner 		.wled_set	= "WLED",
1515db2f26eSSascha Wildner 		.lcd_get	= "\\BKLT",
1525db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
1535db2f26eSSascha Wildner 		.brn_get	= "GPLV",
1545db2f26eSSascha Wildner 		.brn_set	= "SPLV",
1555db2f26eSSascha Wildner 		.disp_get	= "\\ADVG",
1565db2f26eSSascha Wildner 		.disp_set	= "SDSP"
1575db2f26eSSascha Wildner 	},
1585db2f26eSSascha Wildner 	{
1595db2f26eSSascha Wildner 		.name		= "A1x",
1605db2f26eSSascha Wildner 		.mled_set	= "MLED",
1615db2f26eSSascha Wildner 		.lcd_get	= "\\BKLI",
1625db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.ISA.EC0._Q10",
1635db2f26eSSascha Wildner 		.brn_up		= "\\_SB.PCI0.ISA.EC0._Q0E",
1645db2f26eSSascha Wildner 		.brn_dn		= "\\_SB.PCI0.ISA.EC0._Q0F"
1655db2f26eSSascha Wildner 	},
1665db2f26eSSascha Wildner 	{
1675db2f26eSSascha Wildner 		.name		= "A2x",
1685db2f26eSSascha Wildner 		.mled_set	= "MLED",
1695db2f26eSSascha Wildner 		.wled_set	= "WLED",
1705db2f26eSSascha Wildner 		.lcd_get	= "\\BAOF",
1715db2f26eSSascha Wildner 		.lcd_set	= "\\Q10",
1725db2f26eSSascha Wildner 		.brn_get	= "GPLV",
1735db2f26eSSascha Wildner 		.brn_set	= "SPLV",
1745db2f26eSSascha Wildner 		.disp_get	= "\\INFB",
1755db2f26eSSascha Wildner 		.disp_set	= "SDSP"
1765db2f26eSSascha Wildner 	},
1775db2f26eSSascha Wildner 	{
1785db2f26eSSascha Wildner 		.name           = "A3E",
1795db2f26eSSascha Wildner 		.mled_set       = "MLED",
1805db2f26eSSascha Wildner 		.wled_set       = "WLED",
1815db2f26eSSascha Wildner 		.lcd_get        = "\\_SB.PCI0.SBRG.EC0.RPIN(0x67)",
1825db2f26eSSascha Wildner 		.lcd_set        = "\\_SB.PCI0.SBRG.EC0._Q10",
1835db2f26eSSascha Wildner 		.brn_get        = "GPLV",
1845db2f26eSSascha Wildner 		.brn_set        = "SPLV",
1855db2f26eSSascha Wildner 		.disp_get       = "\\_SB.PCI0.P0P2.VGA.GETD",
1865db2f26eSSascha Wildner 		.disp_set       = "SDSP"
1875db2f26eSSascha Wildner 	},
1885db2f26eSSascha Wildner 	{
1895db2f26eSSascha Wildner 		.name           = "A3F",
1905db2f26eSSascha Wildner 		.mled_set       = "MLED",
1915db2f26eSSascha Wildner 		.wled_set       = "WLED",
1925db2f26eSSascha Wildner 		.bled_set       = "BLED",
1935db2f26eSSascha Wildner 		.lcd_get        = "\\_SB.PCI0.SBRG.EC0.RPIN(0x11)",
1945db2f26eSSascha Wildner 		.lcd_set        = "\\_SB.PCI0.SBRG.EC0._Q10",
1955db2f26eSSascha Wildner 		.brn_get        = "GPLV",
1965db2f26eSSascha Wildner 		.brn_set        = "SPLV",
1975db2f26eSSascha Wildner 		.disp_get       = "\\SSTE",
1985db2f26eSSascha Wildner 		.disp_set       = "SDSP"
1995db2f26eSSascha Wildner 	},
2005db2f26eSSascha Wildner 	{
2015db2f26eSSascha Wildner 		.name           = "A3N",
2025db2f26eSSascha Wildner 		.mled_set       = "MLED",
2035db2f26eSSascha Wildner 		.bled_set       = "BLED",
2045db2f26eSSascha Wildner 		.wled_set       = "WLED",
2055db2f26eSSascha Wildner 		.lcd_get        = "\\BKLT",
2065db2f26eSSascha Wildner 		.lcd_set        = "\\_SB.PCI0.SBRG.EC0._Q10",
2075db2f26eSSascha Wildner 		.brn_get        = "GPLV",
2085db2f26eSSascha Wildner 		.brn_set        = "SPLV",
2095db2f26eSSascha Wildner 		.disp_get       = "\\_SB.PCI0.P0P3.VGA.GETD",
2105db2f26eSSascha Wildner 		.disp_set       = "SDSP"
2115db2f26eSSascha Wildner 	},
2125db2f26eSSascha Wildner 	{
2135db2f26eSSascha Wildner 		.name		= "A4D",
2145db2f26eSSascha Wildner 		.mled_set	= "MLED",
2155db2f26eSSascha Wildner 		.brn_up		= "\\_SB_.PCI0.SBRG.EC0._Q0E",
2165db2f26eSSascha Wildner 		.brn_dn		= "\\_SB_.PCI0.SBRG.EC0._Q0F",
2175db2f26eSSascha Wildner 		.brn_get	= "GPLV",
2185db2f26eSSascha Wildner 		.brn_set	= "SPLV",
2195db2f26eSSascha Wildner #ifdef notyet
2205db2f26eSSascha Wildner 		.disp_get	= "\\_SB_.PCI0.SBRG.EC0._Q10",
2215db2f26eSSascha Wildner 		.disp_set	= "\\_SB_.PCI0.SBRG.EC0._Q11"
2225db2f26eSSascha Wildner #endif
2235db2f26eSSascha Wildner 	},
2245db2f26eSSascha Wildner 	{
2255db2f26eSSascha Wildner 		.name		= "A6V",
2265db2f26eSSascha Wildner 		.bled_set	= "BLED",
2275db2f26eSSascha Wildner 		.mled_set	= "MLED",
2285db2f26eSSascha Wildner 		.wled_set	= "WLED",
2295db2f26eSSascha Wildner 		.lcd_get	= NULL,
2305db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
2315db2f26eSSascha Wildner 		.brn_get	= "GPLV",
2325db2f26eSSascha Wildner 		.brn_set	= "SPLV",
2335db2f26eSSascha Wildner 		.disp_get	= "\\_SB.PCI0.P0P3.VGA.GETD",
2345db2f26eSSascha Wildner 		.disp_set	= "SDSP"
2355db2f26eSSascha Wildner 	},
2365db2f26eSSascha Wildner 	{
2375db2f26eSSascha Wildner 		.name		= "A8SR",
2385db2f26eSSascha Wildner 		.bled_set	= "BLED",
2395db2f26eSSascha Wildner 		.mled_set	= "MLED",
2405db2f26eSSascha Wildner 		.wled_set	= "WLED",
2415db2f26eSSascha Wildner 		.lcd_get	= NULL,
2425db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
2435db2f26eSSascha Wildner 		.brn_get	= "GPLV",
2445db2f26eSSascha Wildner 		.brn_set	= "SPLV",
2455db2f26eSSascha Wildner 		.disp_get	= "\\_SB.PCI0.P0P1.VGA.GETD",
2465db2f26eSSascha Wildner 		.disp_set	= "SDSP",
2475db2f26eSSascha Wildner 		.lcdd		= "\\_SB.PCI0.P0P1.VGA.LCDD",
2485db2f26eSSascha Wildner 		.lcdd_n_func	= acpi_asus_lcdd_notify
2495db2f26eSSascha Wildner 	},
2505db2f26eSSascha Wildner 	{
2515db2f26eSSascha Wildner 		.name		= "D1x",
2525db2f26eSSascha Wildner 		.mled_set	= "MLED",
2535db2f26eSSascha Wildner 		.lcd_get	= "\\GP11",
2545db2f26eSSascha Wildner 		.lcd_set	= "\\Q0D",
2555db2f26eSSascha Wildner 		.brn_up		= "\\Q0C",
2565db2f26eSSascha Wildner 		.brn_dn		= "\\Q0B",
2575db2f26eSSascha Wildner 		.disp_get	= "\\INFB",
2585db2f26eSSascha Wildner 		.disp_set	= "SDSP"
2595db2f26eSSascha Wildner 	},
2605db2f26eSSascha Wildner 	{
2615db2f26eSSascha Wildner 		.name		= "G2K",
2625db2f26eSSascha Wildner 		.bled_set	= "BLED",
2635db2f26eSSascha Wildner 		.dled_set	= "DLED",
2645db2f26eSSascha Wildner 		.gled_set	= "GLED",
2655db2f26eSSascha Wildner 		.mled_set	= "MLED",
2665db2f26eSSascha Wildner 		.tled_set	= "TLED",
2675db2f26eSSascha Wildner 		.wled_set	= "WLED",
2685db2f26eSSascha Wildner 		.brn_get	= "GPLV",
2695db2f26eSSascha Wildner 		.brn_set	= "SPLV",
2705db2f26eSSascha Wildner 		.lcd_get	= "GBTL",
2715db2f26eSSascha Wildner 		.lcd_set	= "SBTL",
2725db2f26eSSascha Wildner 		.disp_get	= "\\_SB.PCI0.PCE2.VGA.GETD",
2735db2f26eSSascha Wildner 		.disp_set	= "SDSP",
2745db2f26eSSascha Wildner 	},
2755db2f26eSSascha Wildner 	{
2765db2f26eSSascha Wildner 		.name		= "L2D",
2775db2f26eSSascha Wildner 		.mled_set	= "MLED",
2785db2f26eSSascha Wildner 		.wled_set	= "WLED",
2795db2f26eSSascha Wildner 		.brn_up		= "\\Q0E",
2805db2f26eSSascha Wildner 		.brn_dn		= "\\Q0F",
2815db2f26eSSascha Wildner 		.lcd_get	= "\\SGP0",
2825db2f26eSSascha Wildner 		.lcd_set	= "\\Q10"
2835db2f26eSSascha Wildner 	},
2845db2f26eSSascha Wildner 	{
2855db2f26eSSascha Wildner 		.name		= "L3C",
2865db2f26eSSascha Wildner 		.mled_set	= "MLED",
2875db2f26eSSascha Wildner 		.wled_set	= "WLED",
2885db2f26eSSascha Wildner 		.brn_get	= "GPLV",
2895db2f26eSSascha Wildner 		.brn_set	= "SPLV",
2905db2f26eSSascha Wildner 		.lcd_get	= "\\GL32",
2915db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.PX40.ECD0._Q10"
2925db2f26eSSascha Wildner 	},
2935db2f26eSSascha Wildner 	{
2945db2f26eSSascha Wildner 		.name		= "L3D",
2955db2f26eSSascha Wildner 		.mled_set	= "MLED",
2965db2f26eSSascha Wildner 		.wled_set	= "WLED",
2975db2f26eSSascha Wildner 		.brn_get	= "GPLV",
2985db2f26eSSascha Wildner 		.brn_set	= "SPLV",
2995db2f26eSSascha Wildner 		.lcd_get	= "\\BKLG",
3005db2f26eSSascha Wildner 		.lcd_set	= "\\Q10"
3015db2f26eSSascha Wildner 	},
3025db2f26eSSascha Wildner 	{
3035db2f26eSSascha Wildner 		.name		= "L3H",
3045db2f26eSSascha Wildner 		.mled_set	= "MLED",
3055db2f26eSSascha Wildner 		.wled_set	= "WLED",
3065db2f26eSSascha Wildner 		.brn_get	= "GPLV",
3075db2f26eSSascha Wildner 		.brn_set	= "SPLV",
3085db2f26eSSascha Wildner 		.lcd_get	= "\\_SB.PCI0.PM.PBC",
3095db2f26eSSascha Wildner 		.lcd_set	= "EHK",
3105db2f26eSSascha Wildner 		.disp_get	= "\\_SB.INFB",
3115db2f26eSSascha Wildner 		.disp_set	= "SDSP"
3125db2f26eSSascha Wildner 	},
3135db2f26eSSascha Wildner 	{
3145db2f26eSSascha Wildner 		.name		= "L4R",
3155db2f26eSSascha Wildner 		.mled_set	= "MLED",
3165db2f26eSSascha Wildner 		.wled_set	= "WLED",
3175db2f26eSSascha Wildner 		.brn_get	= "GPLV",
3185db2f26eSSascha Wildner 		.brn_set	= "SPLV",
3195db2f26eSSascha Wildner 		.lcd_get	= "\\_SB.PCI0.SBSM.SEO4",
3205db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
3215db2f26eSSascha Wildner 		.disp_get	= "\\_SB.PCI0.P0P1.VGA.GETD",
3225db2f26eSSascha Wildner 		.disp_set	= "SDSP"
3235db2f26eSSascha Wildner 	},
3245db2f26eSSascha Wildner 	{
3255db2f26eSSascha Wildner 		.name		= "L5x",
3265db2f26eSSascha Wildner 		.mled_set	= "MLED",
3275db2f26eSSascha Wildner 		.tled_set	= "TLED",
3285db2f26eSSascha Wildner 		.lcd_get	= "\\BAOF",
3295db2f26eSSascha Wildner 		.lcd_set	= "\\Q0D",
3305db2f26eSSascha Wildner 		.brn_get	= "GPLV",
3315db2f26eSSascha Wildner 		.brn_set	= "SPLV",
3325db2f26eSSascha Wildner 		.disp_get	= "\\INFB",
3335db2f26eSSascha Wildner 		.disp_set	= "SDSP"
3345db2f26eSSascha Wildner 	},
3355db2f26eSSascha Wildner 	{
3365db2f26eSSascha Wildner 		.name		= "L8L"
3375db2f26eSSascha Wildner 		/* Only has hotkeys, apparently */
3385db2f26eSSascha Wildner 	},
3395db2f26eSSascha Wildner 	{
3405db2f26eSSascha Wildner 		.name		= "M1A",
3415db2f26eSSascha Wildner 		.mled_set	= "MLED",
3425db2f26eSSascha Wildner 		.brn_up		= "\\_SB.PCI0.PX40.EC0.Q0E",
3435db2f26eSSascha Wildner 		.brn_dn		= "\\_SB.PCI0.PX40.EC0.Q0F",
3445db2f26eSSascha Wildner 		.lcd_get	= "\\PNOF",
3455db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.PX40.EC0.Q10"
3465db2f26eSSascha Wildner 	},
3475db2f26eSSascha Wildner 	{
3485db2f26eSSascha Wildner 		.name		= "M2E",
3495db2f26eSSascha Wildner 		.mled_set	= "MLED",
3505db2f26eSSascha Wildner 		.wled_set	= "WLED",
3515db2f26eSSascha Wildner 		.brn_get	= "GPLV",
3525db2f26eSSascha Wildner 		.brn_set	= "SPLV",
3535db2f26eSSascha Wildner 		.lcd_get	= "\\GP06",
3545db2f26eSSascha Wildner 		.lcd_set	= "\\Q10"
3555db2f26eSSascha Wildner 	},
3565db2f26eSSascha Wildner 	{
3575db2f26eSSascha Wildner 		.name		= "M6N",
3585db2f26eSSascha Wildner 		.mled_set	= "MLED",
3595db2f26eSSascha Wildner 		.wled_set	= "WLED",
3605db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
3615db2f26eSSascha Wildner 		.lcd_get	= "\\_SB.BKLT",
3625db2f26eSSascha Wildner 		.brn_set	= "SPLV",
3635db2f26eSSascha Wildner 		.brn_get	= "GPLV",
3645db2f26eSSascha Wildner 		.disp_set	= "SDSP",
3655db2f26eSSascha Wildner 		.disp_get	= "\\SSTE"
3665db2f26eSSascha Wildner 	},
3675db2f26eSSascha Wildner 	{
3685db2f26eSSascha Wildner 		.name		= "M6R",
3695db2f26eSSascha Wildner 		.mled_set	= "MLED",
3705db2f26eSSascha Wildner 		.wled_set	= "WLED",
3715db2f26eSSascha Wildner 		.brn_get	= "GPLV",
3725db2f26eSSascha Wildner 		.brn_set	= "SPLV",
3735db2f26eSSascha Wildner 		.lcd_get	= "\\_SB.PCI0.SBSM.SEO4",
3745db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
3755db2f26eSSascha Wildner 		.disp_get	= "\\SSTE",
3765db2f26eSSascha Wildner 		.disp_set	= "SDSP"
3775db2f26eSSascha Wildner 	},
3785db2f26eSSascha Wildner 	{
3795db2f26eSSascha Wildner 		.name		= "S1x",
3805db2f26eSSascha Wildner 		.mled_set	= "MLED",
3815db2f26eSSascha Wildner 		.wled_set	= "WLED",
3825db2f26eSSascha Wildner 		.lcd_get	= "\\PNOF",
3835db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.PX40.Q10",
3845db2f26eSSascha Wildner 		.brn_get	= "GPLV",
3855db2f26eSSascha Wildner 		.brn_set	= "SPLV"
3865db2f26eSSascha Wildner 	},
3875db2f26eSSascha Wildner 	{
3885db2f26eSSascha Wildner 		.name		= "S2x",
3895db2f26eSSascha Wildner 		.mled_set	= "MLED",
3905db2f26eSSascha Wildner 		.lcd_get	= "\\BKLI",
3915db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.ISA.EC0._Q10",
3925db2f26eSSascha Wildner 		.brn_up		= "\\_SB.PCI0.ISA.EC0._Q0B",
3935db2f26eSSascha Wildner 		.brn_dn		= "\\_SB.PCI0.ISA.EC0._Q0A"
3945db2f26eSSascha Wildner 	},
3955db2f26eSSascha Wildner 	{
3965db2f26eSSascha Wildner 		.name		= "V6V",
3975db2f26eSSascha Wildner 		.bled_set	= "BLED",
3985db2f26eSSascha Wildner 		.tled_set	= "TLED",
3995db2f26eSSascha Wildner 		.wled_set	= "WLED",
4005db2f26eSSascha Wildner 		.lcd_get	= "\\BKLT",
4015db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
4025db2f26eSSascha Wildner 		.brn_get	= "GPLV",
4035db2f26eSSascha Wildner 		.brn_set	= "SPLV",
4045db2f26eSSascha Wildner 		.disp_get	= "\\_SB.PCI0.P0P1.VGA.GETD",
4055db2f26eSSascha Wildner 		.disp_set	= "SDSP"
4065db2f26eSSascha Wildner 	},
4075db2f26eSSascha Wildner 	{
4085db2f26eSSascha Wildner 		.name		= "W5A",
4095db2f26eSSascha Wildner 		.bled_set	= "BLED",
4105db2f26eSSascha Wildner 		.lcd_get	= "\\BKLT",
4115db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.SBRG.EC0._Q10",
4125db2f26eSSascha Wildner 		.brn_get	= "GPLV",
4135db2f26eSSascha Wildner 		.brn_set	= "SPLV",
4145db2f26eSSascha Wildner 		.disp_get	= "\\_SB.PCI0.P0P2.VGA.GETD",
4155db2f26eSSascha Wildner 		.disp_set	= "SDSP"
4165db2f26eSSascha Wildner 	},
4175db2f26eSSascha Wildner 
4185db2f26eSSascha Wildner 	{ .name = NULL }
4195db2f26eSSascha Wildner };
4205db2f26eSSascha Wildner 
4215db2f26eSSascha Wildner /*
4225db2f26eSSascha Wildner  * Samsung P30/P35 laptops have an Asus ATK0100 gadget interface,
4235db2f26eSSascha Wildner  * but they can't be probed quite the same way as Asus laptops.
4245db2f26eSSascha Wildner  */
4255db2f26eSSascha Wildner static struct acpi_asus_model acpi_samsung_models[] = {
4265db2f26eSSascha Wildner 	{
4275db2f26eSSascha Wildner 		.name		= "P30",
4285db2f26eSSascha Wildner 		.wled_set	= "WLED",
4295db2f26eSSascha Wildner 		.brn_up		= "\\_SB.PCI0.LPCB.EC0._Q68",
4305db2f26eSSascha Wildner 		.brn_dn		= "\\_SB.PCI0.LPCB.EC0._Q69",
4315db2f26eSSascha Wildner 		.lcd_get	= "\\BKLT",
4325db2f26eSSascha Wildner 		.lcd_set	= "\\_SB.PCI0.LPCB.EC0._Q0E"
4335db2f26eSSascha Wildner 	},
4345db2f26eSSascha Wildner 
4355db2f26eSSascha Wildner 	{ .name = NULL }
4365db2f26eSSascha Wildner };
4375db2f26eSSascha Wildner 
4385db2f26eSSascha Wildner static void	acpi_asus_eeepc_notify(ACPI_HANDLE h, UINT32 notify, void *context);
4395db2f26eSSascha Wildner 
4405db2f26eSSascha Wildner /*
4415db2f26eSSascha Wildner  * EeePC have an Asus ASUS010 gadget interface,
4425db2f26eSSascha Wildner  * but they can't be probed quite the same way as Asus laptops.
4435db2f26eSSascha Wildner  */
4445db2f26eSSascha Wildner static struct acpi_asus_model acpi_eeepc_models[] = {
4455db2f26eSSascha Wildner 	{
4465db2f26eSSascha Wildner 		.name		= "EEE",
4475db2f26eSSascha Wildner 		.brn_get	= "\\_SB.ATKD.PBLG",
4485db2f26eSSascha Wildner 		.brn_set	= "\\_SB.ATKD.PBLS",
4495db2f26eSSascha Wildner 		.cam_get	= "\\_SB.ATKD.CAMG",
4505db2f26eSSascha Wildner 		.cam_set	= "\\_SB.ATKD.CAMS",
4515db2f26eSSascha Wildner 		.crd_set	= "\\_SB.ATKD.CRDS",
4525db2f26eSSascha Wildner 		.crd_get	= "\\_SB.ATKD.CRDG",
4535db2f26eSSascha Wildner 		.wlan_get	= "\\_SB.ATKD.WLDG",
4545db2f26eSSascha Wildner 		.wlan_set	= "\\_SB.ATKD.WLDS",
4555db2f26eSSascha Wildner 		.n_func		= acpi_asus_eeepc_notify
4565db2f26eSSascha Wildner 	},
4575db2f26eSSascha Wildner 
4585db2f26eSSascha Wildner 	{ .name = NULL }
4595db2f26eSSascha Wildner };
4605db2f26eSSascha Wildner 
4615db2f26eSSascha Wildner static struct {
4625db2f26eSSascha Wildner 	char	*name;
4635db2f26eSSascha Wildner 	char	*description;
4645db2f26eSSascha Wildner 	int	method;
4655db2f26eSSascha Wildner 	int	flags;
4665db2f26eSSascha Wildner } acpi_asus_sysctls[] = {
4675db2f26eSSascha Wildner 	{
4685db2f26eSSascha Wildner 		.name		= "lcd_backlight",
4695db2f26eSSascha Wildner 		.method		= ACPI_ASUS_METHOD_LCD,
4705db2f26eSSascha Wildner 		.description	= "state of the lcd backlight",
4715db2f26eSSascha Wildner 		.flags 		= CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_ANYBODY
4725db2f26eSSascha Wildner 	},
4735db2f26eSSascha Wildner 	{
4745db2f26eSSascha Wildner 		.name		= "lcd_brightness",
4755db2f26eSSascha Wildner 		.method		= ACPI_ASUS_METHOD_BRN,
4765db2f26eSSascha Wildner 		.description	= "brightness of the lcd panel",
4775db2f26eSSascha Wildner 		.flags 		= CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_ANYBODY
4785db2f26eSSascha Wildner 	},
4795db2f26eSSascha Wildner 	{
4805db2f26eSSascha Wildner 		.name		= "video_output",
4815db2f26eSSascha Wildner 		.method		= ACPI_ASUS_METHOD_DISP,
4825db2f26eSSascha Wildner 		.description	= "display output state",
4835db2f26eSSascha Wildner 		.flags 		= CTLTYPE_INT | CTLFLAG_RW
4845db2f26eSSascha Wildner 	},
4855db2f26eSSascha Wildner 	{
4865db2f26eSSascha Wildner 		.name		= "camera",
4875db2f26eSSascha Wildner 		.method		= ACPI_ASUS_METHOD_CAMERA,
4885db2f26eSSascha Wildner 		.description	= "internal camera state",
4895db2f26eSSascha Wildner 		.flags 		= CTLTYPE_INT | CTLFLAG_RW
4905db2f26eSSascha Wildner 	},
4915db2f26eSSascha Wildner 	{
4925db2f26eSSascha Wildner 		.name		= "cardreader",
4935db2f26eSSascha Wildner 		.method		= ACPI_ASUS_METHOD_CARDRD,
4945db2f26eSSascha Wildner 		.description	= "internal card reader state",
4955db2f26eSSascha Wildner 		.flags 		= CTLTYPE_INT | CTLFLAG_RW
4965db2f26eSSascha Wildner 	},
4975db2f26eSSascha Wildner 	{
4985db2f26eSSascha Wildner 		.name		= "wlan",
4995db2f26eSSascha Wildner 		.method		= ACPI_ASUS_METHOD_WLAN,
5005db2f26eSSascha Wildner 		.description	= "wireless lan state",
5015db2f26eSSascha Wildner 		.flags		= CTLTYPE_INT | CTLFLAG_RW
5025db2f26eSSascha Wildner 	},
5035db2f26eSSascha Wildner 
5045db2f26eSSascha Wildner 	{ .name = NULL }
5055db2f26eSSascha Wildner };
5065db2f26eSSascha Wildner 
5075db2f26eSSascha Wildner ACPI_SERIAL_DECL(asus, "ACPI ASUS extras");
5085db2f26eSSascha Wildner 
5095db2f26eSSascha Wildner /* Function prototypes */
5105db2f26eSSascha Wildner static int	acpi_asus_probe(device_t dev);
5115db2f26eSSascha Wildner static int	acpi_asus_attach(device_t dev);
5125db2f26eSSascha Wildner static int	acpi_asus_detach(device_t dev);
5135db2f26eSSascha Wildner 
5145db2f26eSSascha Wildner static void	acpi_asus_led(struct acpi_asus_led *led, int state);
5155db2f26eSSascha Wildner static void	acpi_asus_led_task(struct acpi_asus_led *led, int pending __unused);
5165db2f26eSSascha Wildner 
5175db2f26eSSascha Wildner static int	acpi_asus_sysctl(SYSCTL_HANDLER_ARGS);
5185db2f26eSSascha Wildner static int	acpi_asus_sysctl_init(struct acpi_asus_softc *sc, int method);
5195db2f26eSSascha Wildner static int	acpi_asus_sysctl_get(struct acpi_asus_softc *sc, int method);
5205db2f26eSSascha Wildner static int	acpi_asus_sysctl_set(struct acpi_asus_softc *sc, int method, int val);
5215db2f26eSSascha Wildner 
5225db2f26eSSascha Wildner static void	acpi_asus_notify(ACPI_HANDLE h, UINT32 notify, void *context);
5235db2f26eSSascha Wildner 
5245db2f26eSSascha Wildner static device_method_t acpi_asus_methods[] = {
5255db2f26eSSascha Wildner 	DEVMETHOD(device_probe,  acpi_asus_probe),
5265db2f26eSSascha Wildner 	DEVMETHOD(device_attach, acpi_asus_attach),
5275db2f26eSSascha Wildner 	DEVMETHOD(device_detach, acpi_asus_detach),
5285db2f26eSSascha Wildner 
529d3c9c58eSSascha Wildner 	DEVMETHOD_END
5305db2f26eSSascha Wildner };
5315db2f26eSSascha Wildner 
5325db2f26eSSascha Wildner static driver_t acpi_asus_driver = {
5335db2f26eSSascha Wildner 	"acpi_asus",
5345db2f26eSSascha Wildner 	acpi_asus_methods,
5355db2f26eSSascha Wildner 	sizeof(struct acpi_asus_softc)
5365db2f26eSSascha Wildner };
5375db2f26eSSascha Wildner 
5385db2f26eSSascha Wildner static devclass_t acpi_asus_devclass;
5395db2f26eSSascha Wildner 
5405db2f26eSSascha Wildner DRIVER_MODULE(acpi_asus, acpi, acpi_asus_driver, acpi_asus_devclass, NULL, NULL);
5415db2f26eSSascha Wildner MODULE_DEPEND(acpi_asus, acpi, 1, 1, 1);
542*f2c400e8SImre Vadász MODULE_DEPEND(acpi_asus, led, 1, 1, 1);
5435db2f26eSSascha Wildner 
5445db2f26eSSascha Wildner static int
acpi_asus_probe(device_t dev)5455db2f26eSSascha Wildner acpi_asus_probe(device_t dev)
5465db2f26eSSascha Wildner {
5475db2f26eSSascha Wildner 	struct acpi_asus_model	*model;
5485db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
5495db2f26eSSascha Wildner 	struct sbuf		*sb;
5505db2f26eSSascha Wildner 	ACPI_BUFFER		Buf;
5515db2f26eSSascha Wildner 	ACPI_OBJECT		Arg, *Obj;
5525db2f26eSSascha Wildner 	ACPI_OBJECT_LIST	Args;
5535db2f26eSSascha Wildner 	static char		*asus_ids[] = { "ATK0100", "ASUS010", NULL };
5545db2f26eSSascha Wildner 	char *rstr;
5555db2f26eSSascha Wildner 
5565db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
5575db2f26eSSascha Wildner 
5585db2f26eSSascha Wildner 	if (acpi_disabled("asus"))
5595db2f26eSSascha Wildner 		return (ENXIO);
5605db2f26eSSascha Wildner 	ACPI_SERIAL_INIT(asus);
5615db2f26eSSascha Wildner 	rstr = ACPI_ID_PROBE(device_get_parent(dev), dev, asus_ids);
5625db2f26eSSascha Wildner 	if (rstr == NULL) {
5635db2f26eSSascha Wildner 		return (ENXIO);
5645db2f26eSSascha Wildner 	}
5655db2f26eSSascha Wildner 
5665db2f26eSSascha Wildner 	sc = device_get_softc(dev);
5675db2f26eSSascha Wildner 	sc->dev = dev;
5685db2f26eSSascha Wildner 	sc->handle = acpi_get_handle(dev);
5695db2f26eSSascha Wildner 
5705db2f26eSSascha Wildner 	Arg.Type = ACPI_TYPE_INTEGER;
5715db2f26eSSascha Wildner 	Arg.Integer.Value = 0;
5725db2f26eSSascha Wildner 
5735db2f26eSSascha Wildner 	Args.Count = 1;
5745db2f26eSSascha Wildner 	Args.Pointer = &Arg;
5755db2f26eSSascha Wildner 
5765db2f26eSSascha Wildner 	Buf.Pointer = NULL;
5775db2f26eSSascha Wildner 	Buf.Length = ACPI_ALLOCATE_BUFFER;
5785db2f26eSSascha Wildner 
5795db2f26eSSascha Wildner 	AcpiEvaluateObject(sc->handle, "INIT", &Args, &Buf);
5805db2f26eSSascha Wildner 	Obj = Buf.Pointer;
5815db2f26eSSascha Wildner 
5825db2f26eSSascha Wildner 	/*
5835db2f26eSSascha Wildner 	 * The Samsung P30 returns a null-pointer from INIT, we
5845db2f26eSSascha Wildner 	 * can identify it from the 'ODEM' string in the DSDT.
5855db2f26eSSascha Wildner 	 */
5865db2f26eSSascha Wildner 	if (Obj->String.Pointer == NULL) {
5875db2f26eSSascha Wildner 		ACPI_STATUS		status;
5885db2f26eSSascha Wildner 		ACPI_TABLE_HEADER	th;
5895db2f26eSSascha Wildner 
5905db2f26eSSascha Wildner 		status = AcpiGetTableHeader(ACPI_SIG_DSDT, 0, &th);
5915db2f26eSSascha Wildner 		if (ACPI_FAILURE(status)) {
5925db2f26eSSascha Wildner 			device_printf(dev, "Unsupported (Samsung?) laptop\n");
5935db2f26eSSascha Wildner 			AcpiOsFree(Buf.Pointer);
5945db2f26eSSascha Wildner 			return (ENXIO);
5955db2f26eSSascha Wildner 		}
5965db2f26eSSascha Wildner 
5975db2f26eSSascha Wildner 		if (strncmp("ODEM", th.OemTableId, 4) == 0) {
5985db2f26eSSascha Wildner 			sc->model = &acpi_samsung_models[0];
5995db2f26eSSascha Wildner 			device_set_desc(dev, "Samsung P30 Laptop Extras");
6005db2f26eSSascha Wildner 			AcpiOsFree(Buf.Pointer);
6015db2f26eSSascha Wildner 			return (0);
6025db2f26eSSascha Wildner 		}
6035db2f26eSSascha Wildner 
6045db2f26eSSascha Wildner 		/* EeePC */
6055db2f26eSSascha Wildner 		if (strncmp("ASUS010", rstr, 7) == 0) {
6065db2f26eSSascha Wildner 			sc->model = &acpi_eeepc_models[0];
6075db2f26eSSascha Wildner 			device_set_desc(dev, "ASUS EeePC");
6085db2f26eSSascha Wildner 			AcpiOsFree(Buf.Pointer);
6095db2f26eSSascha Wildner 			return (0);
6105db2f26eSSascha Wildner 		}
6115db2f26eSSascha Wildner 	}
6125db2f26eSSascha Wildner 
6135db2f26eSSascha Wildner 	sb = sbuf_new_auto();
6145db2f26eSSascha Wildner 	if (sb == NULL)
6155db2f26eSSascha Wildner 		return (ENOMEM);
6165db2f26eSSascha Wildner 
6175db2f26eSSascha Wildner 	/*
6185db2f26eSSascha Wildner 	 * Asus laptops are simply identified by name, easy!
6195db2f26eSSascha Wildner 	 */
6205db2f26eSSascha Wildner 	for (model = acpi_asus_models; model->name != NULL; model++) {
6215db2f26eSSascha Wildner 		if (strncmp(Obj->String.Pointer, model->name, 3) == 0) {
6225db2f26eSSascha Wildner 
6235db2f26eSSascha Wildner good:
6245db2f26eSSascha Wildner 			sbuf_printf(sb, "Asus %s Laptop Extras",
6255db2f26eSSascha Wildner 			    Obj->String.Pointer);
6265db2f26eSSascha Wildner 			sbuf_finish(sb);
6275db2f26eSSascha Wildner 
6285db2f26eSSascha Wildner 			sc->model = model;
6295db2f26eSSascha Wildner 			device_set_desc_copy(dev, sbuf_data(sb));
6305db2f26eSSascha Wildner 
6315db2f26eSSascha Wildner 			sbuf_delete(sb);
6325db2f26eSSascha Wildner 			AcpiOsFree(Buf.Pointer);
6335db2f26eSSascha Wildner 			return (0);
6345db2f26eSSascha Wildner 		}
6355db2f26eSSascha Wildner 
6365db2f26eSSascha Wildner 		/*
6375db2f26eSSascha Wildner 		 * Some models look exactly the same as other models, but have
6385db2f26eSSascha Wildner 		 * their own ids.  If we spot these, set them up with the same
6395db2f26eSSascha Wildner 		 * details as the models they're like, possibly dealing with
6405db2f26eSSascha Wildner 		 * small differences.
6415db2f26eSSascha Wildner 		 *
6425db2f26eSSascha Wildner 		 * XXX: there must be a prettier way to do this!
6435db2f26eSSascha Wildner 		 */
6445db2f26eSSascha Wildner 		else if (strncmp(model->name, "xxN", 3) == 0 &&
6455db2f26eSSascha Wildner 		    (strncmp(Obj->String.Pointer, "M3N", 3) == 0 ||
6465db2f26eSSascha Wildner 		     strncmp(Obj->String.Pointer, "S1N", 3) == 0))
6475db2f26eSSascha Wildner 			goto good;
6485db2f26eSSascha Wildner 		else if (strncmp(model->name, "A1x", 3) == 0 &&
6495db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "A1", 2) == 0)
6505db2f26eSSascha Wildner 			goto good;
6515db2f26eSSascha Wildner 		else if (strncmp(model->name, "A2x", 3) == 0 &&
6525db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "A2", 2) == 0)
6535db2f26eSSascha Wildner 			goto good;
6545db2f26eSSascha Wildner 		else if (strncmp(model->name, "A3F", 3) == 0 &&
6555db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "A6F", 3) == 0)
6565db2f26eSSascha Wildner 			goto good;
6575db2f26eSSascha Wildner 		else if (strncmp(model->name, "D1x", 3) == 0 &&
6585db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "D1", 2) == 0)
6595db2f26eSSascha Wildner 			goto good;
6605db2f26eSSascha Wildner 		else if (strncmp(model->name, "L3H", 3) == 0 &&
6615db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "L2E", 3) == 0)
6625db2f26eSSascha Wildner 			goto good;
6635db2f26eSSascha Wildner 		else if (strncmp(model->name, "L5x", 3) == 0 &&
6645db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "L5", 2) == 0)
6655db2f26eSSascha Wildner 			goto good;
6665db2f26eSSascha Wildner 		else if (strncmp(model->name, "M2E", 3) == 0 &&
6675db2f26eSSascha Wildner 		    (strncmp(Obj->String.Pointer, "M2", 2) == 0 ||
6685db2f26eSSascha Wildner 		     strncmp(Obj->String.Pointer, "L4E", 3) == 0))
6695db2f26eSSascha Wildner 			goto good;
6705db2f26eSSascha Wildner 		else if (strncmp(model->name, "S1x", 3) == 0 &&
6715db2f26eSSascha Wildner 		    (strncmp(Obj->String.Pointer, "L8", 2) == 0 ||
6725db2f26eSSascha Wildner 		     strncmp(Obj->String.Pointer, "S1", 2) == 0))
6735db2f26eSSascha Wildner 			goto good;
6745db2f26eSSascha Wildner 		else if (strncmp(model->name, "S2x", 3) == 0 &&
6755db2f26eSSascha Wildner 		    (strncmp(Obj->String.Pointer, "J1", 2) == 0 ||
6765db2f26eSSascha Wildner 		     strncmp(Obj->String.Pointer, "S2", 2) == 0))
6775db2f26eSSascha Wildner 			goto good;
6785db2f26eSSascha Wildner 
6795db2f26eSSascha Wildner 		/* L2B is like L3C but has no lcd_get method */
6805db2f26eSSascha Wildner 		else if (strncmp(model->name, "L3C", 3) == 0 &&
6815db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "L2B", 3) == 0) {
6825db2f26eSSascha Wildner 			model->lcd_get = NULL;
6835db2f26eSSascha Wildner 			goto good;
6845db2f26eSSascha Wildner 		}
6855db2f26eSSascha Wildner 
6865db2f26eSSascha Wildner 		/* A3G is like M6R but with a different lcd_get method */
6875db2f26eSSascha Wildner 		else if (strncmp(model->name, "M6R", 3) == 0 &&
6885db2f26eSSascha Wildner 		    strncmp(Obj->String.Pointer, "A3G", 3) == 0) {
6895db2f26eSSascha Wildner 			model->lcd_get = "\\BLFG";
6905db2f26eSSascha Wildner 			goto good;
6915db2f26eSSascha Wildner 		}
6925db2f26eSSascha Wildner 
6935db2f26eSSascha Wildner 		/* M2N and W1N are like xxN with added WLED */
6945db2f26eSSascha Wildner 		else if (strncmp(model->name, "xxN", 3) == 0 &&
6955db2f26eSSascha Wildner 		    (strncmp(Obj->String.Pointer, "M2N", 3) == 0 ||
6965db2f26eSSascha Wildner 		     strncmp(Obj->String.Pointer, "W1N", 3) == 0)) {
6975db2f26eSSascha Wildner 			model->wled_set = "WLED";
6985db2f26eSSascha Wildner 			goto good;
6995db2f26eSSascha Wildner 		}
7005db2f26eSSascha Wildner 
7015db2f26eSSascha Wildner 		/* M5N and S5N are like xxN without MLED */
7025db2f26eSSascha Wildner 		else if (strncmp(model->name, "xxN", 3) == 0 &&
7035db2f26eSSascha Wildner 		    (strncmp(Obj->String.Pointer, "M5N", 3) == 0 ||
7045db2f26eSSascha Wildner 		     strncmp(Obj->String.Pointer, "S5N", 3) == 0)) {
7055db2f26eSSascha Wildner 			model->mled_set = NULL;
7065db2f26eSSascha Wildner 			goto good;
7075db2f26eSSascha Wildner 		}
7085db2f26eSSascha Wildner 	}
7095db2f26eSSascha Wildner 
7105db2f26eSSascha Wildner 	sbuf_printf(sb, "Unsupported Asus laptop: %s\n", Obj->String.Pointer);
7115db2f26eSSascha Wildner 	sbuf_finish(sb);
7125db2f26eSSascha Wildner 
7135db2f26eSSascha Wildner 	device_printf(dev, "%s", sbuf_data(sb));
7145db2f26eSSascha Wildner 
7155db2f26eSSascha Wildner 	sbuf_delete(sb);
7165db2f26eSSascha Wildner 	AcpiOsFree(Buf.Pointer);
7175db2f26eSSascha Wildner 
7185db2f26eSSascha Wildner 	return (ENXIO);
7195db2f26eSSascha Wildner }
7205db2f26eSSascha Wildner 
7215db2f26eSSascha Wildner static int
acpi_asus_attach(device_t dev)7225db2f26eSSascha Wildner acpi_asus_attach(device_t dev)
7235db2f26eSSascha Wildner {
7245db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
7255db2f26eSSascha Wildner 	struct acpi_softc	*acpi_sc;
7265db2f26eSSascha Wildner 
7275db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
7285db2f26eSSascha Wildner 
7295db2f26eSSascha Wildner 	sc = device_get_softc(dev);
7305db2f26eSSascha Wildner 	acpi_sc = acpi_device_get_parent_softc(dev);
7315db2f26eSSascha Wildner 
7325db2f26eSSascha Wildner 	/* Build sysctl tree */
7335db2f26eSSascha Wildner 	sysctl_ctx_init(&sc->sysctl_ctx);
7345db2f26eSSascha Wildner 	sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx,
7355db2f26eSSascha Wildner 	    SYSCTL_CHILDREN(acpi_sc->acpi_sysctl_tree),
7365db2f26eSSascha Wildner 	    OID_AUTO, "asus", CTLFLAG_RD, 0, "");
7375db2f26eSSascha Wildner 
7385db2f26eSSascha Wildner 	/* Hook up nodes */
7395db2f26eSSascha Wildner 	for (int i = 0; acpi_asus_sysctls[i].name != NULL; i++) {
7405db2f26eSSascha Wildner 		if (!acpi_asus_sysctl_init(sc, acpi_asus_sysctls[i].method))
7415db2f26eSSascha Wildner 			continue;
7425db2f26eSSascha Wildner 
7435db2f26eSSascha Wildner 		SYSCTL_ADD_PROC(&sc->sysctl_ctx,
7445db2f26eSSascha Wildner 		    SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO,
7455db2f26eSSascha Wildner 		    acpi_asus_sysctls[i].name,
7465db2f26eSSascha Wildner 		    acpi_asus_sysctls[i].flags,
7475db2f26eSSascha Wildner 		    sc, i, acpi_asus_sysctl, "I",
7485db2f26eSSascha Wildner 		    acpi_asus_sysctls[i].description);
7495db2f26eSSascha Wildner 	}
7505db2f26eSSascha Wildner 
7515db2f26eSSascha Wildner 	/* Attach leds */
7525db2f26eSSascha Wildner 	if (sc->model->bled_set) {
7535db2f26eSSascha Wildner 		sc->s_bled.busy = 0;
7545db2f26eSSascha Wildner 		sc->s_bled.sc = sc;
7555db2f26eSSascha Wildner 		sc->s_bled.type = ACPI_ASUS_LED_BLED;
7565db2f26eSSascha Wildner 		sc->s_bled.cdev =
7575db2f26eSSascha Wildner 		    led_create_state((led_t *)acpi_asus_led, &sc->s_bled,
7585db2f26eSSascha Wildner 			"bled", 1);
7595db2f26eSSascha Wildner 	}
7605db2f26eSSascha Wildner 
7615db2f26eSSascha Wildner 	if (sc->model->dled_set) {
7625db2f26eSSascha Wildner 		sc->s_dled.busy = 0;
7635db2f26eSSascha Wildner 		sc->s_dled.sc = sc;
7645db2f26eSSascha Wildner 		sc->s_dled.type = ACPI_ASUS_LED_DLED;
7655db2f26eSSascha Wildner 		sc->s_dled.cdev =
7665db2f26eSSascha Wildner 		    led_create((led_t *)acpi_asus_led, &sc->s_dled, "dled");
7675db2f26eSSascha Wildner 	}
7685db2f26eSSascha Wildner 
7695db2f26eSSascha Wildner 	if (sc->model->gled_set) {
7705db2f26eSSascha Wildner 		sc->s_gled.busy = 0;
7715db2f26eSSascha Wildner 		sc->s_gled.sc = sc;
7725db2f26eSSascha Wildner 		sc->s_gled.type = ACPI_ASUS_LED_GLED;
7735db2f26eSSascha Wildner 		sc->s_gled.cdev =
7745db2f26eSSascha Wildner 		    led_create((led_t *)acpi_asus_led, &sc->s_gled, "gled");
7755db2f26eSSascha Wildner 	}
7765db2f26eSSascha Wildner 
7775db2f26eSSascha Wildner 	if (sc->model->mled_set) {
7785db2f26eSSascha Wildner 		sc->s_mled.busy = 0;
7795db2f26eSSascha Wildner 		sc->s_mled.sc = sc;
7805db2f26eSSascha Wildner 		sc->s_mled.type = ACPI_ASUS_LED_MLED;
7815db2f26eSSascha Wildner 		sc->s_mled.cdev =
7825db2f26eSSascha Wildner 		    led_create((led_t *)acpi_asus_led, &sc->s_mled, "mled");
7835db2f26eSSascha Wildner 	}
7845db2f26eSSascha Wildner 
7855db2f26eSSascha Wildner 	if (sc->model->tled_set) {
7865db2f26eSSascha Wildner 		sc->s_tled.busy = 0;
7875db2f26eSSascha Wildner 		sc->s_tled.sc = sc;
7885db2f26eSSascha Wildner 		sc->s_tled.type = ACPI_ASUS_LED_TLED;
7895db2f26eSSascha Wildner 		sc->s_tled.cdev =
7905db2f26eSSascha Wildner 		    led_create_state((led_t *)acpi_asus_led, &sc->s_tled,
7915db2f26eSSascha Wildner 			"tled", 1);
7925db2f26eSSascha Wildner 	}
7935db2f26eSSascha Wildner 
7945db2f26eSSascha Wildner 	if (sc->model->wled_set) {
7955db2f26eSSascha Wildner 		sc->s_wled.busy = 0;
7965db2f26eSSascha Wildner 		sc->s_wled.sc = sc;
7975db2f26eSSascha Wildner 		sc->s_wled.type = ACPI_ASUS_LED_WLED;
7985db2f26eSSascha Wildner 		sc->s_wled.cdev =
7995db2f26eSSascha Wildner 		    led_create_state((led_t *)acpi_asus_led, &sc->s_wled,
8005db2f26eSSascha Wildner 			"wled", 1);
8015db2f26eSSascha Wildner 	}
8025db2f26eSSascha Wildner 
8035db2f26eSSascha Wildner 	/* Activate hotkeys */
8045db2f26eSSascha Wildner 	AcpiEvaluateObject(sc->handle, "BSTS", NULL, NULL);
8055db2f26eSSascha Wildner 
8065db2f26eSSascha Wildner 	/* Handle notifies */
8075db2f26eSSascha Wildner 	if (sc->model->n_func == NULL)
8085db2f26eSSascha Wildner 		sc->model->n_func = acpi_asus_notify;
8095db2f26eSSascha Wildner 
8105db2f26eSSascha Wildner 	AcpiInstallNotifyHandler(sc->handle, ACPI_SYSTEM_NOTIFY,
8115db2f26eSSascha Wildner 	    sc->model->n_func, dev);
8125db2f26eSSascha Wildner 
8135db2f26eSSascha Wildner 	/* Find and hook the 'LCDD' object */
8145db2f26eSSascha Wildner 	if (sc->model->lcdd != NULL && sc->model->lcdd_n_func != NULL) {
8155db2f26eSSascha Wildner 		ACPI_STATUS res;
8165db2f26eSSascha Wildner 
8175db2f26eSSascha Wildner 		sc->lcdd_handle = NULL;
8185db2f26eSSascha Wildner 		res = AcpiGetHandle((sc->model->lcdd[0] == '\\' ?
8195db2f26eSSascha Wildner 		    NULL : sc->handle), sc->model->lcdd, &(sc->lcdd_handle));
8205db2f26eSSascha Wildner 		if (ACPI_SUCCESS(res)) {
8215db2f26eSSascha Wildner 			AcpiInstallNotifyHandler((sc->lcdd_handle),
8225db2f26eSSascha Wildner 			    ACPI_DEVICE_NOTIFY, sc->model->lcdd_n_func, dev);
8235db2f26eSSascha Wildner 	    	} else {
8245db2f26eSSascha Wildner 	    		kprintf("%s: unable to find LCD device '%s'\n",
8255db2f26eSSascha Wildner 	    		    __func__, sc->model->lcdd);
8265db2f26eSSascha Wildner 	    	}
8275db2f26eSSascha Wildner 	}
8285db2f26eSSascha Wildner 
8295db2f26eSSascha Wildner 	return (0);
8305db2f26eSSascha Wildner }
8315db2f26eSSascha Wildner 
8325db2f26eSSascha Wildner static int
acpi_asus_detach(device_t dev)8335db2f26eSSascha Wildner acpi_asus_detach(device_t dev)
8345db2f26eSSascha Wildner {
8355db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
8365db2f26eSSascha Wildner 
8375db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
8385db2f26eSSascha Wildner 
8395db2f26eSSascha Wildner 	sc = device_get_softc(dev);
8405db2f26eSSascha Wildner 	/* Turn the lights off */
8415db2f26eSSascha Wildner 	if (sc->model->bled_set)
8425db2f26eSSascha Wildner 		led_destroy(sc->s_bled.cdev);
8435db2f26eSSascha Wildner 
8445db2f26eSSascha Wildner 	if (sc->model->dled_set)
8455db2f26eSSascha Wildner 		led_destroy(sc->s_dled.cdev);
8465db2f26eSSascha Wildner 
8475db2f26eSSascha Wildner 	if (sc->model->gled_set)
8485db2f26eSSascha Wildner 		led_destroy(sc->s_gled.cdev);
8495db2f26eSSascha Wildner 
8505db2f26eSSascha Wildner 	if (sc->model->mled_set)
8515db2f26eSSascha Wildner 		led_destroy(sc->s_mled.cdev);
8525db2f26eSSascha Wildner 
8535db2f26eSSascha Wildner 	if (sc->model->tled_set)
8545db2f26eSSascha Wildner 		led_destroy(sc->s_tled.cdev);
8555db2f26eSSascha Wildner 
8565db2f26eSSascha Wildner 	if (sc->model->wled_set)
8575db2f26eSSascha Wildner 		led_destroy(sc->s_wled.cdev);
8585db2f26eSSascha Wildner 
8595db2f26eSSascha Wildner 	/* Remove notify handler */
8605db2f26eSSascha Wildner 	AcpiRemoveNotifyHandler(sc->handle, ACPI_SYSTEM_NOTIFY,
8615db2f26eSSascha Wildner 	    acpi_asus_notify);
8625db2f26eSSascha Wildner 
8635db2f26eSSascha Wildner 	if (sc->lcdd_handle) {
8645db2f26eSSascha Wildner 		KASSERT(sc->model->lcdd_n_func != NULL,
8655db2f26eSSascha Wildner 		    ("model->lcdd_n_func is NULL, but lcdd_handle is non-zero"));
8665db2f26eSSascha Wildner 		AcpiRemoveNotifyHandler((sc->lcdd_handle),
8675db2f26eSSascha Wildner 		    ACPI_DEVICE_NOTIFY, sc->model->lcdd_n_func);
8685db2f26eSSascha Wildner 	}
8695db2f26eSSascha Wildner 
8705db2f26eSSascha Wildner 	/* Free sysctl tree */
8715db2f26eSSascha Wildner 	sysctl_ctx_free(&sc->sysctl_ctx);
8725db2f26eSSascha Wildner 
8735db2f26eSSascha Wildner 	return (0);
8745db2f26eSSascha Wildner }
8755db2f26eSSascha Wildner 
8765db2f26eSSascha Wildner static void
acpi_asus_led_task(struct acpi_asus_led * led,int pending __unused)8775db2f26eSSascha Wildner acpi_asus_led_task(struct acpi_asus_led *led, int pending __unused)
8785db2f26eSSascha Wildner {
8795db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
8805db2f26eSSascha Wildner 	char			*method;
8815db2f26eSSascha Wildner 	int			state;
8825db2f26eSSascha Wildner 
8835db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
8845db2f26eSSascha Wildner 
8855db2f26eSSascha Wildner 	sc = led->sc;
8865db2f26eSSascha Wildner 
8875db2f26eSSascha Wildner 	switch (led->type) {
8885db2f26eSSascha Wildner 	case ACPI_ASUS_LED_BLED:
8895db2f26eSSascha Wildner 		method = sc->model->bled_set;
8905db2f26eSSascha Wildner 		state = led->state;
8915db2f26eSSascha Wildner 		break;
8925db2f26eSSascha Wildner 	case ACPI_ASUS_LED_DLED:
8935db2f26eSSascha Wildner 		method = sc->model->dled_set;
8945db2f26eSSascha Wildner 		state = led->state;
8955db2f26eSSascha Wildner 		break;
8965db2f26eSSascha Wildner 	case ACPI_ASUS_LED_GLED:
8975db2f26eSSascha Wildner 		method = sc->model->gled_set;
8985db2f26eSSascha Wildner 		state = led->state + 1;	/* 1: off, 2: on */
8995db2f26eSSascha Wildner 		break;
9005db2f26eSSascha Wildner 	case ACPI_ASUS_LED_MLED:
9015db2f26eSSascha Wildner 		method = sc->model->mled_set;
9025db2f26eSSascha Wildner 		state = !led->state;	/* inverted */
9035db2f26eSSascha Wildner 		break;
9045db2f26eSSascha Wildner 	case ACPI_ASUS_LED_TLED:
9055db2f26eSSascha Wildner 		method = sc->model->tled_set;
9065db2f26eSSascha Wildner 		state = led->state;
9075db2f26eSSascha Wildner 		break;
9085db2f26eSSascha Wildner 	case ACPI_ASUS_LED_WLED:
9095db2f26eSSascha Wildner 		method = sc->model->wled_set;
9105db2f26eSSascha Wildner 		state = led->state;
9115db2f26eSSascha Wildner 		break;
9125db2f26eSSascha Wildner 	default:
9135db2f26eSSascha Wildner 		kprintf("acpi_asus_led: invalid LED type %d\n",
9145db2f26eSSascha Wildner 		    (int)led->type);
9155db2f26eSSascha Wildner 		return;
9165db2f26eSSascha Wildner 	}
9175db2f26eSSascha Wildner 
9185db2f26eSSascha Wildner 	acpi_SetInteger(sc->handle, method, state);
9195db2f26eSSascha Wildner 	led->busy = 0;
9205db2f26eSSascha Wildner }
9215db2f26eSSascha Wildner 
9225db2f26eSSascha Wildner static void
acpi_asus_led(struct acpi_asus_led * led,int state)9235db2f26eSSascha Wildner acpi_asus_led(struct acpi_asus_led *led, int state)
9245db2f26eSSascha Wildner {
9255db2f26eSSascha Wildner 
9265db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
9275db2f26eSSascha Wildner 
9285db2f26eSSascha Wildner 	if (led->busy)
9295db2f26eSSascha Wildner 		return;
9305db2f26eSSascha Wildner 
9315db2f26eSSascha Wildner 	led->busy = 1;
9325db2f26eSSascha Wildner 	led->state = state;
9335db2f26eSSascha Wildner 
9345db2f26eSSascha Wildner 	AcpiOsExecute(OSL_NOTIFY_HANDLER, (void *)acpi_asus_led_task, led);
9355db2f26eSSascha Wildner }
9365db2f26eSSascha Wildner 
9375db2f26eSSascha Wildner static int
acpi_asus_sysctl(SYSCTL_HANDLER_ARGS)9385db2f26eSSascha Wildner acpi_asus_sysctl(SYSCTL_HANDLER_ARGS)
9395db2f26eSSascha Wildner {
9405db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
9415db2f26eSSascha Wildner 	int			arg;
9425db2f26eSSascha Wildner 	int			error = 0;
9435db2f26eSSascha Wildner 	int			function;
9445db2f26eSSascha Wildner 	int			method;
9455db2f26eSSascha Wildner 
9465db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
9475db2f26eSSascha Wildner 
9485db2f26eSSascha Wildner 	sc = (struct acpi_asus_softc *)oidp->oid_arg1;
9495db2f26eSSascha Wildner 	function = oidp->oid_arg2;
9505db2f26eSSascha Wildner 	method = acpi_asus_sysctls[function].method;
9515db2f26eSSascha Wildner 
9525db2f26eSSascha Wildner 	ACPI_SERIAL_BEGIN(asus);
9535db2f26eSSascha Wildner 	arg = acpi_asus_sysctl_get(sc, method);
9545db2f26eSSascha Wildner 	error = sysctl_handle_int(oidp, &arg, 0, req);
9555db2f26eSSascha Wildner 
9565db2f26eSSascha Wildner 	/* Sanity check */
9575db2f26eSSascha Wildner 	if (error != 0 || req->newptr == NULL)
9585db2f26eSSascha Wildner 		goto out;
9595db2f26eSSascha Wildner 
9605db2f26eSSascha Wildner 	/* Update */
9615db2f26eSSascha Wildner 	error = acpi_asus_sysctl_set(sc, method, arg);
9625db2f26eSSascha Wildner 
9635db2f26eSSascha Wildner out:
9645db2f26eSSascha Wildner 	ACPI_SERIAL_END(asus);
9655db2f26eSSascha Wildner 	return (error);
9665db2f26eSSascha Wildner }
9675db2f26eSSascha Wildner 
9685db2f26eSSascha Wildner static int
acpi_asus_sysctl_get(struct acpi_asus_softc * sc,int method)9695db2f26eSSascha Wildner acpi_asus_sysctl_get(struct acpi_asus_softc *sc, int method)
9705db2f26eSSascha Wildner {
9715db2f26eSSascha Wildner 	int val = 0;
9725db2f26eSSascha Wildner 
9735db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
9745db2f26eSSascha Wildner 	ACPI_SERIAL_ASSERT(asus);
9755db2f26eSSascha Wildner 
9765db2f26eSSascha Wildner 	switch (method) {
9775db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_BRN:
9785db2f26eSSascha Wildner 		val = sc->s_brn;
9795db2f26eSSascha Wildner 		break;
9805db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_DISP:
9815db2f26eSSascha Wildner 		val = sc->s_disp;
9825db2f26eSSascha Wildner 		break;
9835db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_LCD:
9845db2f26eSSascha Wildner 		val = sc->s_lcd;
9855db2f26eSSascha Wildner 		break;
9865db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_CAMERA:
9875db2f26eSSascha Wildner 		val = sc->s_cam;
9885db2f26eSSascha Wildner 		break;
9895db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_CARDRD:
9905db2f26eSSascha Wildner 		val = sc->s_crd;
9915db2f26eSSascha Wildner 		break;
9925db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_WLAN:
9935db2f26eSSascha Wildner 		val = sc->s_wlan;
9945db2f26eSSascha Wildner 		break;
9955db2f26eSSascha Wildner 	}
9965db2f26eSSascha Wildner 
9975db2f26eSSascha Wildner 	return (val);
9985db2f26eSSascha Wildner }
9995db2f26eSSascha Wildner 
10005db2f26eSSascha Wildner static int
acpi_asus_sysctl_set(struct acpi_asus_softc * sc,int method,int arg)10015db2f26eSSascha Wildner acpi_asus_sysctl_set(struct acpi_asus_softc *sc, int method, int arg)
10025db2f26eSSascha Wildner {
10035db2f26eSSascha Wildner 	ACPI_STATUS		status = AE_OK;
10045db2f26eSSascha Wildner 	ACPI_OBJECT_LIST 	acpiargs;
10055db2f26eSSascha Wildner 	ACPI_OBJECT		acpiarg[1];
10065db2f26eSSascha Wildner 
10075db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
10085db2f26eSSascha Wildner 	ACPI_SERIAL_ASSERT(asus);
10095db2f26eSSascha Wildner 
10105db2f26eSSascha Wildner 	acpiargs.Count = 1;
10115db2f26eSSascha Wildner 	acpiargs.Pointer = acpiarg;
10125db2f26eSSascha Wildner 	acpiarg[0].Type = ACPI_TYPE_INTEGER;
10135db2f26eSSascha Wildner 	acpiarg[0].Integer.Value = arg;
10145db2f26eSSascha Wildner 
10155db2f26eSSascha Wildner 	switch (method) {
10165db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_BRN:
10175db2f26eSSascha Wildner 		if (arg < 0 || arg > 15)
10185db2f26eSSascha Wildner 			return (EINVAL);
10195db2f26eSSascha Wildner 
10205db2f26eSSascha Wildner 		if (sc->model->brn_set)
10215db2f26eSSascha Wildner 			status = acpi_SetInteger(sc->handle,
10225db2f26eSSascha Wildner 			    sc->model->brn_set, arg);
10235db2f26eSSascha Wildner 		else {
10245db2f26eSSascha Wildner 			while (arg != 0) {
10255db2f26eSSascha Wildner 				status = AcpiEvaluateObject(sc->handle,
10265db2f26eSSascha Wildner 				    (arg > 0) ?  sc->model->brn_up :
10275db2f26eSSascha Wildner 				    sc->model->brn_dn, NULL, NULL);
10285db2f26eSSascha Wildner 				(arg > 0) ? arg-- : arg++;
10295db2f26eSSascha Wildner 			}
10305db2f26eSSascha Wildner 		}
10315db2f26eSSascha Wildner 
10325db2f26eSSascha Wildner 		if (ACPI_SUCCESS(status))
10335db2f26eSSascha Wildner 			sc->s_brn = arg;
10345db2f26eSSascha Wildner 
10355db2f26eSSascha Wildner 		break;
10365db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_DISP:
10375db2f26eSSascha Wildner 		if (arg < 0 || arg > 7)
10385db2f26eSSascha Wildner 			return (EINVAL);
10395db2f26eSSascha Wildner 
10405db2f26eSSascha Wildner 		status = acpi_SetInteger(sc->handle,
10415db2f26eSSascha Wildner 		    sc->model->disp_set, arg);
10425db2f26eSSascha Wildner 
10435db2f26eSSascha Wildner 		if (ACPI_SUCCESS(status))
10445db2f26eSSascha Wildner 			sc->s_disp = arg;
10455db2f26eSSascha Wildner 
10465db2f26eSSascha Wildner 		break;
10475db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_LCD:
10485db2f26eSSascha Wildner 		if (arg < 0 || arg > 1)
10495db2f26eSSascha Wildner 			return (EINVAL);
10505db2f26eSSascha Wildner 
10515db2f26eSSascha Wildner 		if (strncmp(sc->model->name, "L3H", 3) != 0)
10525db2f26eSSascha Wildner 			status = AcpiEvaluateObject(sc->handle,
10535db2f26eSSascha Wildner 			    sc->model->lcd_set, NULL, NULL);
10545db2f26eSSascha Wildner 		else
10555db2f26eSSascha Wildner 			status = acpi_SetInteger(sc->handle,
10565db2f26eSSascha Wildner 			    sc->model->lcd_set, 0x7);
10575db2f26eSSascha Wildner 
10585db2f26eSSascha Wildner 		if (ACPI_SUCCESS(status))
10595db2f26eSSascha Wildner 			sc->s_lcd = arg;
10605db2f26eSSascha Wildner 
10615db2f26eSSascha Wildner 		break;
10625db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_CAMERA:
10635db2f26eSSascha Wildner 		if (arg < 0 || arg > 1)
10645db2f26eSSascha Wildner 			return (EINVAL);
10655db2f26eSSascha Wildner 
10665db2f26eSSascha Wildner 		status = AcpiEvaluateObject(sc->handle,
10675db2f26eSSascha Wildner 		    sc->model->cam_set, &acpiargs, NULL);
10685db2f26eSSascha Wildner 
10695db2f26eSSascha Wildner 		if (ACPI_SUCCESS(status))
10705db2f26eSSascha Wildner 			sc->s_cam = arg;
10715db2f26eSSascha Wildner 		break;
10725db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_CARDRD:
10735db2f26eSSascha Wildner 		if (arg < 0 || arg > 1)
10745db2f26eSSascha Wildner 			return (EINVAL);
10755db2f26eSSascha Wildner 
10765db2f26eSSascha Wildner 		status = AcpiEvaluateObject(sc->handle,
10775db2f26eSSascha Wildner 		    sc->model->crd_set, &acpiargs, NULL);
10785db2f26eSSascha Wildner 
10795db2f26eSSascha Wildner 		if (ACPI_SUCCESS(status))
10805db2f26eSSascha Wildner 			sc->s_crd = arg;
10815db2f26eSSascha Wildner 		break;
10825db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_WLAN:
10835db2f26eSSascha Wildner 		if (arg < 0 || arg > 1)
10845db2f26eSSascha Wildner 			return (EINVAL);
10855db2f26eSSascha Wildner 
10865db2f26eSSascha Wildner 		status = AcpiEvaluateObject(sc->handle,
10875db2f26eSSascha Wildner 		    sc->model->wlan_set, &acpiargs, NULL);
10885db2f26eSSascha Wildner 
10895db2f26eSSascha Wildner 		if (ACPI_SUCCESS(status))
10905db2f26eSSascha Wildner 			sc->s_wlan = arg;
10915db2f26eSSascha Wildner 		break;
10925db2f26eSSascha Wildner 	}
10935db2f26eSSascha Wildner 
10945db2f26eSSascha Wildner 	return (0);
10955db2f26eSSascha Wildner }
10965db2f26eSSascha Wildner 
10975db2f26eSSascha Wildner static int
acpi_asus_sysctl_init(struct acpi_asus_softc * sc,int method)10985db2f26eSSascha Wildner acpi_asus_sysctl_init(struct acpi_asus_softc *sc, int method)
10995db2f26eSSascha Wildner {
11005db2f26eSSascha Wildner 	ACPI_STATUS	status;
11015db2f26eSSascha Wildner 
11025db2f26eSSascha Wildner 	switch (method) {
11035db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_BRN:
11045db2f26eSSascha Wildner 		if (sc->model->brn_get) {
11055db2f26eSSascha Wildner 			/* GPLV/SPLV models */
11065db2f26eSSascha Wildner 			status = acpi_GetInteger(sc->handle,
11075db2f26eSSascha Wildner 			    sc->model->brn_get, &sc->s_brn);
11085db2f26eSSascha Wildner 			if (ACPI_SUCCESS(status))
11095db2f26eSSascha Wildner 				return (TRUE);
11105db2f26eSSascha Wildner 		} else if (sc->model->brn_up) {
11115db2f26eSSascha Wildner 			/* Relative models */
11125db2f26eSSascha Wildner 			status = AcpiEvaluateObject(sc->handle,
11135db2f26eSSascha Wildner 			    sc->model->brn_up, NULL, NULL);
11145db2f26eSSascha Wildner 			if (ACPI_FAILURE(status))
11155db2f26eSSascha Wildner 				return (FALSE);
11165db2f26eSSascha Wildner 
11175db2f26eSSascha Wildner 			status = AcpiEvaluateObject(sc->handle,
11185db2f26eSSascha Wildner 			    sc->model->brn_dn, NULL, NULL);
11195db2f26eSSascha Wildner 			if (ACPI_FAILURE(status))
11205db2f26eSSascha Wildner 				return (FALSE);
11215db2f26eSSascha Wildner 
11225db2f26eSSascha Wildner 			return (TRUE);
11235db2f26eSSascha Wildner 		}
11245db2f26eSSascha Wildner 		return (FALSE);
11255db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_DISP:
11265db2f26eSSascha Wildner 		if (sc->model->disp_get) {
11275db2f26eSSascha Wildner 			status = acpi_GetInteger(sc->handle,
11285db2f26eSSascha Wildner 			    sc->model->disp_get, &sc->s_disp);
11295db2f26eSSascha Wildner 			if (ACPI_SUCCESS(status))
11305db2f26eSSascha Wildner 				return (TRUE);
11315db2f26eSSascha Wildner 		}
11325db2f26eSSascha Wildner 		return (FALSE);
11335db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_LCD:
11345db2f26eSSascha Wildner 		if (sc->model->lcd_get) {
11355db2f26eSSascha Wildner 			if (strncmp(sc->model->name, "L3H", 3) == 0) {
11365db2f26eSSascha Wildner 				ACPI_BUFFER		Buf;
11375db2f26eSSascha Wildner 				ACPI_OBJECT		Arg[2], Obj;
11385db2f26eSSascha Wildner 				ACPI_OBJECT_LIST	Args;
11395db2f26eSSascha Wildner 
11405db2f26eSSascha Wildner 				/* L3H is a bit special */
11415db2f26eSSascha Wildner 				Arg[0].Type = ACPI_TYPE_INTEGER;
11425db2f26eSSascha Wildner 				Arg[0].Integer.Value = 0x02;
11435db2f26eSSascha Wildner 				Arg[1].Type = ACPI_TYPE_INTEGER;
11445db2f26eSSascha Wildner 				Arg[1].Integer.Value = 0x03;
11455db2f26eSSascha Wildner 
11465db2f26eSSascha Wildner 				Args.Count = 2;
11475db2f26eSSascha Wildner 				Args.Pointer = Arg;
11485db2f26eSSascha Wildner 
11495db2f26eSSascha Wildner 				Buf.Length = sizeof(Obj);
11505db2f26eSSascha Wildner 				Buf.Pointer = &Obj;
11515db2f26eSSascha Wildner 
11525db2f26eSSascha Wildner 				status = AcpiEvaluateObject(sc->handle,
11535db2f26eSSascha Wildner 				    sc->model->lcd_get, &Args, &Buf);
11545db2f26eSSascha Wildner 				if (ACPI_SUCCESS(status) &&
11555db2f26eSSascha Wildner 				    Obj.Type == ACPI_TYPE_INTEGER) {
11565db2f26eSSascha Wildner 					sc->s_lcd = Obj.Integer.Value >> 8;
11575db2f26eSSascha Wildner 					return (TRUE);
11585db2f26eSSascha Wildner 				}
11595db2f26eSSascha Wildner 			} else {
11605db2f26eSSascha Wildner 				status = acpi_GetInteger(sc->handle,
11615db2f26eSSascha Wildner 				    sc->model->lcd_get, &sc->s_lcd);
11625db2f26eSSascha Wildner 				if (ACPI_SUCCESS(status))
11635db2f26eSSascha Wildner 					return (TRUE);
11645db2f26eSSascha Wildner 			}
11655db2f26eSSascha Wildner 		}
11665db2f26eSSascha Wildner 		return (FALSE);
11675db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_CAMERA:
11685db2f26eSSascha Wildner 		if (sc->model->cam_get) {
11695db2f26eSSascha Wildner 			status = acpi_GetInteger(sc->handle,
11705db2f26eSSascha Wildner 			    sc->model->cam_get, &sc->s_cam);
11715db2f26eSSascha Wildner 			if (ACPI_SUCCESS(status))
11725db2f26eSSascha Wildner 				return (TRUE);
11735db2f26eSSascha Wildner 		}
11745db2f26eSSascha Wildner 		return (FALSE);
11755db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_CARDRD:
11765db2f26eSSascha Wildner 		if (sc->model->crd_get) {
11775db2f26eSSascha Wildner 			status = acpi_GetInteger(sc->handle,
11785db2f26eSSascha Wildner 			    sc->model->crd_get, &sc->s_crd);
11795db2f26eSSascha Wildner 			if (ACPI_SUCCESS(status))
11805db2f26eSSascha Wildner 				return (TRUE);
11815db2f26eSSascha Wildner 		}
11825db2f26eSSascha Wildner 		return (FALSE);
11835db2f26eSSascha Wildner 	case ACPI_ASUS_METHOD_WLAN:
11845db2f26eSSascha Wildner 		if (sc->model->wlan_get) {
11855db2f26eSSascha Wildner 			status = acpi_GetInteger(sc->handle,
11865db2f26eSSascha Wildner 			    sc->model->wlan_get, &sc->s_wlan);
11875db2f26eSSascha Wildner 			if (ACPI_SUCCESS(status))
11885db2f26eSSascha Wildner 				return (TRUE);
11895db2f26eSSascha Wildner 		}
11905db2f26eSSascha Wildner 		return (FALSE);
11915db2f26eSSascha Wildner 	}
11925db2f26eSSascha Wildner 	return (FALSE);
11935db2f26eSSascha Wildner }
11945db2f26eSSascha Wildner 
11955db2f26eSSascha Wildner static void
acpi_asus_notify(ACPI_HANDLE h,UINT32 notify,void * context)11965db2f26eSSascha Wildner acpi_asus_notify(ACPI_HANDLE h, UINT32 notify, void *context)
11975db2f26eSSascha Wildner {
11985db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
11995db2f26eSSascha Wildner 	struct acpi_softc	*acpi_sc;
12005db2f26eSSascha Wildner 
12015db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
12025db2f26eSSascha Wildner 
12035db2f26eSSascha Wildner 	sc = device_get_softc((device_t)context);
12045db2f26eSSascha Wildner 	acpi_sc = acpi_device_get_parent_softc(sc->dev);
12055db2f26eSSascha Wildner 
12065db2f26eSSascha Wildner 	ACPI_SERIAL_BEGIN(asus);
12075db2f26eSSascha Wildner 	if ((notify & ~0x10) <= 15) {
12085db2f26eSSascha Wildner 		sc->s_brn = notify & ~0x10;
12095db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "Brightness increased\n");
12105db2f26eSSascha Wildner 	} else if ((notify & ~0x20) <= 15) {
12115db2f26eSSascha Wildner 		sc->s_brn = notify & ~0x20;
12125db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "Brightness decreased\n");
12135db2f26eSSascha Wildner 	} else if (notify == 0x33) {
12145db2f26eSSascha Wildner 		sc->s_lcd = 1;
12155db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "LCD turned on\n");
12165db2f26eSSascha Wildner 	} else if (notify == 0x34) {
12175db2f26eSSascha Wildner 		sc->s_lcd = 0;
12185db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "LCD turned off\n");
12195db2f26eSSascha Wildner 	} else if (notify == 0x86) {
12205db2f26eSSascha Wildner 		acpi_asus_sysctl_set(sc, ACPI_ASUS_METHOD_BRN, sc->s_brn-1);
12215db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "Brightness decreased\n");
12225db2f26eSSascha Wildner 	} else if (notify == 0x87) {
12235db2f26eSSascha Wildner 		acpi_asus_sysctl_set(sc, ACPI_ASUS_METHOD_BRN, sc->s_brn+1);
12245db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "Brightness increased\n");
12255db2f26eSSascha Wildner 	} else {
12265db2f26eSSascha Wildner 		/* Notify devd(8) */
12275db2f26eSSascha Wildner 		acpi_UserNotify("ASUS", h, notify);
12285db2f26eSSascha Wildner 	}
12295db2f26eSSascha Wildner 	ACPI_SERIAL_END(asus);
12305db2f26eSSascha Wildner }
12315db2f26eSSascha Wildner 
12325db2f26eSSascha Wildner static void
acpi_asus_lcdd_notify(ACPI_HANDLE h,UINT32 notify,void * context)12335db2f26eSSascha Wildner acpi_asus_lcdd_notify(ACPI_HANDLE h, UINT32 notify, void *context)
12345db2f26eSSascha Wildner {
12355db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
12365db2f26eSSascha Wildner 	struct acpi_softc	*acpi_sc;
12375db2f26eSSascha Wildner 
12385db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
12395db2f26eSSascha Wildner 
12405db2f26eSSascha Wildner 	sc = device_get_softc((device_t)context);
12415db2f26eSSascha Wildner 	acpi_sc = acpi_device_get_parent_softc(sc->dev);
12425db2f26eSSascha Wildner 
12435db2f26eSSascha Wildner 	ACPI_SERIAL_BEGIN(asus);
12445db2f26eSSascha Wildner 	switch (notify) {
12455db2f26eSSascha Wildner 	case 0x87:
12465db2f26eSSascha Wildner 		acpi_asus_sysctl_set(sc, ACPI_ASUS_METHOD_BRN, sc->s_brn-1);
12475db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "Brightness decreased\n");
12485db2f26eSSascha Wildner 		break;
12495db2f26eSSascha Wildner 	case 0x86:
12505db2f26eSSascha Wildner 		acpi_asus_sysctl_set(sc, ACPI_ASUS_METHOD_BRN, sc->s_brn+1);
12515db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc, "Brightness increased\n");
12525db2f26eSSascha Wildner 		break;
1253d1fb95bdSSascha Wildner 	default:
1254d1fb95bdSSascha Wildner 		device_printf(sc->dev, "unknown notify: %#x\n", notify);
1255d1fb95bdSSascha Wildner 		break;
12565db2f26eSSascha Wildner 	}
12575db2f26eSSascha Wildner 	ACPI_SERIAL_END(asus);
12585db2f26eSSascha Wildner }
12595db2f26eSSascha Wildner 
12605db2f26eSSascha Wildner static void
acpi_asus_eeepc_notify(ACPI_HANDLE h,UINT32 notify,void * context)12615db2f26eSSascha Wildner acpi_asus_eeepc_notify(ACPI_HANDLE h, UINT32 notify, void *context)
12625db2f26eSSascha Wildner {
12635db2f26eSSascha Wildner 	struct acpi_asus_softc	*sc;
12645db2f26eSSascha Wildner 	struct acpi_softc	*acpi_sc;
12655db2f26eSSascha Wildner 
12665db2f26eSSascha Wildner 	ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
12675db2f26eSSascha Wildner 
12685db2f26eSSascha Wildner 	sc = device_get_softc((device_t)context);
12695db2f26eSSascha Wildner 	acpi_sc = acpi_device_get_parent_softc(sc->dev);
12705db2f26eSSascha Wildner 
12715db2f26eSSascha Wildner 	ACPI_SERIAL_BEGIN(asus);
12725db2f26eSSascha Wildner 	if ((notify & ~0x20) <= 15) {
12735db2f26eSSascha Wildner 		sc->s_brn = notify & ~0x20;
12745db2f26eSSascha Wildner 		ACPI_VPRINT(sc->dev, acpi_sc,
12755db2f26eSSascha Wildner 		    "Brightness increased/decreased\n");
12765db2f26eSSascha Wildner 	} else {
12775db2f26eSSascha Wildner 		/* Notify devd(8) */
12785db2f26eSSascha Wildner 		acpi_UserNotify("ASUS-Eee", h, notify);
12795db2f26eSSascha Wildner 	}
12805db2f26eSSascha Wildner 	ACPI_SERIAL_END(asus);
12815db2f26eSSascha Wildner }
1282