OLD | NEW |
(Empty) | |
| 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. |
| 4 |
| 5 #include "device/generic_sensor/platform_sensor_ambient_light_mac.h" |
| 6 |
| 7 #include <stdint.h> |
| 8 |
| 9 #include <IOKit/IOMessage.h> |
| 10 |
| 11 #include "base/bind.h" |
| 12 #include "device/base/synchronization/shared_memory_seqlock_buffer.h" |
| 13 #include "device/generic_sensor/generic_sensor_consts.h" |
| 14 #include "device/generic_sensor/platform_sensor_provider_mac.h" |
| 15 |
| 16 namespace { |
| 17 |
| 18 // Convert the value returned by the ambient light LMU sensor on Mac |
| 19 // hardware to a lux value. |
| 20 double LMUvalueToLux(uint64_t raw_value) { |
| 21 // Conversion formula from regression. |
| 22 // https://bugzilla.mozilla.org/show_bug.cgi?id=793728 |
| 23 // Let x = raw_value, then |
| 24 // lux = -2.978303814*(10^-27)*x^4 + 2.635687683*(10^-19)*x^3 - |
| 25 // 3.459747434*(10^-12)*x^2 + 3.905829689*(10^-5)*x - 0.1932594532 |
| 26 |
| 27 static const long double k4 = pow(10.L, -7); |
| 28 static const long double k3 = pow(10.L, -4); |
| 29 static const long double k2 = pow(10.L, -2); |
| 30 static const long double k1 = pow(10.L, 5); |
| 31 long double scaled_value = raw_value / k1; |
| 32 |
| 33 long double lux_value = |
| 34 (-3 * k4 * pow(scaled_value, 4)) + (2.6 * k3 * pow(scaled_value, 3)) + |
| 35 (-3.4 * k2 * pow(scaled_value, 2)) + (3.9 * scaled_value) - 0.19; |
| 36 |
| 37 double lux = ceil(static_cast<double>(lux_value)); |
| 38 return lux > 0 ? lux : 0; |
| 39 } |
| 40 |
| 41 } // namespace |
| 42 |
| 43 namespace device { |
| 44 |
| 45 enum LmuFunctionIndex { |
| 46 kGetSensorReadingID = 0, // getSensorReading(int *, int *) |
| 47 }; |
| 48 |
| 49 PlatformSensorAmbientLightMac::PlatformSensorAmbientLightMac( |
| 50 mojo::ScopedSharedBufferMapping mapping, |
| 51 PlatformSensorProvider* provider) |
| 52 : PlatformSensor(mojom::SensorType::AMBIENT_LIGHT, |
| 53 std::move(mapping), |
| 54 provider), |
| 55 light_sensor_port_(nullptr), |
| 56 current_lux_(0.0) {} |
| 57 |
| 58 PlatformSensorAmbientLightMac::~PlatformSensorAmbientLightMac() = default; |
| 59 |
| 60 mojom::ReportingMode PlatformSensorAmbientLightMac::GetReportingMode() { |
| 61 return mojom::ReportingMode::ON_CHANGE; |
| 62 } |
| 63 |
| 64 bool PlatformSensorAmbientLightMac::CheckSensorConfiguration( |
| 65 const PlatformSensorConfiguration& configuration) { |
| 66 return configuration.frequency() > 0 && |
| 67 configuration.frequency() <= |
| 68 mojom::SensorConfiguration::kMaxAllowedFrequency; |
| 69 } |
| 70 |
| 71 PlatformSensorConfiguration |
| 72 PlatformSensorAmbientLightMac::GetDefaultConfiguration() { |
| 73 PlatformSensorConfiguration default_configuration; |
| 74 default_configuration.set_frequency(kDefaultAmbientLightFrequencyHz); |
| 75 return default_configuration; |
| 76 } |
| 77 |
| 78 void PlatformSensorAmbientLightMac::IOServiceCallback(void* context, |
| 79 io_service_t service, |
| 80 natural_t message_type, |
| 81 void* message_argument) { |
| 82 PlatformSensorAmbientLightMac* sensor = |
| 83 static_cast<PlatformSensorAmbientLightMac*>(context); |
| 84 if (!sensor->ReadAndUpdate()) { |
| 85 sensor->NotifySensorError(); |
| 86 sensor->StopSensor(); |
| 87 } |
| 88 } |
| 89 |
| 90 bool PlatformSensorAmbientLightMac::StartSensor( |
| 91 const PlatformSensorConfiguration& configuration) { |
| 92 // Tested and verified by riju that the following call works on |
| 93 // MacBookPro9,1 : Macbook Pro 15" (Mid 2012 model) |
| 94 // MacBookPro10,1 : Macbook Pro 15" (Retina Display, Early 2013 model). |
| 95 // MacBookPro10,2 : Macbook Pro 13" (Retina Display, Early 2013 model). |
| 96 // MacBookAir5,2 : Macbook Air 13" (Mid 2012 model) (by François Beaufort). |
| 97 // MacBookAir6,2 : Macbook Air 13" (Mid 2013 model). |
| 98 // Testing plans : please download the code and follow the comments :- |
| 99 // https://gist.github.com/riju/74af8c81a665e412d122/ |
| 100 // and add an entry here about the model and the status returned by the code. |
| 101 |
| 102 // Look up a registered IOService object whose class is AppleLMUController. |
| 103 light_sensor_service_.reset(IOServiceGetMatchingService( |
| 104 kIOMasterPortDefault, IOServiceMatching("AppleLMUController"))); |
| 105 |
| 106 // Return early if the ambient light sensor is not present. |
| 107 if (!light_sensor_service_) |
| 108 return false; |
| 109 |
| 110 light_sensor_port_.reset(IONotificationPortCreate(kIOMasterPortDefault)); |
| 111 if (!light_sensor_port_.is_valid()) |
| 112 return false; |
| 113 |
| 114 IONotificationPortSetDispatchQueue( |
| 115 light_sensor_port_.get(), |
| 116 dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)); |
| 117 |
| 118 kern_return_t kr = IOServiceAddInterestNotification( |
| 119 light_sensor_port_.get(), light_sensor_service_, kIOGeneralInterest, |
| 120 IOServiceCallback, this, light_sensor_notification_.InitializeInto()); |
| 121 if (kr != KERN_SUCCESS) |
| 122 return false; |
| 123 |
| 124 kr = IOServiceAddInterestNotification( |
| 125 light_sensor_port_.get(), light_sensor_service_, kIOBusyInterest, |
| 126 IOServiceCallback, this, |
| 127 light_sensor_busy_notification_.InitializeInto()); |
| 128 if (kr != KERN_SUCCESS) |
| 129 return false; |
| 130 |
| 131 kr = IOServiceOpen(light_sensor_service_, mach_task_self(), 0, |
| 132 light_sensor_object_.InitializeInto()); |
| 133 if (kr != KERN_SUCCESS) |
| 134 return false; |
| 135 |
| 136 bool success = ReadAndUpdate(); |
| 137 if (!success) |
| 138 StopSensor(); |
| 139 |
| 140 return success; |
| 141 } |
| 142 |
| 143 void PlatformSensorAmbientLightMac::StopSensor() { |
| 144 light_sensor_port_.reset(); |
| 145 light_sensor_notification_.reset(); |
| 146 light_sensor_busy_notification_.reset(); |
| 147 light_sensor_object_.reset(); |
| 148 light_sensor_service_.reset(); |
| 149 current_lux_ = 0.0; |
| 150 } |
| 151 |
| 152 bool PlatformSensorAmbientLightMac::ReadAndUpdate() { |
| 153 uint32_t scalar_output_count = 2; |
| 154 uint64_t lux_values[2]; |
| 155 kern_return_t kr = IOConnectCallMethod( |
| 156 light_sensor_object_, LmuFunctionIndex::kGetSensorReadingID, nullptr, 0, |
| 157 nullptr, 0, lux_values, &scalar_output_count, nullptr, 0); |
| 158 |
| 159 if (kr != KERN_SUCCESS) |
| 160 return false; |
| 161 |
| 162 uint64_t mean = (lux_values[0] + lux_values[1]) / 2; |
| 163 double lux = LMUvalueToLux(mean); |
| 164 if (lux == current_lux_) |
| 165 return true; |
| 166 current_lux_ = lux; |
| 167 |
| 168 SensorReading reading; |
| 169 reading.timestamp = (base::TimeTicks::Now() - base::TimeTicks()).InSecondsF(); |
| 170 reading.values[0] = current_lux_; |
| 171 UpdateSensorReading(reading, true); |
| 172 return true; |
| 173 } |
| 174 |
| 175 } // namespace device |
OLD | NEW |