diff options
author | Thomas Bushnell <thomas@gnu.org> | 1997-02-25 21:28:37 +0000 |
---|---|---|
committer | Thomas Bushnell <thomas@gnu.org> | 1997-02-25 21:28:37 +0000 |
commit | f07a4c844da9f0ecae5bbee1ab94be56505f26f7 (patch) | |
tree | 12b07c7e578fc1a5f53dbfde2632408491ff2a70 /ipc/mach_port.c | |
download | gnumach-f07a4c844da9f0ecae5bbee1ab94be56505f26f7.tar.gz gnumach-f07a4c844da9f0ecae5bbee1ab94be56505f26f7.tar.bz2 gnumach-f07a4c844da9f0ecae5bbee1ab94be56505f26f7.zip |
Initial source
Diffstat (limited to 'ipc/mach_port.c')
-rw-r--r-- | ipc/mach_port.c | 2505 |
1 files changed, 2505 insertions, 0 deletions
diff --git a/ipc/mach_port.c b/ipc/mach_port.c new file mode 100644 index 00000000..b26c96be --- /dev/null +++ b/ipc/mach_port.c @@ -0,0 +1,2505 @@ +/* + * Mach Operating System + * Copyright (c) 1991,1990,1989 Carnegie Mellon University. + * Copyright (c) 1993,1994 The University of Utah and + * the Computer Systems Laboratory (CSL). + * All rights reserved. + * + * Permission to use, copy, modify and distribute this software and its + * documentation is hereby granted, provided that both the copyright + * notice and this permission notice appear in all copies of the + * software, derivative works or modified versions, and any portions + * thereof, and that both notices appear in supporting documentation. + * + * CARNEGIE MELLON, THE UNIVERSITY OF UTAH AND CSL ALLOW FREE USE OF + * THIS SOFTWARE IN ITS "AS IS" CONDITION, AND DISCLAIM ANY LIABILITY + * OF ANY KIND FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF + * THIS SOFTWARE. + * + * Carnegie Mellon requests users of this software to return to + * + * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU + * School of Computer Science + * Carnegie Mellon University + * Pittsburgh PA 15213-3890 + * + * any improvements or extensions that they make and grant Carnegie Mellon + * the rights to redistribute these changes. + */ +/* + */ +/* + * File: ipc/mach_port.c + * Author: Rich Draves + * Date: 1989 + * + * Exported kernel calls. See mach/mach_port.defs. + */ + +#include <mach_ipc_compat.h> + +#include <mach/port.h> +#include <mach/kern_return.h> +#include <mach/notify.h> +#include <mach/mach_param.h> +#include <mach/vm_param.h> +#include <mach/vm_prot.h> +#ifdef MIGRATING_THREADS +#include <mach/rpc.h> +#include <kern/task.h> +#include <kern/act.h> +#endif /* MIGRATING_THREADS */ +#include <vm/vm_map.h> +#include <vm/vm_kern.h> +#include <vm/vm_user.h> +#include <ipc/ipc_entry.h> +#include <ipc/ipc_space.h> +#include <ipc/ipc_object.h> +#include <ipc/ipc_notify.h> +#include <ipc/ipc_port.h> +#include <ipc/ipc_pset.h> +#include <ipc/ipc_right.h> + + + +/* + * Routine: mach_port_names_helper + * Purpose: + * A helper function for mach_port_names. + */ + +void +mach_port_names_helper( + ipc_port_timestamp_t timestamp, + ipc_entry_t entry, + mach_port_t name, + mach_port_t *names, + mach_port_type_t *types, + ipc_entry_num_t *actualp) +{ + ipc_entry_bits_t bits = entry->ie_bits; + ipc_port_request_index_t request = entry->ie_request; + mach_port_type_t type; + ipc_entry_num_t actual; + + if (bits & MACH_PORT_TYPE_SEND_RIGHTS) { + ipc_port_t port; + boolean_t died; + + port = (ipc_port_t) entry->ie_object; + assert(port != IP_NULL); + + /* + * The timestamp serializes mach_port_names + * with ipc_port_destroy. If the port died, + * but after mach_port_names started, pretend + * that it isn't dead. + */ + + ip_lock(port); + died = (!ip_active(port) && + IP_TIMESTAMP_ORDER(port->ip_timestamp, timestamp)); + ip_unlock(port); + + if (died) { +#if MACH_IPC_COMPAT + if (bits & IE_BITS_COMPAT) + return; +#endif MACH_IPC_COMPAT + + /* pretend this is a dead-name entry */ + + bits &= ~(IE_BITS_TYPE_MASK|IE_BITS_MAREQUEST); + bits |= MACH_PORT_TYPE_DEAD_NAME; + if (request != 0) + bits++; + request = 0; + } + } + + type = IE_BITS_TYPE(bits); +#if MACH_IPC_COMPAT + if (bits & IE_BITS_COMPAT) + type |= MACH_PORT_TYPE_COMPAT; + else +#endif MACH_IPC_COMPAT + if (request != 0) + type |= MACH_PORT_TYPE_DNREQUEST; + if (bits & IE_BITS_MAREQUEST) + type |= MACH_PORT_TYPE_MAREQUEST; + + actual = *actualp; + names[actual] = name; + types[actual] = type; + *actualp = actual+1; +} + +/* + * Routine: mach_port_names [kernel call] + * Purpose: + * Retrieves a list of the rights present in the space, + * along with type information. (Same as returned + * by mach_port_type.) The names are returned in + * no particular order, but they (and the type info) + * are an accurate snapshot of the space. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Arrays of names and types returned. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +mach_port_names( + ipc_space_t space, + mach_port_t **namesp, + mach_msg_type_number_t *namesCnt, + mach_port_type_t **typesp, + mach_msg_type_number_t *typesCnt) +{ + ipc_tree_entry_t tentry; + ipc_entry_t table; + ipc_entry_num_t tsize; + mach_port_index_t index; + ipc_entry_num_t actual; /* this many names */ + ipc_port_timestamp_t timestamp; /* logical time of this operation */ + mach_port_t *names; + mach_port_type_t *types; + kern_return_t kr; + + vm_size_t size; /* size of allocated memory */ + vm_offset_t addr1; /* allocated memory, for names */ + vm_offset_t addr2; /* allocated memory, for types */ + vm_map_copy_t memory1; /* copied-in memory, for names */ + vm_map_copy_t memory2; /* copied-in memory, for types */ + + /* safe simplifying assumption */ + assert_static(sizeof(mach_port_t) == sizeof(mach_port_type_t)); + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + size = 0; + + for (;;) { + ipc_entry_num_t bound; + vm_size_t size_needed; + + is_read_lock(space); + if (!space->is_active) { + is_read_unlock(space); + if (size != 0) { + kmem_free(ipc_kernel_map, addr1, size); + kmem_free(ipc_kernel_map, addr2, size); + } + return KERN_INVALID_TASK; + } + + /* upper bound on number of names in the space */ + + bound = space->is_table_size + space->is_tree_total; + size_needed = round_page(bound * sizeof(mach_port_t)); + + if (size_needed <= size) + break; + + is_read_unlock(space); + + if (size != 0) { + kmem_free(ipc_kernel_map, addr1, size); + kmem_free(ipc_kernel_map, addr2, size); + } + size = size_needed; + + kr = vm_allocate(ipc_kernel_map, &addr1, size, TRUE); + if (kr != KERN_SUCCESS) + return KERN_RESOURCE_SHORTAGE; + + kr = vm_allocate(ipc_kernel_map, &addr2, size, TRUE); + if (kr != KERN_SUCCESS) { + kmem_free(ipc_kernel_map, addr1, size); + return KERN_RESOURCE_SHORTAGE; + } + + /* can't fault while we hold locks */ + + kr = vm_map_pageable(ipc_kernel_map, addr1, addr1 + size, + VM_PROT_READ|VM_PROT_WRITE); + assert(kr == KERN_SUCCESS); + + kr = vm_map_pageable(ipc_kernel_map, addr2, addr2 + size, + VM_PROT_READ|VM_PROT_WRITE); + assert(kr == KERN_SUCCESS); + } + /* space is read-locked and active */ + + names = (mach_port_t *) addr1; + types = (mach_port_type_t *) addr2; + actual = 0; + + timestamp = ipc_port_timestamp(); + + table = space->is_table; + tsize = space->is_table_size; + + for (index = 0; index < tsize; index++) { + ipc_entry_t entry = &table[index]; + ipc_entry_bits_t bits = entry->ie_bits; + + if (IE_BITS_TYPE(bits) != MACH_PORT_TYPE_NONE) { + mach_port_t name = MACH_PORT_MAKEB(index, bits); + + mach_port_names_helper(timestamp, entry, name, + names, types, &actual); + } + } + + for (tentry = ipc_splay_traverse_start(&space->is_tree); + tentry != ITE_NULL; + tentry = ipc_splay_traverse_next(&space->is_tree, FALSE)) { + ipc_entry_t entry = &tentry->ite_entry; + mach_port_t name = tentry->ite_name; + + assert(IE_BITS_TYPE(tentry->ite_bits) != MACH_PORT_TYPE_NONE); + + mach_port_names_helper(timestamp, entry, name, + names, types, &actual); + } + ipc_splay_traverse_finish(&space->is_tree); + is_read_unlock(space); + + if (actual == 0) { + memory1 = VM_MAP_COPY_NULL; + memory2 = VM_MAP_COPY_NULL; + + if (size != 0) { + kmem_free(ipc_kernel_map, addr1, size); + kmem_free(ipc_kernel_map, addr2, size); + } + } else { + vm_size_t size_used; + + size_used = round_page(actual * sizeof(mach_port_t)); + + /* + * Make used memory pageable and get it into + * copied-in form. Free any unused memory. + */ + + kr = vm_map_pageable(ipc_kernel_map, + addr1, addr1 + size_used, + VM_PROT_NONE); + assert(kr == KERN_SUCCESS); + + kr = vm_map_pageable(ipc_kernel_map, + addr2, addr2 + size_used, + VM_PROT_NONE); + assert(kr == KERN_SUCCESS); + + kr = vm_map_copyin(ipc_kernel_map, addr1, size_used, + TRUE, &memory1); + assert(kr == KERN_SUCCESS); + + kr = vm_map_copyin(ipc_kernel_map, addr2, size_used, + TRUE, &memory2); + assert(kr == KERN_SUCCESS); + + if (size_used != size) { + kmem_free(ipc_kernel_map, + addr1 + size_used, size - size_used); + kmem_free(ipc_kernel_map, + addr2 + size_used, size - size_used); + } + } + + *namesp = (mach_port_t *) memory1; + *namesCnt = actual; + *typesp = (mach_port_type_t *) memory2; + *typesCnt = actual; + return KERN_SUCCESS; +} + +/* + * Routine: mach_port_type [kernel call] + * Purpose: + * Retrieves the type of a right in the space. + * The type is a bitwise combination of one or more + * of the following type bits: + * MACH_PORT_TYPE_SEND + * MACH_PORT_TYPE_RECEIVE + * MACH_PORT_TYPE_SEND_ONCE + * MACH_PORT_TYPE_PORT_SET + * MACH_PORT_TYPE_DEAD_NAME + * In addition, the following pseudo-type bits may be present: + * MACH_PORT_TYPE_DNREQUEST + * A dead-name notification is requested. + * MACH_PORT_TYPE_MAREQUEST + * The send/receive right is blocked; + * a msg-accepted notification is outstanding. + * MACH_PORT_TYPE_COMPAT + * This is a compatibility-mode right; + * when the port dies, it will disappear + * instead of turning into a dead-name. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Type is returned. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + */ + +kern_return_t +mach_port_type( + ipc_space_t space, + mach_port_t name, + mach_port_type_t *typep) +{ + mach_port_urefs_t urefs; + ipc_entry_t entry; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return kr; + /* space is write-locked and active */ + + kr = ipc_right_info(space, name, entry, typep, &urefs); + if (kr == KERN_SUCCESS) + is_write_unlock(space); + /* space is unlocked */ + return kr; +} + +/* + * Routine: mach_port_rename [kernel call] + * Purpose: + * Changes the name denoting a right, + * from oname to nname. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS The right is renamed. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The oname doesn't denote a right. + * KERN_INVALID_VALUE The nname isn't a legal name. + * KERN_NAME_EXISTS The nname already denotes a right. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +mach_port_rename( + ipc_space_t space, + mach_port_t oname, + mach_port_t nname) +{ + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (!MACH_PORT_VALID(nname)) + return KERN_INVALID_VALUE; + + return ipc_object_rename(space, oname, nname); +} + +/* + * Routine: mach_port_allocate_name [kernel call] + * Purpose: + * Allocates a right in a space, using a specific name + * for the new right. Possible rights: + * MACH_PORT_RIGHT_RECEIVE + * MACH_PORT_RIGHT_PORT_SET + * MACH_PORT_RIGHT_DEAD_NAME + * + * A new port (allocated with MACH_PORT_RIGHT_RECEIVE) + * has no extant send or send-once rights and no queued + * messages. Its queue limit is MACH_PORT_QLIMIT_DEFAULT + * and its make-send count is 0. It is not a member of + * a port set. It has no registered no-senders or + * port-destroyed notification requests. + * + * A new port set has no members. + * + * A new dead name has one user reference. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS The right is allocated. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_VALUE The name isn't a legal name. + * KERN_INVALID_VALUE "right" isn't a legal kind of right. + * KERN_NAME_EXISTS The name already denotes a right. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +mach_port_allocate_name(space, right, name) + ipc_space_t space; + mach_port_right_t right; + mach_port_t name; +{ + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (!MACH_PORT_VALID(name)) + return KERN_INVALID_VALUE; + + switch (right) { + case MACH_PORT_RIGHT_RECEIVE: { + ipc_port_t port; + + kr = ipc_port_alloc_name(space, name, &port); + if (kr == KERN_SUCCESS) + ip_unlock(port); + break; + } + + case MACH_PORT_RIGHT_PORT_SET: { + ipc_pset_t pset; + + kr = ipc_pset_alloc_name(space, name, &pset); + if (kr == KERN_SUCCESS) + ips_unlock(pset); + break; + } + + case MACH_PORT_RIGHT_DEAD_NAME: + kr = ipc_object_alloc_dead_name(space, name); + break; + + default: + kr = KERN_INVALID_VALUE; + break; + } + + return kr; +} + +/* + * Routine: mach_port_allocate [kernel call] + * Purpose: + * Allocates a right in a space. Like mach_port_allocate_name, + * except that the implementation picks a name for the right. + * The name may be any legal name in the space that doesn't + * currently denote a right. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS The right is allocated. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_VALUE "right" isn't a legal kind of right. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + * KERN_NO_SPACE No room in space for another right. + */ + +kern_return_t +mach_port_allocate(space, right, namep) + ipc_space_t space; + mach_port_right_t right; + mach_port_t *namep; +{ + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + switch (right) { + case MACH_PORT_RIGHT_RECEIVE: { + ipc_port_t port; + + kr = ipc_port_alloc(space, namep, &port); + if (kr == KERN_SUCCESS) + ip_unlock(port); + break; + } + + case MACH_PORT_RIGHT_PORT_SET: { + ipc_pset_t pset; + + kr = ipc_pset_alloc(space, namep, &pset); + if (kr == KERN_SUCCESS) + ips_unlock(pset); + break; + } + + case MACH_PORT_RIGHT_DEAD_NAME: + kr = ipc_object_alloc_dead(space, namep); + break; + + default: + kr = KERN_INVALID_VALUE; + break; + } + + return (kr); +} + +/* + * Routine: mach_port_destroy [kernel call] + * Purpose: + * Cleans up and destroys all rights denoted by a name + * in a space. The destruction of a receive right + * destroys the port, unless a port-destroyed request + * has been made for it; the destruction of a port-set right + * destroys the port set. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS The name is destroyed. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + */ + +kern_return_t +mach_port_destroy( + ipc_space_t space, + mach_port_t name) +{ + ipc_entry_t entry; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return kr; + /* space is write-locked and active */ + + kr = ipc_right_destroy(space, name, entry); /* unlocks space */ + return kr; +} + +/* + * Routine: mach_port_deallocate [kernel call] + * Purpose: + * Deallocates a user reference from a send right, + * send-once right, or a dead-name right. May + * deallocate the right, if this is the last uref, + * and destroy the name, if it doesn't denote + * other rights. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS The uref is deallocated. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT The right isn't correct. + */ + +kern_return_t +mach_port_deallocate( + ipc_space_t space, + mach_port_t name) +{ + ipc_entry_t entry; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return kr; + /* space is write-locked */ + + kr = ipc_right_dealloc(space, name, entry); /* unlocks space */ + return kr; +} + +/* + * Routine: mach_port_get_refs [kernel call] + * Purpose: + * Retrieves the number of user references held by a right. + * Receive rights, port-set rights, and send-once rights + * always have one user reference. Returns zero if the + * name denotes a right, but not the queried right. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Number of urefs returned. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_VALUE "right" isn't a legal value. + * KERN_INVALID_NAME The name doesn't denote a right. + */ + +kern_return_t +mach_port_get_refs( + ipc_space_t space, + mach_port_t name, + mach_port_right_t right, + mach_port_urefs_t *urefsp) +{ + mach_port_type_t type; + mach_port_urefs_t urefs; + ipc_entry_t entry; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (right >= MACH_PORT_RIGHT_NUMBER) + return KERN_INVALID_VALUE; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return kr; + /* space is write-locked and active */ + + kr = ipc_right_info(space, name, entry, &type, &urefs); /* unlocks */ + if (kr != KERN_SUCCESS) + return kr; /* space is unlocked */ + is_write_unlock(space); + + if (type & MACH_PORT_TYPE(right)) + switch (right) { + case MACH_PORT_RIGHT_SEND_ONCE: + assert(urefs == 1); + /* fall-through */ + + case MACH_PORT_RIGHT_PORT_SET: + case MACH_PORT_RIGHT_RECEIVE: + *urefsp = 1; + break; + + case MACH_PORT_RIGHT_DEAD_NAME: + case MACH_PORT_RIGHT_SEND: + assert(urefs > 0); + *urefsp = urefs; + break; + + default: + panic("mach_port_get_refs: strange rights"); + } + else + *urefsp = 0; + + return kr; +} + +/* + * Routine: mach_port_mod_refs + * Purpose: + * Modifies the number of user references held by a right. + * The resulting number of user references must be non-negative. + * If it is zero, the right is deallocated. If the name + * doesn't denote other rights, it is destroyed. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Modified number of urefs. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_VALUE "right" isn't a legal value. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote specified right. + * KERN_INVALID_VALUE Impossible modification to urefs. + * KERN_UREFS_OVERFLOW Urefs would overflow. + */ + +kern_return_t +mach_port_mod_refs( + ipc_space_t space, + mach_port_t name, + mach_port_right_t right, + mach_port_delta_t delta) +{ + ipc_entry_t entry; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (right >= MACH_PORT_RIGHT_NUMBER) + return KERN_INVALID_VALUE; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return kr; + /* space is write-locked and active */ + + kr = ipc_right_delta(space, name, entry, right, delta); /* unlocks */ + return kr; +} + +/* + * Routine: old_mach_port_get_receive_status [kernel call] + * Purpose: + * Compatibility for code written before sequence numbers. + * Retrieves mucho info about a receive right. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved status. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote receive rights. + */ + +kern_return_t +old_mach_port_get_receive_status(space, name, statusp) + ipc_space_t space; + mach_port_t name; + old_mach_port_status_t *statusp; +{ + mach_port_status_t status; + kern_return_t kr; + + kr = mach_port_get_receive_status(space, name, &status); + if (kr != KERN_SUCCESS) + return kr; + + statusp->mps_pset = status.mps_pset; + statusp->mps_mscount = status.mps_mscount; + statusp->mps_qlimit = status.mps_qlimit; + statusp->mps_msgcount = status.mps_msgcount; + statusp->mps_sorights = status.mps_sorights; + statusp->mps_srights = status.mps_srights; + statusp->mps_pdrequest = status.mps_pdrequest; + statusp->mps_nsrequest = status.mps_nsrequest; + + return KERN_SUCCESS; +} + +/* + * Routine: mach_port_set_qlimit [kernel call] + * Purpose: + * Changes a receive right's queue limit. + * The new queue limit must be between 0 and + * MACH_PORT_QLIMIT_MAX, inclusive. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Set queue limit. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote receive rights. + * KERN_INVALID_VALUE Illegal queue limit. + */ + +kern_return_t +mach_port_set_qlimit(space, name, qlimit) + ipc_space_t space; + mach_port_t name; + mach_port_msgcount_t qlimit; +{ + ipc_port_t port; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (qlimit > MACH_PORT_QLIMIT_MAX) + return KERN_INVALID_VALUE; + + kr = ipc_port_translate_receive(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + ipc_port_set_qlimit(port, qlimit); + + ip_unlock(port); + return KERN_SUCCESS; +} + +/* + * Routine: mach_port_set_mscount [kernel call] + * Purpose: + * Changes a receive right's make-send count. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Set make-send count. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote receive rights. + */ + +kern_return_t +mach_port_set_mscount( + ipc_space_t space, + mach_port_t name, + mach_port_mscount_t mscount) +{ + ipc_port_t port; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_port_translate_receive(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + ipc_port_set_mscount(port, mscount); + + ip_unlock(port); + return KERN_SUCCESS; +} + +/* + * Routine: mach_port_set_seqno [kernel call] + * Purpose: + * Changes a receive right's sequence number. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Set sequence number. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote receive rights. + */ + +kern_return_t +mach_port_set_seqno( + ipc_space_t space, + mach_port_t name, + mach_port_seqno_t seqno) +{ + ipc_port_t port; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_port_translate_receive(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + ipc_port_set_seqno(port, seqno); + + ip_unlock(port); + return KERN_SUCCESS; +} + +/* + * Routine: mach_port_gst_helper + * Purpose: + * A helper function for mach_port_get_set_status. + */ + +void +mach_port_gst_helper( + ipc_pset_t pset, + ipc_port_t port, + ipc_entry_num_t maxnames, + mach_port_t *names, + ipc_entry_num_t *actualp) +{ + ipc_pset_t ip_pset; + mach_port_t name; + + assert(port != IP_NULL); + + ip_lock(port); + assert(ip_active(port)); + + name = port->ip_receiver_name; + assert(name != MACH_PORT_NULL); + ip_pset = port->ip_pset; + + ip_unlock(port); + + if (pset == ip_pset) { + ipc_entry_num_t actual = *actualp; + + if (actual < maxnames) + names[actual] = name; + + *actualp = actual+1; + } +} + +/* + * Routine: mach_port_get_set_status [kernel call] + * Purpose: + * Retrieves a list of members in a port set. + * Returns the space's name for each receive right member. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved list of members. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote a port set. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +mach_port_get_set_status( + ipc_space_t space, + mach_port_t name, + mach_port_t **members, + mach_msg_type_number_t *membersCnt) +{ + ipc_entry_num_t actual; /* this many members */ + ipc_entry_num_t maxnames; /* space for this many members */ + kern_return_t kr; + + vm_size_t size; /* size of allocated memory */ + vm_offset_t addr; /* allocated memory */ + vm_map_copy_t memory; /* copied-in memory */ + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + size = PAGE_SIZE; /* initial guess */ + + for (;;) { + ipc_tree_entry_t tentry; + ipc_entry_t entry, table; + ipc_entry_num_t tsize; + mach_port_index_t index; + mach_port_t *names; + ipc_pset_t pset; + + kr = vm_allocate(ipc_kernel_map, &addr, size, TRUE); + if (kr != KERN_SUCCESS) + return KERN_RESOURCE_SHORTAGE; + + /* can't fault while we hold locks */ + + kr = vm_map_pageable(ipc_kernel_map, addr, addr + size, + VM_PROT_READ|VM_PROT_WRITE); + assert(kr == KERN_SUCCESS); + + kr = ipc_right_lookup_read(space, name, &entry); + if (kr != KERN_SUCCESS) { + kmem_free(ipc_kernel_map, addr, size); + return kr; + } + /* space is read-locked and active */ + + if (IE_BITS_TYPE(entry->ie_bits) != MACH_PORT_TYPE_PORT_SET) { + is_read_unlock(space); + kmem_free(ipc_kernel_map, addr, size); + return KERN_INVALID_RIGHT; + } + + pset = (ipc_pset_t) entry->ie_object; + assert(pset != IPS_NULL); + /* the port set must be active */ + + names = (mach_port_t *) addr; + maxnames = size / sizeof(mach_port_t); + actual = 0; + + table = space->is_table; + tsize = space->is_table_size; + + for (index = 0; index < tsize; index++) { + ipc_entry_t ientry = &table[index]; + ipc_entry_bits_t bits = ientry->ie_bits; + + if (bits & MACH_PORT_TYPE_RECEIVE) { + ipc_port_t port = + (ipc_port_t) ientry->ie_object; + + mach_port_gst_helper(pset, port, maxnames, + names, &actual); + } + } + + for (tentry = ipc_splay_traverse_start(&space->is_tree); + tentry != ITE_NULL; + tentry = ipc_splay_traverse_next(&space->is_tree,FALSE)) { + ipc_entry_bits_t bits = tentry->ite_bits; + + assert(IE_BITS_TYPE(bits) != MACH_PORT_TYPE_NONE); + + if (bits & MACH_PORT_TYPE_RECEIVE) { + ipc_port_t port = + (ipc_port_t) tentry->ite_object; + + mach_port_gst_helper(pset, port, maxnames, + names, &actual); + } + } + ipc_splay_traverse_finish(&space->is_tree); + is_read_unlock(space); + + if (actual <= maxnames) + break; + + /* didn't have enough memory; allocate more */ + + kmem_free(ipc_kernel_map, addr, size); + size = round_page(actual * sizeof(mach_port_t)) + PAGE_SIZE; + } + + if (actual == 0) { + memory = VM_MAP_COPY_NULL; + + kmem_free(ipc_kernel_map, addr, size); + } else { + vm_size_t size_used; + + size_used = round_page(actual * sizeof(mach_port_t)); + + /* + * Make used memory pageable and get it into + * copied-in form. Free any unused memory. + */ + + kr = vm_map_pageable(ipc_kernel_map, + addr, addr + size_used, + VM_PROT_NONE); + assert(kr == KERN_SUCCESS); + + kr = vm_map_copyin(ipc_kernel_map, addr, size_used, + TRUE, &memory); + assert(kr == KERN_SUCCESS); + + if (size_used != size) + kmem_free(ipc_kernel_map, + addr + size_used, size - size_used); + } + + *members = (mach_port_t *) memory; + *membersCnt = actual; + return KERN_SUCCESS; +} + +/* + * Routine: mach_port_move_member [kernel call] + * Purpose: + * If after is MACH_PORT_NULL, removes member + * from the port set it is in. Otherwise, adds + * member to after, removing it from any set + * it might already be in. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Moved the port. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME Member didn't denote a right. + * KERN_INVALID_RIGHT Member didn't denote a receive right. + * KERN_INVALID_NAME After didn't denote a right. + * KERN_INVALID_RIGHT After didn't denote a port set right. + * KERN_NOT_IN_SET + * After is MACH_PORT_NULL and Member isn't in a port set. + */ + +kern_return_t +mach_port_move_member( + ipc_space_t space, + mach_port_t member, + mach_port_t after) +{ + ipc_entry_t entry; + ipc_port_t port; + ipc_pset_t nset; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_right_lookup_read(space, member, &entry); + if (kr != KERN_SUCCESS) + return kr; + /* space is read-locked and active */ + + if ((entry->ie_bits & MACH_PORT_TYPE_RECEIVE) == 0) { + is_read_unlock(space); + return KERN_INVALID_RIGHT; + } + + port = (ipc_port_t) entry->ie_object; + assert(port != IP_NULL); + + if (after == MACH_PORT_NULL) + nset = IPS_NULL; + else { + entry = ipc_entry_lookup(space, after); + if (entry == IE_NULL) { + is_read_unlock(space); + return KERN_INVALID_NAME; + } + + if ((entry->ie_bits & MACH_PORT_TYPE_PORT_SET) == 0) { + is_read_unlock(space); + return KERN_INVALID_RIGHT; + } + + nset = (ipc_pset_t) entry->ie_object; + assert(nset != IPS_NULL); + } + + kr = ipc_pset_move(space, port, nset); + /* space is unlocked */ + return kr; +} + +/* + * Routine: mach_port_request_notification [kernel call] + * Purpose: + * Requests a notification. The caller supplies + * a send-once right for the notification to use, + * and the call returns the previously registered + * send-once right, if any. Possible types: + * + * MACH_NOTIFY_PORT_DESTROYED + * Requests a port-destroyed notification + * for a receive right. Sync should be zero. + * MACH_NOTIFY_NO_SENDERS + * Requests a no-senders notification for a + * receive right. If there are currently no + * senders, sync is less than or equal to the + * current make-send count, and a send-once right + * is supplied, then an immediate no-senders + * notification is generated. + * MACH_NOTIFY_DEAD_NAME + * Requests a dead-name notification for a send + * or receive right. If the name is already a + * dead name, sync is non-zero, and a send-once + * right is supplied, then an immediate dead-name + * notification is generated. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Requested a notification. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_VALUE Bad id value. + * KERN_INVALID_NAME Name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote appropriate right. + * KERN_INVALID_CAPABILITY The notify port is dead. + * MACH_NOTIFY_PORT_DESTROYED: + * KERN_INVALID_VALUE Sync isn't zero. + * MACH_NOTIFY_DEAD_NAME: + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + * KERN_INVALID_ARGUMENT Name denotes dead name, but + * sync is zero or notify is IP_NULL. + * KERN_UREFS_OVERFLOW Name denotes dead name, but + * generating immediate notif. would overflow urefs. + */ + +kern_return_t +mach_port_request_notification( + ipc_space_t space, + mach_port_t name, + mach_msg_id_t id, + mach_port_mscount_t sync, + ipc_port_t notify, + ipc_port_t *previousp) +{ + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (notify == IP_DEAD) + return KERN_INVALID_CAPABILITY; + + switch (id) { + case MACH_NOTIFY_PORT_DESTROYED: { + ipc_port_t port, previous; + + if (sync != 0) + return KERN_INVALID_VALUE; + + kr = ipc_port_translate_receive(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + ipc_port_pdrequest(port, notify, &previous); + /* port is unlocked */ + +#if MACH_IPC_COMPAT + /* + * If previous was a send right instead of a send-once + * right, we can't return it in the reply message. + * So destroy it instead. + */ + + if ((previous != IP_NULL) && ip_pdsendp(previous)) { + ipc_port_release_send(ip_pdsend(previous)); + previous = IP_NULL; + } +#endif MACH_IPC_COMPAT + + *previousp = previous; + break; + } + + case MACH_NOTIFY_NO_SENDERS: { + ipc_port_t port; + + kr = ipc_port_translate_receive(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + ipc_port_nsrequest(port, sync, notify, previousp); + /* port is unlocked */ + break; + } + + case MACH_NOTIFY_DEAD_NAME: + kr = ipc_right_dnrequest(space, name, sync != 0, + notify, previousp); + if (kr != KERN_SUCCESS) + return kr; + break; + + default: + return KERN_INVALID_VALUE; + } + + return KERN_SUCCESS; +} + +/* + * Routine: mach_port_insert_right [kernel call] + * Purpose: + * Inserts a right into a space, as if the space + * voluntarily received the right in a message, + * except that the right gets the specified name. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Inserted the right. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_VALUE The name isn't a legal name. + * KERN_NAME_EXISTS The name already denotes a right. + * KERN_INVALID_VALUE Message doesn't carry a port right. + * KERN_INVALID_CAPABILITY Port is null or dead. + * KERN_UREFS_OVERFLOW Urefs limit would be exceeded. + * KERN_RIGHT_EXISTS Space has rights under another name. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +mach_port_insert_right( + ipc_space_t space, + mach_port_t name, + ipc_port_t poly, + mach_msg_type_name_t polyPoly) +{ + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (!MACH_PORT_VALID(name) || + !MACH_MSG_TYPE_PORT_ANY_RIGHT(polyPoly)) + return KERN_INVALID_VALUE; + + if (!IO_VALID(poly)) + return KERN_INVALID_CAPABILITY; + + return ipc_object_copyout_name(space, poly, polyPoly, FALSE, name); +} + +/* + * Routine: mach_port_extract_right [kernel call] + * Purpose: + * Extracts a right from a space, as if the space + * voluntarily sent the right to the caller. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Extracted the right. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_VALUE Requested type isn't a port right. + * KERN_INVALID_NAME Name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote appropriate right. + */ + +kern_return_t +mach_port_extract_right( + ipc_space_t space, + mach_port_t name, + mach_msg_type_name_t msgt_name, + ipc_port_t *poly, + mach_msg_type_name_t *polyPoly) +{ + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + if (!MACH_MSG_TYPE_PORT_ANY(msgt_name)) + return KERN_INVALID_VALUE; + + kr = ipc_object_copyin(space, name, msgt_name, (ipc_object_t *) poly); + + if (kr == KERN_SUCCESS) + *polyPoly = ipc_object_copyin_type(msgt_name); + return kr; +} + +/* + * Routine: mach_port_get_receive_status [kernel call] + * Purpose: + * Retrieves mucho info about a receive right. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved status. + * KERN_INVALID_TASK The space is null. + * KERN_INVALID_TASK The space is dead. + * KERN_INVALID_NAME The name doesn't denote a right. + * KERN_INVALID_RIGHT Name doesn't denote receive rights. + */ + +kern_return_t +mach_port_get_receive_status(space, name, statusp) + ipc_space_t space; + mach_port_t name; + mach_port_status_t *statusp; +{ + ipc_port_t port; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_port_translate_receive(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + if (port->ip_pset != IPS_NULL) { + ipc_pset_t pset = port->ip_pset; + + ips_lock(pset); + if (!ips_active(pset)) { + ipc_pset_remove(pset, port); + ips_check_unlock(pset); + goto no_port_set; + } else { + statusp->mps_pset = pset->ips_local_name; + imq_lock(&pset->ips_messages); + statusp->mps_seqno = port->ip_seqno; + imq_unlock(&pset->ips_messages); + ips_unlock(pset); + assert(MACH_PORT_VALID(statusp->mps_pset)); + } + } else { + no_port_set: + statusp->mps_pset = MACH_PORT_NULL; + imq_lock(&port->ip_messages); + statusp->mps_seqno = port->ip_seqno; + imq_unlock(&port->ip_messages); + } + + statusp->mps_mscount = port->ip_mscount; + statusp->mps_qlimit = port->ip_qlimit; + statusp->mps_msgcount = port->ip_msgcount; + statusp->mps_sorights = port->ip_sorights; + statusp->mps_srights = port->ip_srights > 0; + statusp->mps_pdrequest = port->ip_pdrequest != IP_NULL; + statusp->mps_nsrequest = port->ip_nsrequest != IP_NULL; + ip_unlock(port); + + return KERN_SUCCESS; +} + +#ifdef MIGRATING_THREADS +kern_return_t +mach_port_set_rpcinfo(space, name, rpc_info, rpc_info_count) + ipc_space_t space; + mach_port_t name; + void *rpc_info; + unsigned int rpc_info_count; +{ + ipc_target_t target; + ipc_object_t object; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_object_translate(space, name, + MACH_PORT_RIGHT_PORT_SET, &object); + if (kr == KERN_SUCCESS) + target = &((ipc_pset_t)object)->ips_target; + else { + kr = ipc_object_translate(space, name, + MACH_PORT_RIGHT_RECEIVE, &object); + if (kr != KERN_SUCCESS) + return kr; + target = &((ipc_port_t)object)->ip_target; + } + + /* port/pset is locked and active */ + + kr = port_machine_set_rpcinfo(target, rpc_info, rpc_info_count); + + io_unlock(object); + + return kr; +} + +#if 1 +int sacts, maxsacts; +#endif + +sact_count() +{ + printf("%d server activations in use, %d max\n", sacts, maxsacts); +} + +kern_return_t +mach_port_create_act(task, name, user_stack, user_rbuf, user_rbuf_size, out_act) + task_t task; + mach_port_t name; + vm_offset_t user_stack; + vm_offset_t user_rbuf; + vm_size_t user_rbuf_size; + Act **out_act; +{ + ipc_target_t target; + ipc_space_t space; + ipc_object_t object; + kern_return_t kr; + Act *act; + + if (task == 0) + return KERN_INVALID_TASK; + + /* First create the new activation. */ + kr = act_create(task, user_stack, user_rbuf, user_rbuf_size, &act); + if (kr != KERN_SUCCESS) + return kr; + + space = task->itk_space; + + kr = ipc_object_translate(space, name, + MACH_PORT_RIGHT_PORT_SET, &object); + if (kr == KERN_SUCCESS) + target = &((ipc_pset_t)object)->ips_target; + else { + kr = ipc_object_translate(space, name, + MACH_PORT_RIGHT_RECEIVE, &object); + if (kr != KERN_SUCCESS) { + act_terminate(act); + act_deallocate(act); + return kr; + } + target = &((ipc_port_t)object)->ip_target; + } + + /* port/pset is locked and active */ +#if 0 + printf("act port/pset %08x ipc_target %08x stack %08x act %08x\n", + object, target, user_stack, act); +#endif + + /* Assign the activation to the port's actpool. */ + kr = act_set_target(act, target); + if (kr != KERN_SUCCESS) { + io_unlock(object); + act_terminate(act); + act_deallocate(act); + return kr; + } +#if 0 + printf(" actpool %08x act %08x\n", target->ip_actpool, act); +#endif + + io_unlock(object); + + /* Pass our reference to the activation back to the user. */ + *out_act = act; + +#if 1 + sacts++; + if (sacts > maxsacts) + maxsacts = sacts; + act->mact.pcb->ss.mpsfu_high = 0x69; +#endif + return KERN_SUCCESS; +} + +#ifdef RPCKERNELSIG +kern_return_t +mach_port_set_syscall_right(task, name) + task_t task; + mach_port_t name; +{ + ipc_entry_t entry; + ipc_port_t port; + kern_return_t kr; + + if (task == IS_NULL) + return KERN_INVALID_TASK; + + kr = ipc_right_lookup_write(task, name, &entry); + if (kr != KERN_SUCCESS) { + return kr; + } + + if (!(entry->ie_bits & MACH_PORT_TYPE(MACH_PORT_RIGHT_SEND))) { + is_write_unlock(space); + return KERN_INVALID_RIGHT; + } + + task->syscall_ipc_entry = *entry; + + is_write_unlock(space); + + return KERN_SUCCESS; +} +#endif +#endif /* MIGRATING_THREADS */ + +#if MACH_IPC_COMPAT + +/* + * Routine: port_translate_compat + * Purpose: + * Converts a name to a receive right. + * Conditions: + * Nothing locked. If successful, the port + * is returned locked and active. + * Returns: + * KERN_SUCCESS Port is returned. + * KERN_INVALID_ARGUMENT The space is dead. + * KERN_INVALID_ARGUMENT Name doesn't denote port rights. + * KERN_NOT_RECEIVER Name denotes send, not receive, rights. + * KERN_NOT_RECEIVER Name denotes a send-once right. + * KERN_NOT_RECEIVER Name denotes a dead name. + */ + +kern_return_t +port_translate_compat(space, name, portp) + ipc_space_t space; + mach_port_t name; + ipc_port_t *portp; +{ + ipc_entry_t entry; + mach_port_type_t type; + mach_port_urefs_t urefs; + ipc_port_t port; + kern_return_t kr; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; + /* space is write-locked and active */ + + kr = ipc_right_info(space, name, entry, &type, &urefs); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; /* space is unlocked */ + + if ((type & (MACH_PORT_TYPE_RECEIVE)) == 0) { + is_write_unlock(space); + if (type & MACH_PORT_TYPE_PORT_OR_DEAD) + return KERN_NOT_RECEIVER; + else + return KERN_INVALID_ARGUMENT; + } + + port = (ipc_port_t) entry->ie_object; + assert(port != IP_NULL); + + ip_lock(port); + is_write_unlock(space); + assert(ip_active(port)); + + *portp = port; + return KERN_SUCCESS; +} + +/* + * Routine: convert_port_type + * Purpose: + * Convert a new mach_port_type_t to an old value. + * Note send-once rights and dead names get + * represented as send rights. The extra info + * bits get dropped. + */ + +mach_port_type_t +convert_port_type(type) + mach_port_type_t type; +{ + switch (type & MACH_PORT_TYPE_ALL_RIGHTS) { + case MACH_PORT_TYPE_SEND: + case MACH_PORT_TYPE_SEND_ONCE: + case MACH_PORT_TYPE_DEAD_NAME: + return PORT_TYPE_SEND; + + case MACH_PORT_TYPE_RECEIVE: + case MACH_PORT_TYPE_SEND_RECEIVE: + return PORT_TYPE_RECEIVE_OWN; + + case MACH_PORT_TYPE_PORT_SET: + return PORT_TYPE_SET; + + default: +#if MACH_ASSERT + assert(!"convert_port_type: strange port type"); +#else + panic("convert_port_type: strange port type"); +#endif + } +} + +/* + * Routine: port_names [kernel call] + * Purpose: + * Retrieve all the names in the task's port name space. + * As a (major) convenience, return port type information. + * The port name space includes port sets. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved names. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * Additions: + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +port_names(space, namesp, namesCnt, typesp, typesCnt) + ipc_space_t space; + mach_port_t **namesp; + mach_msg_type_number_t *namesCnt; + mach_port_type_t **typesp; + mach_msg_type_number_t *typesCnt; +{ + kern_return_t kr; + + kr = mach_port_names(space, namesp, namesCnt, typesp, typesCnt); + if (kr == KERN_SUCCESS) { + ipc_entry_num_t actual = (ipc_entry_num_t) *typesCnt; + mach_port_type_t *types; + ipc_entry_num_t i; + + vm_map_copy_t copy = (vm_map_copy_t) *typesp; + vm_offset_t addr; + vm_size_t size = round_page(actual * sizeof(mach_port_type_t)); + + /* convert copy object back to something we can use */ + + kr = vm_map_copyout(ipc_kernel_map, &addr, copy); + if (kr != KERN_SUCCESS) { + vm_map_copy_discard((vm_map_copy_t) *typesp); + vm_map_copy_discard((vm_map_copy_t) *namesp); + return KERN_RESOURCE_SHORTAGE; + } + + types = (mach_port_type_t *) addr; + + for (i = 0; i < actual; i++) + types[i] = convert_port_type(types[i]); + + /* convert memory back into a copy object */ + + kr = vm_map_copyin(ipc_kernel_map, addr, size, + TRUE, ©); + assert(kr == KERN_SUCCESS); + + *typesp = (mach_port_type_t *) copy; + } else if (kr != KERN_RESOURCE_SHORTAGE) + kr = KERN_INVALID_ARGUMENT; + + return kr; +} + +/* + * Routine: port_type [kernel call] + * Purpose: + * Return type of the capability named. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved type. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT The name doesn't denote a right. + */ + +kern_return_t +port_type(space, name, typep) + ipc_space_t space; + mach_port_t name; + mach_port_type_t *typep; +{ + mach_port_type_t type; + kern_return_t kr; + + kr = mach_port_type(space, name, &type); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; + + *typep = convert_port_type(type); + return KERN_SUCCESS; +} + +/* + * Routine: port_rename [kernel call] + * Purpose: + * Change the name of a capability. + * The new name can't be in use. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved type. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT The new name is reserved. + * KERN_NAME_EXISTS The new name already denotes a right. + * KERN_INVALID_ARGUMENT The old name doesn't denote a right. + */ + +kern_return_t +port_rename(space, old_name, new_name) + ipc_space_t space; + mach_port_t old_name; + mach_port_t new_name; +{ + kern_return_t kr; + + kr = mach_port_rename(space, old_name, new_name); + if ((kr != KERN_SUCCESS) && (kr != KERN_NAME_EXISTS)) + kr = KERN_INVALID_ARGUMENT; + + return kr; +} + +/* + * Routine: port_allocate [kernel call] + * Purpose: + * Allocate a new port, giving all rights to "task". + * + * Returns in "port_name" the task's local name for the port. + * Doesn't return a reference to the port. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Allocated a port. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +port_allocate(space, namep) + ipc_space_t space; + mach_port_t *namep; +{ + ipc_port_t port; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_port_alloc_compat(space, namep, &port); + if (kr == KERN_SUCCESS) + ip_unlock(port); + else if (kr != KERN_RESOURCE_SHORTAGE) + kr = KERN_INVALID_ARGUMENT; + + return kr; +} + +/* + * Routine: port_deallocate [kernel call] + * Purpose: + * Delete port rights (send and receive) from a task. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Deallocated the port right. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote a port right. + * Additions: + * KERN_SUCCESS Deallocated a send-once right. + * KERN_SUCCESS Destroyed a dead name. + */ + +kern_return_t +port_deallocate(space, name) + ipc_space_t space; + mach_port_t name; +{ + ipc_entry_t entry; + mach_port_type_t type; + mach_port_urefs_t urefs; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; + /* space is write-locked and active */ + + /* + * We serialize with port destruction with the + * ipc_right_info call, not ipc_right_destroy. + * After ipc_right_info, we pretend that the + * port doesn't get destroyed. + */ + + kr = ipc_right_info(space, name, entry, &type, &urefs); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; /* space is unlocked */ + + if ((type & (MACH_PORT_TYPE_PORT_OR_DEAD)) == 0) { + is_write_unlock(space); + return KERN_INVALID_ARGUMENT; + } + + (void) ipc_right_destroy(space, name, entry); + /* space is unlocked */ + + return KERN_SUCCESS; +} + +/* + * Routine: port_set_backlog [kernel call] + * Purpose: + * Change the queueing backlog on "port_name" to "backlog"; + * the specified "task" must be the current receiver. + * + * Valid backlog values are 0 < backlog <= PORT_BACKLOG_MAX. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Set the backlog. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote a port right. + * KERN_NOT_RECEIVER Name denotes send rights, not receive. + * KERN_INVALID_ARGUMENT Backlog value is invalid. + * Additions: + * KERN_NOT_RECEIVER Name denotes a send-once right. + * KERN_NOT_RECEIVER Name denotes a dead name. + */ + +kern_return_t +port_set_backlog(space, name, backlog) + ipc_space_t space; + mach_port_t name; + int backlog; +{ + ipc_port_t port; + kern_return_t kr; + + if ((space == IS_NULL) || + (backlog <= 0) || + (backlog > PORT_BACKLOG_MAX)) + return KERN_INVALID_ARGUMENT; + + kr = port_translate_compat(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + ipc_port_set_qlimit(port, (mach_port_msgcount_t) backlog); + + ip_unlock(port); + return KERN_SUCCESS; +} + +/* + * Routine: port_set_backup [kernel call] + * Purpose: + * Changes the backup port for the the named port. + * The specified "task" must be the current receiver. + * Returns the old backup port, if any. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Set the backup. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote a port right. + * KERN_NOT_RECEIVER Name denotes send rights, not receive. + * Additions: + * KERN_NOT_RECEIVER Name denotes a send-once right. + * KERN_NOT_RECEIVER Name denotes a dead name. + */ + +kern_return_t +port_set_backup(space, name, backup, previousp) + ipc_space_t space; + mach_port_t name; + ipc_port_t backup; + ipc_port_t *previousp; +{ + ipc_port_t port, previous; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + if (backup == IP_DEAD) + backup = IP_NULL; + else if (backup != IP_NULL) + backup = ip_pdsendm(backup); + + kr = port_translate_compat(space, name, &port); + if (kr != KERN_SUCCESS) + return kr; + /* port is locked and active */ + + ipc_port_pdrequest(port, backup, &previous); + /* port is unlocked */ + + /* + * If previous was a send-once right instead of a send + * right, we can't return it in the reply message. + * So get rid of it in a notification instead. + */ + + if (previous != IP_NULL) { + if (ip_pdsendp(previous)) + previous = ip_pdsend(previous); + else { + ipc_notify_send_once(previous); + previous = IP_NULL; + } + } + + *previousp = previous; + return KERN_SUCCESS; +} + +/* + * Routine: port_status [kernel call] + * Purpose: + * Returns statistics related to "port_name", as seen by "task". + * Only the receiver for a given port will see true message + * counts. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved status. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote a port right. + * Additions: + * KERN_SUCCESS Send-once right. + * KERN_SUCCESS Dead name. + */ + +kern_return_t +port_status(space, name, enabledp, num_msgs, backlog, + ownership, receive_rights) + ipc_space_t space; + mach_port_t name; + mach_port_t *enabledp; + int *num_msgs; + int *backlog; + boolean_t *ownership; + boolean_t *receive_rights; +{ + ipc_entry_t entry; + mach_port_type_t type; + mach_port_urefs_t urefs; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; + /* space is write-locked and active */ + + kr = ipc_right_info(space, name, entry, &type, &urefs); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; /* space is unlocked */ + + if ((type & MACH_PORT_TYPE_PORT_OR_DEAD) == 0) { + is_write_unlock(space); + return KERN_INVALID_ARGUMENT; + } + + if (type & MACH_PORT_TYPE_RECEIVE) { + mach_port_t enabled; + mach_port_msgcount_t qlimit; + mach_port_msgcount_t msgcount; + ipc_port_t port; + + port = (ipc_port_t) entry->ie_object; + assert(port != IP_NULL); + + ip_lock(port); + is_write_unlock(space); + assert(ip_active(port)); + + if (port->ip_pset != IPS_NULL) { + ipc_pset_t pset = port->ip_pset; + + ips_lock(pset); + if (!ips_active(pset)) { + ipc_pset_remove(pset, port); + ips_check_unlock(pset); + enabled = MACH_PORT_NULL; + } else { + enabled = pset->ips_local_name; + ips_unlock(pset); + assert(MACH_PORT_VALID(enabled)); + } + } else + enabled = MACH_PORT_NULL; + + qlimit = port->ip_qlimit; + msgcount = port->ip_msgcount; + ip_unlock(port); + + *ownership = TRUE; + *receive_rights = TRUE; + *enabledp = enabled; + *num_msgs = (int) msgcount; + *backlog = (int) qlimit; + } else { + is_write_unlock(space); + + *ownership = FALSE; + *receive_rights = FALSE; + *enabledp = MACH_PORT_NULL; + *num_msgs = -1; + *backlog = 0; + } + + return KERN_SUCCESS; +} + +/* + * Routine: port_set_allocate [kernel call] + * Purpose: + * Create a new port set, give rights to task, and + * return task's local name for the set. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Allocated a port set. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +port_set_allocate(space, namep) + ipc_space_t space; + mach_port_t *namep; +{ + ipc_pset_t pset; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_pset_alloc(space, namep, &pset); + if (kr == KERN_SUCCESS) + ips_unlock(pset); + else if (kr != KERN_RESOURCE_SHORTAGE) + kr = KERN_INVALID_ARGUMENT; + + return kr; +} + +/* + * Routine: port_set_deallocate [kernel call] + * Purpose: + * Destroys the task's port set. If there are any + * receive rights in the set, they are removed. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Deallocated the port set. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote a port set. + */ + +kern_return_t +port_set_deallocate(space, name) + ipc_space_t space; + mach_port_t name; +{ + ipc_entry_t entry; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return kr; + /* space is write-locked and active */ + + if ((entry->ie_bits & MACH_PORT_TYPE_PORT_SET) == 0) { + is_write_unlock(space); + return KERN_INVALID_ARGUMENT; + } + + kr = ipc_right_destroy(space, name, entry); + /* space is unlocked */ + assert(kr == KERN_SUCCESS); + return kr; +} + +/* + * Routine: port_set_add [kernel call] + * Purpose: + * Moves receive rights into the port set. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Moved the receive right. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT port_name doesn't denote port rights. + * KERN_NOT_RECEIVER port_name doesn't denote receive right. + * KERN_INVALID_ARGUMENT set_name doesn't denote a port set. + * Additions: + * KERN_NOT_RECEIVER port_name denotes a send-once right. + * KERN_NOT_RECEIVER port_name denotes a dead name. + */ + +kern_return_t +port_set_add(space, set_name, port_name) + ipc_space_t space; + mach_port_t set_name; + mach_port_t port_name; +{ + ipc_entry_t entry; + mach_port_type_t type; + mach_port_urefs_t urefs; + ipc_port_t port; + ipc_pset_t pset; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_right_lookup_write(space, port_name, &entry); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; + /* space is write-locked and active */ + + /* use ipc_right_info to check for dead compat entries */ + + kr = ipc_right_info(space, port_name, entry, &type, &urefs); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; /* space is unlocked */ + + if ((type & MACH_PORT_TYPE_RECEIVE) == 0) { + is_write_unlock(space); + if (type & MACH_PORT_TYPE_PORT_OR_DEAD) + return KERN_NOT_RECEIVER; + else + return KERN_INVALID_ARGUMENT; + } + + is_write_to_read_lock(space); + port = (ipc_port_t) entry->ie_object; + assert(port != IP_NULL); + + entry = ipc_entry_lookup(space, set_name); + if ((entry == IE_NULL) || + ((entry->ie_bits & MACH_PORT_TYPE_PORT_SET) == 0)) { + is_read_unlock(space); + return KERN_INVALID_ARGUMENT; + } + + pset = (ipc_pset_t) entry->ie_object; + assert(pset != IPS_NULL); + + kr = ipc_pset_move(space, port, pset); + /* space is unlocked */ + assert(kr == KERN_SUCCESS); + return kr; +} + +/* + * Routine: port_set_remove [kernel call] + * Purpose: + * Removes the receive rights from the set they are in. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Removed the receive right. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote a port right. + * KERN_NOT_RECEIVER Name denotes send rights, not receive. + * KERN_NOT_IN_SET Port isn't in a port set. + * Additions: + * KERN_NOT_RECEIVER Name denotes a send-once right. + * KERN_NOT_RECEIVER Name denotes a dead name. + */ + +kern_return_t +port_set_remove(space, name) + ipc_space_t space; + mach_port_t name; +{ + ipc_entry_t entry; + mach_port_type_t type; + mach_port_urefs_t urefs; + ipc_port_t port; + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_right_lookup_write(space, name, &entry); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; + /* space is write-locked and active */ + + /* use ipc_right_info to check for dead compat entries */ + + kr = ipc_right_info(space, name, entry, &type, &urefs); + if (kr != KERN_SUCCESS) + return KERN_INVALID_ARGUMENT; /* space is unlocked */ + + if ((type & (MACH_PORT_TYPE_RECEIVE)) == 0) { + is_write_unlock(space); + if (type & MACH_PORT_TYPE_PORT_OR_DEAD) + return KERN_NOT_RECEIVER; + else + return KERN_INVALID_ARGUMENT; + } + + is_write_to_read_lock(space); + port = (ipc_port_t) entry->ie_object; + assert(port != IP_NULL); + + kr = ipc_pset_move(space, port, IPS_NULL); + /* space is unlocked */ + return kr; +} + +/* + * Routine: port_set_status [kernel call] + * Purpose: + * Retrieve list of members of a port set. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Retrieved port set status. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote a port set. + * Additions: + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +port_set_status(space, name, members, membersCnt) + ipc_space_t space; + mach_port_t name; + mach_port_t **members; + mach_msg_type_number_t *membersCnt; +{ + kern_return_t kr; + + kr = mach_port_get_set_status(space, name, members, membersCnt); + if ((kr != KERN_SUCCESS) && (kr != KERN_RESOURCE_SHORTAGE)) + kr = KERN_INVALID_ARGUMENT; + + return kr; +} + +/* + * Routine: port_insert_send [kernel call] + * Purpose: + * Inserts send rights to a port into a task, + * at a given name. The name must not be in use. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Inserted send right. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Port is null or dead. + * KERN_INVALID_ARGUMENT Name is reserved. + * KERN_NAME_EXISTS Name already denotes a right. + * KERN_FAILURE Task already has rights for the port. + * Additions: + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +port_insert_send(space, port, name) + ipc_space_t space; + ipc_port_t port; + mach_port_t name; +{ + kern_return_t kr; + + if ((space == IS_NULL) || + !MACH_PORT_VALID(name) || + !IP_VALID(port)) + return KERN_INVALID_ARGUMENT; + + kr = ipc_object_copyout_name_compat(space, (ipc_object_t) port, + MACH_MSG_TYPE_PORT_SEND, name); + switch (kr) { + case KERN_SUCCESS: + case KERN_NAME_EXISTS: + case KERN_RESOURCE_SHORTAGE: + break; + + case KERN_RIGHT_EXISTS: + kr = KERN_FAILURE; + break; + + default: + kr = KERN_INVALID_ARGUMENT; + break; + } + + return kr; +} + +/* + * Routine: port_extract_send [kernel call] + * Purpose: + * Extracts send rights from "task"'s "his_name" port. + * The task is left with no rights for the port. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Extracted send right. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote pure send rights. + */ + +kern_return_t +port_extract_send(space, name, portp) + ipc_space_t space; + mach_port_t name; + ipc_port_t *portp; +{ + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_object_copyin_compat(space, name, + MSG_TYPE_PORT, TRUE, + (ipc_object_t *) portp); + if (kr != KERN_SUCCESS) + kr = KERN_INVALID_ARGUMENT; + + return kr; +} + +/* + * Routine: port_insert_receive [kernel call] + * Purpose: + * Inserts receive/ownership rights to a port into a task, + * at a given name. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Inserted receive right. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Port is null. (Can't be dead.) + * KERN_INVALID_ARGUMENT Name is reserved. + * KERN_NAME_EXISTS Name already denotes a right. + * KERN_FAILURE Task already has rights for the port. + * Additions: + * KERN_RESOURCE_SHORTAGE Couldn't allocate memory. + */ + +kern_return_t +port_insert_receive(space, port, name) + ipc_space_t space; + ipc_port_t port; + mach_port_t name; +{ + kern_return_t kr; + + if ((space == IS_NULL) || + !MACH_PORT_VALID(name) || + !IP_VALID(port)) + return KERN_INVALID_ARGUMENT; + + kr = ipc_object_copyout_name_compat(space, (ipc_object_t) port, + MACH_MSG_TYPE_PORT_RECEIVE, name); + switch (kr) { + case KERN_SUCCESS: + case KERN_NAME_EXISTS: + case KERN_RESOURCE_SHORTAGE: + break; + + case KERN_RIGHT_EXISTS: + kr = KERN_FAILURE; + break; + + default: + kr = KERN_INVALID_ARGUMENT; + break; + } + + return kr; +} + +/* + * Routine: port_extract_receive [kernel call] + * Purpose: + * Extracts receive/ownership rights + * from "task"'s "his_name" port. + * + * The task is left with no rights for the port. + * Conditions: + * Nothing locked. + * Returns: + * KERN_SUCCESS Extracted receive right. + * KERN_INVALID_ARGUMENT Task is null. + * KERN_INVALID_ARGUMENT Task is not active. + * KERN_INVALID_ARGUMENT Name doesn't denote receive rights. + */ + +kern_return_t +port_extract_receive(space, name, portp) + ipc_space_t space; + mach_port_t name; + ipc_port_t *portp; +{ + kern_return_t kr; + + if (space == IS_NULL) + return KERN_INVALID_ARGUMENT; + + kr = ipc_object_copyin_compat(space, name, + MSG_TYPE_PORT_ALL, TRUE, + (ipc_object_t *) portp); + if (kr != KERN_SUCCESS) + kr = KERN_INVALID_ARGUMENT; + + return kr; +} + +#endif MACH_IPC_COMPAT |