2020-09-30 16:53:55 +02:00

424 lines
12 KiB
C++

//+----------------------------------------------------------------------------
//
// Scheduling Agent Service
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1996.
//
// File: events.cxx
//
// Contents: Idle and battery event code.
//
// History: 22-Mar-96 EricB created
//
//-----------------------------------------------------------------------------
#include "..\pch\headers.hxx"
#pragma hdrstop
#include "globals.hxx"
#include "svc_core.hxx"
extern "C"
{
#include "msidle.h"
}
#define SCH_NOIDLE_VALUE TEXT("NoIdle")
BOOL g_fOnBattery;
BOOL g_fIdleInitialized;
HINSTANCE g_hMsidleDll = NULL;
//
// msidle.dll function pointers.
//
_BEGINIDLEDETECTION gpfnBeginIdleDetection;
_ENDIDLEDETECTION gpfnEndIdleDetection;
_SETIDLETIMEOUT gpfnSetIdleTimeout;
_SETIDLENOTIFY gpfnSetIdleNotify;
_SETBUSYNOTIFY gpfnSetBusyNotify;
_GETIDLEMINUTES gpfnGetIdleMinutes;
//+----------------------------------------------------------------------------
//
// Function: OnIdleNotification
//
// Synopsis: Called when the winproc receives idle notifications.
//
// Arguments: [wParam] - indicates whether it is for idle start or end.
//
// Returns: hresults
//
//-----------------------------------------------------------------------------
void WINAPI
OnIdleNotify(DWORD dwState)
{
switch (dwState)
{
case STATE_USER_IDLE_BEGIN:
//
// Received idle notification.
//
if (g_pSched != NULL)
{
schDebugOut((DEB_ITRACE,
"*** OnIdleNotification: entering idle state. ***\n"));
g_pSched->OnIdleEvent(TRUE);
}
break;
case STATE_USER_IDLE_END:
//
// Idle has ended.
//
if (g_pSched != NULL)
{
schDebugOut((DEB_ITRACE,
"*** OnIdleNotification: idle lost. ***\n"));
g_pSched->OnIdleEvent(FALSE);
}
break;
default:
schAssert(0);
break;
}
}
//+----------------------------------------------------------------------------
//
// Function: SetNextIdleNotificationFn
//
// Synopsis: Set the length of time to wait for the next idle notification.
//
// Returns: TRUE for success, and FALSE if unable to make the call.
//
//-----------------------------------------------------------------------------
BOOL
SetNextIdleNotificationFn(WORD wIdleWait)
{
schDebugOut((DEB_ITRACE, "SetNextIdleNotification(%u)\n", wIdleWait));
if (!g_fIdleInitialized)
{
DBG_OUT("Calling SetNextIdleNotification before idle init!");
return FALSE;
}
//
// 0xffff is a flag value meaning that no idle notification is needed.
//
if (wIdleWait == 0xffff)
{
schDebugOut((DEB_IDLE, "Next idle wait is 0xffff, not requesting"
" idle notification\n"));
//
// msidle.dll makes it impossible to turn off idle notification
// completely. SetIdleNotify(FALSE, 0) will do it temporarily,
// but if we have also registered for a loss-of-idle notification,
// then as soon as we get that notification, msidle.dll turns idle
// notification back on automatically.
// So we also set a long idle wait period.
// (It doesn't have to be 0xffff, but it might as well be)
//
gpfnSetIdleTimeout(0xffff, 0);
gpfnSetIdleNotify(FALSE, 0);
return FALSE;
}
schAssert(wIdleWait != 0);
schDebugOut((DEB_IDLE, "Requesting %u-minute idle notification\n",
wIdleWait));
gpfnSetIdleTimeout(wIdleWait, 0);
gpfnSetIdleNotify(TRUE, 0);
return TRUE;
}
//+----------------------------------------------------------------------------
//
// Function: SetIdleLossNotificationFn
//
// Synopsis: Registers for idle loss notification.
//
// Returns: TRUE for success, and FALSE if unable to make the call.
//
//-----------------------------------------------------------------------------
BOOL
SetIdleLossNotificationFn()
{
schDebugOut((DEB_ITRACE, "SetIdleLossNotification()\n"));
if (!g_fIdleInitialized)
{
DBG_OUT("Calling SetIdleLossNotification before idle init!");
return FALSE;
}
schDebugOut((DEB_IDLE, "Requesting idle LOSS notification\n"));
gpfnSetBusyNotify(TRUE, 0);
return TRUE;
}
//+----------------------------------------------------------------------------
//
// Function: GetTimeIdle
//
// Synopsis: Obtains the length of time the machine has been idle.
//
//-----------------------------------------------------------------------------
DWORD
GetTimeIdle(void)
{
DWORD dwMinutes;
if (!g_fIdleInitialized)
{
DBG_OUT("Calling GetTimeIdle before idle init!");
dwMinutes = 0;
}
else
{
dwMinutes = gpfnGetIdleMinutes(0);
}
schDebugOut((DEB_IDLE, "User has been idle for %u minutes\n", dwMinutes));
return dwMinutes;
}
//+----------------------------------------------------------------------------
//
// Function: OnPowerChange
//
// Synopsis: Called when the machine's battery state changes.
//
// Arguments: [fGoingOnBattery] - set to true if going on battery power,
// false if going back on line power.
//
// Returns: hresults
//
//-----------------------------------------------------------------------------
HRESULT
OnPowerChange(BOOL fGoingOnBattery)
{
schDebugOut((DEB_ITRACE, "OnPowerChange: fGoingOnBattery = %s\n",
(fGoingOnBattery) ? "TRUE" : "FALSE"));
//
// Check to see if our battery state has changed, or if this is just
// a battery update
//
if (g_fOnBattery != fGoingOnBattery)
{
g_fOnBattery = fGoingOnBattery;
//
// Signal the main thread to recalculate the next wakeup time, since
// the calculation depends on whether the machine is on batteries.
// Do this by simply signaling the wakeup timer. This will cause a
// recalc.
//
g_pSched->SignalWakeupTimer();
if (fGoingOnBattery)
{
//
// Notify the job processor to kill any jobs with the
// TASK_FLAG_KILL_IF_GOING_ON_BATTERIES flag set.
//
CJobProcessor * pjp;
for (pjp = gpJobProcessorMgr->GetFirstProcessor(); pjp != NULL; )
{
pjp->KillIfFlagSet(TASK_FLAG_KILL_IF_GOING_ON_BATTERIES);
CJobProcessor * pjpNext = pjp->Next();
pjp->Release();
pjp = pjpNext;
}
}
}
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Function: InitBatteryNotification
//
// Synopsis: Initialize the battery event boolean.
//
// Returns: hresults
//
// Notes: Currently only Win95 supports power management.
//
//-----------------------------------------------------------------------------
HRESULT
InitBatteryNotification(void)
{
DWORD dwErr;
//
// Check current battery state and set bool accordingly.
//
SYSTEM_POWER_STATUS PwrStatus;
if (!GetSystemPowerStatus(&PwrStatus))
{
dwErr = GetLastError();
if (dwErr == ERROR_FILE_NOT_FOUND ||
dwErr == ERROR_CALL_NOT_IMPLEMENTED)
{
g_fOnBattery = FALSE;
schDebugOut((DEB_ITRACE,
"InitBatteryNotification: GetSystemPowerStatus"
" returned %u, g_fOnBattery set to FALSE\n",
dwErr));
return S_OK;
}
ERR_OUT("GetSystemPowerStatus", HRESULT_FROM_WIN32(dwErr));
return HRESULT_FROM_WIN32(dwErr);
}
g_fOnBattery = (PwrStatus.ACLineStatus == 0) ? TRUE : FALSE;
schDebugOut((DEB_ITRACE, "InitBatteryNotification: g_fOnBattery = %s\n",
(g_fOnBattery) ? "TRUE" : "FALSE"));
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Function: InitIdleDetection
//
// Synopsis: Called after the message window is created to initialize idle
// detection and hot corners.
//
// Arguments:
//
// Returns: hresults
//
//-----------------------------------------------------------------------------
HRESULT
InitIdleDetection()
{
TRACE_FUNCTION(InitIdleDetection);
DWORD dwErr;
//
// Look in the registry to see if idle detection is disabled.
//
long lErr;
HKEY hSchedKey = NULL;
lErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE, SCH_AGENT_KEY, 0, KEY_READ,
&hSchedKey);
if (lErr == ERROR_SUCCESS)
{
TCHAR tszInit[SCH_MED0BUF_LEN + 1];
DWORD cb = SCH_MED0BUF_LEN * sizeof(TCHAR);
lErr = RegQueryValueEx(hSchedKey, SCH_NOIDLE_VALUE, NULL, NULL,
(LPBYTE)tszInit, &cb);
RegCloseKey(hSchedKey);
if (lErr == ERROR_SUCCESS)
{
//
// The presence of the value is sufficient to disable idle
// detection. g_fIdleInitialized will remain FALSE, resulting
// in all idle operations being skipped.
//
schDebugOut((DEB_ITRACE, "Idle detection is disabled!!!!!!!!\n"));
return S_OK;
}
}
// load msidle.dll
if (g_hMsidleDll == NULL) {
g_hMsidleDll = LoadLibrary(TEXT("MSIDLE.DLL"));
if (g_hMsidleDll == NULL)
{
dwErr = GetLastError();
ERR_OUT("Load of msidle.dll", dwErr);
return HRESULT_FROM_WIN32(dwErr);
}
}
// get entry points
gpfnBeginIdleDetection = (_BEGINIDLEDETECTION)
GetProcAddress(g_hMsidleDll, (LPSTR)3);
gpfnEndIdleDetection = (_ENDIDLEDETECTION)
GetProcAddress(g_hMsidleDll, (LPSTR)4);
gpfnSetIdleTimeout = (_SETIDLETIMEOUT)
GetProcAddress(g_hMsidleDll, (LPSTR)5);
gpfnSetIdleNotify = (_SETIDLENOTIFY)
GetProcAddress(g_hMsidleDll, (LPSTR)6);
gpfnSetBusyNotify = (_SETBUSYNOTIFY)
GetProcAddress(g_hMsidleDll, (LPSTR)7);
gpfnGetIdleMinutes = (_GETIDLEMINUTES)
GetProcAddress(g_hMsidleDll, (LPSTR)8);
if (gpfnBeginIdleDetection == NULL ||
gpfnEndIdleDetection == NULL ||
gpfnSetIdleTimeout == NULL ||
gpfnSetIdleNotify == NULL ||
gpfnSetBusyNotify == NULL ||
gpfnGetIdleMinutes == NULL)
{
dwErr = GetLastError();
ERR_OUT("Getting msidle.dll entry point addresses", dwErr);
goto ErrExit;
}
// call start monitoring
dwErr = gpfnBeginIdleDetection(OnIdleNotify, SCH_DEFAULT_IDLE_TIME, 0);
if (dwErr)
{
ERR_OUT("Making initial idle call", dwErr);
goto ErrExit;
}
g_fIdleInitialized = TRUE;
return S_OK;
ErrExit:
FreeLibrary(g_hMsidleDll);
g_hMsidleDll = NULL;
gpfnBeginIdleDetection = NULL;
gpfnEndIdleDetection = NULL;
gpfnSetIdleTimeout = NULL;
gpfnSetIdleNotify = NULL;
gpfnSetBusyNotify = NULL;
gpfnGetIdleMinutes = NULL;
return HRESULT_FROM_WIN32(dwErr);
}
//+----------------------------------------------------------------------------
//
// Function: EndIdleDetection
//
// Synopsis: Stop idle detection.
//
// Arguments: None.
//
//-----------------------------------------------------------------------------
void
EndIdleDetection()
{
if (gpfnEndIdleDetection != NULL)
{
gpfnEndIdleDetection(0);
}
}