452 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			452 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
| /*****************************************************************
 | |
| 
 | |
| Copyright (c) 1996 Digital Equipment Corporation, Maynard, Massachusetts.
 | |
| 
 | |
| Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
| of this software and associated documentation files (the "Software"), to deal
 | |
| in the Software without restriction, including without limitation the rights
 | |
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | |
| copies of the Software.
 | |
| 
 | |
| The above copyright notice and this permission notice shall be included in
 | |
| all copies or substantial portions of the Software.
 | |
| 
 | |
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 | |
| DIGITAL EQUIPMENT CORPORATION BE LIABLE FOR ANY CLAIM, DAMAGES, INCLUDING,
 | |
| BUT NOT LIMITED TO CONSEQUENTIAL OR INCIDENTAL DAMAGES, OR OTHER LIABILITY,
 | |
| WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
 | |
| IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
| 
 | |
| Except as contained in this notice, the name of Digital Equipment Corporation
 | |
| shall not be used in advertising or otherwise to promote the sale, use or other
 | |
| dealings in this Software without prior written authorization from Digital
 | |
| Equipment Corporation.
 | |
| 
 | |
| ******************************************************************/
 | |
| 
 | |
| #ifdef HAVE_DIX_CONFIG_H
 | |
| #include <dix-config.h>
 | |
| #endif
 | |
| 
 | |
| #include <X11/X.h>
 | |
| #include <X11/Xproto.h>
 | |
| #include "misc.h"
 | |
| #include "os.h"
 | |
| #include "dixstruct.h"
 | |
| #include "extnsionst.h"
 | |
| #include "opaque.h"
 | |
| #include <X11/extensions/dpmsproto.h>
 | |
| #include "dpmsproc.h"
 | |
| #include "extinit.h"
 | |
| #include "scrnintstr.h"
 | |
| #include "windowstr.h"
 | |
| 
 | |
| CARD16 DPMSPowerLevel = 0;
 | |
| Bool DPMSDisabledSwitch = FALSE;
 | |
| CARD32 DPMSStandbyTime = -1;
 | |
| CARD32 DPMSSuspendTime = -1;
 | |
| CARD32 DPMSOffTime = -1;
 | |
| Bool DPMSEnabled;
 | |
| 
 | |
| Bool
 | |
| DPMSSupported(void)
 | |
| {
 | |
|     int i;
 | |
| 
 | |
|     /* For each screen, check if DPMS is supported */
 | |
|     for (i = 0; i < screenInfo.numScreens; i++)
 | |
|         if (screenInfo.screens[i]->DPMS != NULL)
 | |
|             return TRUE;
 | |
| 
 | |
|     for (i = 0; i < screenInfo.numGPUScreens; i++)
 | |
|         if (screenInfo.gpuscreens[i]->DPMS != NULL)
 | |
|             return TRUE;
 | |
| 
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| static Bool
 | |
| isUnblank(int mode)
 | |
| {
 | |
|     switch (mode) {
 | |
|     case SCREEN_SAVER_OFF:
 | |
|     case SCREEN_SAVER_FORCER:
 | |
|         return TRUE;
 | |
|     case SCREEN_SAVER_ON:
 | |
|     case SCREEN_SAVER_CYCLE:
 | |
|         return FALSE;
 | |
|     default:
 | |
|         return TRUE;
 | |
|     }
 | |
| }
 | |
| 
 | |
| int
 | |
| DPMSSet(ClientPtr client, int level)
 | |
| {
 | |
|     int rc, i;
 | |
| 
 | |
|     DPMSPowerLevel = level;
 | |
| 
 | |
|     if (level != DPMSModeOn) {
 | |
|         if (isUnblank(screenIsSaved)) {
 | |
|             rc = dixSaveScreens(client, SCREEN_SAVER_FORCER, ScreenSaverActive);
 | |
|             if (rc != Success)
 | |
|                 return rc;
 | |
|         }
 | |
|     } else if (!isUnblank(screenIsSaved)) {
 | |
|         rc = dixSaveScreens(client, SCREEN_SAVER_OFF, ScreenSaverReset);
 | |
|         if (rc != Success)
 | |
|             return rc;
 | |
|     }
 | |
| 
 | |
|     for (i = 0; i < screenInfo.numScreens; i++)
 | |
|         if (screenInfo.screens[i]->DPMS != NULL)
 | |
|             screenInfo.screens[i]->DPMS(screenInfo.screens[i], level);
 | |
| 
 | |
|     for (i = 0; i < screenInfo.numGPUScreens; i++)
 | |
|         if (screenInfo.gpuscreens[i]->DPMS != NULL)
 | |
|             screenInfo.gpuscreens[i]->DPMS(screenInfo.gpuscreens[i], level);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSGetVersion(ClientPtr client)
 | |
| {
 | |
|     /* REQUEST(xDPMSGetVersionReq); */
 | |
|     xDPMSGetVersionReply rep = {
 | |
|         .type = X_Reply,
 | |
|         .sequenceNumber = client->sequence,
 | |
|         .length = 0,
 | |
|         .majorVersion = DPMSMajorVersion,
 | |
|         .minorVersion = DPMSMinorVersion
 | |
|     };
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSGetVersionReq);
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swaps(&rep.majorVersion);
 | |
|         swaps(&rep.minorVersion);
 | |
|     }
 | |
|     WriteToClient(client, sizeof(xDPMSGetVersionReply), &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSCapable(ClientPtr client)
 | |
| {
 | |
|     /* REQUEST(xDPMSCapableReq); */
 | |
|     xDPMSCapableReply rep = {
 | |
|         .type = X_Reply,
 | |
|         .sequenceNumber = client->sequence,
 | |
|         .length = 0,
 | |
|         .capable = TRUE
 | |
|     };
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSCapableReq);
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|     }
 | |
|     WriteToClient(client, sizeof(xDPMSCapableReply), &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSGetTimeouts(ClientPtr client)
 | |
| {
 | |
|     /* REQUEST(xDPMSGetTimeoutsReq); */
 | |
|     xDPMSGetTimeoutsReply rep = {
 | |
|         .type = X_Reply,
 | |
|         .sequenceNumber = client->sequence,
 | |
|         .length = 0,
 | |
|         .standby = DPMSStandbyTime / MILLI_PER_SECOND,
 | |
|         .suspend = DPMSSuspendTime / MILLI_PER_SECOND,
 | |
|         .off = DPMSOffTime / MILLI_PER_SECOND
 | |
|     };
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSGetTimeoutsReq);
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swaps(&rep.standby);
 | |
|         swaps(&rep.suspend);
 | |
|         swaps(&rep.off);
 | |
|     }
 | |
|     WriteToClient(client, sizeof(xDPMSGetTimeoutsReply), &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSSetTimeouts(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSSetTimeoutsReq);
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSSetTimeoutsReq);
 | |
| 
 | |
|     if ((stuff->off != 0) && (stuff->off < stuff->suspend)) {
 | |
|         client->errorValue = stuff->off;
 | |
|         return BadValue;
 | |
|     }
 | |
|     if ((stuff->suspend != 0) && (stuff->suspend < stuff->standby)) {
 | |
|         client->errorValue = stuff->suspend;
 | |
|         return BadValue;
 | |
|     }
 | |
| 
 | |
|     DPMSStandbyTime = stuff->standby * MILLI_PER_SECOND;
 | |
|     DPMSSuspendTime = stuff->suspend * MILLI_PER_SECOND;
 | |
|     DPMSOffTime = stuff->off * MILLI_PER_SECOND;
 | |
|     SetScreenSaverTimer();
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSEnable(ClientPtr client)
 | |
| {
 | |
|     Bool was_enabled = DPMSEnabled;
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSEnableReq);
 | |
| 
 | |
|     DPMSEnabled = TRUE;
 | |
|     if (!was_enabled)
 | |
|         SetScreenSaverTimer();
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSDisable(ClientPtr client)
 | |
| {
 | |
|     /* REQUEST(xDPMSDisableReq); */
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSDisableReq);
 | |
| 
 | |
|     DPMSSet(client, DPMSModeOn);
 | |
| 
 | |
|     DPMSEnabled = FALSE;
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSForceLevel(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSForceLevelReq);
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSForceLevelReq);
 | |
| 
 | |
|     if (!DPMSEnabled)
 | |
|         return BadMatch;
 | |
| 
 | |
|     if (stuff->level != DPMSModeOn &&
 | |
|         stuff->level != DPMSModeStandby &&
 | |
|         stuff->level != DPMSModeSuspend && stuff->level != DPMSModeOff) {
 | |
|         client->errorValue = stuff->level;
 | |
|         return BadValue;
 | |
|     }
 | |
| 
 | |
|     DPMSSet(client, stuff->level);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSInfo(ClientPtr client)
 | |
| {
 | |
|     /* REQUEST(xDPMSInfoReq); */
 | |
|     xDPMSInfoReply rep = {
 | |
|         .type = X_Reply,
 | |
|         .sequenceNumber = client->sequence,
 | |
|         .length = 0,
 | |
|         .power_level = DPMSPowerLevel,
 | |
|         .state = DPMSEnabled
 | |
|     };
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xDPMSInfoReq);
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swaps(&rep.power_level);
 | |
|     }
 | |
