2 * Copyright (c) 2004, 2005 Philip Paeps <philip@FreeBSD.org>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 #include <sys/cdefs.h>
28 __FBSDID("$FreeBSD$");
31 * Driver for extra ACPI-controlled gadgets (hotkeys, leds, etc) found on
32 * recent Asus (and Medion) laptops. Inspired by the acpi4asus project which
33 * implements these features in the Linux kernel.
35 * <http://sourceforge.net/projects/acpi4asus/>
37 * Currently should support most features, but could use some more testing.
38 * Particularly the display-switching stuff is a bit hairy. If you have an
39 * Asus laptop which doesn't appear to be supported, or strange things happen
40 * when using this driver, please report to <acpi@FreeBSD.org>.
44 #include <sys/param.h>
45 #include <sys/kernel.h>
46 #include <sys/module.h>
50 #include <contrib/dev/acpica/acpi.h>
51 #include <dev/acpica/acpivar.h>
52 #include <dev/led/led.h>
55 #define ACPI_ASUS_METHOD_BRN 1
56 #define ACPI_ASUS_METHOD_DISP 2
57 #define ACPI_ASUS_METHOD_LCD 3
58 #define ACPI_ASUS_METHOD_CAMERA 4
59 #define ACPI_ASUS_METHOD_CARDRD 5
60 #define ACPI_ASUS_METHOD_WLAN 6
62 #define _COMPONENT ACPI_OEM
63 ACPI_MODULE_NAME("ASUS")
65 struct acpi_asus_model {
95 void (*n_func)(ACPI_HANDLE, UINT32, void *);
98 struct acpi_asus_led {
99 struct acpi_asus_softc *sc;
113 struct acpi_asus_softc {
117 struct acpi_asus_model *model;
118 struct sysctl_ctx_list sysctl_ctx;
119 struct sysctl_oid *sysctl_tree;
121 struct acpi_asus_led s_bled;
122 struct acpi_asus_led s_dled;
123 struct acpi_asus_led s_gled;
124 struct acpi_asus_led s_mled;
125 struct acpi_asus_led s_tled;
126 struct acpi_asus_led s_wled;
137 * We can identify Asus laptops from the string they return
138 * as a result of calling the ATK0100 'INIT' method.
140 static struct acpi_asus_model acpi_asus_models[] = {
146 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
149 .disp_get = "\\ADVG",
156 .lcd_set = "\\_SB.PCI0.ISA.EC0._Q10",
157 .brn_up = "\\_SB.PCI0.ISA.EC0._Q0E",
158 .brn_dn = "\\_SB.PCI0.ISA.EC0._Q0F"
168 .disp_get = "\\INFB",
177 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
181 .disp_get = "\\_SB.PCI0.P0P3.VGA.GETD"
186 .brn_up = "\\_SB_.PCI0.SBRG.EC0._Q0E",
187 .brn_dn = "\\_SB_.PCI0.SBRG.EC0._Q0F",
191 .disp_get = "\\_SB_.PCI0.SBRG.EC0._Q10",
192 .disp_set = "\\_SB_.PCI0.SBRG.EC0._Q11"
201 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
204 .disp_get = "\\_SB.PCI0.P0P3.VGA.GETD",
214 .disp_get = "\\INFB",
227 .lcd_get = "\\_SB.PCI0.SBRG.EC0.RPIN",
228 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
229 .disp_get = "\\_SB.PCI0.PCE2.VGA.GETD",
248 .lcd_set = "\\_SB.PCI0.PX40.ECD0._Q10"
265 .lcd_get = "\\_SB.PCI0.PM.PBC",
267 .disp_get = "\\_SB.INFB",
276 .lcd_get = "\\_SB.PCI0.SBSM.SEO4",
277 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
278 .disp_get = "\\_SB.PCI0.P0P1.VGA.GETD",
289 .disp_get = "\\INFB",
294 /* Only has hotkeys, apparently */
299 .brn_up = "\\_SB.PCI0.PX40.EC0.Q0E",
300 .brn_dn = "\\_SB.PCI0.PX40.EC0.Q0F",
302 .lcd_set = "\\_SB.PCI0.PX40.EC0.Q10"
317 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
318 .lcd_get = "\\_SB.BKLT",
330 .lcd_get = "\\_SB.PCI0.SBSM.SEO4",
331 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
332 .disp_get = "\\SSTE",
340 .lcd_set = "\\_SB.PCI0.PX40.Q10",
348 .lcd_set = "\\_SB.PCI0.ISA.EC0._Q10",
349 .brn_up = "\\_SB.PCI0.ISA.EC0._Q0B",
350 .brn_dn = "\\_SB.PCI0.ISA.EC0._Q0A"
358 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
361 .disp_get = "\\_SB.PCI0.P0P1.VGA.GETD",
368 .lcd_set = "\\_SB.PCI0.SBRG.EC0._Q10",
371 .disp_get = "\\_SB.PCI0.P0P2.VGA.GETD",
379 * Samsung P30/P35 laptops have an Asus ATK0100 gadget interface,
380 * but they can't be probed quite the same way as Asus laptops.
382 static struct acpi_asus_model acpi_samsung_models[] = {
386 .brn_up = "\\_SB.PCI0.LPCB.EC0._Q68",
387 .brn_dn = "\\_SB.PCI0.LPCB.EC0._Q69",
389 .lcd_set = "\\_SB.PCI0.LPCB.EC0._Q0E"
395 static void acpi_asus_eeepc_notify(ACPI_HANDLE h, UINT32 notify, void *context);
398 * EeePC have an Asus ASUS010 gadget interface,
399 * but they can't be probed quite the same way as Asus laptops.
401 static struct acpi_asus_model acpi_eeepc_models[] = {
404 .brn_get = "\\_SB.ATKD.PBLG",
405 .brn_set = "\\_SB.ATKD.PBLS",
406 .cam_get = "\\_SB.ATKD.CAMG",
407 .cam_set = "\\_SB.ATKD.CAMS",
408 .crd_set = "\\_SB.ATKD.CRDS",
409 .crd_get = "\\_SB.ATKD.CRDG",
410 .wlan_get = "\\_SB.ATKD.WLDG",
411 .wlan_set = "\\_SB.ATKD.WLDS",
412 .n_func = acpi_asus_eeepc_notify
423 } acpi_asus_sysctls[] = {
425 .name = "lcd_backlight",
426 .method = ACPI_ASUS_METHOD_LCD,
427 .description = "state of the lcd backlight",
428 .flags = CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_ANYBODY
431 .name = "lcd_brightness",
432 .method = ACPI_ASUS_METHOD_BRN,
433 .description = "brightness of the lcd panel",
434 .flags = CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_ANYBODY
437 .name = "video_output",
438 .method = ACPI_ASUS_METHOD_DISP,
439 .description = "display output state",
440 .flags = CTLTYPE_INT | CTLFLAG_RW
444 .method = ACPI_ASUS_METHOD_CAMERA,
445 .description = "internal camera state",
446 .flags = CTLTYPE_INT | CTLFLAG_RW
449 .name = "cardreader",
450 .method = ACPI_ASUS_METHOD_CARDRD,
451 .description = "internal card reader state",
452 .flags = CTLTYPE_INT | CTLFLAG_RW
456 .method = ACPI_ASUS_METHOD_WLAN,
457 .description = "wireless lan state",
458 .flags = CTLTYPE_INT | CTLFLAG_RW
464 ACPI_SERIAL_DECL(asus, "ACPI ASUS extras");
466 /* Function prototypes */
467 static int acpi_asus_probe(device_t dev);
468 static int acpi_asus_attach(device_t dev);
469 static int acpi_asus_detach(device_t dev);
471 static void acpi_asus_led(struct acpi_asus_led *led, int state);
472 static void acpi_asus_led_task(struct acpi_asus_led *led, int pending __unused);
474 static int acpi_asus_sysctl(SYSCTL_HANDLER_ARGS);
475 static int acpi_asus_sysctl_init(struct acpi_asus_softc *sc, int method);
476 static int acpi_asus_sysctl_get(struct acpi_asus_softc *sc, int method);
477 static int acpi_asus_sysctl_set(struct acpi_asus_softc *sc, int method, int val);
479 static void acpi_asus_notify(ACPI_HANDLE h, UINT32 notify, void *context);
481 static device_method_t acpi_asus_methods[] = {
482 DEVMETHOD(device_probe, acpi_asus_probe),
483 DEVMETHOD(device_attach, acpi_asus_attach),
484 DEVMETHOD(device_detach, acpi_asus_detach),
489 static driver_t acpi_asus_driver = {
492 sizeof(struct acpi_asus_softc)
495 static devclass_t acpi_asus_devclass;
497 DRIVER_MODULE(acpi_asus, acpi, acpi_asus_driver, acpi_asus_devclass, 0, 0);
498 MODULE_DEPEND(acpi_asus, acpi, 1, 1, 1);
501 acpi_asus_probe(device_t dev)
503 struct acpi_asus_model *model;
504 struct acpi_asus_softc *sc;
507 ACPI_OBJECT Arg, *Obj;
508 ACPI_OBJECT_LIST Args;
509 static char *asus_ids[] = { "ATK0100", "ASUS010", NULL };
512 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
514 if (acpi_disabled("asus"))
516 rstr = ACPI_ID_PROBE(device_get_parent(dev), dev, asus_ids);
521 sc = device_get_softc(dev);
523 sc->handle = acpi_get_handle(dev);
525 Arg.Type = ACPI_TYPE_INTEGER;
526 Arg.Integer.Value = 0;
532 Buf.Length = ACPI_ALLOCATE_BUFFER;
534 AcpiEvaluateObject(sc->handle, "INIT", &Args, &Buf);
538 * The Samsung P30 returns a null-pointer from INIT, we
539 * can identify it from the 'ODEM' string in the DSDT.
541 if (Obj->String.Pointer == NULL) {
543 ACPI_TABLE_HEADER th;
545 status = AcpiGetTableHeader(ACPI_SIG_DSDT, 0, &th);
546 if (ACPI_FAILURE(status)) {
547 device_printf(dev, "Unsupported (Samsung?) laptop\n");
548 AcpiOsFree(Buf.Pointer);
552 if (strncmp("ODEM", th.OemTableId, 4) == 0) {
553 sc->model = &acpi_samsung_models[0];
554 device_set_desc(dev, "Samsung P30 Laptop Extras");
555 AcpiOsFree(Buf.Pointer);
560 if (strncmp("ASUS010", rstr, 7) == 0) {
561 sc->model = &acpi_eeepc_models[0];
562 device_set_desc(dev, "ASUS EeePC");
563 AcpiOsFree(Buf.Pointer);
568 sb = sbuf_new_auto();
573 * Asus laptops are simply identified by name, easy!
575 for (model = acpi_asus_models; model->name != NULL; model++) {
576 if (strncmp(Obj->String.Pointer, model->name, 3) == 0) {
579 sbuf_printf(sb, "Asus %s Laptop Extras",
580 Obj->String.Pointer);
584 device_set_desc_copy(dev, sbuf_data(sb));
587 AcpiOsFree(Buf.Pointer);
592 * Some models look exactly the same as other models, but have
593 * their own ids. If we spot these, set them up with the same
594 * details as the models they're like, possibly dealing with
597 * XXX: there must be a prettier way to do this!
599 else if (strncmp(model->name, "xxN", 3) == 0 &&
600 (strncmp(Obj->String.Pointer, "M3N", 3) == 0 ||
601 strncmp(Obj->String.Pointer, "S1N", 3) == 0))
603 else if (strncmp(model->name, "A1x", 3) == 0 &&
604 strncmp(Obj->String.Pointer, "A1", 2) == 0)
606 else if (strncmp(model->name, "A2x", 3) == 0 &&
607 strncmp(Obj->String.Pointer, "A2", 2) == 0)
609 else if (strncmp(model->name, "D1x", 3) == 0 &&
610 strncmp(Obj->String.Pointer, "D1", 2) == 0)
612 else if (strncmp(model->name, "L3H", 3) == 0 &&
613 strncmp(Obj->String.Pointer, "L2E", 3) == 0)
615 else if (strncmp(model->name, "L5x", 3) == 0 &&
616 strncmp(Obj->String.Pointer, "L5", 2) == 0)
618 else if (strncmp(model->name, "M2E", 3) == 0 &&
619 (strncmp(Obj->String.Pointer, "M2", 2) == 0 ||
620 strncmp(Obj->String.Pointer, "L4E", 3) == 0))
622 else if (strncmp(model->name, "S1x", 3) == 0 &&
623 (strncmp(Obj->String.Pointer, "L8", 2) == 0 ||
624 strncmp(Obj->String.Pointer, "S1", 2) == 0))
626 else if (strncmp(model->name, "S2x", 3) == 0 &&
627 (strncmp(Obj->String.Pointer, "J1", 2) == 0 ||
628 strncmp(Obj->String.Pointer, "S2", 2) == 0))
631 /* L2B is like L3C but has no lcd_get method */
632 else if (strncmp(model->name, "L3C", 3) == 0 &&
633 strncmp(Obj->String.Pointer, "L2B", 3) == 0) {
634 model->lcd_get = NULL;
638 /* A3G is like M6R but with a different lcd_get method */
639 else if (strncmp(model->name, "M6R", 3) == 0 &&
640 strncmp(Obj->String.Pointer, "A3G", 3) == 0) {
641 model->lcd_get = "\\BLFG";
645 /* M2N and W1N are like xxN with added WLED */
646 else if (strncmp(model->name, "xxN", 3) == 0 &&
647 (strncmp(Obj->String.Pointer, "M2N", 3) == 0 ||
648 strncmp(Obj->String.Pointer, "W1N", 3) == 0)) {
649 model->wled_set = "WLED";
653 /* M5N and S5N are like xxN without MLED */
654 else if (strncmp(model->name, "xxN", 3) == 0 &&
655 (strncmp(Obj->String.Pointer, "M5N", 3) == 0 ||
656 strncmp(Obj->String.Pointer, "S5N", 3) == 0)) {
657 model->mled_set = NULL;
662 sbuf_printf(sb, "Unsupported Asus laptop: %s\n", Obj->String.Pointer);
665 device_printf(dev, sbuf_data(sb));
668 AcpiOsFree(Buf.Pointer);
674 acpi_asus_attach(device_t dev)
676 struct acpi_asus_softc *sc;
677 struct acpi_softc *acpi_sc;
679 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
681 sc = device_get_softc(dev);
682 acpi_sc = acpi_device_get_parent_softc(dev);
684 /* Build sysctl tree */
685 sysctl_ctx_init(&sc->sysctl_ctx);
686 sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx,
687 SYSCTL_CHILDREN(acpi_sc->acpi_sysctl_tree),
688 OID_AUTO, "asus", CTLFLAG_RD, 0, "");
691 for (int i = 0; acpi_asus_sysctls[i].name != NULL; i++) {
692 if (!acpi_asus_sysctl_init(sc, acpi_asus_sysctls[i].method))
695 SYSCTL_ADD_PROC(&sc->sysctl_ctx,
696 SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO,
697 acpi_asus_sysctls[i].name,
698 acpi_asus_sysctls[i].flags,
699 sc, i, acpi_asus_sysctl, "I",
700 acpi_asus_sysctls[i].description);
704 if (sc->model->bled_set) {
707 sc->s_bled.type = ACPI_ASUS_LED_BLED;
709 led_create_state((led_t *)acpi_asus_led, &sc->s_bled,
713 if (sc->model->dled_set) {
716 sc->s_dled.type = ACPI_ASUS_LED_DLED;
718 led_create((led_t *)acpi_asus_led, &sc->s_dled, "dled");
721 if (sc->model->gled_set) {
724 sc->s_gled.type = ACPI_ASUS_LED_GLED;
726 led_create((led_t *)acpi_asus_led, &sc->s_gled, "gled");
729 if (sc->model->mled_set) {
732 sc->s_mled.type = ACPI_ASUS_LED_MLED;
734 led_create((led_t *)acpi_asus_led, &sc->s_mled, "mled");
737 if (sc->model->tled_set) {
740 sc->s_tled.type = ACPI_ASUS_LED_TLED;
742 led_create_state((led_t *)acpi_asus_led, &sc->s_tled,
746 if (sc->model->wled_set) {
749 sc->s_wled.type = ACPI_ASUS_LED_WLED;
751 led_create_state((led_t *)acpi_asus_led, &sc->s_wled,
755 /* Activate hotkeys */
756 AcpiEvaluateObject(sc->handle, "BSTS", NULL, NULL);
758 /* Handle notifies */
759 if (sc->model->n_func == NULL)
760 sc->model->n_func = acpi_asus_notify;
762 AcpiInstallNotifyHandler(sc->handle, ACPI_SYSTEM_NOTIFY,
763 sc->model->n_func, dev);
769 acpi_asus_detach(device_t dev)
771 struct acpi_asus_softc *sc;
773 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
775 sc = device_get_softc(dev);
777 /* Turn the lights off */
778 if (sc->model->bled_set)
779 led_destroy(sc->s_bled.cdev);
781 if (sc->model->dled_set)
782 led_destroy(sc->s_dled.cdev);
784 if (sc->model->gled_set)
785 led_destroy(sc->s_gled.cdev);
787 if (sc->model->mled_set)
788 led_destroy(sc->s_mled.cdev);
790 if (sc->model->tled_set)
791 led_destroy(sc->s_tled.cdev);
793 if (sc->model->wled_set)
794 led_destroy(sc->s_wled.cdev);
796 /* Remove notify handler */
797 AcpiRemoveNotifyHandler(sc->handle, ACPI_SYSTEM_NOTIFY,
800 /* Free sysctl tree */
801 sysctl_ctx_free(&sc->sysctl_ctx);
807 acpi_asus_led_task(struct acpi_asus_led *led, int pending __unused)
809 struct acpi_asus_softc *sc;
813 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
818 case ACPI_ASUS_LED_BLED:
819 method = sc->model->bled_set;
822 case ACPI_ASUS_LED_DLED:
823 method = sc->model->dled_set;
826 case ACPI_ASUS_LED_GLED:
827 method = sc->model->gled_set;
828 state = led->state + 1; /* 1: off, 2: on */
830 case ACPI_ASUS_LED_MLED:
831 method = sc->model->mled_set;
832 state = !led->state; /* inverted */
834 case ACPI_ASUS_LED_TLED:
835 method = sc->model->tled_set;
838 case ACPI_ASUS_LED_WLED:
839 method = sc->model->wled_set;
843 printf("acpi_asus_led: invalid LED type %d\n",
848 acpi_SetInteger(sc->handle, method, state);
853 acpi_asus_led(struct acpi_asus_led *led, int state)
856 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
864 AcpiOsExecute(OSL_NOTIFY_HANDLER, (void *)acpi_asus_led_task, led);
868 acpi_asus_sysctl(SYSCTL_HANDLER_ARGS)
870 struct acpi_asus_softc *sc;
876 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
878 sc = (struct acpi_asus_softc *)oidp->oid_arg1;
879 function = oidp->oid_arg2;
880 method = acpi_asus_sysctls[function].method;
882 ACPI_SERIAL_BEGIN(asus);
883 arg = acpi_asus_sysctl_get(sc, method);
884 error = sysctl_handle_int(oidp, &arg, 0, req);
887 if (error != 0 || req->newptr == NULL)
891 error = acpi_asus_sysctl_set(sc, method, arg);
894 ACPI_SERIAL_END(asus);
899 acpi_asus_sysctl_get(struct acpi_asus_softc *sc, int method)
903 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
904 ACPI_SERIAL_ASSERT(asus);
907 case ACPI_ASUS_METHOD_BRN:
910 case ACPI_ASUS_METHOD_DISP:
913 case ACPI_ASUS_METHOD_LCD:
916 case ACPI_ASUS_METHOD_CAMERA:
919 case ACPI_ASUS_METHOD_CARDRD:
922 case ACPI_ASUS_METHOD_WLAN:
931 acpi_asus_sysctl_set(struct acpi_asus_softc *sc, int method, int arg)
933 ACPI_STATUS status = AE_OK;
934 ACPI_OBJECT_LIST acpiargs;
935 ACPI_OBJECT acpiarg[0];
937 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
938 ACPI_SERIAL_ASSERT(asus);
941 acpiargs.Pointer = acpiarg;
942 acpiarg[0].Type = ACPI_TYPE_INTEGER;
943 acpiarg[0].Integer.Value = arg;
946 case ACPI_ASUS_METHOD_BRN:
947 if (arg < 0 || arg > 15)
950 if (sc->model->brn_set)
951 status = acpi_SetInteger(sc->handle,
952 sc->model->brn_set, arg);
955 status = AcpiEvaluateObject(sc->handle,
956 (arg > 0) ? sc->model->brn_up :
957 sc->model->brn_dn, NULL, NULL);
958 (arg > 0) ? arg-- : arg++;
962 if (ACPI_SUCCESS(status))
966 case ACPI_ASUS_METHOD_DISP:
967 if (arg < 0 || arg > 7)
970 status = acpi_SetInteger(sc->handle,
971 sc->model->disp_set, arg);
973 if (ACPI_SUCCESS(status))
977 case ACPI_ASUS_METHOD_LCD:
978 if (arg < 0 || arg > 1)
981 if (strncmp(sc->model->name, "L3H", 3) != 0)
982 status = AcpiEvaluateObject(sc->handle,
983 sc->model->lcd_set, NULL, NULL);
985 status = acpi_SetInteger(sc->handle,
986 sc->model->lcd_set, 0x7);
988 if (ACPI_SUCCESS(status))
992 case ACPI_ASUS_METHOD_CAMERA:
993 if (arg < 0 || arg > 1)
996 status = AcpiEvaluateObject(sc->handle,
997 sc->model->cam_set, &acpiargs, NULL);
999 if (ACPI_SUCCESS(status))
1002 case ACPI_ASUS_METHOD_CARDRD:
1003 if (arg < 0 || arg > 1)
1006 status = AcpiEvaluateObject(sc->handle,
1007 sc->model->crd_set, &acpiargs, NULL);
1009 if (ACPI_SUCCESS(status))
1012 case ACPI_ASUS_METHOD_WLAN:
1013 if (arg < 0 || arg > 1)
1016 status = AcpiEvaluateObject(sc->handle,
1017 sc->model->wlan_set, &acpiargs, NULL);
1019 if (ACPI_SUCCESS(status))
1028 acpi_asus_sysctl_init(struct acpi_asus_softc *sc, int method)
1033 case ACPI_ASUS_METHOD_BRN:
1034 if (sc->model->brn_get) {
1035 /* GPLV/SPLV models */
1036 status = acpi_GetInteger(sc->handle,
1037 sc->model->brn_get, &sc->s_brn);
1038 if (ACPI_SUCCESS(status))
1040 } else if (sc->model->brn_up) {
1041 /* Relative models */
1042 status = AcpiEvaluateObject(sc->handle,
1043 sc->model->brn_up, NULL, NULL);
1044 if (ACPI_FAILURE(status))
1047 status = AcpiEvaluateObject(sc->handle,
1048 sc->model->brn_dn, NULL, NULL);
1049 if (ACPI_FAILURE(status))
1055 case ACPI_ASUS_METHOD_DISP:
1056 if (sc->model->disp_get) {
1057 status = acpi_GetInteger(sc->handle,
1058 sc->model->disp_get, &sc->s_disp);
1059 if (ACPI_SUCCESS(status))
1063 case ACPI_ASUS_METHOD_LCD:
1064 if (sc->model->lcd_get) {
1065 if (strncmp(sc->model->name, "G2K", 3) == 0) {
1067 ACPI_OBJECT Arg, Obj;
1068 ACPI_OBJECT_LIST Args;
1070 Arg.Type = ACPI_TYPE_INTEGER;
1071 Arg.Integer.Value = 0x11;
1073 Args.Pointer = &Arg;
1074 Buf.Length = sizeof(Obj);
1077 status = AcpiEvaluateObject(sc->handle,
1078 sc->model->lcd_get, &Args, &Buf);
1079 if (ACPI_SUCCESS(status) &&
1080 Obj.Type == ACPI_TYPE_INTEGER) {
1081 sc->s_lcd = Obj.Integer.Value;
1084 } else if (strncmp(sc->model->name, "L3H", 3) == 0) {
1086 ACPI_OBJECT Arg[2], Obj;
1087 ACPI_OBJECT_LIST Args;
1089 /* L3H is a bit special */
1090 Arg[0].Type = ACPI_TYPE_INTEGER;
1091 Arg[0].Integer.Value = 0x02;
1092 Arg[1].Type = ACPI_TYPE_INTEGER;
1093 Arg[1].Integer.Value = 0x03;
1098 Buf.Length = sizeof(Obj);
1101 status = AcpiEvaluateObject(sc->handle,
1102 sc->model->lcd_get, &Args, &Buf);
1103 if (ACPI_SUCCESS(status) &&
1104 Obj.Type == ACPI_TYPE_INTEGER) {
1105 sc->s_lcd = Obj.Integer.Value >> 8;
1109 status = acpi_GetInteger(sc->handle,
1110 sc->model->lcd_get, &sc->s_lcd);
1111 if (ACPI_SUCCESS(status))
1116 case ACPI_ASUS_METHOD_CAMERA:
1117 if (sc->model->cam_get) {
1118 status = acpi_GetInteger(sc->handle,
1119 sc->model->cam_get, &sc->s_cam);
1120 if (ACPI_SUCCESS(status))
1124 case ACPI_ASUS_METHOD_CARDRD:
1125 if (sc->model->crd_get) {
1126 status = acpi_GetInteger(sc->handle,
1127 sc->model->crd_get, &sc->s_crd);
1128 if (ACPI_SUCCESS(status))
1132 case ACPI_ASUS_METHOD_WLAN:
1133 if (sc->model->wlan_get) {
1134 status = acpi_GetInteger(sc->handle,
1135 sc->model->wlan_get, &sc->s_wlan);
1136 if (ACPI_SUCCESS(status))
1145 acpi_asus_notify(ACPI_HANDLE h, UINT32 notify, void *context)
1147 struct acpi_asus_softc *sc;
1148 struct acpi_softc *acpi_sc;
1150 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1152 sc = device_get_softc((device_t)context);
1153 acpi_sc = acpi_device_get_parent_softc(sc->dev);
1155 ACPI_SERIAL_BEGIN(asus);
1156 if ((notify & ~0x10) <= 15) {
1157 sc->s_brn = notify & ~0x10;
1158 ACPI_VPRINT(sc->dev, acpi_sc, "Brightness increased\n");
1159 } else if ((notify & ~0x20) <= 15) {
1160 sc->s_brn = notify & ~0x20;
1161 ACPI_VPRINT(sc->dev, acpi_sc, "Brightness decreased\n");
1162 } else if (notify == 0x33) {
1164 ACPI_VPRINT(sc->dev, acpi_sc, "LCD turned on\n");
1165 } else if (notify == 0x34) {
1167 ACPI_VPRINT(sc->dev, acpi_sc, "LCD turned off\n");
1169 /* Notify devd(8) */
1170 acpi_UserNotify("ASUS", h, notify);
1172 ACPI_SERIAL_END(asus);
1176 acpi_asus_eeepc_notify(ACPI_HANDLE h, UINT32 notify, void *context)
1178 struct acpi_asus_softc *sc;
1179 struct acpi_softc *acpi_sc;
1181 ACPI_FUNCTION_TRACE((char *)(uintptr_t)__func__);
1183 sc = device_get_softc((device_t)context);
1184 acpi_sc = acpi_device_get_parent_softc(sc->dev);
1186 ACPI_SERIAL_BEGIN(asus);
1187 if ((notify & ~0x20) <= 15) {
1188 sc->s_brn = notify & ~0x20;
1189 ACPI_VPRINT(sc->dev, acpi_sc,
1190 "Brightness increased/decreased\n");
1192 /* Notify devd(8) */
1193 acpi_UserNotify("ASUS-Eee", h, notify);
1195 ACPI_SERIAL_END(asus);