Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 // Implementation based on sample code from | 5 // Implementation based on sample code from |
| 6 // http://developer.apple.com/library/mac/#qa/qa1340/_index.html. | 6 // http://developer.apple.com/library/mac/#qa/qa1340/_index.html. |
| 7 | 7 |
| 8 #include "base/system_monitor/system_monitor.h" | 8 #include "base/system_monitor/system_monitor.h" |
| 9 | 9 |
| 10 #import <Carbon/Carbon.h> | |
|
csilv
2012/02/13 05:49:28
use #include instead, #import only for pure obj-c
vandebo (ex-Chrome)
2012/02/14 21:23:23
Done.
| |
| 11 #include <CoreFoundation/CoreFoundation.h> | |
| 10 #include <IOKit/pwr_mgt/IOPMLib.h> | 12 #include <IOKit/pwr_mgt/IOPMLib.h> |
| 11 #include <IOKit/IOMessage.h> | 13 #include <IOKit/IOMessage.h> |
| 12 | 14 |
| 15 #include "base/mac/foundation_util.h" | |
| 16 #include "base/sys_string_conversions.h" | |
| 17 | |
| 13 namespace base { | 18 namespace base { |
| 14 | 19 |
| 15 namespace { | 20 namespace { |
| 16 | 21 |
| 17 io_connect_t g_system_power_io_port = 0; | 22 io_connect_t g_system_power_io_port = 0; |
| 18 IONotificationPortRef g_notification_port_ref = 0; | 23 IONotificationPortRef g_notification_port_ref = 0; |
| 19 io_object_t g_notifier_object = 0; | 24 io_object_t g_notifier_object = 0; |
| 20 | 25 |
| 21 void SystemPowerEventCallback(void*, | 26 void SystemPowerEventCallback(void*, |
| 22 io_service_t service, | 27 io_service_t service, |
| 23 natural_t message_type, | 28 natural_t message_type, |
| 24 void* message_argument) { | 29 void* message_argument) { |
| 25 SystemMonitor* sys_monitor = SystemMonitor::Get(); | 30 SystemMonitor* sys_monitor = SystemMonitor::Get(); |
| 26 DCHECK(sys_monitor); | 31 DCHECK(sys_monitor); |
| 27 switch (message_type) { | 32 switch (message_type) { |
| 28 case kIOMessageSystemWillSleep: | 33 case kIOMessageSystemWillSleep: |
| 29 sys_monitor->ProcessPowerMessage(SystemMonitor::SUSPEND_EVENT); | 34 sys_monitor->ProcessPowerMessage(SystemMonitor::SUSPEND_EVENT); |
| 30 IOAllowPowerChange(g_system_power_io_port, | 35 IOAllowPowerChange(g_system_power_io_port, |
| 31 reinterpret_cast<int>(message_argument)); | 36 reinterpret_cast<int>(message_argument)); |
| 32 break; | 37 break; |
| 33 | 38 |
| 34 case kIOMessageSystemWillPowerOn: | 39 case kIOMessageSystemWillPowerOn: |
| 35 sys_monitor->ProcessPowerMessage(SystemMonitor::RESUME_EVENT); | 40 sys_monitor->ProcessPowerMessage(SystemMonitor::RESUME_EVENT); |
| 36 break; | 41 break; |
| 37 } | 42 } |
| 38 } | 43 } |
| 39 | 44 |
| 45 bool GetDeviceInfo(unsigned long device_number, std::string* name, | |
| 46 FilePath* location) { | |
| 47 ICACopyObjectPropertyDictionaryPB properties_request; | |
| 48 properties_request.object = device_number; | |
| 49 CFDictionaryRef device_properties; | |
| 50 properties_request.theDict = &device_properties; | |
| 51 OSErr ret = ICACopyObjectPropertyDictionary(&properties_request, NULL); | |
| 52 CHECK_EQ(ret, noErr); | |
| 53 | |
| 54 // For now, we only support mass storage media devices. | |
| 55 CFStringRef volume = mac::GetValueFromDictionary<CFStringRef>( | |
| 56 device_properties, CFSTR("volume")); | |
| 57 if (volume == NULL) { | |
| 58 CFRelease(device_properties); | |
| 59 return false; | |
| 60 } | |
| 61 *location = FilePath("/Volumes/").Append(SysCFStringRefToUTF8(volume)); | |
| 62 | |
| 63 CFStringRef device_name = mac::GetValueFromDictionary<CFStringRef>( | |
| 64 device_properties, CFSTR("ICAUserAssignedDeviceNameKey")); | |
| 65 if (device_name == NULL) { | |
| 66 device_name = mac::GetValueFromDictionary<CFStringRef>( | |
| 67 device_properties, CFSTR("ifil")); | |
| 68 } | |
| 69 if (device_name == NULL) { | |
| 70 CFRelease(device_properties); | |
| 71 return false; | |
| 72 } | |
| 73 *name = SysCFStringRefToUTF8(device_name); | |
| 74 | |
| 75 CFRelease(device_properties); | |
| 76 return true; | |
| 77 } | |
| 78 | |
| 79 void MediaDeviceNotificationCallback(CFStringRef notification_type, | |
| 80 CFDictionaryRef notification_dictionary) { | |
| 81 bool attach = false; | |
| 82 if (CFStringCompare(notification_type, kICANotificationTypeDeviceAdded, 0) == | |
| 83 kCFCompareEqualTo) { | |
| 84 attach = true; | |
| 85 } else if (CFStringCompare(notification_type, | |
| 86 kICANotificationTypeDeviceRemoved, 0) != | |
| 87 kCFCompareEqualTo) { | |
| 88 return; | |
| 89 } | |
| 90 CFNumberRef refcon_object = mac::GetValueFromDictionary<CFNumberRef>( | |
| 91 notification_dictionary, kICARefconKey); | |
| 92 unsigned long refcon; | |
| 93 CFNumberGetValue(refcon_object, kCFNumberLongType, &refcon); | |
| 94 SystemMonitor* system_monitor = reinterpret_cast<SystemMonitor*>(refcon); | |
| 95 | |
| 96 CFNumberRef device_number_object = mac::GetValueFromDictionary<CFNumberRef>( | |
| 97 notification_dictionary, kICANotificationDeviceICAObjectKey); | |
| 98 unsigned long device_number; | |
| 99 CFNumberGetValue(device_number_object, kCFNumberIntType, &device_number); | |
| 100 if (attach) { | |
| 101 std::string device_name; | |
| 102 FilePath location; | |
| 103 if (GetDeviceInfo(device_number, &device_name, &location)) { | |
| 104 system_monitor->ProcessMediaDeviceAttached(device_number, device_name, | |
| 105 location); | |
| 106 } | |
| 107 } else { | |
| 108 system_monitor->ProcessMediaDeviceDetached(device_number); | |
| 109 } | |
| 110 } | |
| 111 | |
| 40 } // namespace | 112 } // namespace |
| 41 | 113 |
| 42 // The reason we can't include this code in the constructor is because | 114 // The reason we can't include this code in the constructor is because |
| 43 // PlatformInit() requires an active runloop and the IO port needs to be | 115 // PlatformInit() requires an active runloop and the IO port needs to be |
| 44 // allocated at sandbox initialization time, before there's a runloop. | 116 // allocated at sandbox initialization time, before there's a runloop. |
| 45 // See crbug.com/83783 . | 117 // See crbug.com/83783 . |
| 46 | 118 |
| 47 // static | 119 // static |
| 48 void SystemMonitor::AllocateSystemIOPorts() { | 120 void SystemMonitor::AllocateSystemIOPorts() { |
| 49 DCHECK_EQ(g_system_power_io_port, 0u); | 121 DCHECK_EQ(g_system_power_io_port, 0u); |
| (...skipping 12 matching lines...) Expand all Loading... | |
| 62 // object. | 134 // object. |
| 63 DCHECK_NE(g_system_power_io_port, 0u); | 135 DCHECK_NE(g_system_power_io_port, 0u); |
| 64 if (g_system_power_io_port == 0) | 136 if (g_system_power_io_port == 0) |
| 65 return; | 137 return; |
| 66 | 138 |
| 67 // Add the notification port to the application runloop | 139 // Add the notification port to the application runloop |
| 68 CFRunLoopAddSource( | 140 CFRunLoopAddSource( |
| 69 CFRunLoopGetCurrent(), | 141 CFRunLoopGetCurrent(), |
| 70 IONotificationPortGetRunLoopSource(g_notification_port_ref), | 142 IONotificationPortGetRunLoopSource(g_notification_port_ref), |
| 71 kCFRunLoopCommonModes); | 143 kCFRunLoopCommonModes); |
| 144 | |
| 145 ICARegisterForEventNotificationPB notification_request; | |
| 146 notification_request.header.refcon = reinterpret_cast<unsigned long>(this); | |
| 147 notification_request.objectOfInterest = 0; // Zero means all objects | |
| 148 CFStringRef events_of_interest_array[] = {kICANotificationTypeDeviceAdded, | |
| 149 kICANotificationTypeDeviceRemoved}; | |
| 150 CFArrayRef events_of_interest = | |
| 151 CFArrayCreate(NULL, (const void**)&events_of_interest_array, 2, | |
| 152 &kCFTypeArrayCallBacks); | |
| 153 notification_request.eventsOfInterest = events_of_interest; | |
| 154 notification_request.notificationProc = &MediaDeviceNotificationCallback; | |
| 155 notification_request.options = NULL; | |
| 156 OSErr err = ICARegisterForEventNotification(¬ification_request, NULL); | |
| 157 CHECK_EQ(err, noErr); | |
| 72 } | 158 } |
| 73 | 159 |
| 74 void SystemMonitor::PlatformDestroy() { | 160 void SystemMonitor::PlatformDestroy() { |
| 75 DCHECK_NE(g_system_power_io_port, 0u); | 161 DCHECK_NE(g_system_power_io_port, 0u); |
| 76 if (g_system_power_io_port == 0) | 162 if (g_system_power_io_port == 0) |
| 77 return; | 163 return; |
| 78 | 164 |
| 79 // Remove the sleep notification port from the application runloop | 165 // Remove the sleep notification port from the application runloop |
| 80 CFRunLoopRemoveSource( | 166 CFRunLoopRemoveSource( |
| 81 CFRunLoopGetCurrent(), | 167 CFRunLoopGetCurrent(), |
| 82 IONotificationPortGetRunLoopSource(g_notification_port_ref), | 168 IONotificationPortGetRunLoopSource(g_notification_port_ref), |
| 83 kCFRunLoopCommonModes); | 169 kCFRunLoopCommonModes); |
| 84 | 170 |
| 85 // Deregister for system sleep notifications | 171 // Deregister for system sleep notifications |
| 86 IODeregisterForSystemPower(&g_notifier_object); | 172 IODeregisterForSystemPower(&g_notifier_object); |
| 87 | 173 |
| 88 // IORegisterForSystemPower implicitly opens the Root Power Domain IOService, | 174 // IORegisterForSystemPower implicitly opens the Root Power Domain IOService, |
| 89 // so we close it here. | 175 // so we close it here. |
| 90 IOServiceClose(g_system_power_io_port); | 176 IOServiceClose(g_system_power_io_port); |
| 91 | 177 |
| 92 g_system_power_io_port = 0; | 178 g_system_power_io_port = 0; |
| 93 | 179 |
| 94 // Destroy the notification port allocated by IORegisterForSystemPower. | 180 // Destroy the notification port allocated by IORegisterForSystemPower. |
| 95 IONotificationPortDestroy(g_notification_port_ref); | 181 IONotificationPortDestroy(g_notification_port_ref); |
| 96 } | 182 } |
| 97 | 183 |
| 98 } // namespace base | 184 } // namespace base |
| OLD | NEW |