185 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			185 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
| #ifdef HAVE_XORG_CONFIG_H
 | |
| #include "xorg-config.h"
 | |
| #endif
 | |
| 
 | |
| #include <sys/ioctl.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/socket.h>
 | |
| #include <sys/un.h>
 | |
| #include <unistd.h>
 | |
| #include <fcntl.h>
 | |
| #include <errno.h>
 | |
| 
 | |
| #include "os/log_priv.h"
 | |
| 
 | |
| #include "os.h"
 | |
| #include "xf86_priv.h"
 | |
| #include "xf86Priv.h"
 | |
| #include "xf86_os_support.h"
 | |
| #include "xf86_OSproc.h"
 | |
| 
 | |
| #define ACPI_SOCKET  "/var/run/acpid.socket"
 | |
| 
 | |
| #define ACPI_VIDEO_NOTIFY_SWITCH	0x80
 | |
| #define ACPI_VIDEO_NOTIFY_PROBE		0x81
 | |
| #define ACPI_VIDEO_NOTIFY_CYCLE		0x82
 | |
| #define ACPI_VIDEO_NOTIFY_NEXT_OUTPUT	0x83
 | |
| #define ACPI_VIDEO_NOTIFY_PREV_OUTPUT	0x84
 | |
| 
 | |
| #define ACPI_VIDEO_NOTIFY_CYCLE_BRIGHTNESS	0x85
 | |
| #define	ACPI_VIDEO_NOTIFY_INC_BRIGHTNESS	0x86
 | |
| #define ACPI_VIDEO_NOTIFY_DEC_BRIGHTNESS	0x87
 | |
| #define ACPI_VIDEO_NOTIFY_ZERO_BRIGHTNESS	0x88
 | |
| #define ACPI_VIDEO_NOTIFY_DISPLAY_OFF		0x89
 | |
| 
 | |
| #define ACPI_VIDEO_HEAD_INVALID		(~0u - 1)
 | |
| #define ACPI_VIDEO_HEAD_END		(~0u)
 | |
| 
 | |
| static void lnxCloseACPI(void);
 | |
| static void *ACPIihPtr = NULL;
 | |
| PMClose lnxACPIOpen(void);
 | |
| 
 | |
| /* in milliseconds */
 | |
| #define ACPI_REOPEN_DELAY 1000
 | |
| 
 | |
| static CARD32
 | |
| lnxACPIReopen(OsTimerPtr timer, CARD32 time, void *arg)
 | |
