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