1 /*
2  * Copyright 2023 NXP
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include "sw_isr_common.h"
8 
9 #include <zephyr/sw_isr_table.h>
10 #include <zephyr/spinlock.h>
11 
12 /* an interrupt line can be considered shared only if there's
13  * at least 2 clients using it. As such, enforce the fact that
14  * the maximum number of allowed clients should be at least 2.
15  */
16 BUILD_ASSERT(CONFIG_SHARED_IRQ_MAX_NUM_CLIENTS >= 2,
17 	     "maximum number of clients should be at least 2");
18 
z_shared_isr(const void * data)19 void z_shared_isr(const void *data)
20 {
21 	size_t i;
22 	const struct z_shared_isr_table_entry *entry;
23 	const struct _isr_table_entry *client;
24 
25 	entry = data;
26 
27 	for (i = 0; i < entry->client_num; i++) {
28 		client = &entry->clients[i];
29 
30 		if (client->isr) {
31 			client->isr(client->arg);
32 		}
33 	}
34 }
35 
36 #ifdef CONFIG_DYNAMIC_INTERRUPTS
37 
38 static struct k_spinlock lock;
39 
z_isr_install(unsigned int irq,void (* routine)(const void *),const void * param)40 void z_isr_install(unsigned int irq, void (*routine)(const void *),
41 		   const void *param)
42 {
43 	struct z_shared_isr_table_entry *shared_entry;
44 	struct _isr_table_entry *entry;
45 	struct _isr_table_entry *client;
46 	unsigned int table_idx;
47 	int i;
48 	k_spinlock_key_t key;
49 
50 	table_idx = z_get_sw_isr_table_idx(irq);
51 
52 	/* check for out of bounds table index */
53 	if (table_idx >= IRQ_TABLE_SIZE) {
54 		return;
55 	}
56 
57 	shared_entry = &z_shared_sw_isr_table[table_idx];
58 	entry = &_sw_isr_table[table_idx];
59 
60 	key = k_spin_lock(&lock);
61 
62 	/* have we reached the client limit? */
63 	__ASSERT(shared_entry->client_num < CONFIG_SHARED_IRQ_MAX_NUM_CLIENTS,
64 		 "reached maximum number of clients");
65 
66 	if (entry->isr == z_irq_spurious) {
67 		/* this is the first time a ISR/arg pair is registered
68 		 * for INTID => no need to share it.
69 		 */
70 		entry->isr = routine;
71 		entry->arg = param;
72 
73 		k_spin_unlock(&lock, key);
74 
75 		return;
76 	} else if (entry->isr != z_shared_isr) {
77 		/* INTID is being used by another ISR/arg pair.
78 		 * Push back the ISR/arg pair registered in _sw_isr_table
79 		 * to the list of clients and hijack the pair stored in
80 		 * _sw_isr_table with our own z_shared_isr/shared_entry pair.
81 		 */
82 		shared_entry->clients[shared_entry->client_num].isr = entry->isr;
83 		shared_entry->clients[shared_entry->client_num].arg = entry->arg;
84 
85 		shared_entry->client_num++;
86 
87 		entry->isr = z_shared_isr;
88 		entry->arg = shared_entry;
89 	}
90 
91 	/* don't register the same ISR/arg pair multiple times */
92 	for (i = 0; i < shared_entry->client_num; i++) {
93 		client = &shared_entry->clients[i];
94 
95 		__ASSERT((client->isr == routine && client->arg == param) == false,
96 			 "ISR/arg combination is already registered");
97 	}
98 
99 	shared_entry->clients[shared_entry->client_num].isr = routine;
100 	shared_entry->clients[shared_entry->client_num].arg = param;
101 	shared_entry->client_num++;
102 
103 	k_spin_unlock(&lock, key);
104 }
105 
swap_client_data(struct _isr_table_entry * a,struct _isr_table_entry * b)106 static void swap_client_data(struct _isr_table_entry *a,
107 			     struct _isr_table_entry *b)
108 {
109 	struct _isr_table_entry tmp;
110 
111 	tmp.arg = a->arg;
112 	tmp.isr = a->isr;
113 
114 	a->arg = b->arg;
115 	a->isr = b->isr;
116 
117 	b->arg = tmp.arg;
118 	b->isr = tmp.isr;
119 }
120 
shared_irq_remove_client(struct z_shared_isr_table_entry * shared_entry,int client_idx,unsigned int table_idx)121 static void shared_irq_remove_client(struct z_shared_isr_table_entry *shared_entry,
122 				     int client_idx, unsigned int table_idx)
123 {
124 	int i;
125 
126 	shared_entry->clients[client_idx].isr = NULL;
127 	shared_entry->clients[client_idx].arg = NULL;
128 
129 	/* push back the removed client to the end of the client list */
130 	for (i = client_idx; i <= (int)shared_entry->client_num - 2; i++) {
131 		swap_client_data(&shared_entry->clients[i],
132 				 &shared_entry->clients[i + 1]);
133 	}
134 
135 	shared_entry->client_num--;
136 
137 	/* "unshare" interrupt if there will be a single client left */
138 	if (shared_entry->client_num == 1) {
139 		_sw_isr_table[table_idx].isr = shared_entry->clients[0].isr;
140 		_sw_isr_table[table_idx].arg = shared_entry->clients[0].arg;
141 
142 		shared_entry->clients[0].isr = NULL;
143 		shared_entry->clients[0].arg = NULL;
144 
145 		shared_entry->client_num--;
146 	}
147 }
148 
arch_irq_disconnect_dynamic(unsigned int irq,unsigned int priority,void (* routine)(const void * parameter),const void * parameter,uint32_t flags)149 int __weak arch_irq_disconnect_dynamic(unsigned int irq, unsigned int priority,
150 				       void (*routine)(const void *parameter),
151 				       const void *parameter, uint32_t flags)
152 {
153 	ARG_UNUSED(priority);
154 	ARG_UNUSED(flags);
155 
156 	return z_isr_uninstall(irq, routine, parameter);
157 }
158 
z_isr_uninstall(unsigned int irq,void (* routine)(const void *),const void * parameter)159 int z_isr_uninstall(unsigned int irq,
160 		    void (*routine)(const void *),
161 		    const void *parameter)
162 {
163 	struct z_shared_isr_table_entry *shared_entry;
164 	struct _isr_table_entry *entry;
165 	struct _isr_table_entry *client;
166 	unsigned int table_idx;
167 	size_t i;
168 	k_spinlock_key_t key;
169 
170 	table_idx = z_get_sw_isr_table_idx(irq);
171 
172 	/* check for out of bounds table index */
173 	if (table_idx >= IRQ_TABLE_SIZE) {
174 		return -EINVAL;
175 	}
176 
177 	shared_entry = &z_shared_sw_isr_table[table_idx];
178 	entry = &_sw_isr_table[table_idx];
179 
180 	key = k_spin_lock(&lock);
181 
182 	/* note: it's important that we remove the ISR/arg pair even if
183 	 * the IRQ line is not being shared because z_isr_install() will
184 	 * not overwrite it unless the _sw_isr_table entry for the given
185 	 * IRQ line contains the default pair which is z_irq_spurious/NULL.
186 	 */
187 	if (!shared_entry->client_num) {
188 		if (entry->isr == routine && entry->arg == parameter) {
189 			entry->isr = z_irq_spurious;
190 			entry->arg = NULL;
191 		}
192 
193 		goto out_unlock;
194 	}
195 
196 	for (i = 0; i < shared_entry->client_num; i++) {
197 		client = &shared_entry->clients[i];
198 
199 		if (client->isr == routine && client->arg == parameter) {
200 			/* note: this is the only match we're going to get */
201 			shared_irq_remove_client(shared_entry, i, table_idx);
202 			goto out_unlock;
203 		}
204 	}
205 
206 out_unlock:
207 	k_spin_unlock(&lock, key);
208 	return 0;
209 }
210 
211 #endif /* CONFIG_DYNAMIC_INTERRUPTS */
212