capsules_extra/ambient_light.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129
// Licensed under the Apache License, Version 2.0 or the MIT License.
// SPDX-License-Identifier: Apache-2.0 OR MIT
// Copyright Tock Contributors 2022.
//! Shared userland driver for light sensors.
//!
//! You need a device that provides the `hil::sensors::AmbientLight` trait.
//!
//! ```rust,ignore
//! # use kernel::{hil, static_init};
//!
//! let light = static_init!(
//! capsules::ambient_light::AmbientLight<'static>,
//! capsules::ambient_light::AmbientLight::new(isl29035,
//! board_kernel.create_grant(&grant_cap)));
//! hil::sensors::AmbientLight::set_client(isl29035, ambient_light);
//! ```
use core::cell::Cell;
use kernel::grant::{AllowRoCount, AllowRwCount, Grant, UpcallCount};
use kernel::hil;
use kernel::syscall::{CommandReturn, SyscallDriver};
use kernel::{ErrorCode, ProcessId};
/// Syscall driver number.
use capsules_core::driver;
pub const DRIVER_NUM: usize = driver::NUM::AmbientLight as usize;
/// IDs for subscribed upcalls.
mod upcall {
/// Subscribe to light intensity readings.
///
/// The callback signature is `fn(lux: usize)`, where `lux` is the light
/// intensity in lux (lx).
pub const LIGHT_INTENSITY: usize = 0;
/// Number of upcalls.
pub const COUNT: u8 = 1;
}
/// Per-process metadata
#[derive(Default)]
pub struct App {
pending: bool,
}
pub struct AmbientLight<'a> {
sensor: &'a dyn hil::sensors::AmbientLight<'a>,
command_pending: Cell<bool>,
apps: Grant<App, UpcallCount<{ upcall::COUNT }>, AllowRoCount<0>, AllowRwCount<0>>,
}
impl<'a> AmbientLight<'a> {
pub fn new(
sensor: &'a dyn hil::sensors::AmbientLight<'a>,
grant: Grant<App, UpcallCount<{ upcall::COUNT }>, AllowRoCount<0>, AllowRwCount<0>>,
) -> Self {
Self {
sensor,
command_pending: Cell::new(false),
apps: grant,
}
}
fn enqueue_sensor_reading(&self, processid: ProcessId) -> Result<(), ErrorCode> {
self.apps
.enter(processid, |app, _| {
if app.pending {
Err(ErrorCode::NOMEM)
} else {
app.pending = true;
if !self.command_pending.get() {
self.command_pending.set(true);
let _ = self.sensor.read_light_intensity();
}
Ok(())
}
})
.unwrap_or_else(|err| err.into())
}
}
impl SyscallDriver for AmbientLight<'_> {
/// Initiate light intensity readings
///
/// Sensor readings are coalesced if processes request them concurrently. If
/// multiple processes request have outstanding requests for a sensor
/// reading, only one command will be issued and the result is returned to
/// all subscribed processes.
///
/// ### `command_num`
///
/// - `0`: Check driver presence
/// - `1`: Start a light sensor reading
fn command(
&self,
command_num: usize,
_: usize,
_: usize,
processid: ProcessId,
) -> CommandReturn {
match command_num {
0 => CommandReturn::success(),
1 => {
let _ = self.enqueue_sensor_reading(processid);
CommandReturn::success()
}
_ => CommandReturn::failure(ErrorCode::NOSUPPORT),
}
}
fn allocate_grant(&self, processid: ProcessId) -> Result<(), kernel::process::Error> {
self.apps.enter(processid, |_, _| {})
}
}
impl hil::sensors::AmbientLightClient for AmbientLight<'_> {
fn callback(&self, lux: usize) {
self.command_pending.set(false);
self.apps.each(|_, app, upcalls| {
if app.pending {
app.pending = false;
upcalls
.schedule_upcall(upcall::LIGHT_INTENSITY, (lux, 0, 0))
.ok();
}
});
}
}