aboutsummaryrefslogtreecommitdiff
path: root/include/arch/stm32l4xxx/peripherals/irq.h
blob: 52878ca7471fd0dd26936218a935916a2909649e (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
#ifndef CORE_IRQ_H__
#define CORE_IRQ_H__

#include <stdint.h>

/*
 * Include file for interrupt service routines.
 */

typedef enum {
#define IRQ_RESERVED(n)
#define IRQ(name_, uname, num) \
  IRQ_##uname = num,
#include "arch/stm32l4xxx/peripherals//isrs.inc"
#undef IRQ
#undef IRQ_RESERVED
} interrupt_t;

/* Defines a set of interrupts so they may be enabled all at once. */
typedef struct {
  uint32_t sysirqs; /* System iterrupts. */
  uint32_t irqs[8];
} interrupt_set_t;

inline static void interrupt_set_add(
    interrupt_set_t* interrupt_set, interrupt_t interrupt)
{
  if (interrupt < 16) {
    interrupt_set->sysirqs |= 1 << interrupt;
    return;
  }

  interrupt -= 16;
  int loc = interrupt / 32;
  int off = interrupt % 32;

  interrupt_set->irqs[loc] |= 1 << off;
}

inline static void interrupt_set_remove(
    interrupt_set_t* interrupt_set, interrupt_t interrupt)
{
  if (interrupt < 16) {
    interrupt_set->sysirqs &= ~(1 << interrupt);
    return;
  }

  interrupt -= 16;
  int loc = interrupt / 32;
  int off = interrupt % 32;

  interrupt_set->irqs[loc] &= ~(1 << off);
}

/*
 * The interrupt service routines. These link in the function `main` as the
 * main function.
 */
extern const void* vectors[];

/*
 * Defines an error state. This loops forever and defines a distinct flashing
 * pattern to let the user know an unhandled ISR happened.
 */
void unhandled_isr(uint8_t val);

#define enable_interrupt(val) \
  {interrupt_set_t itrset = { 0 }; \
   interrupt_set_add(&itrset, val); \
    enable_interrupts(&itrset);}

#define disable_interrupt(val) \
  {interrupt_set_t itrset = { 0 }; \
   interrupt_set_add(&itrset, val); \
    disable_interrupts(&itrset);}

/*
 * Enables the provided interrupt. Note that if the interrupt is one of the
 * system interrupts (first 16) this function has no effect because those
 * interrupts are always enabled.
 */
void enable_interrupts(interrupt_set_t* interrupts);

/*
 * Enables the provided interrupt 
 */
void disable_interrupts(interrupt_set_t* interrupts);

#endif /* CORE_IRQ_H_ */