|     WriteToClient(client, sizeof(xDPMSInfoReply), &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcDPMSDispatch(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xReq);
 | |
| 
 | |
|     switch (stuff->data) {
 | |
|     case X_DPMSGetVersion:
 | |
|         return ProcDPMSGetVersion(client);
 | |
|     case X_DPMSCapable:
 | |
|         return ProcDPMSCapable(client);
 | |
|     case X_DPMSGetTimeouts:
 | |
|         return ProcDPMSGetTimeouts(client);
 | |
|     case X_DPMSSetTimeouts:
 | |
|         return ProcDPMSSetTimeouts(client);
 | |
|     case X_DPMSEnable:
 | |
|         return ProcDPMSEnable(client);
 | |
|     case X_DPMSDisable:
 | |
|         return ProcDPMSDisable(client);
 | |
|     case X_DPMSForceLevel:
 | |
|         return ProcDPMSForceLevel(client);
 | |
|     case X_DPMSInfo:
 | |
|         return ProcDPMSInfo(client);
 | |
|     default:
 | |
|         return BadRequest;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSGetVersion(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSGetVersionReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSGetVersionReq);
 | |
|     swaps(&stuff->majorVersion);
 | |
|     swaps(&stuff->minorVersion);
 | |
|     return ProcDPMSGetVersion(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSCapable(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSCapableReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSCapableReq);
 | |
| 
 | |
|     return ProcDPMSCapable(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSGetTimeouts(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSGetTimeoutsReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSGetTimeoutsReq);
 | |
| 
 | |
|     return ProcDPMSGetTimeouts(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSSetTimeouts(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSSetTimeoutsReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSSetTimeoutsReq);
 | |
| 
 | |
|     swaps(&stuff->standby);
 | |
|     swaps(&stuff->suspend);
 | |
|     swaps(&stuff->off);
 | |
|     return ProcDPMSSetTimeouts(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSEnable(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSEnableReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSEnableReq);
 | |
| 
 | |
|     return ProcDPMSEnable(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSDisable(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSDisableReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSDisableReq);
 | |
| 
 | |
|     return ProcDPMSDisable(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSForceLevel(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSForceLevelReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSForceLevelReq);
 | |
| 
 | |
|     swaps(&stuff->level);
 | |
| 
 | |
|     return ProcDPMSForceLevel(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSInfo(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xDPMSInfoReq);
 | |
| 
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xDPMSInfoReq);
 | |
| 
 | |
|     return ProcDPMSInfo(client);
 | |
| }
 | |
| 
 | |
| static int _X_COLD
 | |
| SProcDPMSDispatch(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xReq);
 | |
|     switch (stuff->data) {
 | |
|     case X_DPMSGetVersion:
 | |
|         return SProcDPMSGetVersion(client);
 | |
|     case X_DPMSCapable:
 | |
|         return SProcDPMSCapable(client);
 | |
|     case X_DPMSGetTimeouts:
 | |
|         return SProcDPMSGetTimeouts(client);
 | |
|     case X_DPMSSetTimeouts:
 | |
|         return SProcDPMSSetTimeouts(client);
 | |
|     case X_DPMSEnable:
 | |
|         return SProcDPMSEnable(client);
 | |
|     case X_DPMSDisable:
 | |
|         return SProcDPMSDisable(client);
 | |
|     case X_DPMSForceLevel:
 | |
|         return SProcDPMSForceLevel(client);
 | |
|     case X_DPMSInfo:
 | |
|         return SProcDPMSInfo(client);
 | |
|     default:
 | |
|         return BadRequest;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| DPMSCloseDownExtension(ExtensionEntry *e)
 | |
| {
 | |
|     DPMSSet(serverClient, DPMSModeOn);
 | |
| }
 | |
| 
 | |
| void
 | |
| DPMSExtensionInit(void)
 | |
| {
 | |
| #define CONDITIONALLY_SET_DPMS_TIMEOUT(_timeout_value_)         \
 | |
|     if (_timeout_value_ == -1) { /* not yet set from config */  \
 | |
|         _timeout_value_ = ScreenSaverTime;                      \
 | |
|     }
 | |
| 
 | |
|     CONDITIONALLY_SET_DPMS_TIMEOUT(DPMSStandbyTime)
 | |
|     CONDITIONALLY_SET_DPMS_TIMEOUT(DPMSSuspendTime)
 | |
|     CONDITIONALLY_SET_DPMS_TIMEOUT(DPMSOffTime)
 | |
| 
 | |
|     DPMSPowerLevel = DPMSModeOn;
 | |
|     DPMSEnabled = DPMSSupported();
 | |
| 
 | |
|     if (DPMSEnabled)
 | |
|         AddExtension(DPMSExtensionName, 0, 0,
 | |
|                      ProcDPMSDispatch, SProcDPMSDispatch,
 | |
|                      DPMSCloseDownExtension, StandardMinorOpcode);
 | |
| }
 |