| {
 | |
|     if (lnxACPIOpen()) {
 | |
|         TimerFree(timer);
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     return ACPI_REOPEN_DELAY;
 | |
| }
 | |
| 
 | |
| #define LINE_LENGTH 80
 | |
| 
 | |
| static int
 | |
| lnxACPIGetEventFromOs(int fd, pmEvent * events, int num)
 | |
| {
 | |
|     char ev[LINE_LENGTH];
 | |
|     int n;
 | |
| 
 | |
|     memset(ev, 0, LINE_LENGTH);
 | |
| 
 | |
|     do {
 | |
|         n = read(fd, ev, LINE_LENGTH);
 | |
|     } while ((n == -1) && (errno == EAGAIN || errno == EINTR));
 | |
| 
 | |
|     if (n <= 0) {
 | |
|         lnxCloseACPI();
 | |
|         TimerSet(NULL, 0, ACPI_REOPEN_DELAY, lnxACPIReopen, NULL);
 | |
|         return 0;
 | |
|     }
 | |
|     /* FIXME: this only processes the first read ACPI event & might break
 | |
|      * with interrupted reads. */
 | |
| 
 | |
|     /* Check that we have a video event */
 | |
|     if (!strncmp(ev, "video", 5)) {
 | |
|         char *GFX = NULL;
 | |
|         char *notify = NULL;
 | |
|         char *data = NULL;      /* doesn't appear to be used in the kernel */
 | |
|         unsigned long int notify_l;
 | |
| 
 | |
|         strtok(ev, " ");
 | |
| 
 | |
|         if (!(GFX = strtok(NULL, " ")))
 | |
|             return 0;
 | |
| #if 0
 | |
|         ErrorF("GFX: %s\n", GFX);
 | |
| #endif
 | |
| 
 | |
|         if (!(notify = strtok(NULL, " ")))
 | |
|             return 0;
 | |
|         notify_l = strtoul(notify, NULL, 16);
 | |
| #if 0
 | |
|         ErrorF("notify: 0x%lx\n", notify_l);
 | |
| #endif
 | |
| 
 | |
|         if (!(data = strtok(NULL, " ")))
 | |
|             return 0;
 | |
| #if 0
 | |
|         data_l = strtoul(data, NULL, 16);
 | |
|         ErrorF("data: 0x%lx\n", data_l);
 | |
| #endif
 | |
| 
 | |
|         /* Differentiate between events */
 | |
|         switch (notify_l) {
 | |
|         case ACPI_VIDEO_NOTIFY_SWITCH:
 | |
|         case ACPI_VIDEO_NOTIFY_CYCLE:
 | |
|         case ACPI_VIDEO_NOTIFY_NEXT_OUTPUT:
 | |
|         case ACPI_VIDEO_NOTIFY_PREV_OUTPUT:
 | |
|             events[0] = XF86_APM_CAPABILITY_CHANGED;
 | |
|             return 1;
 | |
|         case ACPI_VIDEO_NOTIFY_PROBE:
 | |
|             return 0;
 | |
|         default:
 | |
|             return 0;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static pmWait
 | |
| lnxACPIConfirmEventToOs(int fd, pmEvent event)
 | |
| {
 | |
|     /* No ability to send back to the kernel in ACPI */
 | |
|     switch (event) {
 | |
|     default:
 | |
|         return PM_NONE;
 | |
|     }
 | |
| }
 | |
| 
 | |
| PMClose
 | |
| lnxACPIOpen(void)
 | |
| {
 | |
|     int fd;
 | |
|     struct sockaddr_un addr;
 | |
|     int r = -1;
 | |
|     static int warned = 0;
 | |
| 
 | |
|     DebugF("ACPI: OSPMOpen called\n");
 | |
|     if (ACPIihPtr || !xf86Info.pmFlag)
 | |
|         return NULL;
 | |
| 
 | |
|     DebugF("ACPI: Opening device\n");
 | |
|     if ((fd = socket(AF_UNIX, SOCK_STREAM, 0)) > -1) {
 | |
|         memset(&addr, 0, sizeof(addr));
 | |
|         addr.sun_family = AF_UNIX;
 | |
|         strcpy(addr.sun_path, ACPI_SOCKET);
 | |
|         if ((r = connect(fd, (struct sockaddr *) &addr, sizeof(addr))) == -1) {
 | |
|             if (!warned)
 | |
|                 LogMessageVerb(X_WARNING, 3, "Open ACPI failed (%s) (%s)\n",
 | |
|                             ACPI_SOCKET, strerror(errno));
 | |
|             warned = 1;
 | |
|             shutdown(fd, 2);
 | |
|             close(fd);
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     xf86PMGetEventFromOs = lnxACPIGetEventFromOs;
 | |
|     xf86PMConfirmEventToOs = lnxACPIConfirmEventToOs;
 | |
|     ACPIihPtr = xf86AddGeneralHandler(fd, xf86HandlePMEvents, NULL);
 | |
|     LogMessageVerb(X_INFO, 3, "Open ACPI successful (%s)\n", ACPI_SOCKET);
 | |
|     warned = 0;
 | |
| 
 | |
|     return lnxCloseACPI;
 | |
| }
 | |
| 
 | |
| static void
 | |
| lnxCloseACPI(void)
 | |
| {
 | |
|     int fd;
 | |
| 
 | |
|     DebugF("ACPI: Closing device\n");
 | |
|     if (ACPIihPtr) {
 | |
|         fd = xf86RemoveGeneralHandler(ACPIihPtr);
 | |
|         shutdown(fd, 2);
 | |
|         close(fd);
 | |
|         ACPIihPtr = NULL;
 | |
|     }
 | |
| }
 |