Changeset 19a1800 in mainline for uspace/lib/c/generic/ipc.c
- Timestamp:
- 2011-03-01T22:20:56Z (14 years ago)
- Branches:
- lfn, master, serial, ticket/834-toolchain-update, topic/msim-upgrade, topic/simplify-dev-export
- Children:
- e24e7b1
- Parents:
- 976f546 (diff), ac8285d (diff)
Note: this is a merge changeset, the changes displayed below correspond to the merge itself.
Use the(diff)
links above to see all the changes relative to each parent. - File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
uspace/lib/c/generic/ipc.c
r976f546 r19a1800 45 45 #include <errno.h> 46 46 #include <adt/list.h> 47 #include <stdio.h>48 #include <unistd.h>49 47 #include <futex.h> 50 #include <kernel/synch/synch.h>51 #include <async.h>52 48 #include <fibril.h> 53 #include <assert.h>54 49 55 50 /** 56 * Structures of this type are used for keeping track of sent asynchronous calls57 * and queing unsent calls.51 * Structures of this type are used for keeping track 52 * of sent asynchronous calls and queing unsent calls. 58 53 */ 59 54 typedef struct { 60 55 link_t list; 61 56 62 57 ipc_async_callback_t callback; 63 58 void *private; 59 64 60 union { 65 61 ipc_callid_t callid; … … 69 65 } msg; 70 66 } u; 71 fid_t fid; /**< Fibril waiting for sending this call. */ 67 68 /** Fibril waiting for sending this call. */ 69 fid_t fid; 72 70 } async_call_t; 73 71 … … 76 74 /** List of asynchronous calls that were not accepted by kernel. 77 75 * 78 * It is protected by async_futex, because if the call cannot be sent into the 79 * kernel, the async framework is used automatically. 76 * Protected by async_futex, because if the call is not accepted 77 * by the kernel, the async framework is used automatically. 78 * 80 79 */ 81 80 LIST_INITIALIZE(queued_calls); … … 83 82 static atomic_t ipc_futex = FUTEX_INITIALIZER; 84 83 85 /** Make a fast synchronous call.86 * 87 * Only three payload arguments can be passed using this function. However, this88 * function is faster than the generic ipc_call_sync_slow() because the payload89 * is passed directly in registers.90 * 91 * @param phoneid 92 * @param method 93 * @param arg1 94 * @param arg2 95 * @param arg3 96 * @param result1 97 * @param result2 98 * @param result3 99 * @param result4 100 * @param result5 101 * 102 * @return Negative values represent errors returned by IPC.103 * Otherwise the RETVAL of the answer is returned.104 * /105 int 106 i pc_call_sync_fast(int phoneid, sysarg_t method, sysarg_t arg1, sysarg_t arg2,107 sysarg_t arg 3, sysarg_t *result1, sysarg_t *result2, sysarg_t *result3,108 sysarg_t *result 4, sysarg_t *result5)84 /** Fast synchronous call. 85 * 86 * Only three payload arguments can be passed using this function. However, 87 * this function is faster than the generic ipc_call_sync_slow() because 88 * the payload is passed directly in registers. 89 * 90 * @param phoneid Phone handle for the call. 91 * @param method Requested method. 92 * @param arg1 Service-defined payload argument. 93 * @param arg2 Service-defined payload argument. 94 * @param arg3 Service-defined payload argument. 95 * @param result1 If non-NULL, the return ARG1 will be stored there. 96 * @param result2 If non-NULL, the return ARG2 will be stored there. 97 * @param result3 If non-NULL, the return ARG3 will be stored there. 98 * @param result4 If non-NULL, the return ARG4 will be stored there. 99 * @param result5 If non-NULL, the return ARG5 will be stored there. 100 * 101 * @return Negative values representing IPC errors. 102 * @return Otherwise the RETVAL of the answer. 103 * 104 */ 105 int ipc_call_sync_fast(int phoneid, sysarg_t method, sysarg_t arg1, 106 sysarg_t arg2, sysarg_t arg3, sysarg_t *result1, sysarg_t *result2, 107 sysarg_t *result3, sysarg_t *result4, sysarg_t *result5) 109 108 { 110 109 ipc_call_t resdata; 111 int callres; 112 113 callres = __SYSCALL6(SYS_IPC_CALL_SYNC_FAST, phoneid, method, arg1, 110 int callres = __SYSCALL6(SYS_IPC_CALL_SYNC_FAST, phoneid, method, arg1, 114 111 arg2, arg3, (sysarg_t) &resdata); 115 112 if (callres) 116 113 return callres; 114 117 115 if (result1) 118 116 *result1 = IPC_GET_ARG1(resdata); … … 125 123 if (result5) 126 124 *result5 = IPC_GET_ARG5(resdata); 127 125 128 126 return IPC_GET_RETVAL(resdata); 129 127 } 130 128 131 /** Make a synchronous call transmitting 5 arguments of payload.129 /** Synchronous call transmitting 5 arguments of payload. 132 130 * 133 131 * @param phoneid Phone handle for the call. … … 144 142 * @param result5 If non-NULL, storage for the fifth return argument. 145 143 * 146 * @return Negative value means IPC error.147 * 148 * 149 */ 150 int 151 ipc_call_sync_slow(int phoneid, sysarg_t imethod, sysarg_t arg1, sysarg_t arg2,152 sysarg_t arg3, sysarg_t arg4, sysarg_t arg5, sysarg_t *result1,153 sysarg_t *result 2, sysarg_t *result3, sysarg_t *result4, sysarg_t *result5)144 * @return Negative values representing IPC errors. 145 * @return Otherwise the RETVAL of the answer. 146 * 147 */ 148 int ipc_call_sync_slow(int phoneid, sysarg_t imethod, sysarg_t arg1, 149 sysarg_t arg2, sysarg_t arg3, sysarg_t arg4, sysarg_t arg5, 150 sysarg_t *result1, sysarg_t *result2, sysarg_t *result3, sysarg_t *result4, 151 sysarg_t *result5) 154 152 { 155 153 ipc_call_t data; … … 181 179 } 182 180 183 /** S yscall to send asynchronous message.181 /** Send asynchronous message via syscall. 184 182 * 185 183 * @param phoneid Phone handle for the call. … … 189 187 * 190 188 */ 191 static ipc_callid_t _ipc_call_async(int phoneid, ipc_call_t *data)189 static ipc_callid_t ipc_call_async_internal(int phoneid, ipc_call_t *data) 192 190 { 193 191 return __SYSCALL2(SYS_IPC_CALL_ASYNC_SLOW, phoneid, (sysarg_t) data); 194 192 } 195 193 196 /** Prolog to ipc_call_async_*() functions. 197 * 198 * @param private Argument for the answer/error callback. 199 * @param callback Answer/error callback. 200 * 201 * @return New, partially initialized async_call structure or NULL. 194 /** Prolog for ipc_call_async_*() functions. 195 * 196 * @param private Argument for the answer/error callback. 197 * @param callback Answer/error callback. 198 * 199 * @return New, partially initialized async_call structure or NULL. 200 * 202 201 */ 203 202 static inline async_call_t *ipc_prepare_async(void *private, 204 203 ipc_async_callback_t callback) 205 204 { 206 async_call_t *call; 207 208 call = malloc(sizeof(*call)); 205 async_call_t *call = 206 (async_call_t *) malloc(sizeof(async_call_t)); 209 207 if (!call) { 210 208 if (callback) 211 209 callback(private, ENOMEM, NULL); 210 212 211 return NULL; 213 212 } 213 214 214 call->callback = callback; 215 215 call->private = private; 216 216 217 217 return call; 218 218 } 219 219 220 /** Epilogue of ipc_call_async_*() functions. 221 * 222 * @param callid Value returned by the SYS_IPC_CALL_ASYNC_* syscall. 223 * @param phoneid Phone handle through which the call was made. 224 * @param call async_call structure returned by ipc_prepare_async(). 225 * @param can_preempt If non-zero, the current fibril can be preempted in this 226 * call. 220 /** Epilog for ipc_call_async_*() functions. 221 * 222 * @param callid Value returned by the SYS_IPC_CALL_ASYNC_* syscall. 223 * @param phoneid Phone handle through which the call was made. 224 * @param call Structure returned by ipc_prepare_async(). 225 * @param can_preempt If true, the current fibril can be preempted 226 * in this call. 227 * 227 228 */ 228 229 static inline void ipc_finish_async(ipc_callid_t callid, int phoneid, 229 async_call_t *call, int can_preempt) 230 { 231 if (!call) { /* Nothing to do regardless if failed or not */ 230 async_call_t *call, bool can_preempt) 231 { 232 if (!call) { 233 /* Nothing to do regardless if failed or not */ 232 234 futex_up(&ipc_futex); 233 235 return; 234 236 } 235 237 236 238 if (callid == (ipc_callid_t) IPC_CALLRET_FATAL) { 237 239 futex_up(&ipc_futex); 240 238 241 /* Call asynchronous handler with error code */ 239 242 if (call->callback) 240 243 call->callback(call->private, ENOENT, NULL); 244 241 245 free(call); 242 246 return; 243 247 } 244 248 245 249 if (callid == (ipc_callid_t) IPC_CALLRET_TEMPORARY) { 246 250 futex_up(&ipc_futex); 247 251 248 252 call->u.msg.phoneid = phoneid; 249 253 250 254 futex_down(&async_futex); 251 255 list_append(&call->list, &queued_calls); 252 256 253 257 if (can_preempt) { 254 258 call->fid = fibril_get_id(); … … 259 263 futex_up(&async_futex); 260 264 } 265 261 266 return; 262 267 } 268 263 269 call->u.callid = callid; 270 264 271 /* Add call to the list of dispatched calls */ 265 272 list_append(&call->list, &dispatched_calls); 266 273 futex_up(&ipc_futex); 267 268 } 269 270 /** Make a fast asynchronous call. 274 } 275 276 /** Fast asynchronous call. 271 277 * 272 278 * This function can only handle four arguments of payload. It is, however, … … 274 280 * 275 281 * Note that this function is a void function. 276 * During normal opertation, answering this call will trigger the callback. 277 * In case of fatal error, call the callback handler with the proper error code. 278 * If the call cannot be temporarily made, queue it. 282 * 283 * During normal operation, answering this call will trigger the callback. 284 * In case of fatal error, the callback handler is called with the proper 285 * error code. If the call cannot be temporarily made, it is queued. 279 286 * 280 287 * @param phoneid Phone handle for the call. … … 286 293 * @param private Argument to be passed to the answer/error callback. 287 294 * @param callback Answer or error callback. 288 * @param can_preempt If non-zero, the current fibril will be preempted in295 * @param can_preempt If true, the current fibril will be preempted in 289 296 * case the kernel temporarily refuses to accept more 290 297 * asynchronous calls. … … 293 300 void ipc_call_async_fast(int phoneid, sysarg_t imethod, sysarg_t arg1, 294 301 sysarg_t arg2, sysarg_t arg3, sysarg_t arg4, void *private, 295 ipc_async_callback_t callback, intcan_preempt)302 ipc_async_callback_t callback, bool can_preempt) 296 303 { 297 304 async_call_t *call = NULL; … … 304 311 305 312 /* 306 * We need to make sure that we get callid before another thread307 * accesses the queue again.313 * We need to make sure that we get callid 314 * before another thread accesses the queue again. 308 315 */ 316 309 317 futex_down(&ipc_futex); 310 318 ipc_callid_t callid = __SYSCALL6(SYS_IPC_CALL_ASYNC_FAST, phoneid, … … 317 325 return; 318 326 } 327 319 328 IPC_SET_IMETHOD(call->u.msg.data, imethod); 320 329 IPC_SET_ARG1(call->u.msg.data, arg1); … … 322 331 IPC_SET_ARG3(call->u.msg.data, arg3); 323 332 IPC_SET_ARG4(call->u.msg.data, arg4); 333 324 334 /* 325 335 * To achieve deterministic behavior, we always zero out the 326 336 * arguments that are beyond the limits of the fast version. 327 337 */ 338 328 339 IPC_SET_ARG5(call->u.msg.data, 0); 329 340 } 341 330 342 ipc_finish_async(callid, phoneid, call, can_preempt); 331 343 } 332 344 333 /** Make an asynchronous call transmitting the entire payload.345 /** Asynchronous call transmitting the entire payload. 334 346 * 335 347 * Note that this function is a void function. 336 * During normal opertation, answering this call will trigger the callback. 337 * In case of fatal error, call the callback handler with the proper error code. 338 * If the call cannot be temporarily made, queue it. 348 * 349 * During normal operation, answering this call will trigger the callback. 350 * In case of fatal error, the callback handler is called with the proper 351 * error code. If the call cannot be temporarily made, it is queued. 339 352 * 340 353 * @param phoneid Phone handle for the call. … … 347 360 * @param private Argument to be passed to the answer/error callback. 348 361 * @param callback Answer or error callback. 349 * @param can_preempt If non-zero, the current fibril will be preempted in362 * @param can_preempt If true, the current fibril will be preempted in 350 363 * case the kernel temporarily refuses to accept more 351 364 * asynchronous calls. … … 354 367 void ipc_call_async_slow(int phoneid, sysarg_t imethod, sysarg_t arg1, 355 368 sysarg_t arg2, sysarg_t arg3, sysarg_t arg4, sysarg_t arg5, void *private, 356 ipc_async_callback_t callback, int can_preempt) 357 { 358 async_call_t *call; 359 ipc_callid_t callid; 360 361 call = ipc_prepare_async(private, callback); 369 ipc_async_callback_t callback, bool can_preempt) 370 { 371 async_call_t *call = ipc_prepare_async(private, callback); 362 372 if (!call) 363 373 return; 364 374 365 375 IPC_SET_IMETHOD(call->u.msg.data, imethod); 366 376 IPC_SET_ARG1(call->u.msg.data, arg1); … … 369 379 IPC_SET_ARG4(call->u.msg.data, arg4); 370 380 IPC_SET_ARG5(call->u.msg.data, arg5); 381 371 382 /* 372 * We need to make sure that we get callid before another thread373 * accesses the queue again.383 * We need to make sure that we get callid 384 * before another threadaccesses the queue again. 374 385 */ 386 375 387 futex_down(&ipc_futex); 376 callid = _ipc_call_async(phoneid, &call->u.msg.data); 377 388 ipc_callid_t callid = 389 ipc_call_async_internal(phoneid, &call->u.msg.data); 390 378 391 ipc_finish_async(callid, phoneid, call, can_preempt); 379 392 } 380 393 381 382 /** Answer a received call - fast version. 394 /** Answer received call (fast version). 383 395 * 384 396 * The fast answer makes use of passing retval and first four arguments in 385 397 * registers. If you need to return more, use the ipc_answer_slow() instead. 386 398 * 387 * @param callid Hash of the call being answered. 388 * @param retval Return value. 389 * @param arg1 First return argument. 390 * @param arg2 Second return argument. 391 * @param arg3 Third return argument. 392 * @param arg4 Fourth return argument. 393 * 394 * @return Zero on success or a value from @ref errno.h on failure. 399 * @param callid Hash of the call being answered. 400 * @param retval Return value. 401 * @param arg1 First return argument. 402 * @param arg2 Second return argument. 403 * @param arg3 Third return argument. 404 * @param arg4 Fourth return argument. 405 * 406 * @return Zero on success. 407 * @return Value from @ref errno.h on failure. 408 * 395 409 */ 396 410 sysarg_t ipc_answer_fast(ipc_callid_t callid, sysarg_t retval, sysarg_t arg1, … … 401 415 } 402 416 403 /** Answer a received call - slow full version. 404 * 405 * @param callid Hash of the call being answered. 406 * @param retval Return value. 407 * @param arg1 First return argument. 408 * @param arg2 Second return argument. 409 * @param arg3 Third return argument. 410 * @param arg4 Fourth return argument. 411 * @param arg5 Fifth return argument. 412 * 413 * @return Zero on success or a value from @ref errno.h on failure. 417 /** Answer received call (entire payload). 418 * 419 * @param callid Hash of the call being answered. 420 * @param retval Return value. 421 * @param arg1 First return argument. 422 * @param arg2 Second return argument. 423 * @param arg3 Third return argument. 424 * @param arg4 Fourth return argument. 425 * @param arg5 Fifth return argument. 426 * 427 * @return Zero on success. 428 * @return Value from @ref errno.h on failure. 429 * 414 430 */ 415 431 sysarg_t ipc_answer_slow(ipc_callid_t callid, sysarg_t retval, sysarg_t arg1, … … 417 433 { 418 434 ipc_call_t data; 419 435 420 436 IPC_SET_RETVAL(data, retval); 421 437 IPC_SET_ARG1(data, arg1); … … 424 440 IPC_SET_ARG4(data, arg4); 425 441 IPC_SET_ARG5(data, arg5); 426 442 427 443 return __SYSCALL2(SYS_IPC_ANSWER_SLOW, callid, (sysarg_t) &data); 428 444 } 429 445 430 431 /** Try to dispatch queued calls from the async queue. */ 432 static void try_dispatch_queued_calls(void) 433 { 434 async_call_t *call; 435 ipc_callid_t callid; 436 446 /** Try to dispatch queued calls from the async queue. 447 * 448 */ 449 static void dispatch_queued_calls(void) 450 { 437 451 /** @todo 438 * Integrate intelligently ipc_futex ,so that it is locked during439 * ipc_call_async_*() ,until it is added to dispatched_calls.452 * Integrate intelligently ipc_futex so that it is locked during 453 * ipc_call_async_*() until it is added to dispatched_calls. 440 454 */ 455 441 456 futex_down(&async_futex); 457 442 458 while (!list_empty(&queued_calls)) { 443 call = list_get_instance(queued_calls.next, async_call_t, list); 444 callid = _ipc_call_async(call->u.msg.phoneid, 445 &call->u.msg.data); 446 if (callid == (ipc_callid_t) IPC_CALLRET_TEMPORARY) { 459 async_call_t *call = 460 list_get_instance(queued_calls.next, async_call_t, list); 461 ipc_callid_t callid = 462 ipc_call_async_internal(call->u.msg.phoneid, &call->u.msg.data); 463 464 if (callid == (ipc_callid_t) IPC_CALLRET_TEMPORARY) 447 465 break; 448 }466 449 467 list_remove(&call->list); 450 468 451 469 futex_up(&async_futex); 470 452 471 if (call->fid) 453 472 fibril_add_ready(call->fid); … … 456 475 if (call->callback) 457 476 call->callback(call->private, ENOENT, NULL); 477 458 478 free(call); 459 479 } else { 460 480 call->u.callid = callid; 481 461 482 futex_down(&ipc_futex); 462 483 list_append(&call->list, &dispatched_calls); 463 484 futex_up(&ipc_futex); 464 485 } 486 465 487 futex_down(&async_futex); 466 488 } 489 467 490 futex_up(&async_futex); 468 491 } 469 492 470 /** Handle areceived answer.493 /** Handle received answer. 471 494 * 472 495 * Find the hash of the answer and call the answer callback. 473 496 * 474 * @todo Make it use hash table. 475 * 476 * @param callid Hash of the received answer. 477 * The answer has the same hash as the request OR'ed with 478 * the IPC_CALLID_ANSWERED bit. 479 * @param data Call data of the answer. 497 * The answer has the same hash as the request OR'ed with 498 * the IPC_CALLID_ANSWERED bit. 499 * 500 * @todo Use hash table. 501 * 502 * @param callid Hash of the received answer. 503 * @param data Call data of the answer. 504 * 480 505 */ 481 506 static void handle_answer(ipc_callid_t callid, ipc_call_t *data) 482 507 { 508 callid &= ~IPC_CALLID_ANSWERED; 509 510 futex_down(&ipc_futex); 511 483 512 link_t *item; 484 async_call_t *call;485 486 callid &= ~IPC_CALLID_ANSWERED;487 488 futex_down(&ipc_futex);489 513 for (item = dispatched_calls.next; item != &dispatched_calls; 490 514 item = item->next) { 491 call = list_get_instance(item, async_call_t, list); 515 async_call_t *call = 516 list_get_instance(item, async_call_t, list); 517 492 518 if (call->u.callid == callid) { 493 519 list_remove(&call->list); 520 494 521 futex_up(&ipc_futex); 522 495 523 if (call->callback) 496 call->callback(call->private, 524 call->callback(call->private, 497 525 IPC_GET_RETVAL(*data), data); 526 498 527 free(call); 499 528 return; 500 529 } 501 530 } 531 502 532 futex_up(&ipc_futex); 503 533 } 504 534 505 506 /** Wait for a first call to come. 507 * 508 * @param call Storage where the incoming call data will be stored.509 * @param usec Timeout in microseconds510 * @param flags Flags passed to SYS_IPC_WAIT (blocking, nonblocking).511 * 512 * @return Hash of the call. Note that certain bits have special513 * meaning. IPC_CALLID_ANSWERED will be set in an answer514 * and IPC_CALLID_NOTIFICATION is used for notifications.515 * 516 */ 517 ipc_callid_t ipc_wait_cycle(ipc_call_t *call, uint32_t usec,int flags)518 { 519 ipc_callid_t callid ;520 521 callid = __SYSCALL3(SYS_IPC_WAIT, (sysarg_t) call, usec, flags);535 /** Wait for first IPC call to come. 536 * 537 * @param call Incoming call storage. 538 * @param usec Timeout in microseconds 539 * @param flags Flags passed to SYS_IPC_WAIT (blocking, nonblocking). 540 * 541 * @return Hash of the call. Note that certain bits have special 542 * meaning: IPC_CALLID_ANSWERED is set in an answer 543 * and IPC_CALLID_NOTIFICATION is used for notifications. 544 * 545 */ 546 ipc_callid_t ipc_wait_cycle(ipc_call_t *call, sysarg_t usec, 547 unsigned int flags) 548 { 549 ipc_callid_t callid = 550 __SYSCALL3(SYS_IPC_WAIT, (sysarg_t) call, usec, flags); 551 522 552 /* Handle received answers */ 523 553 if (callid & IPC_CALLID_ANSWERED) { 524 554 handle_answer(callid, call); 525 try_dispatch_queued_calls();555 dispatch_queued_calls(); 526 556 } 527 557 528 558 return callid; 529 559 } 530 560 531 /** Wait some time for an IPC call. 532 * 533 * The call will return after an answer is received. 534 * 535 * @param call Storage where the incoming call data will be stored. 536 * @param usec Timeout in microseconds. 537 * 538 * @return Hash of the answer. 539 */ 540 ipc_callid_t ipc_wait_for_call_timeout(ipc_call_t *call, uint32_t usec) 561 /** Interrupt one thread of this task from waiting for IPC. 562 * 563 */ 564 void ipc_poke(void) 565 { 566 __SYSCALL0(SYS_IPC_POKE); 567 } 568 569 /** Wait for first IPC call to come. 570 * 571 * Only requests are returned, answers are processed internally. 572 * 573 * @param call Incoming call storage. 574 * @param usec Timeout in microseconds 575 * 576 * @return Hash of the call. 577 * 578 */ 579 ipc_callid_t ipc_wait_for_call_timeout(ipc_call_t *call, sysarg_t usec) 541 580 { 542 581 ipc_callid_t callid; 543 582 544 583 do { 545 584 callid = ipc_wait_cycle(call, usec, SYNCH_FLAGS_NONE); 546 585 } while (callid & IPC_CALLID_ANSWERED); 547 586 548 587 return callid; 549 588 } … … 551 590 /** Check if there is an IPC call waiting to be picked up. 552 591 * 553 * @param call Storage where the incoming call will be stored. 554 * @return Hash of the answer. 592 * Only requests are returned, answers are processed internally. 593 * 594 * @param call Incoming call storage. 595 * 596 * @return Hash of the call. 597 * 555 598 */ 556 599 ipc_callid_t ipc_trywait_for_call(ipc_call_t *call) 557 600 { 558 601 ipc_callid_t callid; 559 602 560 603 do { 561 604 callid = ipc_wait_cycle(call, SYNCH_NO_TIMEOUT, 562 605 SYNCH_FLAGS_NON_BLOCKING); 563 606 } while (callid & IPC_CALLID_ANSWERED); 564 607 565 608 return callid; 566 609 } 567 610 568 /** Interrupt one thread of this task from waiting for IPC. */ 569 void ipc_poke(void) 570 { 571 __SYSCALL0(SYS_IPC_POKE); 572 } 573 574 /** Ask destination to do a callback connection. 575 * 576 * @param phoneid Phone handle used for contacting the other side. 577 * @param arg1 Service-defined argument. 578 * @param arg2 Service-defined argument. 579 * @param arg3 Service-defined argument. 580 * @param phonehash Storage where the library will store an opaque 581 * identifier of the phone that will be used for incoming 582 * calls. This identifier can be used for connection 583 * tracking. 584 * 585 * @return Zero on success or a negative error code. 586 */ 587 int ipc_connect_to_me(int phoneid, int arg1, int arg2, int arg3, 588 sysarg_t *phonehash) 611 /** Request callback connection. 612 * 613 * The @a taskhash and @a phonehash identifiers returned 614 * by the kernel can be used for connection tracking. 615 * 616 * @param phoneid Phone handle used for contacting the other side. 617 * @param arg1 User defined argument. 618 * @param arg2 User defined argument. 619 * @param arg3 User defined argument. 620 * @param taskhash Opaque identifier of the client task. 621 * @param phonehash Opaque identifier of the phone that will 622 * be used for incoming calls. 623 * 624 * @return Zero on success or a negative error code. 625 * 626 */ 627 int ipc_connect_to_me(int phoneid, sysarg_t arg1, sysarg_t arg2, sysarg_t arg3, 628 sysarg_t *taskhash, sysarg_t *phonehash) 589 629 { 590 630 return ipc_call_sync_3_5(phoneid, IPC_M_CONNECT_TO_ME, arg1, arg2, 591 arg3, NULL, NULL, NULL, NULL, phonehash); 592 } 593 594 /** Ask through phone for a new connection to some service. 595 * 596 * @param phoneid Phone handle used for contacting the other side. 597 * @param arg1 User defined argument. 598 * @param arg2 User defined argument. 599 * @param arg3 User defined argument. 600 * 601 * @return New phone handle on success or a negative error code. 602 */ 603 int ipc_connect_me_to(int phoneid, int arg1, int arg2, int arg3) 631 arg3, NULL, NULL, NULL, taskhash, phonehash); 632 } 633 634 /** Request new connection. 635 * 636 * @param phoneid Phone handle used for contacting the other side. 637 * @param arg1 User defined argument. 638 * @param arg2 User defined argument. 639 * @param arg3 User defined argument. 640 * 641 * @return New phone handle on success or a negative error code. 642 * 643 */ 644 int ipc_connect_me_to(int phoneid, sysarg_t arg1, sysarg_t arg2, sysarg_t arg3) 604 645 { 605 646 sysarg_t newphid; 606 int res; 607 608 res = ipc_call_sync_3_5(phoneid, IPC_M_CONNECT_ME_TO, arg1, arg2, arg3, 647 int res = ipc_call_sync_3_5(phoneid, IPC_M_CONNECT_ME_TO, arg1, arg2, arg3, 609 648 NULL, NULL, NULL, NULL, &newphid); 610 649 if (res) 611 650 return res; 651 612 652 return newphid; 613 653 } 614 654 615 /** Ask through phone for a new connection to some service.655 /** Request new connection (blocking) 616 656 * 617 657 * If the connection is not available at the moment, the 618 * call will block. 619 * 620 * @param phoneid Phone handle used for contacting the other side. 621 * @param arg1 User defined argument. 622 * @param arg2 User defined argument. 623 * @param arg3 User defined argument. 624 * 625 * @return New phone handle on success or a negative error code. 626 */ 627 int ipc_connect_me_to_blocking(int phoneid, int arg1, int arg2, int arg3) 658 * call should block. This has to be, however, implemented 659 * on the server side. 660 * 661 * @param phoneid Phone handle used for contacting the other side. 662 * @param arg1 User defined argument. 663 * @param arg2 User defined argument. 664 * @param arg3 User defined argument. 665 * 666 * @return New phone handle on success or a negative error code. 667 * 668 */ 669 int ipc_connect_me_to_blocking(int phoneid, sysarg_t arg1, sysarg_t arg2, 670 sysarg_t arg3) 628 671 { 629 672 sysarg_t newphid; 630 int res; 631 632 res = ipc_call_sync_4_5(phoneid, IPC_M_CONNECT_ME_TO, arg1, arg2, arg3, 673 int res = ipc_call_sync_4_5(phoneid, IPC_M_CONNECT_ME_TO, arg1, arg2, arg3, 633 674 IPC_FLAG_BLOCKING, NULL, NULL, NULL, NULL, &newphid); 634 675 if (res) 635 676 return res; 677 636 678 return newphid; 637 679 } … … 639 681 /** Hang up a phone. 640 682 * 641 * @param phoneid Handle of the phone to be hung up. 642 * 643 * @return Zero on success or a negative error code. 683 * @param phoneid Handle of the phone to be hung up. 684 * 685 * @return Zero on success or a negative error code. 686 * 644 687 */ 645 688 int ipc_hangup(int phoneid) … … 648 691 } 649 692 650 /** Register IRQ notification.651 *652 * @param inr IRQ number.653 * @param devno Device number of the device generating inr.654 * @param method Use this method for notifying me.655 * @param ucode Top-half pseudocode handler.656 *657 * @return Value returned by the kernel.658 */659 int ipc_register_irq(int inr, int devno, int method, irq_code_t *ucode)660 {661 return __SYSCALL4(SYS_IPC_REGISTER_IRQ, inr, devno, method,662 (sysarg_t) ucode);663 }664 665 /** Unregister IRQ notification.666 *667 * @param inr IRQ number.668 * @param devno Device number of the device generating inr.669 *670 * @return Value returned by the kernel.671 */672 int ipc_unregister_irq(int inr, int devno)673 {674 return __SYSCALL2(SYS_IPC_UNREGISTER_IRQ, inr, devno);675 }676 677 693 /** Forward a received call to another destination. 694 * 695 * For non-system methods, the old method, arg1 and arg2 are rewritten 696 * by the new values. For system methods, the new method, arg1 and arg2 697 * are written to the old arg1, arg2 and arg3, respectivelly. Calls with 698 * immutable methods are forwarded verbatim. 678 699 * 679 700 * @param callid Hash of the call to forward. … … 686 707 * @return Zero on success or an error code. 687 708 * 688 * For non-system methods, the old method, arg1 and arg2 are rewritten by the 689 * new values. For system methods, the new method, arg1 and arg2 are written 690 * to the old arg1, arg2 and arg3, respectivelly. Calls with immutable 691 * methods are forwarded verbatim. 692 */ 693 int ipc_forward_fast(ipc_callid_t callid, int phoneid, int imethod, 694 sysarg_t arg1, sysarg_t arg2, int mode) 709 */ 710 int ipc_forward_fast(ipc_callid_t callid, int phoneid, sysarg_t imethod, 711 sysarg_t arg1, sysarg_t arg2, unsigned int mode) 695 712 { 696 713 return __SYSCALL6(SYS_IPC_FORWARD_FAST, callid, phoneid, imethod, arg1, … … 698 715 } 699 716 700 701 int ipc_forward_slow(ipc_callid_t callid, int phoneid, int imethod, 717 int ipc_forward_slow(ipc_callid_t callid, int phoneid, sysarg_t imethod, 702 718 sysarg_t arg1, sysarg_t arg2, sysarg_t arg3, sysarg_t arg4, sysarg_t arg5, 703 int mode)719 unsigned int mode) 704 720 { 705 721 ipc_call_t data; … … 712 728 IPC_SET_ARG5(data, arg5); 713 729 714 return __SYSCALL4(SYS_IPC_FORWARD_SLOW, callid, phoneid, (sysarg_t) &data, mode); 715 } 716 717 /** Wrapper for making IPC_M_SHARE_IN calls. 718 * 719 * @param phoneid Phone that will be used to contact the receiving side. 720 * @param dst Destination address space area base. 721 * @param size Size of the destination address space area. 722 * @param arg User defined argument. 723 * @param flags Storage where the received flags will be stored. Can be 724 * NULL. 725 * 726 * @return Zero on success or a negative error code from errno.h. 730 return __SYSCALL4(SYS_IPC_FORWARD_SLOW, callid, phoneid, (sysarg_t) &data, 731 mode); 732 } 733 734 /** Wrapper for IPC_M_SHARE_IN calls. 735 * 736 * @param phoneid Phone that will be used to contact the receiving side. 737 * @param dst Destination address space area base. 738 * @param size Size of the destination address space area. 739 * @param arg User defined argument. 740 * @param flags Storage for received flags. Can be NULL. 741 * 742 * @return Zero on success or a negative error code from errno.h. 743 * 727 744 */ 728 745 int ipc_share_in_start(int phoneid, void *dst, size_t size, sysarg_t arg, 729 int *flags)746 unsigned int *flags) 730 747 { 731 748 sysarg_t tmp_flags = 0; … … 734 751 735 752 if (flags) 736 *flags = tmp_flags;753 *flags = (unsigned int) tmp_flags; 737 754 738 755 return res; … … 741 758 /** Wrapper for answering the IPC_M_SHARE_IN calls. 742 759 * 743 * This wrapper only makes it more comfortable to answer IPC_M_DATA_READ calls 744 * so that the user doesn't have to remember the meaning of each IPC argument. 745 * 746 * @param callid Hash of the IPC_M_DATA_READ call to answer. 747 * @param src Source address space base. 748 * @param flags Flags to be used for sharing. Bits can be only cleared. 749 * 750 * @return Zero on success or a value from @ref errno.h on failure. 751 */ 752 int ipc_share_in_finalize(ipc_callid_t callid, void *src, int flags) 760 * This wrapper only makes it more comfortable to answer IPC_M_DATA_READ 761 * calls so that the user doesn't have to remember the meaning of each 762 * IPC argument. 763 * 764 * @param callid Hash of the IPC_M_DATA_READ call to answer. 765 * @param src Source address space base. 766 * @param flags Flags to be used for sharing. Bits can be only cleared. 767 * 768 * @return Zero on success or a value from @ref errno.h on failure. 769 * 770 */ 771 int ipc_share_in_finalize(ipc_callid_t callid, void *src, unsigned int flags) 753 772 { 754 773 return ipc_answer_2(callid, EOK, (sysarg_t) src, (sysarg_t) flags); 755 774 } 756 775 757 /** Wrapper for making IPC_M_SHARE_OUT calls. 758 * 759 * @param phoneid Phone that will be used to contact the receiving side. 760 * @param src Source address space area base address. 761 * @param flags Flags to be used for sharing. Bits can be only cleared. 762 * 763 * @return Zero on success or a negative error code from errno.h. 764 */ 765 int ipc_share_out_start(int phoneid, void *src, int flags) 776 /** Wrapper for IPC_M_SHARE_OUT calls. 777 * 778 * @param phoneid Phone that will be used to contact the receiving side. 779 * @param src Source address space area base address. 780 * @param flags Flags to be used for sharing. Bits can be only cleared. 781 * 782 * @return Zero on success or a negative error code from errno.h. 783 * 784 */ 785 int ipc_share_out_start(int phoneid, void *src, unsigned int flags) 766 786 { 767 787 return ipc_call_sync_3_0(phoneid, IPC_M_SHARE_OUT, (sysarg_t) src, 0, … … 771 791 /** Wrapper for answering the IPC_M_SHARE_OUT calls. 772 792 * 773 * This wrapper only makes it more comfortable to answer IPC_M_SHARE_OUT calls 774 * so that the user doesn't have to remember the meaning of each IPC argument. 775 * 776 * @param callid Hash of the IPC_M_DATA_WRITE call to answer. 777 * @param dst Destination address space area base address. 778 * 779 * @return Zero on success or a value from @ref errno.h on failure. 793 * This wrapper only makes it more comfortable to answer IPC_M_SHARE_OUT 794 * calls so that the user doesn't have to remember the meaning of each 795 * IPC argument. 796 * 797 * @param callid Hash of the IPC_M_DATA_WRITE call to answer. 798 * @param dst Destination address space area base address. 799 * 800 * @return Zero on success or a value from @ref errno.h on failure. 801 * 780 802 */ 781 803 int ipc_share_out_finalize(ipc_callid_t callid, void *dst) … … 784 806 } 785 807 786 787 /** Wrapper for making IPC_M_DATA_READ calls. 788 * 789 * @param phoneid Phone that will be used to contact the receiving side.790 * @param dst Address of the beginningof the destination buffer.791 * @param size Size of the destination buffer.792 * 793 * @return Zero on success or a negative error code from errno.h.808 /** Wrapper for IPC_M_DATA_READ calls. 809 * 810 * @param phoneid Phone that will be used to contact the receiving side. 811 * @param dst Address of the beginning of the destination buffer. 812 * @param size Size of the destination buffer. 813 * 814 * @return Zero on success or a negative error code from errno.h. 815 * 794 816 */ 795 817 int ipc_data_read_start(int phoneid, void *dst, size_t size) … … 801 823 /** Wrapper for answering the IPC_M_DATA_READ calls. 802 824 * 803 * This wrapper only makes it more comfortable to answer IPC_M_DATA_READ calls 804 * so that the user doesn't have to remember the meaning of each IPC argument. 805 * 806 * @param callid Hash of the IPC_M_DATA_READ call to answer. 807 * @param src Source address for the IPC_M_DATA_READ call. 808 * @param size Size for the IPC_M_DATA_READ call. Can be smaller than 809 * the maximum size announced by the sender. 810 * 811 * @return Zero on success or a value from @ref errno.h on failure. 825 * This wrapper only makes it more comfortable to answer IPC_M_DATA_READ 826 * calls so that the user doesn't have to remember the meaning of each 827 * IPC argument. 828 * 829 * @param callid Hash of the IPC_M_DATA_READ call to answer. 830 * @param src Source address for the IPC_M_DATA_READ call. 831 * @param size Size for the IPC_M_DATA_READ call. Can be smaller than 832 * the maximum size announced by the sender. 833 * 834 * @return Zero on success or a value from @ref errno.h on failure. 835 * 812 836 */ 813 837 int ipc_data_read_finalize(ipc_callid_t callid, const void *src, size_t size) … … 816 840 } 817 841 818 /** Wrapper for making IPC_M_DATA_WRITE calls. 819 * 820 * @param phoneid Phone that will be used to contact the receiving side. 821 * @param src Address of the beginning of the source buffer. 822 * @param size Size of the source buffer. 823 * 824 * @return Zero on success or a negative error code from errno.h. 842 /** Wrapper for IPC_M_DATA_WRITE calls. 843 * 844 * @param phoneid Phone that will be used to contact the receiving side. 845 * @param src Address of the beginning of the source buffer. 846 * @param size Size of the source buffer. 847 * 848 * @return Zero on success or a negative error code from errno.h. 849 * 825 850 */ 826 851 int ipc_data_write_start(int phoneid, const void *src, size_t size) … … 832 857 /** Wrapper for answering the IPC_M_DATA_WRITE calls. 833 858 * 834 * This wrapper only makes it more comfortable to answer IPC_M_DATA_WRITE calls 835 * so that the user doesn't have to remember the meaning of each IPC argument. 836 * 837 * @param callid Hash of the IPC_M_DATA_WRITE call to answer. 838 * @param dst Final destination address for the IPC_M_DATA_WRITE call. 839 * @param size Final size for the IPC_M_DATA_WRITE call. 840 * 841 * @return Zero on success or a value from @ref errno.h on failure. 859 * This wrapper only makes it more comfortable to answer IPC_M_DATA_WRITE 860 * calls so that the user doesn't have to remember the meaning of each 861 * IPC argument. 862 * 863 * @param callid Hash of the IPC_M_DATA_WRITE call to answer. 864 * @param dst Final destination address for the IPC_M_DATA_WRITE call. 865 * @param size Final size for the IPC_M_DATA_WRITE call. 866 * 867 * @return Zero on success or a value from @ref errno.h on failure. 868 * 842 869 */ 843 870 int ipc_data_write_finalize(ipc_callid_t callid, void *dst, size_t size)
Note:
See TracChangeset
for help on using the changeset viewer.