2 * Copyright (c) 2000-2004 Apple Computer, 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@
29 * @OSF_FREE_COPYRIGHT@
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.
59 * File: ipc/ipc_right.c
63 * Functions to manipulate IPC capabilities.
66 #include <mach/boolean.h>
67 #include <mach/kern_return.h>
68 #include <mach/port.h>
69 #include <mach/message.h>
70 #include <kern/assert.h>
71 #include <kern/misc_protos.h>
73 #include <ipc/ipc_entry.h>
74 #include <ipc/ipc_space.h>
75 #include <ipc/ipc_object.h>
76 #include <ipc/ipc_hash.h>
77 #include <ipc/ipc_port.h>
78 #include <ipc/ipc_pset.h>
79 #include <ipc/ipc_right.h>
80 #include <ipc/ipc_notify.h>
81 #include <ipc/ipc_table.h>
84 * Routine: ipc_right_lookup_write
86 * Finds an entry in a space, given the name.
88 * Nothing locked. If successful, the space is write-locked.
90 * KERN_SUCCESS Found an entry.
91 * KERN_INVALID_TASK The space is dead.
92 * KERN_INVALID_NAME Name doesn't exist in space.
96 ipc_right_lookup_write(
98 mach_port_name_t name
,
103 assert(space
!= IS_NULL
);
105 is_write_lock(space
);
107 if (!space
->is_active
) {
108 is_write_unlock(space
);
109 return KERN_INVALID_TASK
;
112 if ((entry
= ipc_entry_lookup(space
, name
)) == IE_NULL
) {
113 is_write_unlock(space
);
114 return KERN_INVALID_NAME
;
122 * Routine: ipc_right_lookup_two_write
124 * Like ipc_right_lookup except that it returns two
125 * entries for two different names that were looked
126 * up under the same space lock.
128 * Nothing locked. If successful, the space is write-locked.
130 * KERN_INVALID_TASK The space is dead.
131 * KERN_INVALID_NAME Name doesn't exist in space.
135 ipc_right_lookup_two_write(
137 mach_port_name_t name1
,
138 ipc_entry_t
*entryp1
,
139 mach_port_name_t name2
,
140 ipc_entry_t
*entryp2
)
145 assert(space
!= IS_NULL
);
147 is_write_lock(space
);
149 if (!space
->is_active
) {
150 is_write_unlock(space
);
151 return KERN_INVALID_TASK
;
154 if ((entry1
= ipc_entry_lookup(space
, name1
)) == IE_NULL
) {
155 is_write_unlock(space
);
156 return KERN_INVALID_NAME
;
158 if ((entry2
= ipc_entry_lookup(space
, name2
)) == IE_NULL
) {
159 is_write_unlock(space
);
160 return KERN_INVALID_NAME
;
168 * Routine: ipc_right_reverse
170 * Translate (space, object) -> (name, entry).
171 * Only finds send/receive rights.
172 * Returns TRUE if an entry is found; if so,
173 * the object is locked and active.
175 * The space must be locked (read or write) and active.
176 * Nothing else locked.
183 mach_port_name_t
*namep
,
187 mach_port_name_t name
;
190 /* would switch on io_otype to handle multiple types of object */
192 assert(space
->is_active
);
193 assert(io_otype(object
) == IOT_PORT
);
195 port
= (ipc_port_t
) object
;
198 if (!ip_active(port
)) {
204 if (port
->ip_receiver
== space
) {
205 name
= port
->ip_receiver_name
;
206 assert(name
!= MACH_PORT_NULL
);
208 entry
= ipc_entry_lookup(space
, name
);
210 assert(entry
!= IE_NULL
);
211 assert(entry
->ie_bits
& MACH_PORT_TYPE_RECEIVE
);
212 assert(port
== (ipc_port_t
) entry
->ie_object
);
219 if (ipc_hash_lookup(space
, (ipc_object_t
) port
, namep
, entryp
)) {
220 assert((entry
= *entryp
) != IE_NULL
);
221 assert(IE_BITS_TYPE(entry
->ie_bits
) == MACH_PORT_TYPE_SEND
);
222 assert(port
== (ipc_port_t
) entry
->ie_object
);
232 * Routine: ipc_right_dnrequest
234 * Make a dead-name request, returning the previously
235 * registered send-once right. If notify is IP_NULL,
236 * just cancels the previously registered request.
238 * This interacts with the IE_BITS_COMPAT, because they
239 * both use ie_request. If this is a compat entry, then
240 * previous always gets IP_NULL. If notify is IP_NULL,
241 * then the entry remains a compat entry. Otherwise
242 * the real dead-name request is registered and the entry
243 * is no longer a compat entry.
245 * Nothing locked. May allocate memory.
246 * Only consumes/returns refs if successful.
248 * KERN_SUCCESS Made/canceled dead-name request.
249 * KERN_INVALID_TASK The space is dead.
250 * KERN_INVALID_NAME Name doesn't exist in space.
251 * KERN_INVALID_RIGHT Name doesn't denote port/dead rights.
252 * KERN_INVALID_ARGUMENT Name denotes dead name, but
253 * immediate is FALSE or notify is IP_NULL.
254 * KERN_UREFS_OVERFLOW Name denotes dead name, but
255 * generating immediate notif. would overflow urefs.
256 * KERN_RESOURCE_SHORTAGE Couldn't allocate memory.
262 mach_port_name_t name
,
265 ipc_port_t
*previousp
)
271 ipc_entry_bits_t bits
;
274 kr
= ipc_right_lookup_write(space
, name
, &entry
);
275 if (kr
!= KERN_SUCCESS
)
277 /* space is write-locked and active */
278 bits
= entry
->ie_bits
;
279 if (bits
& MACH_PORT_TYPE_PORT_RIGHTS
) {
281 ipc_port_request_index_t request
;
283 port
= (ipc_port_t
) entry
->ie_object
;
284 assert(port
!= IP_NULL
);
286 if (!ipc_right_check(space
, port
, name
, entry
)) {
287 /* port is locked and active */
289 if (notify
== IP_NULL
) {
290 previous
= ipc_right_dncancel_macro(
291 space
, port
, name
, entry
);
294 is_write_unlock(space
);
299 * If a registered soright exists,
300 * want to atomically switch with it.
301 * If ipc_port_dncancel finds us a
302 * soright, then the following
303 * ipc_port_dnrequest will reuse
304 * that slot, so we are guaranteed
305 * not to unlock and retry.
308 previous
= ipc_right_dncancel_macro(space
,
311 kr
= ipc_port_dnrequest(port
, name
, notify
,
313 if (kr
!= KERN_SUCCESS
) {
314 assert(previous
== IP_NULL
);
315 is_write_unlock(space
);
317 kr
= ipc_port_dngrow(port
,
319 /* port is unlocked */
320 if (kr
!= KERN_SUCCESS
)
326 assert(request
!= 0);
329 entry
->ie_request
= request
;
330 is_write_unlock(space
);
335 * Our capability bits were changed by ipc_right_check
336 * because it found an inactive port and removed our
337 * references to it (converting our entry into a dead
338 * one). Reload the bits (and obviously we can't use
339 * the port name anymore).
341 bits
= entry
->ie_bits
;
345 assert(bits
& MACH_PORT_TYPE_DEAD_NAME
);
348 if ((bits
& MACH_PORT_TYPE_DEAD_NAME
) &&
349 immediate
&& (notify
!= IP_NULL
)) {
350 mach_port_urefs_t urefs
= IE_BITS_UREFS(bits
);
352 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
355 if (MACH_PORT_UREFS_OVERFLOW(urefs
, 1)) {
356 is_write_unlock(space
);
357 return KERN_UREFS_OVERFLOW
;
360 (entry
->ie_bits
)++; /* increment urefs */
361 is_write_unlock(space
);
363 ipc_notify_dead_name(notify
, name
);
368 is_write_unlock(space
);
369 if (bits
& MACH_PORT_TYPE_PORT_OR_DEAD
)
370 return KERN_INVALID_ARGUMENT
;
372 return KERN_INVALID_RIGHT
;
375 *previousp
= previous
;
380 * Routine: ipc_right_dncancel
382 * Cancel a dead-name request and return the send-once right.
383 * Afterwards, entry->ie_request == 0.
385 * The space must be write-locked; the port must be locked.
386 * The port must be active; the space doesn't have to be.
391 __unused ipc_space_t space
,
393 mach_port_name_t name
,
396 ipc_port_t dnrequest
;
398 assert(ip_active(port
));
399 assert(port
== (ipc_port_t
) entry
->ie_object
);
401 dnrequest
= ipc_port_dncancel(port
, name
, entry
->ie_request
);
402 entry
->ie_request
= 0;
408 * Routine: ipc_right_inuse
410 * Check if an entry is being used.
411 * Returns TRUE if it is.
413 * The space is write-locked and active.
414 * It is unlocked if the entry is inuse.
420 __unused mach_port_name_t name
,
423 if (IE_BITS_TYPE(entry
->ie_bits
) != MACH_PORT_TYPE_NONE
) {
424 is_write_unlock(space
);
431 * Routine: ipc_right_check
433 * Check if the port has died. If it has,
434 * clean up the entry and return TRUE.
436 * The space is write-locked; the port is not locked.
437 * If returns FALSE, the port is also locked and active.
438 * Otherwise, entry is converted to a dead name, freeing
439 * a reference to port.
446 mach_port_name_t name
,
449 ipc_entry_bits_t bits
;
451 assert(space
->is_active
);
452 assert(port
== (ipc_port_t
) entry
->ie_object
);
459 /* this was either a pure send right or a send-once right */
461 bits
= entry
->ie_bits
;
462 assert((bits
& MACH_PORT_TYPE_RECEIVE
) == 0);
463 assert(IE_BITS_UREFS(bits
) > 0);
465 if (bits
& MACH_PORT_TYPE_SEND
) {
466 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND
);
468 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND_ONCE
);
469 assert(IE_BITS_UREFS(bits
) == 1);
473 ipc_port_release(port
);
475 /* convert entry to dead name */
477 if ((bits
& MACH_PORT_TYPE_SEND
) && !(bits
& MACH_PORT_TYPE_RECEIVE
))
478 ipc_hash_delete(space
, (ipc_object_t
)port
, name
, entry
);
480 bits
= (bits
&~ IE_BITS_TYPE_MASK
) | MACH_PORT_TYPE_DEAD_NAME
;
483 * If there was a notification request outstanding on this
484 * name, and since the port went dead, that notification
485 * must already be on its way up from the port layer. We
486 * don't need the index of the notification port anymore.
488 * JMM - We also add a reference to the entry since the
489 * notification only carries the name and NOT a reference
490 * (or right). This makes for pretty loose reference
491 * counting, since it is only happenstance that we
492 * detected the notification in progress like this.
493 * But most (all?) calls that try to deal with this entry
494 * will also come through here, so the reference gets added
495 * before the entry gets used eventually (I would rather it
496 * be explicit in the notification generation, though)
498 if (entry
->ie_request
!= 0) {
499 assert(IE_BITS_UREFS(bits
) < MACH_PORT_UREFS_MAX
);
500 entry
->ie_request
= 0;
503 entry
->ie_bits
= bits
;
504 entry
->ie_object
= IO_NULL
;
509 * Routine: ipc_right_clean
511 * Cleans up an entry in a dead space.
512 * The entry isn't deallocated or removed
513 * from reverse hash tables.
515 * The space is dead and unlocked.
521 mach_port_name_t name
,
524 ipc_entry_bits_t bits
;
525 mach_port_type_t type
;
527 bits
= entry
->ie_bits
;
528 type
= IE_BITS_TYPE(bits
);
530 assert(!space
->is_active
);
533 * IE_BITS_COMPAT/ipc_right_dncancel doesn't have this
534 * problem, because we check that the port is active. If
535 * we didn't cancel IE_BITS_COMPAT, ipc_port_destroy
536 * would still work, but dead space refs would accumulate
537 * in ip_dnrequests. They would use up slots in
538 * ip_dnrequests and keep the spaces from being freed.
542 case MACH_PORT_TYPE_DEAD_NAME
:
543 assert(entry
->ie_request
== 0);
544 assert(entry
->ie_object
== IO_NULL
);
547 case MACH_PORT_TYPE_PORT_SET
: {
548 ipc_pset_t pset
= (ipc_pset_t
) entry
->ie_object
;
550 assert(entry
->ie_request
== 0);
551 assert(pset
!= IPS_NULL
);
554 assert(ips_active(pset
));
556 ipc_pset_destroy(pset
); /* consumes ref, unlocks */
560 case MACH_PORT_TYPE_SEND
:
561 case MACH_PORT_TYPE_RECEIVE
:
562 case MACH_PORT_TYPE_SEND_RECEIVE
:
563 case MACH_PORT_TYPE_SEND_ONCE
: {
564 ipc_port_t port
= (ipc_port_t
) entry
->ie_object
;
565 ipc_port_t dnrequest
;
566 ipc_port_t nsrequest
= IP_NULL
;
567 mach_port_mscount_t mscount
= 0;
569 assert(port
!= IP_NULL
);
572 if (!ip_active(port
)) {
574 ip_check_unlock(port
);
578 dnrequest
= ipc_right_dncancel_macro(space
, port
,
581 if (type
& MACH_PORT_TYPE_SEND
) {
582 assert(port
->ip_srights
> 0);
583 if (--port
->ip_srights
== 0
585 nsrequest
= port
->ip_nsrequest
;
586 if (nsrequest
!= IP_NULL
) {
587 port
->ip_nsrequest
= IP_NULL
;
588 mscount
= port
->ip_mscount
;
593 if (type
& MACH_PORT_TYPE_RECEIVE
) {
594 assert(port
->ip_receiver_name
== name
);
595 assert(port
->ip_receiver
== space
);
597 ipc_port_clear_receiver(port
);
598 ipc_port_destroy(port
); /* consumes our ref, unlocks */
599 } else if (type
& MACH_PORT_TYPE_SEND_ONCE
) {
600 assert(port
->ip_sorights
> 0);
603 ipc_notify_send_once(port
); /* consumes our ref */
605 assert(port
->ip_receiver
!= space
);
608 ip_unlock(port
); /* port is active */
611 if (nsrequest
!= IP_NULL
)
612 ipc_notify_no_senders(nsrequest
, mscount
);
614 if (dnrequest
!= IP_NULL
)
615 ipc_notify_port_deleted(dnrequest
, name
);
620 panic("ipc_right_clean: strange type");
625 * Routine: ipc_right_destroy
627 * Destroys an entry in a space.
629 * The space is write-locked.
630 * The space must be active.
632 * KERN_SUCCESS The entry was destroyed.
638 mach_port_name_t name
,
641 ipc_entry_bits_t bits
;
642 mach_port_type_t type
;
644 bits
= entry
->ie_bits
;
645 entry
->ie_bits
&= ~IE_BITS_TYPE_MASK
;
646 type
= IE_BITS_TYPE(bits
);
648 assert(space
->is_active
);
651 case MACH_PORT_TYPE_DEAD_NAME
:
652 assert(entry
->ie_request
== 0);
653 assert(entry
->ie_object
== IO_NULL
);
655 ipc_entry_dealloc(space
, name
, entry
);
658 case MACH_PORT_TYPE_PORT_SET
: {
659 ipc_pset_t pset
= (ipc_pset_t
) entry
->ie_object
;
661 assert(entry
->ie_request
== 0);
662 assert(pset
!= IPS_NULL
);
664 entry
->ie_object
= IO_NULL
;
665 ipc_entry_dealloc(space
, name
, entry
);
668 assert(ips_active(pset
));
670 ipc_pset_destroy(pset
); /* consumes ref, unlocks */
674 case MACH_PORT_TYPE_SEND
:
675 case MACH_PORT_TYPE_RECEIVE
:
676 case MACH_PORT_TYPE_SEND_RECEIVE
:
677 case MACH_PORT_TYPE_SEND_ONCE
: {
678 ipc_port_t port
= (ipc_port_t
) entry
->ie_object
;
679 ipc_port_t nsrequest
= IP_NULL
;
680 mach_port_mscount_t mscount
= 0;
681 ipc_port_t dnrequest
;
683 assert(port
!= IP_NULL
);
685 if (type
== MACH_PORT_TYPE_SEND
)
686 ipc_hash_delete(space
, (ipc_object_t
) port
,
691 if (!ip_active(port
)) {
692 assert((type
& MACH_PORT_TYPE_RECEIVE
) == 0);
694 ip_check_unlock(port
);
696 entry
->ie_request
= 0;
697 entry
->ie_object
= IO_NULL
;
698 ipc_entry_dealloc(space
, name
, entry
);
703 dnrequest
= ipc_right_dncancel_macro(space
, port
, name
, entry
);
705 entry
->ie_object
= IO_NULL
;
706 ipc_entry_dealloc(space
, name
, entry
);
708 if (type
& MACH_PORT_TYPE_SEND
) {
709 assert(port
->ip_srights
> 0);
710 if (--port
->ip_srights
== 0) {
711 nsrequest
= port
->ip_nsrequest
;
712 if (nsrequest
!= IP_NULL
) {
713 port
->ip_nsrequest
= IP_NULL
;
714 mscount
= port
->ip_mscount
;
719 if (type
& MACH_PORT_TYPE_RECEIVE
) {
720 assert(ip_active(port
));
721 assert(port
->ip_receiver
== space
);
723 ipc_port_clear_receiver(port
);
724 ipc_port_destroy(port
); /* consumes our ref, unlocks */
725 } else if (type
& MACH_PORT_TYPE_SEND_ONCE
) {
726 assert(port
->ip_sorights
> 0);
729 ipc_notify_send_once(port
); /* consumes our ref */
731 assert(port
->ip_receiver
!= space
);
737 if (nsrequest
!= IP_NULL
)
738 ipc_notify_no_senders(nsrequest
, mscount
);
740 if (dnrequest
!= IP_NULL
)
741 ipc_notify_port_deleted(dnrequest
, name
);
746 panic("ipc_right_destroy: strange type");
753 * Routine: ipc_right_dealloc
755 * Releases a send/send-once/dead-name user ref.
756 * Like ipc_right_delta with a delta of -1,
757 * but looks at the entry to determine the right.
759 * The space is write-locked, and is unlocked upon return.
760 * The space must be active.
762 * KERN_SUCCESS A user ref was released.
763 * KERN_INVALID_RIGHT Entry has wrong type.
769 mach_port_name_t name
,
773 ipc_entry_bits_t bits
;
774 mach_port_type_t type
;
776 bits
= entry
->ie_bits
;
777 type
= IE_BITS_TYPE(bits
);
780 assert(space
->is_active
);
783 case MACH_PORT_TYPE_DEAD_NAME
: {
786 assert(IE_BITS_UREFS(bits
) > 0);
787 assert(entry
->ie_request
== 0);
788 assert(entry
->ie_object
== IO_NULL
);
790 if (IE_BITS_UREFS(bits
) == 1) {
791 ipc_entry_dealloc(space
, name
, entry
);
794 entry
->ie_bits
= bits
-1; /* decrement urefs */
796 is_write_unlock(space
);
800 case MACH_PORT_TYPE_SEND_ONCE
: {
801 ipc_port_t port
, dnrequest
;
803 assert(IE_BITS_UREFS(bits
) == 1);
805 port
= (ipc_port_t
) entry
->ie_object
;
806 assert(port
!= IP_NULL
);
808 if (ipc_right_check(space
, port
, name
, entry
)) {
810 bits
= entry
->ie_bits
;
811 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
814 /* port is locked and active */
816 assert(port
->ip_sorights
> 0);
818 dnrequest
= ipc_right_dncancel_macro(space
, port
, name
, entry
);
821 entry
->ie_object
= IO_NULL
;
822 ipc_entry_dealloc(space
, name
, entry
);
824 is_write_unlock(space
);
826 ipc_notify_send_once(port
);
828 if (dnrequest
!= IP_NULL
)
829 ipc_notify_port_deleted(dnrequest
, name
);
833 case MACH_PORT_TYPE_SEND
: {
835 ipc_port_t dnrequest
= IP_NULL
;
836 ipc_port_t nsrequest
= IP_NULL
;
837 mach_port_mscount_t mscount
= 0;
840 assert(IE_BITS_UREFS(bits
) > 0);
842 port
= (ipc_port_t
) entry
->ie_object
;
843 assert(port
!= IP_NULL
);
845 if (ipc_right_check(space
, port
, name
, entry
)) {
846 bits
= entry
->ie_bits
;
847 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
850 /* port is locked and active */
852 assert(port
->ip_srights
> 0);
854 if (IE_BITS_UREFS(bits
) == 1) {
855 if (--port
->ip_srights
== 0) {
856 nsrequest
= port
->ip_nsrequest
;
857 if (nsrequest
!= IP_NULL
) {
858 port
->ip_nsrequest
= IP_NULL
;
859 mscount
= port
->ip_mscount
;
863 dnrequest
= ipc_right_dncancel_macro(space
, port
,
865 ipc_hash_delete(space
, (ipc_object_t
) port
,
869 entry
->ie_object
= IO_NULL
;
870 ipc_entry_dealloc(space
, name
, entry
);
873 entry
->ie_bits
= bits
-1; /* decrement urefs */
875 /* even if dropped a ref, port is active */
877 is_write_unlock(space
);
879 if (nsrequest
!= IP_NULL
)
880 ipc_notify_no_senders(nsrequest
, mscount
);
882 if (dnrequest
!= IP_NULL
)
883 ipc_notify_port_deleted(dnrequest
, name
);
887 case MACH_PORT_TYPE_SEND_RECEIVE
: {
889 ipc_port_t nsrequest
= IP_NULL
;
890 mach_port_mscount_t mscount
= 0;
892 assert(IE_BITS_UREFS(bits
) > 0);
894 port
= (ipc_port_t
) entry
->ie_object
;
895 assert(port
!= IP_NULL
);
898 assert(ip_active(port
));
899 assert(port
->ip_receiver_name
== name
);
900 assert(port
->ip_receiver
== space
);
901 assert(port
->ip_srights
> 0);
903 if (IE_BITS_UREFS(bits
) == 1) {
904 if (--port
->ip_srights
== 0) {
905 nsrequest
= port
->ip_nsrequest
;
906 if (nsrequest
!= IP_NULL
) {
907 port
->ip_nsrequest
= IP_NULL
;
908 mscount
= port
->ip_mscount
;
912 entry
->ie_bits
= bits
&~ (IE_BITS_UREFS_MASK
|
913 MACH_PORT_TYPE_SEND
);
915 entry
->ie_bits
= bits
-1; /* decrement urefs */
918 is_write_unlock(space
);
920 if (nsrequest
!= IP_NULL
)
921 ipc_notify_no_senders(nsrequest
, mscount
);
926 is_write_unlock(space
);
927 return KERN_INVALID_RIGHT
;
934 * Routine: ipc_right_delta
936 * Modifies the user-reference count for a right.
937 * May deallocate the right, if the count goes to zero.
939 * The space is write-locked, and is unlocked upon return.
940 * The space must be active.
942 * KERN_SUCCESS Count was modified.
943 * KERN_INVALID_RIGHT Entry has wrong type.
944 * KERN_INVALID_VALUE Bad delta for the right.
945 * KERN_UREFS_OVERFLOW OK delta, except would overflow.
951 mach_port_name_t name
,
953 mach_port_right_t right
,
954 mach_port_delta_t delta
)
956 ipc_entry_bits_t bits
;
958 bits
= entry
->ie_bits
;
962 * The following is used (for case MACH_PORT_RIGHT_DEAD_NAME) in the
963 * switch below. It is used to keep track of those cases (in DIPC)
964 * where we have postponed the dropping of a port reference. Since
965 * the dropping of the reference could cause the port to disappear
966 * we postpone doing so when we are holding the space lock.
969 assert(space
->is_active
);
970 assert(right
< MACH_PORT_RIGHT_NUMBER
);
972 /* Rights-specific restrictions and operations. */
975 case MACH_PORT_RIGHT_PORT_SET
: {
978 if ((bits
& MACH_PORT_TYPE_PORT_SET
) == 0)
981 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_PORT_SET
);
982 assert(IE_BITS_UREFS(bits
) == 0);
983 assert(entry
->ie_request
== 0);
991 pset
= (ipc_pset_t
) entry
->ie_object
;
992 assert(pset
!= IPS_NULL
);
996 entry
->ie_object
= IO_NULL
;
997 ipc_entry_dealloc(space
, name
, entry
);
1001 assert(ips_active(pset
));
1002 is_write_unlock(space
);
1004 ipc_pset_destroy(pset
); /* consumes ref, unlocks */
1008 case MACH_PORT_RIGHT_RECEIVE
: {
1010 ipc_port_t dnrequest
= IP_NULL
;
1012 if ((bits
& MACH_PORT_TYPE_RECEIVE
) == 0)
1021 port
= (ipc_port_t
) entry
->ie_object
;
1022 assert(port
!= IP_NULL
);
1025 * The port lock is needed for ipc_right_dncancel;
1026 * otherwise, we wouldn't have to take the lock
1027 * until just before dropping the space lock.
1031 assert(ip_active(port
));
1032 assert(port
->ip_receiver_name
== name
);
1033 assert(port
->ip_receiver
== space
);
1035 if (bits
& MACH_PORT_TYPE_SEND
) {
1036 assert(IE_BITS_TYPE(bits
) ==
1037 MACH_PORT_TYPE_SEND_RECEIVE
);
1038 assert(IE_BITS_UREFS(bits
) > 0);
1039 assert(IE_BITS_UREFS(bits
) < MACH_PORT_UREFS_MAX
);
1040 assert(port
->ip_srights
> 0);
1043 * The remaining send right turns into a
1044 * dead name. Notice we don't decrement
1045 * ip_srights, generate a no-senders notif,
1046 * or use ipc_right_dncancel, because the
1047 * port is destroyed "first".
1049 bits
&= ~IE_BITS_TYPE_MASK
;
1050 bits
|= MACH_PORT_TYPE_DEAD_NAME
;
1051 if (entry
->ie_request
) {
1052 entry
->ie_request
= 0;
1055 entry
->ie_bits
= bits
;
1056 entry
->ie_object
= IO_NULL
;
1058 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_RECEIVE
);
1059 assert(IE_BITS_UREFS(bits
) == 0);
1061 dnrequest
= ipc_right_dncancel_macro(space
, port
,
1063 entry
->ie_object
= IO_NULL
;
1064 ipc_entry_dealloc(space
, name
, entry
);
1066 is_write_unlock(space
);
1068 ipc_port_clear_receiver(port
);
1069 ipc_port_destroy(port
); /* consumes ref, unlocks */
1071 if (dnrequest
!= IP_NULL
)
1072 ipc_notify_port_deleted(dnrequest
, name
);
1076 case MACH_PORT_RIGHT_SEND_ONCE
: {
1077 ipc_port_t port
, dnrequest
;
1079 if ((bits
& MACH_PORT_TYPE_SEND_ONCE
) == 0)
1082 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND_ONCE
);
1083 assert(IE_BITS_UREFS(bits
) == 1);
1085 port
= (ipc_port_t
) entry
->ie_object
;
1086 assert(port
!= IP_NULL
);
1088 if (ipc_right_check(space
, port
, name
, entry
)) {
1089 assert(!(entry
->ie_bits
& MACH_PORT_TYPE_SEND_ONCE
));
1092 /* port is locked and active */
1094 assert(port
->ip_sorights
> 0);
1096 if ((delta
> 0) || (delta
< -1)) {
1106 dnrequest
= ipc_right_dncancel_macro(space
, port
, name
, entry
);
1109 entry
->ie_object
= IO_NULL
;
1110 ipc_entry_dealloc(space
, name
, entry
);
1112 is_write_unlock(space
);
1114 ipc_notify_send_once(port
);
1116 if (dnrequest
!= IP_NULL
)
1117 ipc_notify_port_deleted(dnrequest
, name
);
1121 case MACH_PORT_RIGHT_DEAD_NAME
: {
1122 mach_port_urefs_t urefs
;
1124 if (bits
& MACH_PORT_TYPE_SEND_RIGHTS
) {
1127 port
= (ipc_port_t
) entry
->ie_object
;
1128 assert(port
!= IP_NULL
);
1130 if (!ipc_right_check(space
, port
, name
, entry
)) {
1131 /* port is locked and active */
1135 bits
= entry
->ie_bits
;
1136 } else if ((bits
& MACH_PORT_TYPE_DEAD_NAME
) == 0)
1139 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
1140 assert(IE_BITS_UREFS(bits
) > 0);
1141 assert(entry
->ie_object
== IO_NULL
);
1142 assert(entry
->ie_request
== 0);
1144 urefs
= IE_BITS_UREFS(bits
);
1145 if (MACH_PORT_UREFS_UNDERFLOW(urefs
, delta
))
1147 if (MACH_PORT_UREFS_OVERFLOW(urefs
, delta
))
1148 goto urefs_overflow
;
1150 if ((urefs
+ delta
) == 0) {
1151 ipc_entry_dealloc(space
, name
, entry
);
1154 entry
->ie_bits
= bits
+ delta
;
1156 is_write_unlock(space
);
1161 case MACH_PORT_RIGHT_SEND
: {
1162 mach_port_urefs_t urefs
;
1164 ipc_port_t dnrequest
= IP_NULL
;
1165 ipc_port_t nsrequest
= IP_NULL
;
1166 mach_port_mscount_t mscount
= 0;
1168 if ((bits
& MACH_PORT_TYPE_SEND
) == 0)
1171 /* maximum urefs for send is MACH_PORT_UREFS_MAX-1 */
1173 port
= (ipc_port_t
) entry
->ie_object
;
1174 assert(port
!= IP_NULL
);
1176 if (ipc_right_check(space
, port
, name
, entry
)) {
1177 assert((entry
->ie_bits
& MACH_PORT_TYPE_SEND
) == 0);
1180 /* port is locked and active */
1182 assert(port
->ip_srights
> 0);
1184 urefs
= IE_BITS_UREFS(bits
);
1185 if (MACH_PORT_UREFS_UNDERFLOW(urefs
, delta
)) {
1189 if (MACH_PORT_UREFS_OVERFLOW(urefs
+1, delta
)) {
1191 goto urefs_overflow
;
1194 if ((urefs
+ delta
) == 0) {
1195 if (--port
->ip_srights
== 0) {
1196 nsrequest
= port
->ip_nsrequest
;
1197 if (nsrequest
!= IP_NULL
) {
1198 port
->ip_nsrequest
= IP_NULL
;
1199 mscount
= port
->ip_mscount
;
1203 if (bits
& MACH_PORT_TYPE_RECEIVE
) {
1204 assert(port
->ip_receiver_name
== name
);
1205 assert(port
->ip_receiver
== space
);
1206 assert(IE_BITS_TYPE(bits
) ==
1207 MACH_PORT_TYPE_SEND_RECEIVE
);
1209 entry
->ie_bits
= bits
&~ (IE_BITS_UREFS_MASK
|
1210 MACH_PORT_TYPE_SEND
);
1212 assert(IE_BITS_TYPE(bits
) ==
1213 MACH_PORT_TYPE_SEND
);
1215 dnrequest
= ipc_right_dncancel_macro(space
, port
,
1217 ipc_hash_delete(space
, (ipc_object_t
) port
,
1222 entry
->ie_object
= IO_NULL
;
1223 ipc_entry_dealloc(space
, name
, entry
);
1226 entry
->ie_bits
= bits
+ delta
;
1228 /* even if dropped a ref, port is active */
1230 is_write_unlock(space
);
1232 if (nsrequest
!= IP_NULL
)
1233 ipc_notify_no_senders(nsrequest
, mscount
);
1235 if (dnrequest
!= IP_NULL
)
1236 ipc_notify_port_deleted(dnrequest
, name
);
1241 panic("ipc_right_delta: strange right");
1244 return KERN_SUCCESS
;
1247 is_write_unlock(space
);
1248 return KERN_SUCCESS
;
1251 is_write_unlock(space
);
1252 return KERN_INVALID_RIGHT
;
1255 is_write_unlock(space
);
1256 return KERN_INVALID_VALUE
;
1259 is_write_unlock(space
);
1260 return KERN_UREFS_OVERFLOW
;
1264 * Routine: ipc_right_info
1266 * Retrieves information about the right.
1268 * The space is write-locked, and is unlocked upon return
1269 * if the call is unsuccessful. The space must be active.
1271 * KERN_SUCCESS Retrieved info; space still locked.
1277 mach_port_name_t name
,
1279 mach_port_type_t
*typep
,
1280 mach_port_urefs_t
*urefsp
)
1282 ipc_entry_bits_t bits
;
1283 mach_port_type_t type
;
1284 ipc_port_request_index_t request
;
1286 bits
= entry
->ie_bits
;
1288 if (bits
& MACH_PORT_TYPE_SEND_RIGHTS
) {
1289 ipc_port_t port
= (ipc_port_t
) entry
->ie_object
;
1291 if (ipc_right_check(space
, port
, name
, entry
)) {
1292 bits
= entry
->ie_bits
;
1293 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
1298 type
= IE_BITS_TYPE(bits
);
1299 request
= entry
->ie_request
;
1302 type
|= MACH_PORT_TYPE_DNREQUEST
;
1305 *urefsp
= IE_BITS_UREFS(bits
);
1306 return KERN_SUCCESS
;
1310 * Routine: ipc_right_copyin_check
1312 * Check if a subsequent ipc_right_copyin would succeed.
1314 * The space is locked (read or write) and active.
1318 ipc_right_copyin_check(
1319 __assert_only ipc_space_t space
,
1320 __unused mach_port_name_t name
,
1322 mach_msg_type_name_t msgt_name
)
1324 ipc_entry_bits_t bits
;
1326 bits
= entry
->ie_bits
;
1327 assert(space
->is_active
);
1329 switch (msgt_name
) {
1330 case MACH_MSG_TYPE_MAKE_SEND
:
1331 case MACH_MSG_TYPE_MAKE_SEND_ONCE
:
1332 case MACH_MSG_TYPE_MOVE_RECEIVE
:
1333 if ((bits
& MACH_PORT_TYPE_RECEIVE
) == 0)
1338 case MACH_MSG_TYPE_COPY_SEND
:
1339 case MACH_MSG_TYPE_MOVE_SEND
:
1340 case MACH_MSG_TYPE_MOVE_SEND_ONCE
: {
1344 if (bits
& MACH_PORT_TYPE_DEAD_NAME
)
1347 if ((bits
& MACH_PORT_TYPE_SEND_RIGHTS
) == 0)
1350 port
= (ipc_port_t
) entry
->ie_object
;
1351 assert(port
!= IP_NULL
);
1354 active
= ip_active(port
);
1361 if (msgt_name
== MACH_MSG_TYPE_MOVE_SEND_ONCE
) {
1362 if ((bits
& MACH_PORT_TYPE_SEND_ONCE
) == 0)
1365 if ((bits
& MACH_PORT_TYPE_SEND
) == 0)
1373 panic("ipc_right_copyin_check: strange rights");
1380 * Routine: ipc_right_copyin
1382 * Copyin a capability from a space.
1383 * If successful, the caller gets a ref
1384 * for the resulting object, unless it is IO_DEAD,
1385 * and possibly a send-once right which should
1386 * be used in a port-deleted notification.
1388 * If deadok is not TRUE, the copyin operation
1389 * will fail instead of producing IO_DEAD.
1391 * The entry is never deallocated (except
1392 * when KERN_INVALID_NAME), so the caller
1393 * should deallocate the entry if its type
1394 * is MACH_PORT_TYPE_NONE.
1396 * The space is write-locked and active.
1398 * KERN_SUCCESS Acquired an object, possibly IO_DEAD.
1399 * KERN_INVALID_RIGHT Name doesn't denote correct right.
1405 mach_port_name_t name
,
1407 mach_msg_type_name_t msgt_name
,
1409 ipc_object_t
*objectp
,
1410 ipc_port_t
*sorightp
)
1412 ipc_entry_bits_t bits
;
1414 bits
= entry
->ie_bits
;
1416 assert(space
->is_active
);
1418 switch (msgt_name
) {
1419 case MACH_MSG_TYPE_MAKE_SEND
: {
1422 if ((bits
& MACH_PORT_TYPE_RECEIVE
) == 0)
1425 port
= (ipc_port_t
) entry
->ie_object
;
1426 assert(port
!= IP_NULL
);
1429 assert(ip_active(port
));
1430 assert(port
->ip_receiver_name
== name
);
1431 assert(port
->ip_receiver
== space
);
1438 *objectp
= (ipc_object_t
) port
;
1439 *sorightp
= IP_NULL
;
1443 case MACH_MSG_TYPE_MAKE_SEND_ONCE
: {
1446 if ((bits
& MACH_PORT_TYPE_RECEIVE
) == 0)
1449 port
= (ipc_port_t
) entry
->ie_object
;
1450 assert(port
!= IP_NULL
);
1453 assert(ip_active(port
));
1454 assert(port
->ip_receiver_name
== name
);
1455 assert(port
->ip_receiver
== space
);
1457 port
->ip_sorights
++;
1461 *objectp
= (ipc_object_t
) port
;
1462 *sorightp
= IP_NULL
;
1466 case MACH_MSG_TYPE_MOVE_RECEIVE
: {
1468 ipc_port_t dnrequest
= IP_NULL
;
1470 if ((bits
& MACH_PORT_TYPE_RECEIVE
) == 0)
1473 port
= (ipc_port_t
) entry
->ie_object
;
1474 assert(port
!= IP_NULL
);
1477 assert(ip_active(port
));
1478 assert(port
->ip_receiver_name
== name
);
1479 assert(port
->ip_receiver
== space
);
1481 if (bits
& MACH_PORT_TYPE_SEND
) {
1482 assert(IE_BITS_TYPE(bits
) ==
1483 MACH_PORT_TYPE_SEND_RECEIVE
);
1484 assert(IE_BITS_UREFS(bits
) > 0);
1485 assert(port
->ip_srights
> 0);
1487 ipc_hash_insert(space
, (ipc_object_t
) port
,
1491 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_RECEIVE
);
1492 assert(IE_BITS_UREFS(bits
) == 0);
1494 dnrequest
= ipc_right_dncancel_macro(space
, port
,
1496 entry
->ie_object
= IO_NULL
;
1498 entry
->ie_bits
= bits
&~ MACH_PORT_TYPE_RECEIVE
;
1500 ipc_port_clear_receiver(port
);
1502 port
->ip_receiver_name
= MACH_PORT_NULL
;
1503 port
->ip_destination
= IP_NULL
;
1506 *objectp
= (ipc_object_t
) port
;
1507 *sorightp
= dnrequest
;
1511 case MACH_MSG_TYPE_COPY_SEND
: {
1514 if (bits
& MACH_PORT_TYPE_DEAD_NAME
)
1517 /* allow for dead send-once rights */
1519 if ((bits
& MACH_PORT_TYPE_SEND_RIGHTS
) == 0)
1522 assert(IE_BITS_UREFS(bits
) > 0);
1524 port
= (ipc_port_t
) entry
->ie_object
;
1525 assert(port
!= IP_NULL
);
1527 if (ipc_right_check(space
, port
, name
, entry
)) {
1528 bits
= entry
->ie_bits
;
1531 /* port is locked and active */
1533 if ((bits
& MACH_PORT_TYPE_SEND
) == 0) {
1534 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND_ONCE
);
1535 assert(port
->ip_sorights
> 0);
1541 assert(port
->ip_srights
> 0);
1547 *objectp
= (ipc_object_t
) port
;
1548 *sorightp
= IP_NULL
;
1552 case MACH_MSG_TYPE_MOVE_SEND
: {
1554 ipc_port_t dnrequest
= IP_NULL
;
1556 if (bits
& MACH_PORT_TYPE_DEAD_NAME
)
1559 /* allow for dead send-once rights */
1561 if ((bits
& MACH_PORT_TYPE_SEND_RIGHTS
) == 0)
1564 assert(IE_BITS_UREFS(bits
) > 0);
1566 port
= (ipc_port_t
) entry
->ie_object
;
1567 assert(port
!= IP_NULL
);
1569 if (ipc_right_check(space
, port
, name
, entry
)) {
1570 bits
= entry
->ie_bits
;
1573 /* port is locked and active */
1575 if ((bits
& MACH_PORT_TYPE_SEND
) == 0) {
1576 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND_ONCE
);
1577 assert(port
->ip_sorights
> 0);
1583 assert(port
->ip_srights
> 0);
1585 if (IE_BITS_UREFS(bits
) == 1) {
1586 if (bits
& MACH_PORT_TYPE_RECEIVE
) {
1587 assert(port
->ip_receiver_name
== name
);
1588 assert(port
->ip_receiver
== space
);
1589 assert(IE_BITS_TYPE(bits
) ==
1590 MACH_PORT_TYPE_SEND_RECEIVE
);
1594 assert(IE_BITS_TYPE(bits
) ==
1595 MACH_PORT_TYPE_SEND
);
1597 dnrequest
= ipc_right_dncancel_macro(space
, port
,
1599 ipc_hash_delete(space
, (ipc_object_t
) port
,
1601 entry
->ie_object
= IO_NULL
;
1603 entry
->ie_bits
= bits
&~
1604 (IE_BITS_UREFS_MASK
|MACH_PORT_TYPE_SEND
);
1608 entry
->ie_bits
= bits
-1; /* decrement urefs */
1613 *objectp
= (ipc_object_t
) port
;
1614 *sorightp
= dnrequest
;
1618 case MACH_MSG_TYPE_MOVE_SEND_ONCE
: {
1620 ipc_port_t dnrequest
;
1622 if (bits
& MACH_PORT_TYPE_DEAD_NAME
)
1625 /* allow for dead send rights */
1627 if ((bits
& MACH_PORT_TYPE_SEND_RIGHTS
) == 0)
1630 assert(IE_BITS_UREFS(bits
) > 0);
1632 port
= (ipc_port_t
) entry
->ie_object
;
1633 assert(port
!= IP_NULL
);
1635 if (ipc_right_check(space
, port
, name
, entry
)) {
1636 bits
= entry
->ie_bits
;
1639 /* port is locked and active */
1641 if ((bits
& MACH_PORT_TYPE_SEND_ONCE
) == 0) {
1642 assert(bits
& MACH_PORT_TYPE_SEND
);
1643 assert(port
->ip_srights
> 0);
1649 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND_ONCE
);
1650 assert(IE_BITS_UREFS(bits
) == 1);
1651 assert(port
->ip_sorights
> 0);
1653 dnrequest
= ipc_right_dncancel_macro(space
, port
, name
, entry
);
1656 entry
->ie_object
= IO_NULL
;
1657 entry
->ie_bits
= bits
&~
1658 (IE_BITS_UREFS_MASK
| MACH_PORT_TYPE_SEND_ONCE
);
1660 *objectp
= (ipc_object_t
) port
;
1661 *sorightp
= dnrequest
;
1667 return KERN_INVALID_RIGHT
;
1670 return KERN_SUCCESS
;
1673 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
1674 assert(IE_BITS_UREFS(bits
) > 0);
1675 assert(entry
->ie_request
== 0);
1676 assert(entry
->ie_object
== 0);
1682 *sorightp
= IP_NULL
;
1683 return KERN_SUCCESS
;
1686 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
1687 assert(IE_BITS_UREFS(bits
) > 0);
1688 assert(entry
->ie_request
== 0);
1689 assert(entry
->ie_object
== 0);
1694 if (IE_BITS_UREFS(bits
) == 1) {
1695 bits
&= ~MACH_PORT_TYPE_DEAD_NAME
;
1697 entry
->ie_bits
= bits
-1; /* decrement urefs */
1700 *sorightp
= IP_NULL
;
1701 return KERN_SUCCESS
;
1706 * Routine: ipc_right_copyin_undo
1708 * Undoes the effects of an ipc_right_copyin
1709 * of a send/send-once right that is dead.
1710 * (Object is either IO_DEAD or a dead port.)
1712 * The space is write-locked and active.
1716 ipc_right_copyin_undo(
1718 mach_port_name_t name
,
1720 mach_msg_type_name_t msgt_name
,
1721 ipc_object_t object
,
1724 ipc_entry_bits_t bits
;
1726 bits
= entry
->ie_bits
;
1728 assert(space
->is_active
);
1730 assert((msgt_name
== MACH_MSG_TYPE_MOVE_SEND
) ||
1731 (msgt_name
== MACH_MSG_TYPE_COPY_SEND
) ||
1732 (msgt_name
== MACH_MSG_TYPE_MOVE_SEND_ONCE
));
1734 if (soright
!= IP_NULL
) {
1735 assert((msgt_name
== MACH_MSG_TYPE_MOVE_SEND
) ||
1736 (msgt_name
== MACH_MSG_TYPE_MOVE_SEND_ONCE
));
1737 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_NONE
);
1738 assert(object
!= IO_DEAD
);
1740 entry
->ie_bits
= ((bits
&~ IE_BITS_RIGHT_MASK
) |
1741 MACH_PORT_TYPE_DEAD_NAME
| 2);
1743 } else if (IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_NONE
) {
1744 assert((msgt_name
== MACH_MSG_TYPE_MOVE_SEND
) ||
1745 (msgt_name
== MACH_MSG_TYPE_MOVE_SEND_ONCE
));
1747 entry
->ie_bits
= ((bits
&~ IE_BITS_RIGHT_MASK
) |
1748 MACH_PORT_TYPE_DEAD_NAME
| 1);
1749 } else if (IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
) {
1750 assert(object
== IO_DEAD
);
1751 assert(IE_BITS_UREFS(bits
) > 0);
1753 if (msgt_name
!= MACH_MSG_TYPE_COPY_SEND
) {
1754 assert(IE_BITS_UREFS(bits
) < MACH_PORT_UREFS_MAX
);
1755 entry
->ie_bits
= bits
+1; /* increment urefs */
1758 assert((msgt_name
== MACH_MSG_TYPE_MOVE_SEND
) ||
1759 (msgt_name
== MACH_MSG_TYPE_COPY_SEND
));
1760 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND
);
1761 assert(object
!= IO_DEAD
);
1762 assert(entry
->ie_object
== object
);
1763 assert(IE_BITS_UREFS(bits
) > 0);
1765 if (msgt_name
!= MACH_MSG_TYPE_COPY_SEND
) {
1766 assert(IE_BITS_UREFS(bits
) < MACH_PORT_UREFS_MAX
-1);
1767 entry
->ie_bits
= bits
+1; /* increment urefs */
1771 * May as well convert the entry to a dead name.
1772 * (Or if it is a compat entry, destroy it.)
1775 (void) ipc_right_check(space
, (ipc_port_t
) object
,
1777 /* object is dead so it is not locked */
1780 /* release the reference acquired by copyin */
1782 if (object
!= IO_DEAD
)
1783 ipc_object_release(object
);
1787 * Routine: ipc_right_copyin_two
1789 * Like ipc_right_copyin with MACH_MSG_TYPE_MOVE_SEND
1790 * and deadok == FALSE, except that this moves two
1791 * send rights at once.
1793 * The space is write-locked and active.
1794 * The object is returned with two refs/send rights.
1796 * KERN_SUCCESS Acquired an object.
1797 * KERN_INVALID_RIGHT Name doesn't denote correct right.
1801 ipc_right_copyin_two(
1803 mach_port_name_t name
,
1805 ipc_object_t
*objectp
,
1806 ipc_port_t
*sorightp
)
1808 ipc_entry_bits_t bits
;
1809 mach_port_urefs_t urefs
;
1811 ipc_port_t dnrequest
= IP_NULL
;
1813 assert(space
->is_active
);
1815 bits
= entry
->ie_bits
;
1817 if ((bits
& MACH_PORT_TYPE_SEND
) == 0)
1820 urefs
= IE_BITS_UREFS(bits
);
1824 port
= (ipc_port_t
) entry
->ie_object
;
1825 assert(port
!= IP_NULL
);
1827 if (ipc_right_check(space
, port
, name
, entry
)) {
1830 /* port is locked and active */
1832 assert(port
->ip_srights
> 0);
1835 if (bits
& MACH_PORT_TYPE_RECEIVE
) {
1836 assert(port
->ip_receiver_name
== name
);
1837 assert(port
->ip_receiver
== space
);
1838 assert(IE_BITS_TYPE(bits
) ==
1839 MACH_PORT_TYPE_SEND_RECEIVE
);
1845 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND
);
1847 dnrequest
= ipc_right_dncancel_macro(space
, port
,
1852 ipc_hash_delete(space
, (ipc_object_t
) port
,
1854 entry
->ie_object
= IO_NULL
;
1856 entry
->ie_bits
= bits
&~ (IE_BITS_UREFS_MASK
|MACH_PORT_TYPE_SEND
);
1858 port
->ip_srights
+= 2;
1861 entry
->ie_bits
= bits
-2; /* decrement urefs */
1865 *objectp
= (ipc_object_t
) port
;
1866 *sorightp
= dnrequest
;
1867 return KERN_SUCCESS
;
1870 return KERN_INVALID_RIGHT
;
1874 * Routine: ipc_right_copyout
1876 * Copyout a capability to a space.
1877 * If successful, consumes a ref for the object.
1879 * Always succeeds when given a newly-allocated entry,
1880 * because user-reference overflow isn't a possibility.
1882 * If copying out the object would cause the user-reference
1883 * count in the entry to overflow, and overflow is TRUE,
1884 * then instead the user-reference count is left pegged
1885 * to its maximum value and the copyout succeeds anyway.
1887 * The space is write-locked and active.
1888 * The object is locked and active.
1889 * The object is unlocked; the space isn't.
1891 * KERN_SUCCESS Copied out capability.
1892 * KERN_UREFS_OVERFLOW User-refs would overflow;
1893 * guaranteed not to happen with a fresh entry
1894 * or if overflow=TRUE was specified.
1900 mach_port_name_t name
,
1902 mach_msg_type_name_t msgt_name
,
1904 ipc_object_t object
)
1906 ipc_entry_bits_t bits
;
1909 bits
= entry
->ie_bits
;
1911 assert(IO_VALID(object
));
1912 assert(io_otype(object
) == IOT_PORT
);
1913 assert(io_active(object
));
1914 assert(entry
->ie_object
== object
);
1916 port
= (ipc_port_t
) object
;
1918 switch (msgt_name
) {
1919 case MACH_MSG_TYPE_PORT_SEND_ONCE
:
1921 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_NONE
);
1922 assert(port
->ip_sorights
> 0);
1924 /* transfer send-once right and ref to entry */
1927 entry
->ie_bits
= bits
| (MACH_PORT_TYPE_SEND_ONCE
| 1);
1930 case MACH_MSG_TYPE_PORT_SEND
:
1931 assert(port
->ip_srights
> 0);
1933 if (bits
& MACH_PORT_TYPE_SEND
) {
1934 mach_port_urefs_t urefs
= IE_BITS_UREFS(bits
);
1936 assert(port
->ip_srights
> 1);
1938 assert(urefs
< MACH_PORT_UREFS_MAX
);
1940 if (urefs
+1 == MACH_PORT_UREFS_MAX
) {
1942 /* leave urefs pegged to maximum */
1947 return KERN_SUCCESS
;
1951 return KERN_UREFS_OVERFLOW
;
1957 } else if (bits
& MACH_PORT_TYPE_RECEIVE
) {
1958 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_RECEIVE
);
1959 assert(IE_BITS_UREFS(bits
) == 0);
1961 /* transfer send right to entry */
1965 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_NONE
);
1966 assert(IE_BITS_UREFS(bits
) == 0);
1968 /* transfer send right and ref to entry */
1971 /* entry is locked holding ref, so can use port */
1973 ipc_hash_insert(space
, (ipc_object_t
) port
,
1977 entry
->ie_bits
= (bits
| MACH_PORT_TYPE_SEND
) + 1;
1980 case MACH_MSG_TYPE_PORT_RECEIVE
: {
1983 assert(port
->ip_mscount
== 0);
1984 assert(port
->ip_receiver_name
== MACH_PORT_NULL
);
1985 dest
= port
->ip_destination
;
1987 port
->ip_receiver_name
= name
;
1988 port
->ip_receiver
= space
;
1990 assert((bits
& MACH_PORT_TYPE_RECEIVE
) == 0);
1992 if (bits
& MACH_PORT_TYPE_SEND
) {
1993 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_SEND
);
1994 assert(IE_BITS_UREFS(bits
) > 0);
1995 assert(port
->ip_srights
> 0);
2000 /* entry is locked holding ref, so can use port */
2002 ipc_hash_delete(space
, (ipc_object_t
) port
,
2005 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_NONE
);
2006 assert(IE_BITS_UREFS(bits
) == 0);
2008 /* transfer ref to entry */
2011 entry
->ie_bits
= bits
| MACH_PORT_TYPE_RECEIVE
;
2013 if (dest
!= IP_NULL
)
2014 ipc_port_release(dest
);
2019 panic("ipc_right_copyout: strange rights");
2022 return KERN_SUCCESS
;
2026 * Routine: ipc_right_rename
2028 * Transfer an entry from one name to another.
2029 * The old entry is deallocated.
2031 * The space is write-locked and active.
2032 * The new entry is unused. Upon return,
2033 * the space is unlocked.
2035 * KERN_SUCCESS Moved entry to new name.
2041 mach_port_name_t oname
,
2043 mach_port_name_t nname
,
2046 ipc_port_request_index_t request
= oentry
->ie_request
;
2047 ipc_entry_bits_t bits
= oentry
->ie_bits
;
2048 ipc_object_t object
= oentry
->ie_object
;
2050 assert(space
->is_active
);
2051 assert(oname
!= nname
);
2054 * If IE_BITS_COMPAT, we can't allow the entry to be renamed
2055 * if the port is dead. (This would foil ipc_port_destroy.)
2056 * Instead we should fail because oentry shouldn't exist.
2057 * Note IE_BITS_COMPAT implies ie_request != 0.
2063 assert(bits
& MACH_PORT_TYPE_PORT_RIGHTS
);
2064 port
= (ipc_port_t
) object
;
2065 assert(port
!= IP_NULL
);
2067 if (ipc_right_check(space
, port
, oname
, oentry
)) {
2070 bits
= oentry
->ie_bits
;
2071 assert(IE_BITS_TYPE(bits
) == MACH_PORT_TYPE_DEAD_NAME
);
2072 assert(oentry
->ie_request
== 0);
2074 /* port is locked and active */
2076 ipc_port_dnrename(port
, request
, oname
, nname
);
2078 oentry
->ie_request
= 0;
2082 /* initialize nentry before letting ipc_hash_insert see it */
2084 assert((nentry
->ie_bits
& IE_BITS_RIGHT_MASK
) == 0);
2085 nentry
->ie_bits
|= bits
& IE_BITS_RIGHT_MASK
;
2086 nentry
->ie_request
= request
;
2087 nentry
->ie_object
= object
;
2089 switch (IE_BITS_TYPE(bits
)) {
2090 case MACH_PORT_TYPE_SEND
: {
2093 port
= (ipc_port_t
) object
;
2094 assert(port
!= IP_NULL
);
2096 /* remember, there are no other share entries possible */
2097 /* or we can't do the rename. Therefore we do not need */
2098 /* to check the other subspaces */
2099 ipc_hash_delete(space
, (ipc_object_t
) port
, oname
, oentry
);
2100 ipc_hash_insert(space
, (ipc_object_t
) port
, nname
, nentry
);
2104 case MACH_PORT_TYPE_RECEIVE
:
2105 case MACH_PORT_TYPE_SEND_RECEIVE
: {
2108 port
= (ipc_port_t
) object
;
2109 assert(port
!= IP_NULL
);
2112 assert(ip_active(port
));
2113 assert(port
->ip_receiver_name
== oname
);
2114 assert(port
->ip_receiver
== space
);
2116 port
->ip_receiver_name
= nname
;
2121 case MACH_PORT_TYPE_PORT_SET
: {
2124 pset
= (ipc_pset_t
) object
;
2125 assert(pset
!= IPS_NULL
);
2128 assert(ips_active(pset
));
2129 assert(pset
->ips_local_name
== oname
);
2131 pset
->ips_local_name
= nname
;
2136 case MACH_PORT_TYPE_SEND_ONCE
:
2137 case MACH_PORT_TYPE_DEAD_NAME
:
2141 panic("ipc_right_rename: strange rights");
2144 assert(oentry
->ie_request
== 0);
2145 oentry
->ie_object
= IO_NULL
;
2146 ipc_entry_dealloc(space
, oname
, oentry
);
2147 is_write_unlock(space
);
2149 return KERN_SUCCESS
;