| Index: device/generic_sensor/platform_sensor_ambient_light_mac.cc
|
| diff --git a/device/generic_sensor/platform_sensor_ambient_light_mac.cc b/device/generic_sensor/platform_sensor_ambient_light_mac.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..8a72acf7d6d6108614916bc20c3fc7be626f956e
|
| --- /dev/null
|
| +++ b/device/generic_sensor/platform_sensor_ambient_light_mac.cc
|
| @@ -0,0 +1,175 @@
|
| +// Copyright 2016 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "device/generic_sensor/platform_sensor_ambient_light_mac.h"
|
| +
|
| +#include <stdint.h>
|
| +
|
| +#include <IOKit/IOMessage.h>
|
| +
|
| +#include "base/bind.h"
|
| +#include "device/base/synchronization/shared_memory_seqlock_buffer.h"
|
| +#include "device/generic_sensor/generic_sensor_consts.h"
|
| +#include "device/generic_sensor/platform_sensor_provider_mac.h"
|
| +
|
| +namespace {
|
| +
|
| +// Convert the value returned by the ambient light LMU sensor on Mac
|
| +// hardware to a lux value.
|
| +double LMUvalueToLux(uint64_t raw_value) {
|
| + // Conversion formula from regression.
|
| + // https://bugzilla.mozilla.org/show_bug.cgi?id=793728
|
| + // Let x = raw_value, then
|
| + // lux = -2.978303814*(10^-27)*x^4 + 2.635687683*(10^-19)*x^3 -
|
| + // 3.459747434*(10^-12)*x^2 + 3.905829689*(10^-5)*x - 0.1932594532
|
| +
|
| + static const long double k4 = pow(10.L, -7);
|
| + static const long double k3 = pow(10.L, -4);
|
| + static const long double k2 = pow(10.L, -2);
|
| + static const long double k1 = pow(10.L, 5);
|
| + long double scaled_value = raw_value / k1;
|
| +
|
| + long double lux_value =
|
| + (-3 * k4 * pow(scaled_value, 4)) + (2.6 * k3 * pow(scaled_value, 3)) +
|
| + (-3.4 * k2 * pow(scaled_value, 2)) + (3.9 * scaled_value) - 0.19;
|
| +
|
| + double lux = ceil(static_cast<double>(lux_value));
|
| + return lux > 0 ? lux : 0;
|
| +}
|
| +
|
| +} // namespace
|
| +
|
| +namespace device {
|
| +
|
| +enum LmuFunctionIndex {
|
| + kGetSensorReadingID = 0, // getSensorReading(int *, int *)
|
| +};
|
| +
|
| +PlatformSensorAmbientLightMac::PlatformSensorAmbientLightMac(
|
| + mojo::ScopedSharedBufferMapping mapping,
|
| + PlatformSensorProvider* provider)
|
| + : PlatformSensor(mojom::SensorType::AMBIENT_LIGHT,
|
| + std::move(mapping),
|
| + provider),
|
| + light_sensor_port_(nullptr),
|
| + current_lux_(0.0) {}
|
| +
|
| +PlatformSensorAmbientLightMac::~PlatformSensorAmbientLightMac() = default;
|
| +
|
| +mojom::ReportingMode PlatformSensorAmbientLightMac::GetReportingMode() {
|
| + return mojom::ReportingMode::ON_CHANGE;
|
| +}
|
| +
|
| +bool PlatformSensorAmbientLightMac::CheckSensorConfiguration(
|
| + const PlatformSensorConfiguration& configuration) {
|
| + return configuration.frequency() > 0 &&
|
| + configuration.frequency() <=
|
| + mojom::SensorConfiguration::kMaxAllowedFrequency;
|
| +}
|
| +
|
| +PlatformSensorConfiguration
|
| +PlatformSensorAmbientLightMac::GetDefaultConfiguration() {
|
| + PlatformSensorConfiguration default_configuration;
|
| + default_configuration.set_frequency(kDefaultAmbientLightFrequencyHz);
|
| + return default_configuration;
|
| +}
|
| +
|
| +void PlatformSensorAmbientLightMac::IOServiceCallback(void* context,
|
| + io_service_t service,
|
| + natural_t message_type,
|
| + void* message_argument) {
|
| + PlatformSensorAmbientLightMac* sensor =
|
| + static_cast<PlatformSensorAmbientLightMac*>(context);
|
| + if (!sensor->ReadAndUpdate()) {
|
| + sensor->NotifySensorError();
|
| + sensor->StopSensor();
|
| + }
|
| +}
|
| +
|
| +bool PlatformSensorAmbientLightMac::StartSensor(
|
| + const PlatformSensorConfiguration& configuration) {
|
| + // Tested and verified by riju that the following call works on
|
| + // MacBookPro9,1 : Macbook Pro 15" (Mid 2012 model)
|
| + // MacBookPro10,1 : Macbook Pro 15" (Retina Display, Early 2013 model).
|
| + // MacBookPro10,2 : Macbook Pro 13" (Retina Display, Early 2013 model).
|
| + // MacBookAir5,2 : Macbook Air 13" (Mid 2012 model) (by François Beaufort).
|
| + // MacBookAir6,2 : Macbook Air 13" (Mid 2013 model).
|
| + // Testing plans : please download the code and follow the comments :-
|
| + // https://gist.github.com/riju/74af8c81a665e412d122/
|
| + // and add an entry here about the model and the status returned by the code.
|
| +
|
| + // Look up a registered IOService object whose class is AppleLMUController.
|
| + light_sensor_service_.reset(IOServiceGetMatchingService(
|
| + kIOMasterPortDefault, IOServiceMatching("AppleLMUController")));
|
| +
|
| + // Return early if the ambient light sensor is not present.
|
| + if (!light_sensor_service_)
|
| + return false;
|
| +
|
| + light_sensor_port_.reset(IONotificationPortCreate(kIOMasterPortDefault));
|
| + if (!light_sensor_port_.is_valid())
|
| + return false;
|
| +
|
| + IONotificationPortSetDispatchQueue(
|
| + light_sensor_port_.get(),
|
| + dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0));
|
| +
|
| + kern_return_t kr = IOServiceAddInterestNotification(
|
| + light_sensor_port_.get(), light_sensor_service_, kIOGeneralInterest,
|
| + IOServiceCallback, this, light_sensor_notification_.InitializeInto());
|
| + if (kr != KERN_SUCCESS)
|
| + return false;
|
| +
|
| + kr = IOServiceAddInterestNotification(
|
| + light_sensor_port_.get(), light_sensor_service_, kIOBusyInterest,
|
| + IOServiceCallback, this,
|
| + light_sensor_busy_notification_.InitializeInto());
|
| + if (kr != KERN_SUCCESS)
|
| + return false;
|
| +
|
| + kr = IOServiceOpen(light_sensor_service_, mach_task_self(), 0,
|
| + light_sensor_object_.InitializeInto());
|
| + if (kr != KERN_SUCCESS)
|
| + return false;
|
| +
|
| + bool success = ReadAndUpdate();
|
| + if (!success)
|
| + StopSensor();
|
| +
|
| + return success;
|
| +}
|
| +
|
| +void PlatformSensorAmbientLightMac::StopSensor() {
|
| + light_sensor_port_.reset();
|
| + light_sensor_notification_.reset();
|
| + light_sensor_busy_notification_.reset();
|
| + light_sensor_object_.reset();
|
| + light_sensor_service_.reset();
|
| + current_lux_ = 0.0;
|
| +}
|
| +
|
| +bool PlatformSensorAmbientLightMac::ReadAndUpdate() {
|
| + uint32_t scalar_output_count = 2;
|
| + uint64_t lux_values[2];
|
| + kern_return_t kr = IOConnectCallMethod(
|
| + light_sensor_object_, LmuFunctionIndex::kGetSensorReadingID, nullptr, 0,
|
| + nullptr, 0, lux_values, &scalar_output_count, nullptr, 0);
|
| +
|
| + if (kr != KERN_SUCCESS)
|
| + return false;
|
| +
|
| + uint64_t mean = (lux_values[0] + lux_values[1]) / 2;
|
| + double lux = LMUvalueToLux(mean);
|
| + if (lux == current_lux_)
|
| + return true;
|
| + current_lux_ = lux;
|
| +
|
| + SensorReading reading;
|
| + reading.timestamp = (base::TimeTicks::Now() - base::TimeTicks()).InSecondsF();
|
| + reading.values[0] = current_lux_;
|
| + UpdateSensorReading(reading, true);
|
| + return true;
|
| +}
|
| +
|
| +} // namespace device
|
|
|