aboutsummaryrefslogtreecommitdiff
path: root/examples/nested.rs
blob: fda2a7d431dd424cd5a3229401fa242f1d08b683 (plain) (blame)
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
//! Nesting claims and how the preemption threshold works
//!
//! If you run this program you'll hit the breakpoints as indicated by the
//! letters in the comments: A, then B, then C, etc.

#![deny(unsafe_code)]
#![feature(const_fn)]
#![feature(proc_macro)]
#![no_std]

#[macro_use(task)]
extern crate cortex_m_rtfm as rtfm;
extern crate stm32f103xx;

use stm32f103xx::Interrupt;
use rtfm::{app, Resource, Threshold};

app! {
    device: stm32f103xx,

    resources: {
        static LOW: u64 = 0;
        static HIGH: u64 = 0;
    },

    tasks: {
        EXTI0: {
            enabled: true,
            priority: 1,
            resources: [LOW, HIGH],
        },

        EXTI1: {
            enabled: true,
            priority: 2,
            resources: [LOW],
        },

        EXTI2: {
            enabled: true,
            priority: 3,
            resources: [HIGH],
        },
    },
}

fn init(_p: init::Peripherals, _r: init::Resources) {}

fn idle() -> ! {
    // sets task `exti0` as pending
    //
    // because `exti0` has higher priority than `idle` it will be executed
    // immediately
    rtfm::set_pending(Interrupt::EXTI0); // ~> exti0

    loop {
        rtfm::wfi();
    }
}

task!(EXTI0, exti0);

fn exti0(t: &mut Threshold, r: EXTI0::Resources) {
    // because this task has a priority of 1 the preemption threshold is also 1

    let mut low = r.LOW;
    let mut high = r.HIGH;

    // A
    rtfm::bkpt();

    // because `exti1` has higher priority than `exti0` it can preempt it
    rtfm::set_pending(Interrupt::EXTI1); // ~> exti1

    // a claim creates a critical section
    low.claim_mut(t, |_low, t| {
        // this claim increases the preemption threshold to 2
        // just high enough to not race with task `exti1` for access to the
        // `LOW` resource

        // C
        rtfm::bkpt();

        // now `exti1` can't preempt this task because its priority is equal to
        // the current preemption threshold
        rtfm::set_pending(Interrupt::EXTI1);

        // but `exti2` can, because its priority is higher than the current
        // preemption threshold
        rtfm::set_pending(Interrupt::EXTI2); // ~> exti2

        // E
        rtfm::bkpt();

        // claims can be nested
        high.claim_mut(t, |_high, _| {
            // This claim increases the preemption threshold to 3

            // now `exti2` can't preempt this task
            rtfm::set_pending(Interrupt::EXTI2);

            // F
            rtfm::bkpt();
        });

        // upon leaving the critical section the preemption threshold drops to 2
        // and `exti2` immediately preempts this task
        // ~> exti2
    });

    // once again the preemption threshold drops to 1
    // now the pending `exti1` can preempt this task
    // ~> exti1
}

task!(EXTI1, exti1);

fn exti1(_t: &mut Threshold, _r: EXTI1::Resources) {
    // B, H
    rtfm::bkpt();
}

task!(EXTI2, exti2);

fn exti2(_t: &mut Threshold, _r: EXTI2::Resources) {
    // D, G
    rtfm::bkpt();
}