capsules_extra/buzzer_pwm.rs
1// Licensed under the Apache License, Version 2.0 or the MIT License.
2// SPDX-License-Identifier: Apache-2.0 OR MIT
3// Copyright Tock Contributors 2022.
4
5//! Service capsule for a buzzer that uses a PWM pin.
6//!
7//! ## Instantiation
8//!
9//! Instantiate the capsule for use as a service capsule, using a virtual pwm buzzer
10//! and a virtual alarm. For example:
11//!
12//! ```rust,ignore
13//! # use kernel::static_init;
14//!
15//! let mux_pwm = static_init!(
16//! capsules::virtual_pwm::MuxPwm<'static, nrf52833::pwm::Pwm>,
17//! capsules::virtual_pwm::MuxPwm::new(&base_peripherals.pwm0)
18//! );
19//! let virtual_pwm_buzzer = static_init!(
20//! capsules::virtual_pwm::PwmPinUser<'static, nrf52833::pwm::Pwm>,
21//! capsules::virtual_pwm::PwmPinUser::new(
22//! mux_pwm,
23//! nrf52833::pinmux::Pinmux::new(SPEAKER_PIN as u32)
24//! )
25//! );
26//! virtual_pwm_buzzer.add_to_mux();
27//!
28//! let virtual_alarm_buzzer = static_init!(
29//! capsules::virtual_alarm::VirtualMuxAlarm<'static, nrf52833::rtc::Rtc>,
30//! capsules::virtual_alarm::VirtualMuxAlarm::new(mux_alarm)
31//! );
32//! virtual_alarm_buzzer.setup();
33//!
34//! let pwm_buzzer = static_init!(
35//! capsules::buzzer_pwm::PwmBuzzer<
36//! 'static,
37//! capsules::virtual_alarm::VirtualMuxAlarm<'static, nrf52833::rtc::Rtc>,
38//! capsules::virtual_pwm::PwmPinUser<'static, nrf52833::pwm::Pwm>,
39//! >,
40//! capsules::buzzer_pwm::PwmBuzzer::new(
41//! virtual_pwm_buzzer,
42//! virtual_alarm_buzzer,
43//! capsules::buzzer_pwm::DEFAULT_MAX_BUZZ_TIME_MS,
44//! )
45//! );
46//!
47//! pwm_buzzer.set_client(buzzer);
48//!
49//! virtual_alarm_buzzer.set_alarm_client(pwm_buzzer);
50//!
51//! ```
52
53use core::cmp;
54
55use kernel::hil;
56use kernel::hil::buzzer::BuzzerClient;
57use kernel::hil::time::Frequency;
58use kernel::utilities::cells::OptionalCell;
59use kernel::ErrorCode;
60
61/// Standard max buzz time.
62pub const DEFAULT_MAX_BUZZ_TIME_MS: usize = 5000;
63
64pub struct PwmBuzzer<'a, A: hil::time::Alarm<'a>, P: hil::pwm::PwmPin> {
65 /// The underlying PWM generator to make the buzzer buzz.
66 pwm_pin: &'a P,
67 /// Alarm to stop the PWM after some time.
68 alarm: &'a A,
69 /// Max buzz time.
70 max_duration_ms: usize,
71 /// The client currently using the service capsule.
72 client: OptionalCell<&'a dyn BuzzerClient>,
73}
74
75impl<'a, A: hil::time::Alarm<'a>, P: hil::pwm::PwmPin> PwmBuzzer<'a, A, P> {
76 pub fn new(pwm_pin: &'a P, alarm: &'a A, max_duration_ms: usize) -> PwmBuzzer<'a, A, P> {
77 PwmBuzzer {
78 pwm_pin,
79 alarm,
80 client: OptionalCell::empty(),
81 max_duration_ms,
82 }
83 }
84}
85
86impl<'a, A: hil::time::Alarm<'a>, P: hil::pwm::PwmPin> hil::buzzer::Buzzer<'a>
87 for PwmBuzzer<'a, A, P>
88{
89 fn set_client(&self, client: &'a dyn BuzzerClient) {
90 self.client.replace(client);
91 }
92
93 fn buzz(&self, frequency_hz: usize, duration_ms: usize) -> Result<(), ErrorCode> {
94 let duration_ms_cmp = cmp::min(duration_ms, self.max_duration_ms);
95 self.pwm_pin
96 .start(frequency_hz, self.pwm_pin.get_maximum_duty_cycle() / 2)?;
97
98 // Set an alarm for the given duration.
99 let interval = (duration_ms_cmp as u32) * <A::Frequency>::frequency() / 1000;
100 self.alarm
101 .set_alarm(self.alarm.now(), A::Ticks::from(interval));
102 Ok(())
103 }
104
105 fn stop(&self) -> Result<(), ErrorCode> {
106 // Disarm the current alarm and instantly fire another.
107 self.alarm.disarm()?;
108 // This method was used to reduce the size of the code.
109 self.alarm.set_alarm(self.alarm.now(), A::Ticks::from(0));
110 Ok(())
111 }
112}
113
114impl<'a, A: hil::time::Alarm<'a>, P: hil::pwm::PwmPin> hil::time::AlarmClient
115 for PwmBuzzer<'a, A, P>
116{
117 fn alarm(&self) {
118 // Stop the pin output and signal that the buzzer has finished
119 // playing.
120 self.client
121 .map(|buzz_client| buzz_client.buzzer_done(self.pwm_pin.stop()));
122 }
123}