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
//! [BCM2711 PWM][1]
//!
//! [1]: https://datasheets.raspberrypi.com/bcm2711/bcm2711-peripherals.pdf#%5B%7B%22num%22%3A130%2C%22gen%22%3A0%7D%2C%7B%22name%22%3A%22XYZ%22%7D%2C115%2C841.89%2Cnull%5D
use tock_registers::{
    register_bitfields, register_structs,
    registers::{ReadWrite, WriteOnly},
};

use crate::Vpa;

/// The base address of the PWM0 instance of [the PWM register block](Registers).
pub const BASE_PWM0: Vpa = Vpa(0x4_7e20_c000);
/// The base address of the PWM1 instance of [the PWM register block](Registers).
pub const BASE_PWM1: Vpa = Vpa(0x4_7e20_c800);

register_structs! {
    pub Registers {
        /// PWM control
        (0x00 => pub ctl: ReadWrite<u32, CTL::Register>),
        /// PWM status
        (0x04 => pub sta: ReadWrite<u32, STA::Register>),
        /// PMA DMA configuration
        (0x08 => pub dmac: ReadWrite<u32, DMAC::Register>),
        (0x0c => _pad0),
        /// PWM channel 1 range
        (0x10 => pub rng1: ReadWrite<u32>),
        /// PWM channel 1 data
        (0x14 => pub dat1: ReadWrite<u32>),
        /// PWM FIFO input
        (0x18 => pub fif1: WriteOnly<u32>),
        (0x1c => _pad1),
        /// PWM channel 2 range
        (0x20 => pub rng2: ReadWrite<u32>),
        /// PWM channel 2 data
        (0x24 => pub dat2: ReadWrite<u32>),
        (0x28 => @END),
    }
}

register_bitfields! {u32,
    pub CTL [
        /// Channel 1 enable
        PWEN1 OFFSET(0) NUMBITS(1) [],
        /// Channel 1 mode
        MODE1 OFFSET(1) NUMBITS(1) [
            Pwm = 0,
            Serialiser = 1,
        ],
        /// Channel 1 repeat last data
        RPTL1 OFFSET(2) NUMBITS(1) [],
        /// Channel 1 silence bit
        SBIT1 OFFSET(3) NUMBITS(1) [],
        /// Channel 1 polarity
        POLA1 OFFSET(4) NUMBITS(1) [
            ActiveHigh = 1,
            ActiveLow = 0,
        ],
        /// Channel 1 use FIFO
        USEF1 OFFSET(5) NUMBITS(1) [],
        /// Clear FIFO
        CLRF OFFSET(6) NUMBITS(1) [],
        /// Channel 1 M/S enable
        MSEN1 OFFSET(7) NUMBITS(1) [
            Pwm = 0,
            Ms = 1,
        ],
        /// Channel 2 enable
        PWEN2 OFFSET(8) NUMBITS(1) [],
        /// Channel 2 mode
        MODE2 OFFSET(9) NUMBITS(1) [
            Pwm = 0,
            Serialiser = 1,
        ],
        /// Channel 1 repeat last data
        RPTL2 OFFSET(10) NUMBITS(1) [],
        /// Channel 2 silence bit
        SBIT2 OFFSET(11) NUMBITS(1) [],
        /// Channel 2 polarity
        POLA2 OFFSET(12) NUMBITS(1) [
            ActiveHigh = 1,
            ActiveLow = 0,
        ],
        /// Channel 2 use FIFO
        USEF2 OFFSET(13) NUMBITS(1) [],
        /// Channel 2 M/S enable
        MSEN2 OFFSET(15) NUMBITS(1) [
            Pwm = 0,
            Ms = 1,
        ],
    ]
}

register_bitfields! {u32,
    pub STA [
        /// FIFO full flag (RO)
        FULL1 OFFSET(0) NUMBITS(1) [],
        /// FIFO empty flag (RO)
        EMPT1 OFFSET(1) NUMBITS(1) [],
        /// FIFO write error flag (W1C)
        WERR1 OFFSET(2) NUMBITS(1) [],
        /// FIFO read error flag (W1C)
        RERR1 OFFSET(3) NUMBITS(1) [],
        /// Channel 1 gap occurred flag (W1C)
        GAPO1 OFFSET(4) NUMBITS(1) [],
        /// Channel 2 gap occurred flag (W1C)
        GAPO2 OFFSET(5) NUMBITS(1) [],
        /// Bus error flag (W1C)
        BERR OFFSET(8) NUMBITS(1) [],
        /// Channel 1 state (RO)
        STA1 OFFSET(9) NUMBITS(1) [],
        /// Channel 2 state (RO)
        STA2 OFFSET(10) NUMBITS(1) [],
    ]
}

register_bitfields! {u32,
    pub DMAC [
        /// DMA threshold for DREQ signal
        DREQ OFFSET(0) NUMBITS(8) [],
        /// DMA threshold for PANIC signal
        PANIC OFFSET(8) NUMBITS(8) [],
        /// DMA enable
        ENAB OFFSET(31) NUMBITS(1) [],
    ]
}