source: mainline/kernel/generic/src/ipc/irq.c@ 63d8f43

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 63d8f43 was 63d8f43, checked in by Jakub Jermar <jakub@…>, 8 years ago

Allocate the kobjects referenced by capabilities dynamically

  • Property mode set to 100644
File size: 14.5 KB
RevLine 
[162f919]1/*
[df4ed85]2 * Copyright (c) 2006 Ondrej Palkovsky
3 * Copyright (c) 2006 Jakub Jermar
[162f919]4 * All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 *
10 * - Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * - Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * - The name of the author may not be used to endorse or promote products
16 * derived from this software without specific prior written permission.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29
[cc73a8a1]30/** @addtogroup genericipc
[b45c443]31 * @{
32 */
[da1bafb]33
[cc73a8a1]34/**
35 * @file
36 * @brief IRQ notification framework.
[bdc5c516]37 *
[8820544]38 * This framework allows applications to subscribe to receive a notification
[a5d0143]39 * when an interrupt is detected. The application may provide a simple
40 * 'top-half' handler as part of its registration, which can perform simple
41 * operations (read/write port/memory, add information to notification IPC
42 * message).
[bdc5c516]43 *
44 * The structure of a notification message is as follows:
[8820544]45 * - IMETHOD: interface and method as set by the SYS_IPC_IRQ_SUBSCRIBE syscall
[56c167c]46 * - ARG1: payload modified by a 'top-half' handler (scratch[1])
47 * - ARG2: payload modified by a 'top-half' handler (scratch[2])
48 * - ARG3: payload modified by a 'top-half' handler (scratch[3])
49 * - ARG4: payload modified by a 'top-half' handler (scratch[4])
50 * - ARG5: payload modified by a 'top-half' handler (scratch[5])
[43752b6]51 * - in_phone_hash: interrupt counter (may be needed to assure correct order
[228e490]52 * in multithreaded drivers)
[bdc5c516]53 */
54
[162f919]55#include <arch.h>
[63e27ef]56#include <assert.h>
[162f919]57#include <mm/slab.h>
[a996ae31]58#include <mm/page.h>
59#include <mm/km.h>
[162f919]60#include <errno.h>
[2b017ba]61#include <ddi/irq.h>
[162f919]62#include <ipc/ipc.h>
63#include <ipc/irq.h>
[e3c762cd]64#include <syscall/copy.h>
[d0c5901]65#include <console/console.h>
[253f35a1]66#include <print.h>
[a996ae31]67#include <macros.h>
[3f74275]68#include <cap/cap.h>
[a996ae31]69
70static void ranges_unmap(irq_pio_range_t *ranges, size_t rangecount)
71{
[56c167c]72 for (size_t i = 0; i < rangecount; i++) {
[472d813]73#ifdef IO_SPACE_BOUNDARY
[a996ae31]74 if ((void *) ranges[i].base >= IO_SPACE_BOUNDARY)
[472d813]75#endif
[a996ae31]76 km_unmap(ranges[i].base, ranges[i].size);
77 }
78}
79
80static int ranges_map_and_apply(irq_pio_range_t *ranges, size_t rangecount,
81 irq_cmd_t *cmds, size_t cmdcount)
82{
83 /* Copy the physical base addresses aside. */
[56c167c]84 uintptr_t *pbase = malloc(rangecount * sizeof(uintptr_t), 0);
85 for (size_t i = 0; i < rangecount; i++)
[a996ae31]86 pbase[i] = ranges[i].base;
[56c167c]87
[a996ae31]88 /* Map the PIO ranges into the kernel virtual address space. */
[56c167c]89 for (size_t i = 0; i < rangecount; i++) {
[472d813]90#ifdef IO_SPACE_BOUNDARY
[a996ae31]91 if ((void *) ranges[i].base < IO_SPACE_BOUNDARY)
92 continue;
[472d813]93#endif
[a996ae31]94 ranges[i].base = km_map(pbase[i], ranges[i].size,
95 PAGE_READ | PAGE_WRITE | PAGE_KERNEL | PAGE_NOT_CACHEABLE);
96 if (!ranges[i].base) {
97 ranges_unmap(ranges, i);
98 free(pbase);
99 return ENOMEM;
100 }
101 }
[56c167c]102
[a5d0143]103 /* Rewrite the IRQ code addresses from physical to kernel virtual. */
[56c167c]104 for (size_t i = 0; i < cmdcount; i++) {
[a996ae31]105 uintptr_t addr;
[f2bbe8c]106 size_t size;
[56c167c]107
[a996ae31]108 /* Process only commands that use an address. */
109 switch (cmds[i].cmd) {
110 case CMD_PIO_READ_8:
[56c167c]111 case CMD_PIO_WRITE_8:
112 case CMD_PIO_WRITE_A_8:
[f2bbe8c]113 size = 1;
114 break;
[56c167c]115 case CMD_PIO_READ_16:
116 case CMD_PIO_WRITE_16:
117 case CMD_PIO_WRITE_A_16:
[f2bbe8c]118 size = 2;
119 break;
[56c167c]120 case CMD_PIO_READ_32:
121 case CMD_PIO_WRITE_32:
122 case CMD_PIO_WRITE_A_32:
[f2bbe8c]123 size = 4;
[a996ae31]124 break;
125 default:
126 /* Move onto the next command. */
127 continue;
128 }
[56c167c]129
[a996ae31]130 addr = (uintptr_t) cmds[i].addr;
131
[56c167c]132 size_t j;
[a996ae31]133 for (j = 0; j < rangecount; j++) {
134 /* Find the matching range. */
[f2bbe8c]135 if (!iswithin(pbase[j], ranges[j].size, addr, size))
[a996ae31]136 continue;
[56c167c]137
[a996ae31]138 /* Switch the command to a kernel virtual address. */
139 addr -= pbase[j];
140 addr += ranges[j].base;
[56c167c]141
[a996ae31]142 cmds[i].addr = (void *) addr;
143 break;
[bd8c6537]144 }
[56c167c]145
[bd8c6537]146 if (j == rangecount) {
147 /*
148 * The address used in this command is outside of all
149 * defined ranges.
150 */
151 ranges_unmap(ranges, rangecount);
152 free(pbase);
153 return EINVAL;
154 }
[a996ae31]155 }
[56c167c]156
[a996ae31]157 free(pbase);
158 return EOK;
159}
[162f919]160
[a5d0143]161/** Statically check the top-half IRQ code
[8486c07]162 *
[a5d0143]163 * Check the top-half IRQ code for invalid or unsafe constructs.
[8486c07]164 *
165 */
166static int code_check(irq_cmd_t *cmds, size_t cmdcount)
167{
168 for (size_t i = 0; i < cmdcount; i++) {
169 /*
170 * Check for accepted ranges.
171 */
172 if (cmds[i].cmd >= CMD_LAST)
173 return EINVAL;
174
175 if (cmds[i].srcarg >= IPC_CALL_LEN)
176 return EINVAL;
177
178 if (cmds[i].dstarg >= IPC_CALL_LEN)
179 return EINVAL;
180
181 switch (cmds[i].cmd) {
182 case CMD_PREDICATE:
183 /*
184 * Check for control flow overflow.
185 * Note that jumping just beyond the last
186 * command is a correct behaviour.
187 */
188 if (i + cmds[i].value > cmdcount)
189 return EINVAL;
190
191 break;
192 default:
193 break;
194 }
195 }
196
197 return EOK;
198}
199
[a5d0143]200/** Free the top-half IRQ code.
[8b243f2]201 *
[a5d0143]202 * @param code Pointer to the top-half IRQ code.
[da1bafb]203 *
[8b243f2]204 */
[162f919]205static void code_free(irq_code_t *code)
206{
207 if (code) {
[a996ae31]208 ranges_unmap(code->ranges, code->rangecount);
209 free(code->ranges);
[162f919]210 free(code->cmds);
211 free(code);
212 }
213}
214
[a5d0143]215/** Copy the top-half IRQ code from userspace into the kernel.
[8b243f2]216 *
[a5d0143]217 * @param ucode Userspace address of the top-half IRQ code.
[da1bafb]218 *
[a5d0143]219 * @return Kernel address of the copied IRQ code.
[8b243f2]220 *
221 */
222static irq_code_t *code_from_uspace(irq_code_t *ucode)
[162f919]223{
[a996ae31]224 irq_pio_range_t *ranges = NULL;
225 irq_cmd_t *cmds = NULL;
[56c167c]226
[da1bafb]227 irq_code_t *code = malloc(sizeof(*code), 0);
228 int rc = copy_from_uspace(code, ucode, sizeof(*code));
[a996ae31]229 if (rc != EOK)
230 goto error;
[162f919]231
[a996ae31]232 if ((code->rangecount > IRQ_MAX_RANGE_COUNT) ||
233 (code->cmdcount > IRQ_MAX_PROG_SIZE))
234 goto error;
[da1bafb]235
[a996ae31]236 ranges = malloc(sizeof(code->ranges[0]) * code->rangecount, 0);
237 rc = copy_from_uspace(ranges, code->ranges,
238 sizeof(code->ranges[0]) * code->rangecount);
239 if (rc != EOK)
240 goto error;
[56c167c]241
[a996ae31]242 cmds = malloc(sizeof(code->cmds[0]) * code->cmdcount, 0);
243 rc = copy_from_uspace(cmds, code->cmds,
[8b243f2]244 sizeof(code->cmds[0]) * code->cmdcount);
[a996ae31]245 if (rc != EOK)
246 goto error;
[8486c07]247
248 rc = code_check(cmds, code->cmdcount);
249 if (rc != EOK)
250 goto error;
251
[a996ae31]252 rc = ranges_map_and_apply(ranges, code->rangecount, cmds,
253 code->cmdcount);
254 if (rc != EOK)
255 goto error;
[56c167c]256
[a996ae31]257 code->ranges = ranges;
258 code->cmds = cmds;
[56c167c]259
[162f919]260 return code;
[56c167c]261
[a996ae31]262error:
263 if (cmds)
264 free(cmds);
[56c167c]265
[a996ae31]266 if (ranges)
267 free(ranges);
[56c167c]268
[a996ae31]269 free(code);
270 return NULL;
[162f919]271}
272
[8820544]273/** Subscribe an answerbox as a receiving end for IRQ notifications.
[2b017ba]274 *
[56c167c]275 * @param box Receiving answerbox.
276 * @param inr IRQ number.
[a5d0143]277 * @param imethod Interface and method to be associated with the notification.
278 * @param ucode Uspace pointer to top-half IRQ code.
[56c167c]279 *
[3f74275]280 * @return IRQ capability handle.
[e9d15d9]281 * @return Negative error code.
[2b017ba]282 *
283 */
[24abb85d]284int ipc_irq_subscribe(answerbox_t *box, inr_t inr, sysarg_t imethod,
285 irq_code_t *ucode)
[162f919]286{
[96b02eb9]287 sysarg_t key[] = {
[24abb85d]288 [IRQ_HT_KEY_INR] = (sysarg_t) inr,
289 [IRQ_HT_KEY_MODE] = (sysarg_t) IRQ_HT_MODE_NO_CLAIM
[cecb0789]290 };
[56c167c]291
[78ffb70]292 if ((inr < 0) || (inr > last_inr))
293 return ELIMIT;
[c822026]294
[da1bafb]295 irq_code_t *code;
[162f919]296 if (ucode) {
297 code = code_from_uspace(ucode);
298 if (!code)
299 return EBADMEM;
[da1bafb]300 } else
[162f919]301 code = NULL;
[c822026]302
[cecb0789]303 /*
[e9d15d9]304 * Allocate and populate the IRQ kernel object.
[cecb0789]305 */
[3f74275]306 int handle = cap_alloc(TASK);
307 if (handle < 0)
308 return handle;
[63d8f43]309
310 irq_t *irq = (irq_t *) malloc(sizeof(irq_t), FRAME_ATOMIC);
311 if (!irq) {
312 cap_free(TASK, handle);
313 return ENOMEM;
314 }
315
[3f74275]316 cap_t *cap = cap_get_current(handle, CAP_TYPE_ALLOCATED);
317 assert(cap);
[9306cd7]318
[cecb0789]319 irq_initialize(irq);
320 irq->inr = inr;
321 irq->claim = ipc_irq_top_half_claim;
[691eb52]322 irq->handler = ipc_irq_top_half_handler;
[4874c2d]323 irq->notif_cfg.notify = true;
[2b017ba]324 irq->notif_cfg.answerbox = box;
[228e490]325 irq->notif_cfg.imethod = imethod;
[2b017ba]326 irq->notif_cfg.code = code;
327 irq->notif_cfg.counter = 0;
[c822026]328
[63d8f43]329 cap->kobject = (void *) irq;
330
[cecb0789]331 /*
[a5d0143]332 * Insert the IRQ structure into the uspace IRQ hash table and retype
333 * the capability. By retyping the capability inside the critical
334 * section, we make sure another thread cannot attempt to unregister the
335 * IRQ before it is inserted into the hash table.
[cecb0789]336 */
[da1bafb]337 irq_spinlock_lock(&irq_uspace_hash_table_lock, true);
338 irq_spinlock_lock(&irq->lock, false);
339
[9306cd7]340 cap->type = CAP_TYPE_IRQ;
[cecb0789]341 hash_table_insert(&irq_uspace_hash_table, key, &irq->link);
[c822026]342
[da1bafb]343 irq_spinlock_unlock(&irq->lock, false);
344 irq_spinlock_unlock(&irq_uspace_hash_table_lock, true);
345
[3f74275]346 return handle;
[cecb0789]347}
348
[8820544]349/** Unsubscribe task from IRQ notification.
[cecb0789]350 *
[3f74275]351 * @param box Answerbox associated with the notification.
352 * @param handle IRQ capability handle.
[56c167c]353 *
354 * @return EOK on success or a negative error code.
355 *
[cecb0789]356 */
[3f74275]357int ipc_irq_unsubscribe(answerbox_t *box, int handle)
[cecb0789]358{
[9306cd7]359 irq_spinlock_lock(&TASK->lock, true);
[3f74275]360 cap_t *cap = cap_get_current(handle, CAP_TYPE_IRQ);
[9306cd7]361 if (!cap) {
362 irq_spinlock_unlock(&TASK->lock, true);
[cecb0789]363 return ENOENT;
[9306cd7]364 }
[a5d0143]365 /* Make sure only one thread can win the race to unsubscribe. */
[9306cd7]366 cap->type = CAP_TYPE_ALLOCATED;
367 irq_spinlock_unlock(&TASK->lock, true);
368
[63d8f43]369 irq_t *irq = (irq_t *) cap->kobject;
[9306cd7]370
[e9d15d9]371 irq_spinlock_lock(&irq_uspace_hash_table_lock, true);
372 irq_spinlock_lock(&irq->lock, false);
[cecb0789]373
[63e27ef]374 assert(irq->notif_cfg.answerbox == box);
[cecb0789]375
376 /* Remove the IRQ from the uspace IRQ hash table. */
[e9d15d9]377 hash_table_remove_item(&irq_uspace_hash_table, &irq->link);
[cecb0789]378
[e9d15d9]379 /* irq->lock unlocked by the hash table remove_callback */
[da1bafb]380 irq_spinlock_unlock(&irq_uspace_hash_table_lock, true);
[cecb0789]381
[a5d0143]382 /* Free up the IRQ code and associated structures. */
[3cc070d]383 code_free(irq->notif_cfg.code);
384
[3f74275]385 /* Free up the IRQ capability and the underlying kernel object. */
[63d8f43]386 free(cap->kobject);
[3f74275]387 cap_free(TASK, handle);
[cecb0789]388
389 return EOK;
390}
391
[8b243f2]392/** Add a call to the proper answerbox queue.
[2b017ba]393 *
[da1bafb]394 * Assume irq->lock is locked and interrupts disabled.
395 *
396 * @param irq IRQ structure referencing the target answerbox.
397 * @param call IRQ notification call.
[874621f]398 *
[2b017ba]399 */
400static void send_call(irq_t *irq, call_t *call)
[874621f]401{
[da1bafb]402 irq_spinlock_lock(&irq->notif_cfg.answerbox->irq_lock, false);
[cfaa35a]403 list_append(&call->ab_link, &irq->notif_cfg.answerbox->irq_notifs);
[da1bafb]404 irq_spinlock_unlock(&irq->notif_cfg.answerbox->irq_lock, false);
405
[2b017ba]406 waitq_wakeup(&irq->notif_cfg.answerbox->wq, WAKEUP_FIRST);
[874621f]407}
408
[a5d0143]409/** Apply the top-half IRQ code to find out whether to accept the IRQ or not.
[874621f]410 *
[da1bafb]411 * @param irq IRQ structure.
412 *
[a5d0143]413 * @return IRQ_ACCEPT if the interrupt is accepted by the IRQ code.
414 * @return IRQ_DECLINE if the interrupt is not accepted byt the IRQ code.
[cecb0789]415 *
[874621f]416 */
[cecb0789]417irq_ownership_t ipc_irq_top_half_claim(irq_t *irq)
[874621f]418{
[cecb0789]419 irq_code_t *code = irq->notif_cfg.code;
[da1bafb]420 uint32_t *scratch = irq->notif_cfg.scratch;
[cecb0789]421
422 if (!irq->notif_cfg.notify)
423 return IRQ_DECLINE;
424
425 if (!code)
426 return IRQ_DECLINE;
427
[01e39cbe]428 for (size_t i = 0; i < code->cmdcount; i++) {
[da1bafb]429 uintptr_t srcarg = code->cmds[i].srcarg;
430 uintptr_t dstarg = code->cmds[i].dstarg;
[874621f]431
[cecb0789]432 switch (code->cmds[i].cmd) {
433 case CMD_PIO_READ_8:
[8486c07]434 scratch[dstarg] =
435 pio_read_8((ioport8_t *) code->cmds[i].addr);
[cecb0789]436 break;
437 case CMD_PIO_READ_16:
[8486c07]438 scratch[dstarg] =
439 pio_read_16((ioport16_t *) code->cmds[i].addr);
[cecb0789]440 break;
441 case CMD_PIO_READ_32:
[8486c07]442 scratch[dstarg] =
443 pio_read_32((ioport32_t *) code->cmds[i].addr);
[cecb0789]444 break;
445 case CMD_PIO_WRITE_8:
446 pio_write_8((ioport8_t *) code->cmds[i].addr,
447 (uint8_t) code->cmds[i].value);
448 break;
449 case CMD_PIO_WRITE_16:
450 pio_write_16((ioport16_t *) code->cmds[i].addr,
451 (uint16_t) code->cmds[i].value);
452 break;
453 case CMD_PIO_WRITE_32:
454 pio_write_32((ioport32_t *) code->cmds[i].addr,
455 (uint32_t) code->cmds[i].value);
456 break;
[9cdac5a]457 case CMD_PIO_WRITE_A_8:
[8486c07]458 pio_write_8((ioport8_t *) code->cmds[i].addr,
459 (uint8_t) scratch[srcarg]);
[9cdac5a]460 break;
461 case CMD_PIO_WRITE_A_16:
[8486c07]462 pio_write_16((ioport16_t *) code->cmds[i].addr,
463 (uint16_t) scratch[srcarg]);
[9cdac5a]464 break;
465 case CMD_PIO_WRITE_A_32:
[8486c07]466 pio_write_32((ioport32_t *) code->cmds[i].addr,
467 (uint32_t) scratch[srcarg]);
468 break;
469 case CMD_LOAD:
470 scratch[dstarg] = code->cmds[i].value;
[9cdac5a]471 break;
[8486c07]472 case CMD_AND:
473 scratch[dstarg] = scratch[srcarg] &
474 code->cmds[i].value;
[cecb0789]475 break;
476 case CMD_PREDICATE:
[8486c07]477 if (scratch[srcarg] == 0)
[cecb0789]478 i += code->cmds[i].value;
[8486c07]479
[cecb0789]480 break;
481 case CMD_ACCEPT:
482 return IRQ_ACCEPT;
483 case CMD_DECLINE:
484 default:
485 return IRQ_DECLINE;
486 }
[874621f]487 }
[01e39cbe]488
[cecb0789]489 return IRQ_DECLINE;
[874621f]490}
491
[cecb0789]492/* IRQ top-half handler.
[162f919]493 *
[2b017ba]494 * We expect interrupts to be disabled and the irq->lock already held.
[8b243f2]495 *
[da1bafb]496 * @param irq IRQ structure.
497 *
[162f919]498 */
[cecb0789]499void ipc_irq_top_half_handler(irq_t *irq)
[162f919]500{
[63e27ef]501 assert(irq);
[56c167c]502
[63e27ef]503 assert(interrupts_disabled());
504 assert(irq_spinlock_locked(&irq->lock));
[da1bafb]505
[2b017ba]506 if (irq->notif_cfg.answerbox) {
[da1bafb]507 call_t *call = ipc_call_alloc(FRAME_ATOMIC);
[cecb0789]508 if (!call)
[d8f7362]509 return;
[cecb0789]510
[162f919]511 call->flags |= IPC_CALL_NOTIF;
[43752b6]512 /* Put a counter to the message */
[0c1a5d8a]513 call->priv = ++irq->notif_cfg.counter;
[da1bafb]514
[43752b6]515 /* Set up args */
[228e490]516 IPC_SET_IMETHOD(call->data, irq->notif_cfg.imethod);
[cecb0789]517 IPC_SET_ARG1(call->data, irq->notif_cfg.scratch[1]);
518 IPC_SET_ARG2(call->data, irq->notif_cfg.scratch[2]);
519 IPC_SET_ARG3(call->data, irq->notif_cfg.scratch[3]);
520 IPC_SET_ARG4(call->data, irq->notif_cfg.scratch[4]);
521 IPC_SET_ARG5(call->data, irq->notif_cfg.scratch[5]);
[da1bafb]522
[2b017ba]523 send_call(irq, call);
[162f919]524 }
525}
526
[cecb0789]527/** Send notification message.
[874621f]528 *
[da1bafb]529 * @param irq IRQ structure.
530 * @param a1 Driver-specific payload argument.
531 * @param a2 Driver-specific payload argument.
532 * @param a3 Driver-specific payload argument.
533 * @param a4 Driver-specific payload argument.
534 * @param a5 Driver-specific payload argument.
535 *
[162f919]536 */
[96b02eb9]537void ipc_irq_send_msg(irq_t *irq, sysarg_t a1, sysarg_t a2, sysarg_t a3,
538 sysarg_t a4, sysarg_t a5)
[162f919]539{
[da1bafb]540 irq_spinlock_lock(&irq->lock, true);
541
[cecb0789]542 if (irq->notif_cfg.answerbox) {
[da1bafb]543 call_t *call = ipc_call_alloc(FRAME_ATOMIC);
[cecb0789]544 if (!call) {
[da1bafb]545 irq_spinlock_unlock(&irq->lock, true);
[cecb0789]546 return;
[b14e35f2]547 }
[da1bafb]548
[cecb0789]549 call->flags |= IPC_CALL_NOTIF;
550 /* Put a counter to the message */
551 call->priv = ++irq->notif_cfg.counter;
[da1bafb]552
[228e490]553 IPC_SET_IMETHOD(call->data, irq->notif_cfg.imethod);
[cecb0789]554 IPC_SET_ARG1(call->data, a1);
555 IPC_SET_ARG2(call->data, a2);
556 IPC_SET_ARG3(call->data, a3);
557 IPC_SET_ARG4(call->data, a4);
558 IPC_SET_ARG5(call->data, a5);
559
560 send_call(irq, call);
[b14e35f2]561 }
[da1bafb]562
563 irq_spinlock_unlock(&irq->lock, true);
[162f919]564}
[b45c443]565
[cc73a8a1]566/** @}
[b45c443]567 */
Note: See TracBrowser for help on using the repository browser.