rewrited buzzer for 2,5kHz Buzzer and use PWM

icreased SPI frequency to 3.8MHz
This commit is contained in:
PC_WSL 2025-05-08 15:58:47 +02:00
parent b3b0176b46
commit ee4726d5a2
5 changed files with 94 additions and 58 deletions

1
Cargo.lock generated
View File

@ -423,6 +423,7 @@ dependencies = [
"gpio", "gpio",
"log", "log",
"regex", "regex",
"rgb",
"rocket", "rocket",
"rppal", "rppal",
"rust-embed", "rust-embed",

View File

@ -1,56 +1,61 @@
use rppal::gpio::{Gpio, OutputPin}; use rppal::pwm::{Pwm, Channel, Polarity};
use std::{error::Error, time};
use tokio::time::sleep; use tokio::time::sleep;
use std::{error::Error, time::Duration};
pub struct GPIOBuzzer { pub struct GPIOBuzzer {
pin: OutputPin, pwm: Pwm,
} }
impl GPIOBuzzer { impl GPIOBuzzer {
pub fn new(pin_num: u8) -> Result<Self, Box<dyn Error>> { /// Create a new GPIOBuzzer instance.
let gpio = Gpio::new()?; /// 0.5 duty cyle
let pin = gpio.get(pin_num)?.into_output(); /// # Arguments
/// * "channel" - PWM channel for buzzer PWM0 = GPIO 12 / PWM1 = GPIO 13
pub fn new(channel: Channel) -> Result<Self, Box<dyn Error>> {
// Enable with dummy values; we'll set frequency/duty in the tone method
let duty_cycle:f64 = 0.5;
let pwm = Pwm::with_frequency(channel, 1000.0, duty_cycle, Polarity::Normal, true)?;
pwm.disable()?; // Start disabled
Ok(GPIOBuzzer { pin }) Ok(GPIOBuzzer { pwm })
} }
/// Emits a sound on a passive buzzer. /// Play a tone using hardware PWM on supported GPIO pins.
async fn modulated_tone(&mut self, carrier_hz: u32, sound_hz: u32, duration_ms: u64) { ///
let carrier_period = /// # Arguments
time::Duration::from_micros((1_000_000.0 / carrier_hz as f64 / 2.0) as u64); /// * `frequency` - Frequency in Hz.
let mod_period = 1_000.0 / sound_hz as f64; // in ms /// * `duration_ms` - Duration in milliseconds.
let total_cycles = duration_ms as f64 / mod_period; async fn modulated_tone(&mut self, frequency: f64, duration_ms: u64) -> Result<(), Box<dyn Error>> {
self.pwm.set_frequency(frequency, 0.5)?; // 50% duty cycle (square wave)
for _ in 0..total_cycles as u64 { self.pwm.enable()?;
// Modulation on: Carrier on for mod_period / 2 sleep(Duration::from_millis(duration_ms)).await;
let cycles_on = (carrier_hz as f64 * (mod_period / 2.0) / 1000.0) as u64; self.pwm.disable()?;
for _ in 0..cycles_on { Ok(())
self.pin.set_high();
sleep(carrier_period).await;
self.pin.set_low();
sleep(carrier_period).await;
}
// Modulation off: Carrier on for mod_period / 2
let pause = time::Duration::from_millis((mod_period / 2.0) as u64);
sleep(pause).await;
}
}
pub async fn beep_ack(&mut self) {
// carrier = 2300 Hz, sound = 440 Hz, duration = 1 sec
self.modulated_tone(2300, 659, 400).await;
self.modulated_tone(2300, 784,100).await;
} }
pub async fn beep_nak(&mut self) { pub async fn beep_ack(&mut self) -> Result<(), Box<dyn Error>>{
// carrier = 2300 Hz, sound = 440 Hz, duration = 1 sec let sleep_ms: u64 = 100;
self.modulated_tone(2300, 659, 400).await; self.modulated_tone(750.0, 100).await?;
self.modulated_tone(2300, 523, 100).await; sleep(Duration::from_millis(sleep_ms)).await;
self.modulated_tone(1200.0,100).await?;
sleep(Duration::from_millis(sleep_ms)).await;
self.modulated_tone(2300.0,100).await?;
Ok(())
} }
pub async fn beep_unnkown(&mut self) { pub async fn beep_nak(&mut self) -> Result<(), Box<dyn Error>>{
self.modulated_tone(2300, 784, 150).await; self.modulated_tone(2300.0,100).await?;
self.modulated_tone(2300, 659, 150).await; self.modulated_tone(2300.0,100).await?;
self.modulated_tone(2300, 500, 150).await; Ok(())
}
pub async fn beep_unnkown(&mut self) -> Result<(), Box<dyn Error>>{
let sleep_ms: u64 = 100;
self.modulated_tone(2300.0,100).await?;
sleep(Duration::from_millis(sleep_ms)).await;
self.modulated_tone(2300.0,100).await?;
sleep(Duration::from_millis(sleep_ms)).await;
self.modulated_tone(2300.0,100).await?;
Ok(())
} }
} }

View File

@ -15,9 +15,9 @@ pub enum NamedColor {
impl Into<Rgb<u8>> for NamedColor { impl Into<Rgb<u8>> for NamedColor {
fn into(self) -> Rgb<u8> { fn into(self) -> Rgb<u8> {
match self { match self {
NamedColor::Red => Rgb { r: 255, g: 0, b: 0 }, NamedColor::Red => Rgb { r: 150, g: 0, b: 0 },
NamedColor::Green => Rgb { r: 0, g: 255, b: 0 }, NamedColor::Green => Rgb { r: 0, g: 150, b: 0 },
NamedColor::Blue => Rgb { r: 0, g: 0, b: 255 }, NamedColor::Blue => Rgb { r: 0, g: 0, b: 150 },
NamedColor::White => Rgb { NamedColor::White => Rgb {
r: 255, r: 255,
g: 255, g: 255,

View File

@ -1,27 +1,34 @@
use std::time::Duration;
use rppal::spi::{Bus, Mode, SlaveSelect, Spi}; use rppal::spi::{Bus, Mode, SlaveSelect, Spi};
use smart_leds::{RGB8, SmartLedsWrite}; use smart_leds::SmartLedsWrite;
use tokio::time::sleep;
use ws2812_spi::Ws2812; use ws2812_spi::Ws2812;
use crate::color::NamedColor;
pub struct Led { pub struct Led {
controller: Ws2812<Spi>, controller: Ws2812<Spi>,
} }
const STATUS_DURATION: Duration = Duration::from_secs(2); // 2s sleep for all status led signals
impl Led { impl Led {
pub fn new() -> Result<Self, Box<dyn std::error::Error>> { pub fn new() -> Result<Self, Box<dyn std::error::Error>> {
let spi = Spi::new(Bus::Spi0, SlaveSelect::Ss0, 3_000_000, Mode::Mode0)?; let spi = Spi::new(Bus::Spi0, SlaveSelect::Ss0, 3_800_000, Mode::Mode0)?;
let controller = Ws2812::new(spi); let controller = Ws2812::new(spi);
Ok(Led { controller }) Ok(Led { controller })
} }
pub fn turn_green_on(&mut self) -> Result<(), Box<dyn std::error::Error>> { pub async fn turn_green_on_1s(&mut self) -> Result<(), Box<dyn std::error::Error>> {
let green = [RGB8 { r: 0, g: 255, b: 0 }]; self.controller.write(NamedColor::Green.into_iter())?;
self.controller.write(green.iter().cloned())?; sleep(STATUS_DURATION).await;
self.controller.write(NamedColor::Off.into_iter())?;
Ok(()) Ok(())
} }
pub fn turn_off(&mut self) -> Result<(), Box<dyn std::error::Error>> { pub fn turn_off(&mut self) -> Result<(), Box<dyn std::error::Error>> {
let off = [RGB8 { r: 0, g: 0, b: 0 }]; self.controller.write(NamedColor::Off.into_iter())?;
self.controller.write(off.iter().cloned())?;
Ok(()) Ok(())
} }
} }

View File

@ -1,8 +1,9 @@
use buzzer::GPIOBuzzer; use buzzer::GPIOBuzzer;
use color::NamedColor;
use id_store::IDStore; use id_store::IDStore;
use log::{LevelFilter, error, info, warn}; use led::Led;
use log::{LevelFilter, debug, error, info, warn};
use pm3::run_pm3; use pm3::run_pm3;
use rppal::pwm::Channel;
use simplelog::{ConfigBuilder, SimpleLogger}; use simplelog::{ConfigBuilder, SimpleLogger};
use std::{env, error::Error, sync::Arc}; use std::{env, error::Error, sync::Arc};
use tokio::{ use tokio::{
@ -20,7 +21,7 @@ mod pm3;
mod webserver; mod webserver;
const STORE_PATH: &str = "./data.json"; const STORE_PATH: &str = "./data.json";
const BUZZER_PIN: u8 = 26; const PWM_CHANNEL_BUZZER: Channel = Channel::Pwm0; //PWM0 = GPIO18/Physical pin 12
fn setup_logger() { fn setup_logger() {
let log_level = env::var("LOG_LEVEL") let log_level = env::var("LOG_LEVEL")
@ -70,8 +71,12 @@ async fn main() -> Result<(), Box<dyn Error>> {
IDStore::new() IDStore::new()
}; };
debug!("created store sucessfully");
let store: Arc<Mutex<IDStore>> = Arc::new(Mutex::new(raw_store)); let store: Arc<Mutex<IDStore>> = Arc::new(Mutex::new(raw_store));
let gpio_buzzer: Arc<Mutex<GPIOBuzzer>> = Arc::new(Mutex::new(buzzer::GPIOBuzzer::new(BUZZER_PIN)?)); let gpio_buzzer: Arc<Mutex<GPIOBuzzer>> =
Arc::new(Mutex::new(GPIOBuzzer::new(PWM_CHANNEL_BUZZER)?));
let status_led: Arc<Mutex<Led>> = Arc::new(Mutex::new(Led::new()?));
let channel_store = store.clone(); let channel_store = store.clone();
tokio::spawn(async move { tokio::spawn(async move {
@ -82,13 +87,31 @@ async fn main() -> Result<(), Box<dyn Error>> {
.add_id(id_store::TallyID(tally_id_string)) .add_id(id_store::TallyID(tally_id_string))
{ {
info!("Added new id to current day"); info!("Added new id to current day");
// led.set_named_color_time(NamedColor::Green, 1); //led is green for 1 sec status_led
gpio_buzzer.lock().await.beep_ack().await; .lock()
.await
.turn_green_on_1s()
.await
.unwrap_or_else(|e| {
error!("Failed to blink LED {}", e);
});
gpio_buzzer
.lock()
.await
.beep_ack()
.await
.unwrap_or_else(|e| error!("Failed to beep Ack {}", e));
if let Err(e) = channel_store.lock().await.export_json(STORE_PATH).await { if let Err(e) = channel_store.lock().await.export_json(STORE_PATH).await {
error!("Failed to save id store to file: {}", e); error!("Failed to save id store to file: {}", e);
// TODO: How to handle a failure to save ? // TODO: How to handle a failure to save ?
gpio_buzzer.lock().await.beep_nak().await; gpio_buzzer
.lock()
.await
.beep_nak()
.await
.unwrap_or_else(|e| error!("Failed to beep Nack {}", e));
} }
} }
} }