2 * Copyright (c) 2000-2007 Apple Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
32 * Mach Operating System
33 * Copyright (c) 1991,1990,1989 Carnegie Mellon University
34 * All Rights Reserved.
36 * Permission to use, copy, modify and distribute this software and its
37 * documentation is hereby granted, provided that both the copyright
38 * notice and this permission notice appear in all copies of the
39 * software, derivative works or modified versions, and any portions
40 * thereof, and that both notices appear in supporting documentation.
42 * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
43 * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
44 * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
46 * Carnegie Mellon requests users of this software to return to
48 * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
49 * School of Computer Science
50 * Carnegie Mellon University
51 * Pittsburgh PA 15213-3890
53 * any improvements or extensions that they make and grant Carnegie Mellon
54 * the rights to redistribute these changes.
57 * NOTICE: This file was modified by McAfee Research in 2004 to introduce
58 * support for mandatory and extensible security protections. This notice
59 * is included in support of clause 2.2 (b) of the Apple Public License,
61 * Copyright (c) 2005 SPARTA, Inc.
66 * File: ipc/mach_msg.c
70 * Exported message traps. See mach/message.h.
73 #include <mach/mach_types.h>
74 #include <mach/kern_return.h>
75 #include <mach/port.h>
76 #include <mach/message.h>
77 #include <mach/mig_errors.h>
78 #include <mach/mach_traps.h>
80 #include <kern/kern_types.h>
81 #include <kern/assert.h>
82 #include <kern/counters.h>
83 #include <kern/cpu_number.h>
84 #include <kern/ipc_kobject.h>
85 #include <kern/ipc_mig.h>
86 #include <kern/task.h>
87 #include <kern/thread.h>
88 #include <kern/sched_prim.h>
89 #include <kern/exception.h>
90 #include <kern/misc_protos.h>
91 #include <kern/kalloc.h>
92 #include <kern/processor.h>
93 #include <kern/syscall_subr.h>
94 #include <kern/policy_internal.h>
96 #include <vm/vm_map.h>
98 #include <ipc/ipc_types.h>
99 #include <ipc/ipc_kmsg.h>
100 #include <ipc/ipc_mqueue.h>
101 #include <ipc/ipc_object.h>
102 #include <ipc/ipc_notify.h>
103 #include <ipc/ipc_port.h>
104 #include <ipc/ipc_pset.h>
105 #include <ipc/ipc_space.h>
106 #include <ipc/ipc_entry.h>
107 #include <ipc/ipc_importance.h>
108 #include <ipc/ipc_voucher.h>
110 #include <machine/machine_routines.h>
111 #include <security/mac_mach_internal.h>
113 #include <sys/kdebug.h>
116 #define offsetof(type, member) ((size_t)(&((type *)0)->member))
117 #endif /* offsetof */
120 * Forward declarations - kernel internal routines
123 mach_msg_return_t
mach_msg_send(
124 mach_msg_header_t
*msg
,
125 mach_msg_option_t option
,
126 mach_msg_size_t send_size
,
127 mach_msg_timeout_t send_timeout
,
128 mach_port_name_t notify
);
130 mach_msg_return_t
mach_msg_receive(
131 mach_msg_header_t
*msg
,
132 mach_msg_option_t option
,
133 mach_msg_size_t rcv_size
,
134 mach_port_name_t rcv_name
,
135 mach_msg_timeout_t rcv_timeout
,
136 void (*continuation
)(mach_msg_return_t
),
137 mach_msg_size_t slist_size
);
140 mach_msg_return_t
msg_receive_error(
142 mach_msg_option_t option
,
143 mach_vm_address_t rcv_addr
,
144 mach_msg_size_t rcv_size
,
145 mach_port_seqno_t seqno
,
147 mach_msg_size_t
*out_size
);
149 static mach_msg_return_t
150 mach_msg_rcv_link_special_reply_port(
151 ipc_port_t special_reply_port
,
152 mach_port_name_t dest_name_port
);
155 mach_msg_rcv_unlink_special_reply_port(void);
157 security_token_t KERNEL_SECURITY_TOKEN
= KERNEL_SECURITY_TOKEN_VALUE
;
158 audit_token_t KERNEL_AUDIT_TOKEN
= KERNEL_AUDIT_TOKEN_VALUE
;
160 mach_msg_format_0_trailer_t trailer_template
= {
161 /* mach_msg_trailer_type_t */ MACH_MSG_TRAILER_FORMAT_0
,
162 /* mach_msg_trailer_size_t */ MACH_MSG_TRAILER_MINIMUM_SIZE
,
163 /* mach_port_seqno_t */ 0,
164 /* security_token_t */ KERNEL_SECURITY_TOKEN_VALUE
168 * Routine: mach_msg_send [Kernel Internal]
170 * Routine for kernel-task threads to send a message.
172 * Unlike mach_msg_send_from_kernel(), this routine
173 * looks port names up in the kernel's port namespace
174 * and copies in the kernel virtual memory (instead
175 * of taking a vm_map_copy_t pointer for OOL descriptors).
179 * MACH_MSG_SUCCESS Sent the message.
180 * MACH_SEND_MSG_TOO_SMALL Message smaller than a header.
181 * MACH_SEND_NO_BUFFER Couldn't allocate buffer.
182 * MACH_SEND_INVALID_DATA Couldn't copy message data.
183 * MACH_SEND_INVALID_HEADER
184 * Illegal value in the message header bits.
185 * MACH_SEND_INVALID_DEST The space is dead.
186 * MACH_SEND_INVALID_NOTIFY Bad notify port.
187 * MACH_SEND_INVALID_DEST Can't copyin destination port.
188 * MACH_SEND_INVALID_REPLY Can't copyin reply port.
189 * MACH_SEND_TIMED_OUT Timeout expired without delivery.
190 * MACH_SEND_INTERRUPTED Delivery interrupted.
195 mach_msg_header_t
*msg
,
196 mach_msg_option_t option
,
197 mach_msg_size_t send_size
,
198 mach_msg_timeout_t send_timeout
,
199 mach_msg_priority_t override
)
201 ipc_space_t space
= current_space();
202 vm_map_t map
= current_map();
204 mach_msg_return_t mr
;
205 mach_msg_size_t msg_and_trailer_size
;
206 mach_msg_max_trailer_t
*trailer
;
208 if ((send_size
& 3) ||
209 send_size
< sizeof(mach_msg_header_t
) ||
210 (send_size
< sizeof(mach_msg_base_t
) && (msg
->msgh_bits
& MACH_MSGH_BITS_COMPLEX
)))
211 return MACH_SEND_MSG_TOO_SMALL
;
213 if (send_size
> MACH_MSG_SIZE_MAX
- MAX_TRAILER_SIZE
)
214 return MACH_SEND_TOO_LARGE
;
216 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_START
);
218 msg_and_trailer_size
= send_size
+ MAX_TRAILER_SIZE
;
220 kmsg
= ipc_kmsg_alloc(msg_and_trailer_size
);
222 if (kmsg
== IKM_NULL
) {
223 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_END
, MACH_SEND_NO_BUFFER
);
224 return MACH_SEND_NO_BUFFER
;
227 KERNEL_DEBUG_CONSTANT(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_LINK
) | DBG_FUNC_NONE
,
228 (uintptr_t)0, /* this should only be called from the kernel! */
229 VM_KERNEL_ADDRPERM((uintptr_t)kmsg
),
232 (void) memcpy((void *) kmsg
->ikm_header
, (const void *) msg
, send_size
);
234 kmsg
->ikm_header
->msgh_size
= send_size
;
237 * reserve for the trailer the largest space (MAX_TRAILER_SIZE)
238 * However, the internal size field of the trailer (msgh_trailer_size)
239 * is initialized to the minimum (sizeof(mach_msg_trailer_t)), to optimize
240 * the cases where no implicit data is requested.
242 trailer
= (mach_msg_max_trailer_t
*) ((vm_offset_t
)kmsg
->ikm_header
+ send_size
);
243 trailer
->msgh_sender
= current_thread()->task
->sec_token
;
244 trailer
->msgh_audit
= current_thread()->task
->audit_token
;
245 trailer
->msgh_trailer_type
= MACH_MSG_TRAILER_FORMAT_0
;
246 trailer
->msgh_trailer_size
= MACH_MSG_TRAILER_MINIMUM_SIZE
;
248 mr
= ipc_kmsg_copyin(kmsg
, space
, map
, override
, &option
);
250 if (mr
!= MACH_MSG_SUCCESS
) {
252 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_END
, mr
);
256 mr
= ipc_kmsg_send(kmsg
, option
, send_timeout
);
258 if (mr
!= MACH_MSG_SUCCESS
) {
259 mr
|= ipc_kmsg_copyout_pseudo(kmsg
, space
, map
, MACH_MSG_BODY_NULL
);
260 (void) memcpy((void *) msg
, (const void *) kmsg
->ikm_header
,
261 kmsg
->ikm_header
->msgh_size
);
263 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_END
, mr
);
270 * message header as seen at user-space
271 * (for MACH_RCV_LARGE/IDENTITY updating)
275 mach_msg_bits_t msgh_bits
;
276 mach_msg_size_t msgh_size
;
277 mach_port_name_t msgh_remote_port
;
278 mach_port_name_t msgh_local_port
;
279 mach_msg_size_t msgh_reserved
;
280 mach_msg_id_t msgh_id
;
281 } mach_msg_user_header_t
;
284 * Routine: mach_msg_receive_results
290 * MACH_MSG_SUCCESS Received a message.
291 * MACH_RCV_INVALID_NAME The name doesn't denote a right,
292 * or the denoted right is not receive or port set.
293 * MACH_RCV_IN_SET Receive right is a member of a set.
294 * MACH_RCV_TOO_LARGE Message wouldn't fit into buffer.
295 * MACH_RCV_TIMED_OUT Timeout expired without a message.
296 * MACH_RCV_INTERRUPTED Reception interrupted.
297 * MACH_RCV_PORT_DIED Port/set died while receiving.
298 * MACH_RCV_PORT_CHANGED Port moved into set while receiving.
299 * MACH_RCV_INVALID_DATA Couldn't copy to user buffer.
300 * MACH_RCV_INVALID_NOTIFY Bad notify port.
301 * MACH_RCV_HEADER_ERROR
305 mach_msg_receive_results(
306 mach_msg_size_t
*sizep
)
308 thread_t self
= current_thread();
309 ipc_space_t space
= current_space();
310 vm_map_t map
= current_map();
312 ipc_object_t object
= self
->ith_object
;
313 mach_msg_return_t mr
= self
->ith_state
;
314 mach_vm_address_t rcv_addr
= self
->ith_msg_addr
;
315 mach_msg_size_t rcv_size
= self
->ith_rsize
;
316 mach_msg_option_t option
= self
->ith_option
;
317 ipc_kmsg_t kmsg
= self
->ith_kmsg
;
318 mach_port_seqno_t seqno
= self
->ith_seqno
;
320 mach_msg_trailer_size_t trailer_size
;
321 mach_msg_size_t size
= 0;
323 /* unlink the special_reply_port before releasing reference to object */
324 mach_msg_rcv_unlink_special_reply_port();
327 if (mr
!= MACH_MSG_SUCCESS
) {
329 if (mr
== MACH_RCV_TOO_LARGE
) {
332 * If the receive operation occurs with MACH_RCV_LARGE set
333 * then no message was extracted from the queue, and the size
334 * and (optionally) receiver names were the only thing captured.
335 * Just copyout the size (and optional port name) in a fake
338 if (option
& MACH_RCV_LARGE
) {
340 if ((option
& MACH_RCV_STACK
) == 0 &&
341 rcv_size
>= offsetof(mach_msg_user_header_t
, msgh_reserved
)) {
344 * We need to inform the user-level code that it needs more
345 * space. The value for how much space was returned in the
346 * msize save area instead of the message (which was left on
349 if (option
& MACH_RCV_LARGE_IDENTITY
) {
350 if (copyout((char *) &self
->ith_receiver_name
,
351 rcv_addr
+ offsetof(mach_msg_user_header_t
, msgh_local_port
),
352 sizeof(mach_port_name_t
)))
353 mr
= MACH_RCV_INVALID_DATA
;
355 if (copyout((char *) &self
->ith_msize
,
356 rcv_addr
+ offsetof(mach_msg_user_header_t
, msgh_size
),
357 sizeof(mach_msg_size_t
)))
358 mr
= MACH_RCV_INVALID_DATA
;
362 /* discard importance in message */
363 ipc_importance_clean(kmsg
);
365 if (msg_receive_error(kmsg
, option
, rcv_addr
, rcv_size
, seqno
, space
, &size
)
366 == MACH_RCV_INVALID_DATA
)
367 mr
= MACH_RCV_INVALID_DATA
;
376 /* MACH_MSG_SUCCESS */
378 #if IMPORTANCE_INHERITANCE
380 /* adopt/transform any importance attributes carried in the message */
381 ipc_importance_receive(kmsg
, option
);
383 #endif /* IMPORTANCE_INHERITANCE */
385 /* auto redeem the voucher in the message */
386 ipc_voucher_receive_postprocessing(kmsg
, option
);
388 trailer_size
= ipc_kmsg_add_trailer(kmsg
, space
, option
, self
, seqno
, FALSE
,
389 kmsg
->ikm_header
->msgh_remote_port
->ip_context
);
391 mr
= ipc_kmsg_copyout(kmsg
, space
, map
, MACH_MSG_BODY_NULL
, option
);
393 if (mr
!= MACH_MSG_SUCCESS
) {
395 /* already received importance, so have to undo that here */
396 ipc_importance_unreceive(kmsg
, option
);
398 /* if we had a body error copyout what we have, otherwise a simple header/trailer */
399 if ((mr
&~ MACH_MSG_MASK
) == MACH_RCV_BODY_ERROR
) {
400 if (ipc_kmsg_put(kmsg
, option
, rcv_addr
, rcv_size
, trailer_size
, &size
) == MACH_RCV_INVALID_DATA
)
401 mr
= MACH_RCV_INVALID_DATA
;
403 if (msg_receive_error(kmsg
, option
, rcv_addr
, rcv_size
, seqno
, space
, &size
)
404 == MACH_RCV_INVALID_DATA
)
405 mr
= MACH_RCV_INVALID_DATA
;
408 /* capture ksmg QoS values to the thread continuation state */
409 self
->ith_qos
= kmsg
->ikm_qos
;
410 self
->ith_qos_override
= kmsg
->ikm_qos_override
;
411 mr
= ipc_kmsg_put(kmsg
, option
, rcv_addr
, rcv_size
, trailer_size
, &size
);
418 #ifndef _PTHREAD_PRIORITY_EVENT_MANAGER_FLAG
419 #define _PTHREAD_PRIORITY_EVENT_MANAGER_FLAG 0x02000000 /* pthread event manager bit */
421 #ifndef _PTHREAD_PRIORITY_OVERCOMMIT_FLAG
422 #define _PTHREAD_PRIORITY_OVERCOMMIT_FLAG 0x80000000 /* request overcommit threads */
424 #ifndef _PTHREAD_PRIORITY_QOS_CLASS_MASK
425 #define _PTHREAD_PRIORITY_QOS_CLASS_MASK 0x003fff00 /* QoS class mask */
428 /* JMM - this needs to invoke a pthread function to compute this */
430 mach_msg_priority_combine(mach_msg_priority_t msg_qos
,
431 mach_msg_priority_t recv_qos
)
433 mach_msg_priority_t overcommit
;
434 mach_msg_priority_t no_oc_qos
;
435 mach_msg_priority_t res
;
437 assert(msg_qos
< _PTHREAD_PRIORITY_EVENT_MANAGER_FLAG
);
438 overcommit
= recv_qos
& _PTHREAD_PRIORITY_OVERCOMMIT_FLAG
;
439 no_oc_qos
= recv_qos
& ~overcommit
;
440 res
= (no_oc_qos
> msg_qos
) ? no_oc_qos
: msg_qos
;
447 * Routine: mach_msg_receive [Kernel Internal]
449 * Routine for kernel-task threads to actively receive a message.
451 * Unlike being dispatched to by ipc_kobject_server() or the
452 * reply part of mach_msg_rpc_from_kernel(), this routine
453 * looks up the receive port name in the kernel's port
454 * namespace and copies out received port rights to that namespace
455 * as well. Out-of-line memory is copied out the kernel's
456 * address space (rather than just providing the vm_map_copy_t).
460 * MACH_MSG_SUCCESS Received a message.
461 * See <mach/message.h> for list of MACH_RCV_XXX errors.
465 mach_msg_header_t
*msg
,
466 mach_msg_option_t option
,
467 mach_msg_size_t rcv_size
,
468 mach_port_name_t rcv_name
,
469 mach_msg_timeout_t rcv_timeout
,
470 void (*continuation
)(mach_msg_return_t
),
471 __unused mach_msg_size_t slist_size
)
473 thread_t self
= current_thread();
474 ipc_space_t space
= current_space();
477 mach_msg_return_t mr
;
479 mr
= ipc_mqueue_copyin(space
, rcv_name
, &mqueue
, &object
);
480 if (mr
!= MACH_MSG_SUCCESS
) {
483 /* hold ref for object */
485 self
->ith_msg_addr
= CAST_DOWN(mach_vm_address_t
, msg
);
486 self
->ith_object
= object
;
487 self
->ith_rsize
= rcv_size
;
489 self
->ith_option
= option
;
490 self
->ith_continuation
= continuation
;
491 self
->ith_knote
= ITH_KNOTE_NULL
;
493 ipc_mqueue_receive(mqueue
, option
, rcv_size
, rcv_timeout
, THREAD_ABORTSAFE
);
494 if ((option
& MACH_RCV_TIMEOUT
) && rcv_timeout
== 0)
495 thread_poll_yield(self
);
496 return mach_msg_receive_results(NULL
);
500 mach_msg_receive_continue(void)
502 mach_msg_return_t mr
;
503 thread_t self
= current_thread();
505 if (self
->ith_state
== MACH_PEEK_READY
)
506 mr
= MACH_PEEK_READY
;
508 mr
= mach_msg_receive_results(NULL
);
509 (*self
->ith_continuation
)(mr
);
514 * Routine: mach_msg_overwrite_trap [mach trap]
516 * Possibly send a message; possibly receive a message.
520 * All of mach_msg_send and mach_msg_receive error codes.
524 mach_msg_overwrite_trap(
525 struct mach_msg_overwrite_trap_args
*args
)
527 mach_vm_address_t msg_addr
= args
->msg
;
528 mach_msg_option_t option
= args
->option
;
529 mach_msg_size_t send_size
= args
->send_size
;
530 mach_msg_size_t rcv_size
= args
->rcv_size
;
531 mach_port_name_t rcv_name
= args
->rcv_name
;
532 mach_msg_timeout_t msg_timeout
= args
->timeout
;
533 mach_msg_priority_t override
= args
->override
;
534 mach_vm_address_t rcv_msg_addr
= args
->rcv_msg
;
535 __unused mach_port_seqno_t temp_seqno
= 0;
537 mach_msg_return_t mr
= MACH_MSG_SUCCESS
;
538 vm_map_t map
= current_map();
540 /* Only accept options allowed by the user */
541 option
&= MACH_MSG_OPTION_USER
;
543 if (option
& MACH_SEND_MSG
) {
544 ipc_space_t space
= current_space();
547 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_START
);
549 mr
= ipc_kmsg_get(msg_addr
, send_size
, &kmsg
);
551 if (mr
!= MACH_MSG_SUCCESS
) {
552 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_END
, mr
);
556 KERNEL_DEBUG_CONSTANT(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_LINK
) | DBG_FUNC_NONE
,
558 VM_KERNEL_ADDRPERM((uintptr_t)kmsg
),
562 mr
= ipc_kmsg_copyin(kmsg
, space
, map
, override
, &option
);
564 if (mr
!= MACH_MSG_SUCCESS
) {
566 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_END
, mr
);
570 mr
= ipc_kmsg_send(kmsg
, option
, msg_timeout
);
572 if (mr
!= MACH_MSG_SUCCESS
) {
573 mr
|= ipc_kmsg_copyout_pseudo(kmsg
, space
, map
, MACH_MSG_BODY_NULL
);
574 (void) ipc_kmsg_put(kmsg
, option
, msg_addr
, send_size
, 0, NULL
);
575 KDBG(MACHDBG_CODE(DBG_MACH_IPC
,MACH_IPC_KMSG_INFO
) | DBG_FUNC_END
, mr
);
581 if (option
& MACH_RCV_MSG
) {
582 thread_t self
= current_thread();
583 ipc_space_t space
= current_space();
587 mr
= ipc_mqueue_copyin(space
, rcv_name
, &mqueue
, &object
);
588 if (mr
!= MACH_MSG_SUCCESS
) {
591 /* hold ref for object */
593 if ((option
& MACH_RCV_SYNC_WAIT
) && !(option
& MACH_SEND_SYNC_OVERRIDE
)) {
594 ipc_port_t special_reply_port
;
595 __IGNORE_WCASTALIGN(special_reply_port
= (ipc_port_t
) object
);
596 /* link the special reply port to the destination */
597 mr
= mach_msg_rcv_link_special_reply_port(special_reply_port
,
598 (mach_port_name_t
)override
);
599 if (mr
!= MACH_MSG_SUCCESS
) {
605 if (rcv_msg_addr
!= (mach_vm_address_t
)0)
606 self
->ith_msg_addr
= rcv_msg_addr
;
608 self
->ith_msg_addr
= msg_addr
;
609 self
->ith_object
= object
;
610 self
->ith_rsize
= rcv_size
;
612 self
->ith_option
= option
;
613 self
->ith_receiver_name
= MACH_PORT_NULL
;
614 self
->ith_continuation
= thread_syscall_return
;
615 self
->ith_knote
= ITH_KNOTE_NULL
;
617 ipc_mqueue_receive(mqueue
, option
, rcv_size
, msg_timeout
, THREAD_ABORTSAFE
);
618 if ((option
& MACH_RCV_TIMEOUT
) && msg_timeout
== 0)
619 thread_poll_yield(self
);
620 return mach_msg_receive_results(NULL
);
623 return MACH_MSG_SUCCESS
;
627 * Routine: mach_msg_rcv_link_special_reply_port
629 * Link the special reply port(rcv right) to the
630 * other end of the sync ipc channel.
636 static mach_msg_return_t
637 mach_msg_rcv_link_special_reply_port(
638 ipc_port_t special_reply_port
,
639 mach_port_name_t dest_name_port
)
641 ipc_port_t dest_port
= IP_NULL
;
645 if (current_thread()->ith_special_reply_port
!= special_reply_port
) {
646 return MACH_RCV_INVALID_NOTIFY
;
649 /* Copyin the destination port */
650 if (!MACH_PORT_VALID(dest_name_port
)) {
651 return MACH_RCV_INVALID_NOTIFY
;
654 kr
= ipc_object_copyin(current_space(),
655 dest_name_port
, MACH_MSG_TYPE_COPY_SEND
,
656 (ipc_object_t
*) &dest_port
);
659 * The receive right of dest port might have gone away,
660 * do not fail the receive in that case.
662 if (kr
== KERN_SUCCESS
&& IP_VALID(dest_port
)) {
664 /* Get the effective qos of the thread */
665 qos
= proc_get_effective_thread_policy(current_thread(), TASK_POLICY_QOS
);
667 ipc_port_link_special_reply_port_with_qos(special_reply_port
,
670 /* release the send right */
671 ipc_port_release_send(dest_port
);
673 return MACH_MSG_SUCCESS
;
677 * Routine: mach_msg_rcv_unlink_special_reply_port
679 * Unlink the special reply port to the other end
680 * of the sync ipc channel.
687 mach_msg_rcv_unlink_special_reply_port(void)
689 thread_t self
= current_thread();
690 ipc_port_t special_reply_port
= self
->ith_special_reply_port
;
691 mach_msg_option_t option
= self
->ith_option
;
693 if ((special_reply_port
== IP_NULL
) ||
694 !(option
& MACH_RCV_SYNC_WAIT
)) {
698 ipc_port_unlink_special_reply_port(special_reply_port
,
699 IPC_PORT_UNLINK_SR_ALLOW_SYNC_QOS_LINKAGE
);
703 * Routine: mach_msg_trap [mach trap]
705 * Possibly send a message; possibly receive a message.
709 * All of mach_msg_send and mach_msg_receive error codes.
714 struct mach_msg_overwrite_trap_args
*args
)
717 args
->rcv_msg
= (mach_vm_address_t
)0;
719 kr
= mach_msg_overwrite_trap(args
);
725 * Routine: msg_receive_error [internal]
727 * Builds a minimal header/trailer and copies it to
728 * the user message buffer. Invoked when in the case of a
729 * MACH_RCV_TOO_LARGE or MACH_RCV_BODY_ERROR error.
732 * size - maximum buffer size on input,
733 * actual copied-out size on output
735 * MACH_MSG_SUCCESS minimal header/trailer copied
736 * MACH_RCV_INVALID_DATA copyout to user buffer failed
742 mach_msg_option_t option
,
743 mach_vm_address_t rcv_addr
,
744 mach_msg_size_t rcv_size
,
745 mach_port_seqno_t seqno
,
747 mach_msg_size_t
*sizep
)
749 mach_vm_address_t context
;
750 mach_msg_trailer_size_t trailer_size
;
751 mach_msg_max_trailer_t
*trailer
;
753 context
= kmsg
->ikm_header
->msgh_remote_port
->ip_context
;
756 * Copy out the destination port in the message.
757 * Destroy all other rights and memory in the message.
759 ipc_kmsg_copyout_dest(kmsg
, space
);
762 * Build a minimal message with the requested trailer.
764 trailer
= (mach_msg_max_trailer_t
*)
765 ((vm_offset_t
)kmsg
->ikm_header
+
766 round_msg(sizeof(mach_msg_header_t
)));
767 kmsg
->ikm_header
->msgh_size
= sizeof(mach_msg_header_t
);
768 bcopy( (char *)&trailer_template
,
770 sizeof(trailer_template
));
772 trailer_size
= ipc_kmsg_add_trailer(kmsg
, space
,
773 option
, current_thread(), seqno
,
777 * Copy the message to user space and return the size
778 * (note that ipc_kmsg_put may also adjust the actual
779 * size copied out to user-space).
781 if (ipc_kmsg_put(kmsg
, option
, rcv_addr
, rcv_size
, trailer_size
, sizep
) == MACH_RCV_INVALID_DATA
)
782 return(MACH_RCV_INVALID_DATA
);
784 return(MACH_MSG_SUCCESS
);