VirtualBox

source: vbox/trunk/src/VBox/Main/darwin/HostPowerDarwin.cpp@ 15633

Last change on this file since 15633 was 14527, checked in by vboxsync, 16 years ago

Main-OSX: added initial power change notification

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 5.9 KB
Line 
1/** @file
2 *
3 * VirtualBox interface to host's power notification service
4 */
5
6/*
7 * Copyright (C) 2008 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#include "HostPower.h"
23#include "Logging.h"
24
25#include <IOKit/IOMessage.h>
26
27HostPowerServiceDarwin::HostPowerServiceDarwin (VirtualBox *aVirtualBox)
28 : HostPowerService (aVirtualBox),
29 mThread (NULL),
30 mRootPort (MACH_PORT_NULL),
31 mNotifyPort (nil),
32 mRunLoop (nil)
33{
34 /* Create the new worker thread. */
35 int rc = RTThreadCreate (&mThread, HostPowerServiceDarwin::powerChangeNotificationThread, this, 65536,
36 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "MainPower");
37
38 if (RT_FAILURE (rc))
39 LogFlow (("RTThreadCreate failed with %Rrc\n", rc));
40}
41
42HostPowerServiceDarwin::~HostPowerServiceDarwin()
43{
44 /* Jump out of the run loop. */
45 CFRunLoopStop (mRunLoop);
46 /* Remove the sleep notification port from the application runloop. */
47 CFRunLoopRemoveSource (CFRunLoopGetCurrent(),
48 IONotificationPortGetRunLoopSource (mNotifyPort),
49 kCFRunLoopCommonModes);
50 /* Deregister for system sleep notifications. */
51 IODeregisterForSystemPower (&mNotifierObject);
52 /* IORegisterForSystemPower implicitly opens the Root Power Domain
53 * IOService so we close it here. */
54 IOServiceClose (mRootPort);
55 /* Destroy the notification port allocated by IORegisterForSystemPower */
56 IONotificationPortDestroy (mNotifyPort);
57}
58
59DECLCALLBACK(int) HostPowerServiceDarwin::powerChangeNotificationThread (RTTHREAD ThreadSelf, void *pInstance)
60{
61 HostPowerServiceDarwin *pPowerObj = static_cast<HostPowerServiceDarwin *> (pInstance);
62
63// OSErr retCode = AEInstallEventHandler(kAEMacPowerMgtEvt, kAEMacLowPowerSaveData,
64// HostPowerServiceDarwin::lowPowerEventHandler,
65// pPowerObj, false);
66
67 /* Register to receive system sleep notifications */
68 pPowerObj->mRootPort = IORegisterForSystemPower (pPowerObj, &pPowerObj->mNotifyPort,
69 HostPowerServiceDarwin::powerChangeNotificationHandler,
70 &pPowerObj->mNotifierObject);
71 if (pPowerObj->mRootPort == MACH_PORT_NULL)
72 {
73 LogFlow (("IORegisterForSystemPower failed\n"));
74 return VERR_NOT_SUPPORTED;
75 }
76 pPowerObj->mRunLoop = CFRunLoopGetCurrent();
77 /* Add the notification port to the application runloop */
78 CFRunLoopAddSource (pPowerObj->mRunLoop,
79 IONotificationPortGetRunLoopSource (pPowerObj->mNotifyPort),
80 kCFRunLoopCommonModes);
81 /* Start the run loop. This blocks. */
82 CFRunLoopRun();
83
84 return VINF_SUCCESS;
85}
86
87void HostPowerServiceDarwin::powerChangeNotificationHandler (void *pData, io_service_t service, natural_t messageType, void *pMessageArgument)
88{
89 HostPowerServiceDarwin *pPowerObj = static_cast<HostPowerServiceDarwin *> (pData);
90// printf( "messageType %08lx, arg %08lx\n", (long unsigned int)messageType, (long unsigned int)pMessageArgument );
91
92 switch (messageType)
93 {
94 case kIOMessageCanSystemSleep:
95 {
96 /* Idle sleep is about to kick in. This message will not be
97 * sent for forced sleep. Applications have a chance to prevent
98 * sleep by calling IOCancelPowerChange. Most applications
99 * should not prevent idle sleep. Power Management waits up to
100 * 30 seconds for you to either allow or deny idle sleep. If
101 * you don't acknowledge this power change by calling either
102 * IOAllowPowerChange or IOCancelPowerChange, the system will
103 * wait 30 seconds then go to sleep. */
104 IOAllowPowerChange (pPowerObj->mRootPort, reinterpret_cast<long> (pMessageArgument));
105 break;
106 }
107 case kIOMessageSystemWillSleep:
108 {
109 /* The system will go for sleep. */
110 pPowerObj->notify (HostPowerEvent_Suspend);
111 /* If you do not call IOAllowPowerChange or IOCancelPowerChange to
112 * acknowledge this message, sleep will be delayed by 30 seconds.
113 * NOTE: If you call IOCancelPowerChange to deny sleep it returns
114 * kIOReturnSuccess, however the system WILL still go to sleep. */
115 IOAllowPowerChange (pPowerObj->mRootPort, reinterpret_cast<long> (pMessageArgument));
116 break;
117 }
118 case kIOMessageSystemWillPowerOn:
119 {
120 /* System has started the wake up process. */
121 break;
122 }
123 case kIOMessageSystemHasPoweredOn:
124 {
125 /* System has finished the wake up process. */
126 pPowerObj->notify (HostPowerEvent_Resume);
127 break;
128 }
129 default:
130 break;
131 }
132}
133
134OSErr HostPowerServiceDarwin::lowPowerEventHandler (const AppleEvent *event, AppleEvent *replyEvent, long data)
135{
136 printf ("low power event\n");
137 LogRel (("low power event\n"));
138// HostPowerServiceDarwin *pPowerObj = reinterpret_cast<HostPowerServiceDarwin *> (data);
139// pPowerObj->notify(HostPowerEvent_BatteryLow);
140 return noErr;
141}
142
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette