Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(310)

Side by Side Diff: device/generic_sensor/platform_sensor_ambient_light_mac.cc

Issue 2865263002: Move //device/generic_sensor to be part of the internal implementation of the Device Service. (Closed)
Patch Set: code rebase Created 3 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(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
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698