Fix handling of duty cycle while PWM channels is disabled

Fixes #390
This commit is contained in:
Jan Niehusmann 2022-07-26 08:13:35 +00:00
parent 32411b8652
commit c0c72c4db3

View file

@ -572,6 +572,7 @@ pub struct Channel<S: SliceId, M: SliceMode, C: ChannelId> {
slice_mode: PhantomData<M>, slice_mode: PhantomData<M>,
channel_id: PhantomData<C>, channel_id: PhantomData<C>,
duty_cycle: u16, duty_cycle: u16,
enabled: bool,
} }
impl<S: SliceId, M: SliceMode, C: ChannelId> Channel<S, M, C> { impl<S: SliceId, M: SliceMode, C: ChannelId> Channel<S, M, C> {
@ -580,7 +581,8 @@ impl<S: SliceId, M: SliceMode, C: ChannelId> Channel<S, M, C> {
regs: Registers::new(), regs: Registers::new(),
slice_mode: PhantomData, slice_mode: PhantomData,
channel_id: PhantomData, channel_id: PhantomData,
duty_cycle, duty_cycle, // stores the duty cycle while the channel is disabled
enabled: true,
} }
} }
} }
@ -593,16 +595,26 @@ impl<S: SliceId, M: SliceMode> PwmPin for Channel<S, M, A> {
/// We cant disable the channel without disturbing the other channel. /// We cant disable the channel without disturbing the other channel.
/// So this just sets the duty cycle to zero /// So this just sets the duty cycle to zero
fn disable(&mut self) { fn disable(&mut self) {
if self.enabled {
self.duty_cycle = self.regs.read_cc_a(); self.duty_cycle = self.regs.read_cc_a();
}
self.enabled = false;
self.regs.write_cc_a(0) self.regs.write_cc_a(0)
} }
fn enable(&mut self) { fn enable(&mut self) {
if !self.enabled {
self.enabled = true;
self.regs.write_cc_a(self.duty_cycle) self.regs.write_cc_a(self.duty_cycle)
} }
}
fn get_duty(&self) -> Self::Duty { fn get_duty(&self) -> Self::Duty {
if self.enabled {
self.regs.read_cc_a() self.regs.read_cc_a()
} else {
self.duty_cycle
}
} }
fn get_max_duty(&self) -> Self::Duty { fn get_max_duty(&self) -> Self::Duty {
@ -610,9 +622,12 @@ impl<S: SliceId, M: SliceMode> PwmPin for Channel<S, M, A> {
} }
fn set_duty(&mut self, duty: Self::Duty) { fn set_duty(&mut self, duty: Self::Duty) {
self.duty_cycle = duty;
if self.enabled {
self.regs.write_cc_a(duty) self.regs.write_cc_a(duty)
} }
} }
}
impl<S: SliceId, M: SliceMode> PwmPin for Channel<S, M, B> { impl<S: SliceId, M: SliceMode> PwmPin for Channel<S, M, B> {
type Duty = u16; type Duty = u16;
@ -620,16 +635,26 @@ impl<S: SliceId, M: SliceMode> PwmPin for Channel<S, M, B> {
/// We cant disable the channel without disturbing the other channel. /// We cant disable the channel without disturbing the other channel.
/// So this just sets the duty cycle to zero /// So this just sets the duty cycle to zero
fn disable(&mut self) { fn disable(&mut self) {
if self.enabled {
self.duty_cycle = self.regs.read_cc_b(); self.duty_cycle = self.regs.read_cc_b();
}
self.enabled = false;
self.regs.write_cc_b(0) self.regs.write_cc_b(0)
} }
fn enable(&mut self) { fn enable(&mut self) {
if !self.enabled {
self.enabled = true;
self.regs.write_cc_b(self.duty_cycle) self.regs.write_cc_b(self.duty_cycle)
} }
}
fn get_duty(&self) -> Self::Duty { fn get_duty(&self) -> Self::Duty {
if self.enabled {
self.regs.read_cc_b() self.regs.read_cc_b()
} else {
self.duty_cycle
}
} }
fn get_max_duty(&self) -> Self::Duty { fn get_max_duty(&self) -> Self::Duty {
@ -637,9 +662,12 @@ impl<S: SliceId, M: SliceMode> PwmPin for Channel<S, M, B> {
} }
fn set_duty(&mut self, duty: Self::Duty) { fn set_duty(&mut self, duty: Self::Duty) {
self.duty_cycle = duty;
if self.enabled {
self.regs.write_cc_b(duty) self.regs.write_cc_b(duty)
} }
} }
}
impl<S: SliceId, M: SliceMode + ValidSliceMode<S>> Channel<S, M, A> { impl<S: SliceId, M: SliceMode + ValidSliceMode<S>> Channel<S, M, A> {
/// Capture a gpio pin and use it as pwm output for channel A /// Capture a gpio pin and use it as pwm output for channel A