2 * Copyright (c) 1998-2014 Apple Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
30 #include <libkern/c++/OSKext.h>
31 #include <IOKit/IOKitServer.h>
32 #include <IOKit/IOKitKeysPrivate.h>
33 #include <IOKit/IOUserClient.h>
34 #include <IOKit/IOService.h>
35 #include <IOKit/IORegistryEntry.h>
36 #include <IOKit/IOCatalogue.h>
37 #include <IOKit/IOMemoryDescriptor.h>
38 #include <IOKit/IOBufferMemoryDescriptor.h>
39 #include <IOKit/IOLib.h>
40 #include <IOKit/IOBSD.h>
41 #include <IOKit/IOStatisticsPrivate.h>
42 #include <IOKit/IOTimeStamp.h>
43 #include <IOKit/system.h>
44 #include <libkern/OSDebug.h>
46 #include <sys/kauth.h>
47 #include <sys/codesign.h>
54 #include <security/mac_framework.h>
56 #include <sys/kauth.h>
60 #endif /* CONFIG_MACF */
62 #include <IOKit/assert.h>
64 #include "IOServicePrivate.h"
65 #include "IOKitKernelInternal.h"
67 #define SCALAR64(x) ((io_user_scalar_t)((unsigned int)x))
68 #define SCALAR32(x) ((uint32_t )x)
69 #define ARG32(x) ((void *)(uintptr_t)SCALAR32(x))
70 #define REF64(x) ((io_user_reference_t)((UInt64)(x)))
71 #define REF32(x) ((int)(x))
75 kIOUCAsync0Flags
= 3ULL,
76 kIOUCAsync64Flag
= 1ULL,
77 kIOUCAsyncErrorLoggedFlag
= 2ULL
82 #define IOStatisticsRegisterCounter() \
84 reserved->counter = IOStatistics::registerUserClient(this); \
87 #define IOStatisticsUnregisterCounter() \
90 IOStatistics::unregisterUserClient(reserved->counter); \
93 #define IOStatisticsClientCall() \
95 IOStatistics::countUserClientCall(client); \
100 #define IOStatisticsRegisterCounter()
101 #define IOStatisticsUnregisterCounter()
102 #define IOStatisticsClientCall()
104 #endif /* IOKITSTATS */
106 #if DEVELOPMENT || DEBUG
108 #define FAKE_STACK_FRAME(a) \
109 const void ** __frameptr; \
110 const void * __retaddr; \
111 __frameptr = (typeof(__frameptr)) __builtin_frame_address(0); \
112 __retaddr = __frameptr[1]; \
115 #define FAKE_STACK_FRAME_END() \
116 __frameptr[1] = __retaddr;
118 #else /* DEVELOPMENT || DEBUG */
120 #define FAKE_STACK_FRAME(a)
121 #define FAKE_STACK_FRAME_END()
123 #endif /* DEVELOPMENT || DEBUG */
125 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
127 // definitions we should get from osfmk
129 //typedef struct ipc_port * ipc_port_t;
130 typedef natural_t ipc_kobject_type_t
;
132 #define IKOT_IOKIT_SPARE 27
133 #define IKOT_IOKIT_CONNECT 29
134 #define IKOT_IOKIT_OBJECT 30
138 extern ipc_port_t
iokit_alloc_object_port( io_object_t obj
,
139 ipc_kobject_type_t type
);
141 extern kern_return_t
iokit_destroy_object_port( ipc_port_t port
);
143 extern mach_port_name_t
iokit_make_send_right( task_t task
,
144 io_object_t obj
, ipc_kobject_type_t type
);
146 extern kern_return_t
iokit_mod_send_right( task_t task
, mach_port_name_t name
, mach_port_delta_t delta
);
148 extern io_object_t
iokit_lookup_connect_ref(io_object_t clientRef
, ipc_space_t task
);
150 extern io_object_t
iokit_lookup_connect_ref_current_task(io_object_t clientRef
);
152 extern ipc_port_t master_device_port
;
154 extern void iokit_retain_port( ipc_port_t port
);
155 extern void iokit_release_port( ipc_port_t port
);
156 extern void iokit_release_port_send( ipc_port_t port
);
158 extern kern_return_t
iokit_switch_object_port( ipc_port_t port
, io_object_t obj
, ipc_kobject_type_t type
);
160 #include <mach/mach_traps.h>
161 #include <vm/vm_map.h>
166 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
168 // IOMachPort maps OSObjects to ports, avoiding adding an ivar to OSObject.
170 class IOMachPort
: public OSObject
172 OSDeclareDefaultStructors(IOMachPort
)
179 static IOMachPort
* portForObject( OSObject
* obj
,
180 ipc_kobject_type_t type
);
181 static bool noMoreSendersForObject( OSObject
* obj
,
182 ipc_kobject_type_t type
, mach_port_mscount_t
* mscount
);
183 static void releasePortForObject( OSObject
* obj
,
184 ipc_kobject_type_t type
);
185 static void setHoldDestroy( OSObject
* obj
, ipc_kobject_type_t type
);
187 static OSDictionary
* dictForType( ipc_kobject_type_t type
);
189 static mach_port_name_t
makeSendRightForTask( task_t task
,
190 io_object_t obj
, ipc_kobject_type_t type
);
192 virtual void free() APPLE_KEXT_OVERRIDE
;
195 #define super OSObject
196 OSDefineMetaClassAndStructors(IOMachPort
, OSObject
)
198 static IOLock
* gIOObjectPortLock
;
200 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
202 // not in dictForType() for debugging ease
203 static OSDictionary
* gIOObjectPorts
;
204 static OSDictionary
* gIOConnectPorts
;
206 OSDictionary
* IOMachPort::dictForType( ipc_kobject_type_t type
)
208 OSDictionary
** dict
;
210 if( IKOT_IOKIT_OBJECT
== type
)
211 dict
= &gIOObjectPorts
;
212 else if( IKOT_IOKIT_CONNECT
== type
)
213 dict
= &gIOConnectPorts
;
218 *dict
= OSDictionary::withCapacity( 1 );
223 IOMachPort
* IOMachPort::portForObject ( OSObject
* obj
,
224 ipc_kobject_type_t type
)
226 IOMachPort
* inst
= 0;
229 IOTakeLock( gIOObjectPortLock
);
233 dict
= dictForType( type
);
237 if( (inst
= (IOMachPort
*)
238 dict
->getObject( (const OSSymbol
*) obj
))) {
244 inst
= new IOMachPort
;
245 if( inst
&& !inst
->init()) {
250 inst
->port
= iokit_alloc_object_port( obj
, type
);
253 dict
->setObject( (const OSSymbol
*) obj
, inst
);
263 IOUnlock( gIOObjectPortLock
);
268 bool IOMachPort::noMoreSendersForObject( OSObject
* obj
,
269 ipc_kobject_type_t type
, mach_port_mscount_t
* mscount
)
272 IOMachPort
* machPort
;
274 bool destroyed
= true;
276 IOTakeLock( gIOObjectPortLock
);
278 if( (dict
= dictForType( type
))) {
281 machPort
= (IOMachPort
*) dict
->getObject( (const OSSymbol
*) obj
);
283 destroyed
= (machPort
->mscount
<= *mscount
);
284 if (!destroyed
) *mscount
= machPort
->mscount
;
287 if ((IKOT_IOKIT_CONNECT
== type
) && (uc
= OSDynamicCast(IOUserClient
, obj
)))
291 dict
->removeObject( (const OSSymbol
*) obj
);
297 IOUnlock( gIOObjectPortLock
);
302 void IOMachPort::releasePortForObject( OSObject
* obj
,
303 ipc_kobject_type_t type
)
306 IOMachPort
* machPort
;
308 assert(IKOT_IOKIT_CONNECT
!= type
);
310 IOTakeLock( gIOObjectPortLock
);
312 if( (dict
= dictForType( type
))) {
314 machPort
= (IOMachPort
*) dict
->getObject( (const OSSymbol
*) obj
);
315 if( machPort
&& !machPort
->holdDestroy
)
316 dict
->removeObject( (const OSSymbol
*) obj
);
320 IOUnlock( gIOObjectPortLock
);
323 void IOMachPort::setHoldDestroy( OSObject
* obj
, ipc_kobject_type_t type
)
326 IOMachPort
* machPort
;
328 IOLockLock( gIOObjectPortLock
);
330 if( (dict
= dictForType( type
))) {
331 machPort
= (IOMachPort
*) dict
->getObject( (const OSSymbol
*) obj
);
333 machPort
->holdDestroy
= true;
336 IOLockUnlock( gIOObjectPortLock
);
339 void IOUserClient::destroyUserReferences( OSObject
* obj
)
341 IOMachPort::releasePortForObject( obj
, IKOT_IOKIT_OBJECT
);
344 // IOMachPort::releasePortForObject( obj, IKOT_IOKIT_CONNECT );
348 IOTakeLock( gIOObjectPortLock
);
351 if( (dict
= IOMachPort::dictForType( IKOT_IOKIT_CONNECT
)))
354 port
= (IOMachPort
*) dict
->getObject( (const OSSymbol
*) obj
);
358 if ((uc
= OSDynamicCast(IOUserClient
, obj
)))
363 dict
->setObject((const OSSymbol
*) uc
->mappings
, port
);
364 iokit_switch_object_port(port
->port
, uc
->mappings
, IKOT_IOKIT_CONNECT
);
366 uc
->mappings
->release();
370 dict
->removeObject( (const OSSymbol
*) obj
);
374 IOUnlock( gIOObjectPortLock
);
377 mach_port_name_t
IOMachPort::makeSendRightForTask( task_t task
,
378 io_object_t obj
, ipc_kobject_type_t type
)
380 return( iokit_make_send_right( task
, obj
, type
));
383 void IOMachPort::free( void )
386 iokit_destroy_object_port( port
);
390 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
392 class IOUserIterator
: public OSIterator
394 OSDeclareDefaultStructors(IOUserIterator
)
396 OSObject
* userIteratorObject
;
399 static IOUserIterator
* withIterator(OSIterator
* iter
);
400 virtual bool init( void ) APPLE_KEXT_OVERRIDE
;
401 virtual void free() APPLE_KEXT_OVERRIDE
;
403 virtual void reset() APPLE_KEXT_OVERRIDE
;
404 virtual bool isValid() APPLE_KEXT_OVERRIDE
;
405 virtual OSObject
* getNextObject() APPLE_KEXT_OVERRIDE
;
408 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
410 class IOUserNotification
: public IOUserIterator
412 OSDeclareDefaultStructors(IOUserNotification
)
414 #define holdNotify userIteratorObject
418 virtual void free() APPLE_KEXT_OVERRIDE
;
420 virtual void setNotification( IONotifier
* obj
);
422 virtual void reset() APPLE_KEXT_OVERRIDE
;
423 virtual bool isValid() APPLE_KEXT_OVERRIDE
;
426 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
428 OSDefineMetaClassAndStructors( IOUserIterator
, OSIterator
)
431 IOUserIterator::withIterator(OSIterator
* iter
)
435 if (!iter
) return (0);
437 me
= new IOUserIterator
;
438 if (me
&& !me
->init())
444 me
->userIteratorObject
= iter
;
450 IOUserIterator::init( void )
452 if (!OSObject::init()) return (false);
454 lock
= IOLockAlloc();
462 IOUserIterator::free()
464 if (userIteratorObject
) userIteratorObject
->release();
465 if (lock
) IOLockFree(lock
);
470 IOUserIterator::reset()
473 assert(OSDynamicCast(OSIterator
, userIteratorObject
));
474 ((OSIterator
*)userIteratorObject
)->reset();
479 IOUserIterator::isValid()
484 assert(OSDynamicCast(OSIterator
, userIteratorObject
));
485 ret
= ((OSIterator
*)userIteratorObject
)->isValid();
492 IOUserIterator::getNextObject()
497 assert(OSDynamicCast(OSIterator
, userIteratorObject
));
498 ret
= ((OSIterator
*)userIteratorObject
)->getNextObject();
504 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
507 // functions called from osfmk/device/iokit_rpc.c
510 iokit_add_reference( io_object_t obj
)
517 iokit_remove_reference( io_object_t obj
)
524 iokit_add_connect_reference( io_object_t obj
)
530 if ((uc
= OSDynamicCast(IOUserClient
, obj
))) OSIncrementAtomic(&uc
->__ipc
);
536 iokit_remove_connect_reference( io_object_t obj
)
539 bool finalize
= false;
543 if ((uc
= OSDynamicCast(IOUserClient
, obj
)))
545 if (1 == OSDecrementAtomic(&uc
->__ipc
) && uc
->isInactive())
547 IOLockLock(gIOObjectPortLock
);
548 if ((finalize
= uc
->__ipcFinal
)) uc
->__ipcFinal
= false;
549 IOLockUnlock(gIOObjectPortLock
);
551 if (finalize
) uc
->scheduleFinalize(true);
558 IOUserClient::finalizeUserReferences(OSObject
* obj
)
563 if ((uc
= OSDynamicCast(IOUserClient
, obj
)))
565 IOLockLock(gIOObjectPortLock
);
566 if ((uc
->__ipcFinal
= (0 != uc
->__ipc
))) ok
= false;
567 IOLockUnlock(gIOObjectPortLock
);
573 iokit_port_for_object( io_object_t obj
, ipc_kobject_type_t type
)
575 IOMachPort
* machPort
;
578 if( (machPort
= IOMachPort::portForObject( obj
, type
))) {
580 port
= machPort
->port
;
582 iokit_retain_port( port
);
593 iokit_client_died( io_object_t obj
, ipc_port_t
/* port */,
594 ipc_kobject_type_t type
, mach_port_mscount_t
* mscount
)
596 IOUserClient
* client
;
598 IOUserNotification
* notify
;
600 if( !IOMachPort::noMoreSendersForObject( obj
, type
, mscount
))
601 return( kIOReturnNotReady
);
603 if( IKOT_IOKIT_CONNECT
== type
)
605 if( (client
= OSDynamicCast( IOUserClient
, obj
)))
607 IOStatisticsClientCall();
608 client
->clientDied();
611 else if( IKOT_IOKIT_OBJECT
== type
)
613 if( (map
= OSDynamicCast( IOMemoryMap
, obj
)))
615 else if( (notify
= OSDynamicCast( IOUserNotification
, obj
)))
616 notify
->setNotification( 0 );
619 return( kIOReturnSuccess
);
624 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
626 class IOServiceUserNotification
: public IOUserNotification
628 OSDeclareDefaultStructors(IOServiceUserNotification
)
631 mach_msg_header_t msgHdr
;
632 OSNotificationHeader64 notifyHeader
;
635 enum { kMaxOutstanding
= 1024 };
640 OSObject
* lastEntry
;
646 virtual bool init( mach_port_t port
, natural_t type
,
647 void * reference
, vm_size_t referenceSize
,
649 virtual void free() APPLE_KEXT_OVERRIDE
;
651 static bool _handler( void * target
,
652 void * ref
, IOService
* newService
, IONotifier
* notifier
);
653 virtual bool handler( void * ref
, IOService
* newService
);
655 virtual OSObject
* getNextObject() APPLE_KEXT_OVERRIDE
;
658 class IOServiceMessageUserNotification
: public IOUserNotification
660 OSDeclareDefaultStructors(IOServiceMessageUserNotification
)
663 mach_msg_header_t msgHdr
;
664 mach_msg_body_t msgBody
;
665 mach_msg_port_descriptor_t ports
[1];
666 OSNotificationHeader64 notifyHeader
__attribute__ ((packed
));
677 virtual bool init( mach_port_t port
, natural_t type
,
678 void * reference
, vm_size_t referenceSize
,
682 virtual void free() APPLE_KEXT_OVERRIDE
;
684 static IOReturn
_handler( void * target
, void * ref
,
685 UInt32 messageType
, IOService
* provider
,
686 void * messageArgument
, vm_size_t argSize
);
687 virtual IOReturn
handler( void * ref
,
688 UInt32 messageType
, IOService
* provider
,
689 void * messageArgument
, vm_size_t argSize
);
691 virtual OSObject
* getNextObject() APPLE_KEXT_OVERRIDE
;
694 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
697 #define super IOUserIterator
698 OSDefineMetaClass( IOUserNotification
, IOUserIterator
)
699 OSDefineAbstractStructors( IOUserNotification
, IOUserIterator
)
701 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
703 void IOUserNotification::free( void )
707 assert(OSDynamicCast(IONotifier
, holdNotify
));
708 ((IONotifier
*)holdNotify
)->remove();
711 // can't be in handler now
717 void IOUserNotification::setNotification( IONotifier
* notify
)
719 OSObject
* previousNotify
;
721 IOLockLock( gIOObjectPortLock
);
723 previousNotify
= holdNotify
;
726 IOLockUnlock( gIOObjectPortLock
);
730 assert(OSDynamicCast(IONotifier
, previousNotify
));
731 ((IONotifier
*)previousNotify
)->remove();
735 void IOUserNotification::reset()
740 bool IOUserNotification::isValid()
745 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
748 #define super IOUserNotification
749 OSDefineMetaClassAndStructors(IOServiceUserNotification
, IOUserNotification
)
751 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
753 bool IOServiceUserNotification::init( mach_port_t port
, natural_t type
,
754 void * reference
, vm_size_t referenceSize
,
760 newSet
= OSArray::withCapacity( 1 );
764 if (referenceSize
> sizeof(OSAsyncReference64
))
767 msgSize
= sizeof(PingMsg
) - sizeof(OSAsyncReference64
) + referenceSize
;
768 pingMsg
= (PingMsg
*) IOMalloc( msgSize
);
772 bzero( pingMsg
, msgSize
);
774 pingMsg
->msgHdr
.msgh_remote_port
= port
;
775 pingMsg
->msgHdr
.msgh_bits
= MACH_MSGH_BITS(
776 MACH_MSG_TYPE_COPY_SEND
/*remote*/,
777 MACH_MSG_TYPE_MAKE_SEND
/*local*/);
778 pingMsg
->msgHdr
.msgh_size
= msgSize
;
779 pingMsg
->msgHdr
.msgh_id
= kOSNotificationMessageID
;
781 pingMsg
->notifyHeader
.size
= 0;
782 pingMsg
->notifyHeader
.type
= type
;
783 bcopy( reference
, pingMsg
->notifyHeader
.reference
, referenceSize
);
788 void IOServiceUserNotification::free( void )
793 OSObject
* _lastEntry
;
797 _lastEntry
= lastEntry
;
802 if( _pingMsg
&& _msgSize
) {
803 if (_pingMsg
->msgHdr
.msgh_remote_port
) {
804 iokit_release_port_send(_pingMsg
->msgHdr
.msgh_remote_port
);
806 IOFree(_pingMsg
, _msgSize
);
810 _lastEntry
->release();
816 bool IOServiceUserNotification::_handler( void * target
,
817 void * ref
, IOService
* newService
, IONotifier
* notifier
)
819 return( ((IOServiceUserNotification
*) target
)->handler( ref
, newService
));
822 bool IOServiceUserNotification::handler( void * ref
,
823 IOService
* newService
)
827 ipc_port_t port
= NULL
;
828 bool sendPing
= false;
832 count
= newSet
->getCount();
833 if( count
< kMaxOutstanding
) {
835 newSet
->setObject( newService
);
836 if( (sendPing
= (armed
&& (0 == count
))))
842 if( kIOServiceTerminatedNotificationType
== pingMsg
->notifyHeader
.type
)
843 IOMachPort::setHoldDestroy( newService
, IKOT_IOKIT_OBJECT
);
846 if( (port
= iokit_port_for_object( this, IKOT_IOKIT_OBJECT
) ))
847 pingMsg
->msgHdr
.msgh_local_port
= port
;
849 pingMsg
->msgHdr
.msgh_local_port
= NULL
;
851 kr
= mach_msg_send_from_kernel_with_options( &pingMsg
->msgHdr
,
852 pingMsg
->msgHdr
.msgh_size
,
853 (MACH_SEND_MSG
| MACH_SEND_ALWAYS
| MACH_SEND_IMPORTANCE
),
856 iokit_release_port( port
);
858 if( (KERN_SUCCESS
!= kr
) && !ipcLogged
)
861 IOLog("%s: mach_msg_send_from_kernel_proper(0x%x)\n", __PRETTY_FUNCTION__
, kr
);
868 OSObject
* IOServiceUserNotification::getNextObject()
872 OSObject
* releaseEntry
;
876 releaseEntry
= lastEntry
;
877 count
= newSet
->getCount();
879 result
= newSet
->getObject( count
- 1 );
881 newSet
->removeObject( count
- 1);
890 if (releaseEntry
) releaseEntry
->release();
895 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
897 OSDefineMetaClassAndStructors(IOServiceMessageUserNotification
, IOUserNotification
)
899 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
901 bool IOServiceMessageUserNotification::init( mach_port_t port
, natural_t type
,
902 void * reference
, vm_size_t referenceSize
, vm_size_t extraSize
,
908 if (referenceSize
> sizeof(OSAsyncReference64
))
911 clientIs64
= client64
;
913 owningPID
= proc_selfpid();
915 extraSize
+= sizeof(IOServiceInterestContent64
);
916 msgSize
= sizeof(PingMsg
) - sizeof(OSAsyncReference64
) + referenceSize
;
917 pingMsg
= (PingMsg
*) IOMalloc( msgSize
);
921 bzero( pingMsg
, msgSize
);
923 pingMsg
->msgHdr
.msgh_remote_port
= port
;
924 pingMsg
->msgHdr
.msgh_bits
= MACH_MSGH_BITS_COMPLEX
926 MACH_MSG_TYPE_COPY_SEND
/*remote*/,
927 MACH_MSG_TYPE_MAKE_SEND
/*local*/);
928 pingMsg
->msgHdr
.msgh_size
= msgSize
;
929 pingMsg
->msgHdr
.msgh_id
= kOSNotificationMessageID
;
931 pingMsg
->msgBody
.msgh_descriptor_count
= 1;
933 pingMsg
->ports
[0].name
= 0;
934 pingMsg
->ports
[0].disposition
= MACH_MSG_TYPE_MAKE_SEND
;
935 pingMsg
->ports
[0].type
= MACH_MSG_PORT_DESCRIPTOR
;
937 pingMsg
->notifyHeader
.size
= extraSize
;
938 pingMsg
->notifyHeader
.type
= type
;
939 bcopy( reference
, pingMsg
->notifyHeader
.reference
, referenceSize
);
944 void IOServiceMessageUserNotification::free( void )
954 if( _pingMsg
&& _msgSize
) {
955 if (_pingMsg
->msgHdr
.msgh_remote_port
) {
956 iokit_release_port_send(_pingMsg
->msgHdr
.msgh_remote_port
);
958 IOFree( _pingMsg
, _msgSize
);
962 IOReturn
IOServiceMessageUserNotification::_handler( void * target
, void * ref
,
963 UInt32 messageType
, IOService
* provider
,
964 void * argument
, vm_size_t argSize
)
966 return( ((IOServiceMessageUserNotification
*) target
)->handler(
967 ref
, messageType
, provider
, argument
, argSize
));
970 IOReturn
IOServiceMessageUserNotification::handler( void * ref
,
971 UInt32 messageType
, IOService
* provider
,
972 void * messageArgument
, vm_size_t callerArgSize
)
974 enum { kLocalMsgSize
= 0x100 };
975 uint64_t stackMsg
[kLocalMsgSize
/ sizeof(uint64_t)];
979 vm_size_t thisMsgSize
;
980 ipc_port_t thisPort
, providerPort
;
981 struct PingMsg
* thisMsg
;
982 IOServiceInterestContent64
* data
;
984 if (kIOMessageCopyClientID
== messageType
)
986 *((void **) messageArgument
) = OSNumber::withNumber(owningPID
, 32);
987 return (kIOReturnSuccess
);
990 if (callerArgSize
== 0)
992 if (clientIs64
) argSize
= sizeof(data
->messageArgument
[0]);
993 else argSize
= sizeof(uint32_t);
997 argSize
= callerArgSize
;
998 if( argSize
> kIOUserNotifyMaxMessageSize
)
999 argSize
= kIOUserNotifyMaxMessageSize
;
1002 // adjust message size for ipc restrictions
1004 type
= pingMsg
->notifyHeader
.type
;
1005 type
&= ~(kIOKitNoticationMsgSizeMask
<< kIOKitNoticationTypeSizeAdjShift
);
1006 type
|= ((argSize
& kIOKitNoticationMsgSizeMask
) << kIOKitNoticationTypeSizeAdjShift
);
1007 argSize
= (argSize
+ kIOKitNoticationMsgSizeMask
) & ~kIOKitNoticationMsgSizeMask
;
1009 thisMsgSize
= msgSize
1010 + sizeof( IOServiceInterestContent64
)
1011 - sizeof( data
->messageArgument
)
1014 if (thisMsgSize
> sizeof(stackMsg
))
1016 allocMsg
= IOMalloc(thisMsgSize
);
1017 if (!allocMsg
) return (kIOReturnNoMemory
);
1018 thisMsg
= (typeof(thisMsg
)) allocMsg
;
1023 thisMsg
= (typeof(thisMsg
)) stackMsg
;
1026 bcopy(pingMsg
, thisMsg
, msgSize
);
1027 thisMsg
->notifyHeader
.type
= type
;
1028 data
= (IOServiceInterestContent64
*) (((uint8_t *) thisMsg
) + msgSize
);
1029 // == pingMsg->notifyHeader.content;
1030 data
->messageType
= messageType
;
1032 if (callerArgSize
== 0)
1034 data
->messageArgument
[0] = (io_user_reference_t
) messageArgument
;
1037 data
->messageArgument
[0] |= (data
->messageArgument
[0] << 32);
1042 bcopy( messageArgument
, data
->messageArgument
, callerArgSize
);
1043 bzero((void *)(((uintptr_t) &data
->messageArgument
[0]) + callerArgSize
), argSize
- callerArgSize
);
1046 thisMsg
->notifyHeader
.type
= type
;
1047 thisMsg
->msgHdr
.msgh_size
= thisMsgSize
;
1049 providerPort
= iokit_port_for_object( provider
, IKOT_IOKIT_OBJECT
);
1050 thisMsg
->ports
[0].name
= providerPort
;
1051 thisPort
= iokit_port_for_object( this, IKOT_IOKIT_OBJECT
);
1052 thisMsg
->msgHdr
.msgh_local_port
= thisPort
;
1054 kr
= mach_msg_send_from_kernel_with_options( &thisMsg
->msgHdr
,
1055 thisMsg
->msgHdr
.msgh_size
,
1056 (MACH_SEND_MSG
| MACH_SEND_ALWAYS
| MACH_SEND_IMPORTANCE
),
1059 iokit_release_port( thisPort
);
1061 iokit_release_port( providerPort
);
1064 IOFree(allocMsg
, thisMsgSize
);
1066 if((KERN_SUCCESS
!= kr
) && !ipcLogged
)
1069 IOLog("%s: mach_msg_send_from_kernel_proper (0x%x)\n", __PRETTY_FUNCTION__
, kr
);
1072 return( kIOReturnSuccess
);
1075 OSObject
* IOServiceMessageUserNotification::getNextObject()
1080 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1083 #define super IOService
1084 OSDefineMetaClassAndAbstractStructors( IOUserClient
, IOService
)
1086 IOLock
* gIOUserClientOwnersLock
;
1088 void IOUserClient::initialize( void )
1090 gIOObjectPortLock
= IOLockAlloc();
1091 gIOUserClientOwnersLock
= IOLockAlloc();
1092 assert(gIOObjectPortLock
&& gIOUserClientOwnersLock
);
1095 void IOUserClient::setAsyncReference(OSAsyncReference asyncRef
,
1096 mach_port_t wakePort
,
1097 void *callback
, void *refcon
)
1099 asyncRef
[kIOAsyncReservedIndex
] = ((uintptr_t) wakePort
)
1100 | (kIOUCAsync0Flags
& asyncRef
[kIOAsyncReservedIndex
]);
1101 asyncRef
[kIOAsyncCalloutFuncIndex
] = (uintptr_t) callback
;
1102 asyncRef
[kIOAsyncCalloutRefconIndex
] = (uintptr_t) refcon
;
1105 void IOUserClient::setAsyncReference64(OSAsyncReference64 asyncRef
,
1106 mach_port_t wakePort
,
1107 mach_vm_address_t callback
, io_user_reference_t refcon
)
1109 asyncRef
[kIOAsyncReservedIndex
] = ((io_user_reference_t
) wakePort
)
1110 | (kIOUCAsync0Flags
& asyncRef
[kIOAsyncReservedIndex
]);
1111 asyncRef
[kIOAsyncCalloutFuncIndex
] = (io_user_reference_t
) callback
;
1112 asyncRef
[kIOAsyncCalloutRefconIndex
] = refcon
;
1115 void IOUserClient::setAsyncReference64(OSAsyncReference64 asyncRef
,
1116 mach_port_t wakePort
,
1117 mach_vm_address_t callback
, io_user_reference_t refcon
, task_t task
)
1119 setAsyncReference64(asyncRef
, wakePort
, callback
, refcon
);
1120 if (vm_map_is_64bit(get_task_map(task
))) {
1121 asyncRef
[kIOAsyncReservedIndex
] |= kIOUCAsync64Flag
;
1125 static OSDictionary
* CopyConsoleUser(UInt32 uid
)
1128 OSDictionary
* user
= 0;
1130 if ((array
= OSDynamicCast(OSArray
,
1131 IORegistryEntry::getRegistryRoot()->copyProperty(gIOConsoleUsersKey
))))
1133 for (unsigned int idx
= 0;
1134 (user
= OSDynamicCast(OSDictionary
, array
->getObject(idx
)));
1138 if ((num
= OSDynamicCast(OSNumber
, user
->getObject(gIOConsoleSessionUIDKey
)))
1139 && (uid
== num
->unsigned32BitValue())) {
1149 static OSDictionary
* CopyUserOnConsole(void)
1152 OSDictionary
* user
= 0;
1154 if ((array
= OSDynamicCast(OSArray
,
1155 IORegistryEntry::getRegistryRoot()->copyProperty(gIOConsoleUsersKey
))))
1157 for (unsigned int idx
= 0;
1158 (user
= OSDynamicCast(OSDictionary
, array
->getObject(idx
)));
1161 if (kOSBooleanTrue
== user
->getObject(gIOConsoleSessionOnConsoleKey
))
1172 IOReturn
IOUserClient::clientHasAuthorization( task_t task
,
1173 IOService
* service
)
1177 p
= (proc_t
) get_bsdtask_info(task
);
1180 uint64_t authorizationID
;
1182 authorizationID
= proc_uniqueid(p
);
1183 if (authorizationID
)
1185 if (service
->getAuthorizationID() == authorizationID
)
1187 return (kIOReturnSuccess
);
1192 return (kIOReturnNotPermitted
);
1195 IOReturn
IOUserClient::clientHasPrivilege( void * securityToken
,
1196 const char * privilegeName
)
1199 security_token_t token
;
1200 mach_msg_type_number_t count
;
1202 OSDictionary
* user
;
1206 if (!strncmp(privilegeName
, kIOClientPrivilegeForeground
,
1207 sizeof(kIOClientPrivilegeForeground
)))
1209 if (task_is_gpu_denied(current_task()))
1210 return (kIOReturnNotPrivileged
);
1212 return (kIOReturnSuccess
);
1215 if (!strncmp(privilegeName
, kIOClientPrivilegeConsoleSession
,
1216 sizeof(kIOClientPrivilegeConsoleSession
)))
1221 task
= (task_t
) securityToken
;
1223 task
= current_task();
1224 p
= (proc_t
) get_bsdtask_info(task
);
1225 kr
= kIOReturnNotPrivileged
;
1227 if (p
&& (cred
= kauth_cred_proc_ref(p
)))
1229 user
= CopyUserOnConsole();
1233 if ((num
= OSDynamicCast(OSNumber
, user
->getObject(gIOConsoleSessionAuditIDKey
)))
1234 && (cred
->cr_audit
.as_aia_p
->ai_asid
== (au_asid_t
) num
->unsigned32BitValue()))
1236 kr
= kIOReturnSuccess
;
1240 kauth_cred_unref(&cred
);
1245 if ((secureConsole
= !strncmp(privilegeName
, kIOClientPrivilegeSecureConsoleProcess
,
1246 sizeof(kIOClientPrivilegeSecureConsoleProcess
))))
1247 task
= (task_t
)((IOUCProcessToken
*)securityToken
)->token
;
1249 task
= (task_t
)securityToken
;
1251 count
= TASK_SECURITY_TOKEN_COUNT
;
1252 kr
= task_info( task
, TASK_SECURITY_TOKEN
, (task_info_t
) &token
, &count
);
1254 if (KERN_SUCCESS
!= kr
)
1256 else if (!strncmp(privilegeName
, kIOClientPrivilegeAdministrator
,
1257 sizeof(kIOClientPrivilegeAdministrator
))) {
1258 if (0 != token
.val
[0])
1259 kr
= kIOReturnNotPrivileged
;
1260 } else if (!strncmp(privilegeName
, kIOClientPrivilegeLocalUser
,
1261 sizeof(kIOClientPrivilegeLocalUser
))) {
1262 user
= CopyConsoleUser(token
.val
[0]);
1266 kr
= kIOReturnNotPrivileged
;
1267 } else if (secureConsole
|| !strncmp(privilegeName
, kIOClientPrivilegeConsoleUser
,
1268 sizeof(kIOClientPrivilegeConsoleUser
))) {
1269 user
= CopyConsoleUser(token
.val
[0]);
1271 if (user
->getObject(gIOConsoleSessionOnConsoleKey
) != kOSBooleanTrue
)
1272 kr
= kIOReturnNotPrivileged
;
1273 else if ( secureConsole
) {
1274 OSNumber
* pid
= OSDynamicCast(OSNumber
, user
->getObject(gIOConsoleSessionSecureInputPIDKey
));
1275 if ( pid
&& pid
->unsigned32BitValue() != ((IOUCProcessToken
*)securityToken
)->pid
)
1276 kr
= kIOReturnNotPrivileged
;
1281 kr
= kIOReturnNotPrivileged
;
1283 kr
= kIOReturnUnsupported
;
1288 OSObject
* IOUserClient::copyClientEntitlement( task_t task
,
1289 const char * entitlement
)
1291 #define MAX_ENTITLEMENTS_LEN (128 * 1024)
1295 char procname
[MAXCOMLEN
+ 1] = "";
1297 void *entitlements_blob
= NULL
;
1298 char *entitlements_data
= NULL
;
1299 OSObject
*entitlements_obj
= NULL
;
1300 OSDictionary
*entitlements
= NULL
;
1301 OSString
*errorString
= NULL
;
1302 OSObject
*value
= NULL
;
1304 p
= (proc_t
)get_bsdtask_info(task
);
1308 proc_name(pid
, procname
, (int)sizeof(procname
));
1310 if (cs_entitlements_blob_get(p
, &entitlements_blob
, &len
) != 0)
1313 if (len
<= offsetof(CS_GenericBlob
, data
))
1317 * Per <rdar://problem/11593877>, enforce a limit on the amount of XML
1318 * we'll try to parse in the kernel.
1320 len
-= offsetof(CS_GenericBlob
, data
);
1321 if (len
> MAX_ENTITLEMENTS_LEN
) {
1322 IOLog("failed to parse entitlements for %s[%u]: %lu bytes of entitlements exceeds maximum of %u\n", procname
, pid
, len
, MAX_ENTITLEMENTS_LEN
);
1327 * OSUnserializeXML() expects a nul-terminated string, but that isn't
1328 * what is stored in the entitlements blob. Copy the string and
1331 entitlements_data
= (char *)IOMalloc(len
+ 1);
1332 if (entitlements_data
== NULL
)
1334 memcpy(entitlements_data
, ((CS_GenericBlob
*)entitlements_blob
)->data
, len
);
1335 entitlements_data
[len
] = '\0';
1337 entitlements_obj
= OSUnserializeXML(entitlements_data
, len
+ 1, &errorString
);
1338 if (errorString
!= NULL
) {
1339 IOLog("failed to parse entitlements for %s[%u]: %s\n", procname
, pid
, errorString
->getCStringNoCopy());
1342 if (entitlements_obj
== NULL
)
1345 entitlements
= OSDynamicCast(OSDictionary
, entitlements_obj
);
1346 if (entitlements
== NULL
)
1349 /* Fetch the entitlement value from the dictionary. */
1350 value
= entitlements
->getObject(entitlement
);
1355 if (entitlements_data
!= NULL
)
1356 IOFree(entitlements_data
, len
+ 1);
1357 if (entitlements_obj
!= NULL
)
1358 entitlements_obj
->release();
1359 if (errorString
!= NULL
)
1360 errorString
->release();
1364 bool IOUserClient::init()
1366 if (getPropertyTable() || super::init())
1372 bool IOUserClient::init(OSDictionary
* dictionary
)
1374 if (getPropertyTable() || super::init(dictionary
))
1380 bool IOUserClient::initWithTask(task_t owningTask
,
1384 if (getPropertyTable() || super::init())
1390 bool IOUserClient::initWithTask(task_t owningTask
,
1393 OSDictionary
* properties
)
1397 ok
= super::init( properties
);
1398 ok
&= initWithTask( owningTask
, securityID
, type
);
1403 bool IOUserClient::reserve()
1406 reserved
= IONew(ExpansionData
, 1);
1411 setTerminateDefer(NULL
, true);
1412 IOStatisticsRegisterCounter();
1417 struct IOUserClientOwner
1420 queue_chain_t taskLink
;
1422 queue_chain_t ucLink
;
1426 IOUserClient::registerOwner(task_t task
)
1428 IOUserClientOwner
* owner
;
1432 IOLockLock(gIOUserClientOwnersLock
);
1435 ret
= kIOReturnSuccess
;
1437 if (!owners
.next
) queue_init(&owners
);
1440 queue_iterate(&owners
, owner
, IOUserClientOwner
*, ucLink
)
1442 if (task
!= owner
->task
) continue;
1449 owner
= IONew(IOUserClientOwner
, 1);
1450 if (!newOwner
) ret
= kIOReturnNoMemory
;
1455 queue_enter_first(&owners
, owner
, IOUserClientOwner
*, ucLink
);
1456 queue_enter_first(task_io_user_clients(task
), owner
, IOUserClientOwner
*, taskLink
);
1460 IOLockUnlock(gIOUserClientOwnersLock
);
1466 IOUserClient::noMoreSenders(void)
1468 IOUserClientOwner
* owner
;
1470 IOLockLock(gIOUserClientOwnersLock
);
1474 while (!queue_empty(&owners
))
1476 owner
= (IOUserClientOwner
*)(void *) queue_first(&owners
);
1477 queue_remove(task_io_user_clients(owner
->task
), owner
, IOUserClientOwner
*, taskLink
);
1478 queue_remove(&owners
, owner
, IOUserClientOwner
*, ucLink
);
1479 IODelete(owner
, IOUserClientOwner
, 1);
1481 owners
.next
= owners
.prev
= NULL
;
1484 IOLockUnlock(gIOUserClientOwnersLock
);
1487 extern "C" kern_return_t
1488 iokit_task_terminate(task_t task
)
1490 IOUserClientOwner
* owner
;
1491 IOUserClient
* dead
;
1493 queue_head_t
* taskque
;
1495 IOLockLock(gIOUserClientOwnersLock
);
1497 taskque
= task_io_user_clients(task
);
1499 while (!queue_empty(taskque
))
1501 owner
= (IOUserClientOwner
*)(void *) queue_first(taskque
);
1503 queue_remove(taskque
, owner
, IOUserClientOwner
*, taskLink
);
1504 queue_remove(&uc
->owners
, owner
, IOUserClientOwner
*, ucLink
);
1505 if (queue_empty(&uc
->owners
))
1508 IOLog("destroying out of band connect for %s\n", uc
->getName());
1509 // now using the uc queue head as a singly linked queue,
1510 // leaving .next as NULL to mark it empty
1511 uc
->owners
.next
= NULL
;
1512 uc
->owners
.prev
= (queue_entry_t
) dead
;
1515 IODelete(owner
, IOUserClientOwner
, 1);
1518 IOLockUnlock(gIOUserClientOwnersLock
);
1523 dead
= (IOUserClient
*)(void *) dead
->owners
.prev
;
1524 uc
->owners
.prev
= NULL
;
1525 if (uc
->sharedInstance
|| !uc
->closed
) uc
->clientDied();
1529 return (KERN_SUCCESS
);
1532 void IOUserClient::free()
1534 if( mappings
) mappings
->release();
1536 IOStatisticsUnregisterCounter();
1538 assert(!owners
.next
);
1539 assert(!owners
.prev
);
1541 if (reserved
) IODelete(reserved
, ExpansionData
, 1);
1546 IOReturn
IOUserClient::clientDied( void )
1548 IOReturn ret
= kIOReturnNotReady
;
1550 if (sharedInstance
|| OSCompareAndSwap8(0, 1, &closed
))
1552 ret
= clientClose();
1558 IOReturn
IOUserClient::clientClose( void )
1560 return( kIOReturnUnsupported
);
1563 IOService
* IOUserClient::getService( void )
1568 IOReturn
IOUserClient::registerNotificationPort(
1569 mach_port_t
/* port */,
1571 UInt32
/* refCon */)
1573 return( kIOReturnUnsupported
);
1576 IOReturn
IOUserClient::registerNotificationPort(
1579 io_user_reference_t refCon
)
1581 return (registerNotificationPort(port
, type
, (UInt32
) refCon
));
1584 IOReturn
IOUserClient::getNotificationSemaphore( UInt32 notification_type
,
1585 semaphore_t
* semaphore
)
1587 return( kIOReturnUnsupported
);
1590 IOReturn
IOUserClient::connectClient( IOUserClient
* /* client */ )
1592 return( kIOReturnUnsupported
);
1595 IOReturn
IOUserClient::clientMemoryForType( UInt32 type
,
1596 IOOptionBits
* options
,
1597 IOMemoryDescriptor
** memory
)
1599 return( kIOReturnUnsupported
);
1603 IOMemoryMap
* IOUserClient::mapClientMemory(
1606 IOOptionBits mapFlags
,
1607 IOVirtualAddress atAddress
)
1613 IOMemoryMap
* IOUserClient::mapClientMemory64(
1616 IOOptionBits mapFlags
,
1617 mach_vm_address_t atAddress
)
1620 IOOptionBits options
= 0;
1621 IOMemoryDescriptor
* memory
= 0;
1622 IOMemoryMap
* map
= 0;
1624 err
= clientMemoryForType( (UInt32
) type
, &options
, &memory
);
1626 if( memory
&& (kIOReturnSuccess
== err
)) {
1628 FAKE_STACK_FRAME(getMetaClass());
1630 options
= (options
& ~kIOMapUserOptionsMask
)
1631 | (mapFlags
& kIOMapUserOptionsMask
);
1632 map
= memory
->createMappingInTask( task
, atAddress
, options
);
1635 FAKE_STACK_FRAME_END();
1641 IOReturn
IOUserClient::exportObjectToClient(task_t task
,
1642 OSObject
*obj
, io_object_t
*clientObj
)
1644 mach_port_name_t name
;
1646 name
= IOMachPort::makeSendRightForTask( task
, obj
, IKOT_IOKIT_OBJECT
);
1648 *(mach_port_name_t
*)clientObj
= name
;
1649 return kIOReturnSuccess
;
1652 IOExternalMethod
* IOUserClient::getExternalMethodForIndex( UInt32
/* index */)
1657 IOExternalAsyncMethod
* IOUserClient::getExternalAsyncMethodForIndex( UInt32
/* index */)
1662 IOExternalTrap
* IOUserClient::
1663 getExternalTrapForIndex(UInt32 index
)
1668 #pragma clang diagnostic push
1669 #pragma clang diagnostic ignored "-Wdeprecated-declarations"
1671 // Suppressing the deprecated-declarations warning. Avoiding the use of deprecated
1672 // functions can break clients of kexts implementing getExternalMethodForIndex()
1673 IOExternalMethod
* IOUserClient::
1674 getTargetAndMethodForIndex(IOService
**targetP
, UInt32 index
)
1676 IOExternalMethod
*method
= getExternalMethodForIndex(index
);
1679 *targetP
= (IOService
*) method
->object
;
1684 IOExternalAsyncMethod
* IOUserClient::
1685 getAsyncTargetAndMethodForIndex(IOService
** targetP
, UInt32 index
)
1687 IOExternalAsyncMethod
*method
= getExternalAsyncMethodForIndex(index
);
1690 *targetP
= (IOService
*) method
->object
;
1695 IOExternalTrap
* IOUserClient::
1696 getTargetAndTrapForIndex(IOService
** targetP
, UInt32 index
)
1698 IOExternalTrap
*trap
= getExternalTrapForIndex(index
);
1701 *targetP
= trap
->object
;
1706 #pragma clang diagnostic pop
1708 IOReturn
IOUserClient::releaseAsyncReference64(OSAsyncReference64 reference
)
1711 port
= (mach_port_t
) (reference
[0] & ~kIOUCAsync0Flags
);
1713 if (MACH_PORT_NULL
!= port
)
1714 iokit_release_port_send(port
);
1716 return (kIOReturnSuccess
);
1719 IOReturn
IOUserClient::releaseNotificationPort(mach_port_t port
)
1721 if (MACH_PORT_NULL
!= port
)
1722 iokit_release_port_send(port
);
1724 return (kIOReturnSuccess
);
1727 IOReturn
IOUserClient::sendAsyncResult(OSAsyncReference reference
,
1728 IOReturn result
, void *args
[], UInt32 numArgs
)
1730 OSAsyncReference64 reference64
;
1731 io_user_reference_t args64
[kMaxAsyncArgs
];
1734 if (numArgs
> kMaxAsyncArgs
)
1735 return kIOReturnMessageTooLarge
;
1737 for (idx
= 0; idx
< kOSAsyncRef64Count
; idx
++)
1738 reference64
[idx
] = REF64(reference
[idx
]);
1740 for (idx
= 0; idx
< numArgs
; idx
++)
1741 args64
[idx
] = REF64(args
[idx
]);
1743 return (sendAsyncResult64(reference64
, result
, args64
, numArgs
));
1746 IOReturn
IOUserClient::sendAsyncResult64WithOptions(OSAsyncReference64 reference
,
1747 IOReturn result
, io_user_reference_t args
[], UInt32 numArgs
, IOOptionBits options
)
1749 return _sendAsyncResult64(reference
, result
, args
, numArgs
, options
);
1752 IOReturn
IOUserClient::sendAsyncResult64(OSAsyncReference64 reference
,
1753 IOReturn result
, io_user_reference_t args
[], UInt32 numArgs
)
1755 return _sendAsyncResult64(reference
, result
, args
, numArgs
, 0);
1758 IOReturn
IOUserClient::_sendAsyncResult64(OSAsyncReference64 reference
,
1759 IOReturn result
, io_user_reference_t args
[], UInt32 numArgs
, IOOptionBits options
)
1763 mach_msg_header_t msgHdr
;
1768 OSNotificationHeader notifyHdr
;
1769 IOAsyncCompletionContent asyncContent
;
1770 uint32_t args
[kMaxAsyncArgs
];
1774 OSNotificationHeader64 notifyHdr
;
1775 IOAsyncCompletionContent asyncContent
;
1776 io_user_reference_t args
[kMaxAsyncArgs
] __attribute__ ((packed
));
1781 mach_port_t replyPort
;
1784 // If no reply port, do nothing.
1785 replyPort
= (mach_port_t
) (reference
[0] & ~kIOUCAsync0Flags
);
1786 if (replyPort
== MACH_PORT_NULL
)
1787 return kIOReturnSuccess
;
1789 if (numArgs
> kMaxAsyncArgs
)
1790 return kIOReturnMessageTooLarge
;
1792 bzero(&replyMsg
, sizeof(replyMsg
));
1793 replyMsg
.msgHdr
.msgh_bits
= MACH_MSGH_BITS(MACH_MSG_TYPE_COPY_SEND
/*remote*/,
1795 replyMsg
.msgHdr
.msgh_remote_port
= replyPort
;
1796 replyMsg
.msgHdr
.msgh_local_port
= 0;
1797 replyMsg
.msgHdr
.msgh_id
= kOSNotificationMessageID
;
1798 if (kIOUCAsync64Flag
& reference
[0])
1800 replyMsg
.msgHdr
.msgh_size
=
1801 sizeof(replyMsg
.msgHdr
) + sizeof(replyMsg
.m
.msg64
)
1802 - (kMaxAsyncArgs
- numArgs
) * sizeof(io_user_reference_t
);
1803 replyMsg
.m
.msg64
.notifyHdr
.size
= sizeof(IOAsyncCompletionContent
)
1804 + numArgs
* sizeof(io_user_reference_t
);
1805 replyMsg
.m
.msg64
.notifyHdr
.type
= kIOAsyncCompletionNotificationType
;
1806 bcopy(reference
, replyMsg
.m
.msg64
.notifyHdr
.reference
, sizeof(OSAsyncReference64
));
1808 replyMsg
.m
.msg64
.asyncContent
.result
= result
;
1810 bcopy(args
, replyMsg
.m
.msg64
.args
, numArgs
* sizeof(io_user_reference_t
));
1816 replyMsg
.msgHdr
.msgh_size
=
1817 sizeof(replyMsg
.msgHdr
) + sizeof(replyMsg
.m
.msg32
)
1818 - (kMaxAsyncArgs
- numArgs
) * sizeof(uint32_t);
1820 replyMsg
.m
.msg32
.notifyHdr
.size
= sizeof(IOAsyncCompletionContent
)
1821 + numArgs
* sizeof(uint32_t);
1822 replyMsg
.m
.msg32
.notifyHdr
.type
= kIOAsyncCompletionNotificationType
;
1824 for (idx
= 0; idx
< kOSAsyncRefCount
; idx
++)
1825 replyMsg
.m
.msg32
.notifyHdr
.reference
[idx
] = REF32(reference
[idx
]);
1827 replyMsg
.m
.msg32
.asyncContent
.result
= result
;
1829 for (idx
= 0; idx
< numArgs
; idx
++)
1830 replyMsg
.m
.msg32
.args
[idx
] = REF32(args
[idx
]);
1833 if ((options
& kIOUserNotifyOptionCanDrop
) != 0) {
1834 kr
= mach_msg_send_from_kernel_with_options( &replyMsg
.msgHdr
,
1835 replyMsg
.msgHdr
.msgh_size
, MACH_SEND_TIMEOUT
, MACH_MSG_TIMEOUT_NONE
);
1837 /* Fail on full queue. */
1838 kr
= mach_msg_send_from_kernel_proper( &replyMsg
.msgHdr
,
1839 replyMsg
.msgHdr
.msgh_size
);
1841 if ((KERN_SUCCESS
!= kr
) && (MACH_SEND_TIMED_OUT
!= kr
) && !(kIOUCAsyncErrorLoggedFlag
& reference
[0]))
1843 reference
[0] |= kIOUCAsyncErrorLoggedFlag
;
1844 IOLog("%s: mach_msg_send_from_kernel_proper(0x%x)\n", __PRETTY_FUNCTION__
, kr
);
1850 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1854 #define CHECK(cls,obj,out) \
1856 if( !(out = OSDynamicCast( cls, obj))) \
1857 return( kIOReturnBadArgument )
1859 #define CHECKLOCKED(cls,obj,out) \
1860 IOUserIterator * oIter; \
1862 if( !(oIter = OSDynamicCast(IOUserIterator, obj))) \
1863 return (kIOReturnBadArgument); \
1864 if( !(out = OSDynamicCast(cls, oIter->userIteratorObject))) \
1865 return (kIOReturnBadArgument)
1867 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1869 // Create a vm_map_copy_t or kalloc'ed data for memory
1870 // to be copied out. ipc will free after the copyout.
1872 static kern_return_t
copyoutkdata( const void * data
, vm_size_t len
,
1873 io_buf_ptr_t
* buf
)
1878 err
= vm_map_copyin( kernel_map
, CAST_USER_ADDR_T(data
), len
,
1879 false /* src_destroy */, ©
);
1881 assert( err
== KERN_SUCCESS
);
1882 if( err
== KERN_SUCCESS
)
1883 *buf
= (char *) copy
;
1888 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1890 /* Routine io_server_version */
1891 kern_return_t
is_io_server_version(
1892 mach_port_t master_port
,
1895 *version
= IOKIT_SERVER_VERSION
;
1896 return (kIOReturnSuccess
);
1899 /* Routine io_object_get_class */
1900 kern_return_t
is_io_object_get_class(
1902 io_name_t className
)
1904 const OSMetaClass
* my_obj
= NULL
;
1907 return( kIOReturnBadArgument
);
1909 my_obj
= object
->getMetaClass();
1911 return (kIOReturnNotFound
);
1914 strlcpy( className
, my_obj
->getClassName(), sizeof(io_name_t
));
1916 return( kIOReturnSuccess
);
1919 /* Routine io_object_get_superclass */
1920 kern_return_t
is_io_object_get_superclass(
1921 mach_port_t master_port
,
1923 io_name_t class_name
)
1925 const OSMetaClass
* my_obj
= NULL
;
1926 const OSMetaClass
* superclass
= NULL
;
1927 const OSSymbol
*my_name
= NULL
;
1928 const char *my_cstr
= NULL
;
1930 if (!obj_name
|| !class_name
)
1931 return (kIOReturnBadArgument
);
1933 if( master_port
!= master_device_port
)
1934 return( kIOReturnNotPrivileged
);
1936 my_name
= OSSymbol::withCString(obj_name
);
1939 my_obj
= OSMetaClass::getMetaClassWithName(my_name
);
1943 superclass
= my_obj
->getSuperClass();
1947 return( kIOReturnNotFound
);
1950 my_cstr
= superclass
->getClassName();
1953 strlcpy(class_name
, my_cstr
, sizeof(io_name_t
));
1954 return( kIOReturnSuccess
);
1956 return (kIOReturnNotFound
);
1959 /* Routine io_object_get_bundle_identifier */
1960 kern_return_t
is_io_object_get_bundle_identifier(
1961 mach_port_t master_port
,
1963 io_name_t bundle_name
)
1965 const OSMetaClass
* my_obj
= NULL
;
1966 const OSSymbol
*my_name
= NULL
;
1967 const OSSymbol
*identifier
= NULL
;
1968 const char *my_cstr
= NULL
;
1970 if (!obj_name
|| !bundle_name
)
1971 return (kIOReturnBadArgument
);
1973 if( master_port
!= master_device_port
)
1974 return( kIOReturnNotPrivileged
);
1976 my_name
= OSSymbol::withCString(obj_name
);
1979 my_obj
= OSMetaClass::getMetaClassWithName(my_name
);
1984 identifier
= my_obj
->getKmodName();
1987 return( kIOReturnNotFound
);
1990 my_cstr
= identifier
->getCStringNoCopy();
1992 strlcpy(bundle_name
, identifier
->getCStringNoCopy(), sizeof(io_name_t
));
1993 return( kIOReturnSuccess
);
1996 return (kIOReturnBadArgument
);
1999 /* Routine io_object_conforms_to */
2000 kern_return_t
is_io_object_conforms_to(
2002 io_name_t className
,
2003 boolean_t
*conforms
)
2006 return( kIOReturnBadArgument
);
2008 *conforms
= (0 != object
->metaCast( className
));
2010 return( kIOReturnSuccess
);
2013 /* Routine io_object_get_retain_count */
2014 kern_return_t
is_io_object_get_retain_count(
2016 uint32_t *retainCount
)
2019 return( kIOReturnBadArgument
);
2021 *retainCount
= object
->getRetainCount();
2022 return( kIOReturnSuccess
);
2025 /* Routine io_iterator_next */
2026 kern_return_t
is_io_iterator_next(
2027 io_object_t iterator
,
2028 io_object_t
*object
)
2033 CHECK( OSIterator
, iterator
, iter
);
2035 obj
= iter
->getNextObject();
2039 ret
= kIOReturnSuccess
;
2041 ret
= kIOReturnNoDevice
;
2046 /* Routine io_iterator_reset */
2047 kern_return_t
is_io_iterator_reset(
2048 io_object_t iterator
)
2050 CHECK( OSIterator
, iterator
, iter
);
2054 return( kIOReturnSuccess
);
2057 /* Routine io_iterator_is_valid */
2058 kern_return_t
is_io_iterator_is_valid(
2059 io_object_t iterator
,
2060 boolean_t
*is_valid
)
2062 CHECK( OSIterator
, iterator
, iter
);
2064 *is_valid
= iter
->isValid();
2066 return( kIOReturnSuccess
);
2070 static kern_return_t
internal_io_service_match_property_table(
2071 io_service_t _service
,
2072 const char * matching
,
2073 mach_msg_type_number_t matching_size
,
2076 CHECK( IOService
, _service
, service
);
2080 OSDictionary
* dict
;
2082 assert(matching_size
);
2083 obj
= OSUnserializeXML(matching
, matching_size
);
2085 if( (dict
= OSDynamicCast( OSDictionary
, obj
))) {
2086 *matches
= service
->passiveMatch( dict
);
2087 kr
= kIOReturnSuccess
;
2089 kr
= kIOReturnBadArgument
;
2097 /* Routine io_service_match_property_table */
2098 kern_return_t
is_io_service_match_property_table(
2099 io_service_t service
,
2100 io_string_t matching
,
2101 boolean_t
*matches
)
2103 return (kIOReturnUnsupported
);
2107 /* Routine io_service_match_property_table_ool */
2108 kern_return_t
is_io_service_match_property_table_ool(
2109 io_object_t service
,
2110 io_buf_ptr_t matching
,
2111 mach_msg_type_number_t matchingCnt
,
2112 kern_return_t
*result
,
2113 boolean_t
*matches
)
2117 vm_map_offset_t map_data
;
2119 kr
= vm_map_copyout( kernel_map
, &map_data
, (vm_map_copy_t
) matching
);
2120 data
= CAST_DOWN(vm_offset_t
, map_data
);
2122 if( KERN_SUCCESS
== kr
) {
2123 // must return success after vm_map_copyout() succeeds
2124 *result
= internal_io_service_match_property_table(service
,
2125 (const char *)data
, matchingCnt
, matches
);
2126 vm_deallocate( kernel_map
, data
, matchingCnt
);
2132 /* Routine io_service_match_property_table_bin */
2133 kern_return_t
is_io_service_match_property_table_bin(
2134 io_object_t service
,
2135 io_struct_inband_t matching
,
2136 mach_msg_type_number_t matchingCnt
,
2139 return (internal_io_service_match_property_table(service
, matching
, matchingCnt
, matches
));
2142 static kern_return_t
internal_io_service_get_matching_services(
2143 mach_port_t master_port
,
2144 const char * matching
,
2145 mach_msg_type_number_t matching_size
,
2146 io_iterator_t
*existing
)
2150 OSDictionary
* dict
;
2152 if( master_port
!= master_device_port
)
2153 return( kIOReturnNotPrivileged
);
2155 assert(matching_size
);
2156 obj
= OSUnserializeXML(matching
, matching_size
);
2158 if( (dict
= OSDynamicCast( OSDictionary
, obj
))) {
2159 *existing
= IOUserIterator::withIterator(IOService::getMatchingServices( dict
));
2160 kr
= kIOReturnSuccess
;
2162 kr
= kIOReturnBadArgument
;
2170 /* Routine io_service_get_matching_services */
2171 kern_return_t
is_io_service_get_matching_services(
2172 mach_port_t master_port
,
2173 io_string_t matching
,
2174 io_iterator_t
*existing
)
2176 return (kIOReturnUnsupported
);
2179 /* Routine io_service_get_matching_services_ool */
2180 kern_return_t
is_io_service_get_matching_services_ool(
2181 mach_port_t master_port
,
2182 io_buf_ptr_t matching
,
2183 mach_msg_type_number_t matchingCnt
,
2184 kern_return_t
*result
,
2185 io_object_t
*existing
)
2189 vm_map_offset_t map_data
;
2191 kr
= vm_map_copyout( kernel_map
, &map_data
, (vm_map_copy_t
) matching
);
2192 data
= CAST_DOWN(vm_offset_t
, map_data
);
2194 if( KERN_SUCCESS
== kr
) {
2195 // must return success after vm_map_copyout() succeeds
2196 // and mig will copy out objects on success
2198 *result
= internal_io_service_get_matching_services(master_port
,
2199 (const char *) data
, matchingCnt
, existing
);
2200 vm_deallocate( kernel_map
, data
, matchingCnt
);
2206 /* Routine io_service_get_matching_services_bin */
2207 kern_return_t
is_io_service_get_matching_services_bin(
2208 mach_port_t master_port
,
2209 io_struct_inband_t matching
,
2210 mach_msg_type_number_t matchingCnt
,
2211 io_object_t
*existing
)
2213 return (internal_io_service_get_matching_services(master_port
, matching
, matchingCnt
, existing
));
2217 static kern_return_t
internal_io_service_get_matching_service(
2218 mach_port_t master_port
,
2219 const char * matching
,
2220 mach_msg_type_number_t matching_size
,
2221 io_service_t
*service
)
2225 OSDictionary
* dict
;
2227 if( master_port
!= master_device_port
)
2228 return( kIOReturnNotPrivileged
);
2230 assert(matching_size
);
2231 obj
= OSUnserializeXML(matching
, matching_size
);
2233 if( (dict
= OSDynamicCast( OSDictionary
, obj
))) {
2234 *service
= IOService::copyMatchingService( dict
);
2235 kr
= *service
? kIOReturnSuccess
: kIOReturnNotFound
;
2237 kr
= kIOReturnBadArgument
;
2245 /* Routine io_service_get_matching_service */
2246 kern_return_t
is_io_service_get_matching_service(
2247 mach_port_t master_port
,
2248 io_string_t matching
,
2249 io_service_t
*service
)
2251 return (kIOReturnUnsupported
);
2254 /* Routine io_service_get_matching_services_ool */
2255 kern_return_t
is_io_service_get_matching_service_ool(
2256 mach_port_t master_port
,
2257 io_buf_ptr_t matching
,
2258 mach_msg_type_number_t matchingCnt
,
2259 kern_return_t
*result
,
2260 io_object_t
*service
)
2264 vm_map_offset_t map_data
;
2266 kr
= vm_map_copyout( kernel_map
, &map_data
, (vm_map_copy_t
) matching
);
2267 data
= CAST_DOWN(vm_offset_t
, map_data
);
2269 if( KERN_SUCCESS
== kr
) {
2270 // must return success after vm_map_copyout() succeeds
2271 // and mig will copy out objects on success
2273 *result
= internal_io_service_get_matching_service(master_port
,
2274 (const char *) data
, matchingCnt
, service
);
2275 vm_deallocate( kernel_map
, data
, matchingCnt
);
2281 /* Routine io_service_get_matching_service_bin */
2282 kern_return_t
is_io_service_get_matching_service_bin(
2283 mach_port_t master_port
,
2284 io_struct_inband_t matching
,
2285 mach_msg_type_number_t matchingCnt
,
2286 io_object_t
*service
)
2288 return (internal_io_service_get_matching_service(master_port
, matching
, matchingCnt
, service
));
2291 static kern_return_t
internal_io_service_add_notification(
2292 mach_port_t master_port
,
2293 io_name_t notification_type
,
2294 const char * matching
,
2295 size_t matching_size
,
2298 vm_size_t referenceSize
,
2300 io_object_t
* notification
)
2302 IOServiceUserNotification
* userNotify
= 0;
2303 IONotifier
* notify
= 0;
2304 const OSSymbol
* sym
;
2305 OSDictionary
* dict
;
2307 unsigned long int userMsgType
;
2309 if( master_port
!= master_device_port
)
2310 return( kIOReturnNotPrivileged
);
2313 err
= kIOReturnNoResources
;
2315 if( !(sym
= OSSymbol::withCString( notification_type
)))
2316 err
= kIOReturnNoResources
;
2318 assert(matching_size
);
2319 dict
= OSDynamicCast(OSDictionary
, OSUnserializeXML(matching
, matching_size
));
2321 err
= kIOReturnBadArgument
;
2325 if( (sym
== gIOPublishNotification
)
2326 || (sym
== gIOFirstPublishNotification
))
2327 userMsgType
= kIOServicePublishNotificationType
;
2328 else if( (sym
== gIOMatchedNotification
)
2329 || (sym
== gIOFirstMatchNotification
))
2330 userMsgType
= kIOServiceMatchedNotificationType
;
2331 else if( sym
== gIOTerminatedNotification
)
2332 userMsgType
= kIOServiceTerminatedNotificationType
;
2334 userMsgType
= kLastIOKitNotificationType
;
2336 userNotify
= new IOServiceUserNotification
;
2338 if( userNotify
&& !userNotify
->init( port
, userMsgType
,
2339 reference
, referenceSize
, client64
)) {
2340 iokit_release_port_send(port
);
2341 userNotify
->release();
2347 notify
= IOService::addMatchingNotification( sym
, dict
,
2348 &userNotify
->_handler
, userNotify
);
2350 *notification
= userNotify
;
2351 userNotify
->setNotification( notify
);
2352 err
= kIOReturnSuccess
;
2354 err
= kIOReturnUnsupported
;
2367 /* Routine io_service_add_notification */
2368 kern_return_t
is_io_service_add_notification(
2369 mach_port_t master_port
,
2370 io_name_t notification_type
,
2371 io_string_t matching
,
2373 io_async_ref_t reference
,
2374 mach_msg_type_number_t referenceCnt
,
2375 io_object_t
* notification
)
2377 return (kIOReturnUnsupported
);
2380 /* Routine io_service_add_notification_64 */
2381 kern_return_t
is_io_service_add_notification_64(
2382 mach_port_t master_port
,
2383 io_name_t notification_type
,
2384 io_string_t matching
,
2385 mach_port_t wake_port
,
2386 io_async_ref64_t reference
,
2387 mach_msg_type_number_t referenceCnt
,
2388 io_object_t
*notification
)
2390 return (kIOReturnUnsupported
);
2393 /* Routine io_service_add_notification_bin */
2394 kern_return_t is_io_service_add_notification_bin
2396 mach_port_t master_port
,
2397 io_name_t notification_type
,
2398 io_struct_inband_t matching
,
2399 mach_msg_type_number_t matchingCnt
,
2400 mach_port_t wake_port
,
2401 io_async_ref_t reference
,
2402 mach_msg_type_number_t referenceCnt
,
2403 io_object_t
*notification
)
2405 return (internal_io_service_add_notification(master_port
, notification_type
,
2406 matching
, matchingCnt
, wake_port
, &reference
[0], sizeof(io_async_ref_t
),
2407 false, notification
));
2410 /* Routine io_service_add_notification_bin_64 */
2411 kern_return_t is_io_service_add_notification_bin_64
2413 mach_port_t master_port
,
2414 io_name_t notification_type
,
2415 io_struct_inband_t matching
,
2416 mach_msg_type_number_t matchingCnt
,
2417 mach_port_t wake_port
,
2418 io_async_ref64_t reference
,
2419 mach_msg_type_number_t referenceCnt
,
2420 io_object_t
*notification
)
2422 return (internal_io_service_add_notification(master_port
, notification_type
,
2423 matching
, matchingCnt
, wake_port
, &reference
[0], sizeof(io_async_ref64_t
),
2424 true, notification
));
2427 static kern_return_t
internal_io_service_add_notification_ool(
2428 mach_port_t master_port
,
2429 io_name_t notification_type
,
2430 io_buf_ptr_t matching
,
2431 mach_msg_type_number_t matchingCnt
,
2432 mach_port_t wake_port
,
2434 vm_size_t referenceSize
,
2436 kern_return_t
*result
,
2437 io_object_t
*notification
)
2441 vm_map_offset_t map_data
;
2443 kr
= vm_map_copyout( kernel_map
, &map_data
, (vm_map_copy_t
) matching
);
2444 data
= CAST_DOWN(vm_offset_t
, map_data
);
2446 if( KERN_SUCCESS
== kr
) {
2447 // must return success after vm_map_copyout() succeeds
2448 // and mig will copy out objects on success
2450 *result
= internal_io_service_add_notification( master_port
, notification_type
,
2451 (char *) data
, matchingCnt
, wake_port
, reference
, referenceSize
, client64
, notification
);
2452 vm_deallocate( kernel_map
, data
, matchingCnt
);
2458 /* Routine io_service_add_notification_ool */
2459 kern_return_t
is_io_service_add_notification_ool(
2460 mach_port_t master_port
,
2461 io_name_t notification_type
,
2462 io_buf_ptr_t matching
,
2463 mach_msg_type_number_t matchingCnt
,
2464 mach_port_t wake_port
,
2465 io_async_ref_t reference
,
2466 mach_msg_type_number_t referenceCnt
,
2467 kern_return_t
*result
,
2468 io_object_t
*notification
)
2470 return (internal_io_service_add_notification_ool(master_port
, notification_type
,
2471 matching
, matchingCnt
, wake_port
, &reference
[0], sizeof(io_async_ref_t
),
2472 false, result
, notification
));
2475 /* Routine io_service_add_notification_ool_64 */
2476 kern_return_t
is_io_service_add_notification_ool_64(
2477 mach_port_t master_port
,
2478 io_name_t notification_type
,
2479 io_buf_ptr_t matching
,
2480 mach_msg_type_number_t matchingCnt
,
2481 mach_port_t wake_port
,
2482 io_async_ref64_t reference
,
2483 mach_msg_type_number_t referenceCnt
,
2484 kern_return_t
*result
,
2485 io_object_t
*notification
)
2487 return (internal_io_service_add_notification_ool(master_port
, notification_type
,
2488 matching
, matchingCnt
, wake_port
, &reference
[0], sizeof(io_async_ref64_t
),
2489 true, result
, notification
));
2492 /* Routine io_service_add_notification_old */
2493 kern_return_t
is_io_service_add_notification_old(
2494 mach_port_t master_port
,
2495 io_name_t notification_type
,
2496 io_string_t matching
,
2498 // for binary compatibility reasons, this must be natural_t for ILP32
2500 io_object_t
* notification
)
2502 return( is_io_service_add_notification( master_port
, notification_type
,
2503 matching
, port
, &ref
, 1, notification
));
2507 static kern_return_t
internal_io_service_add_interest_notification(
2508 io_object_t _service
,
2509 io_name_t type_of_interest
,
2512 vm_size_t referenceSize
,
2514 io_object_t
* notification
)
2517 IOServiceMessageUserNotification
* userNotify
= 0;
2518 IONotifier
* notify
= 0;
2519 const OSSymbol
* sym
;
2522 CHECK( IOService
, _service
, service
);
2524 err
= kIOReturnNoResources
;
2525 if( (sym
= OSSymbol::withCString( type_of_interest
))) do {
2527 userNotify
= new IOServiceMessageUserNotification
;
2529 if( userNotify
&& !userNotify
->init( port
, kIOServiceMessageNotificationType
,
2530 reference
, referenceSize
,
2531 kIOUserNotifyMaxMessageSize
,
2533 iokit_release_port_send(port
);
2534 userNotify
->release();
2540 notify
= service
->registerInterest( sym
,
2541 &userNotify
->_handler
, userNotify
);
2543 *notification
= userNotify
;
2544 userNotify
->setNotification( notify
);
2545 err
= kIOReturnSuccess
;
2547 err
= kIOReturnUnsupported
;
2556 /* Routine io_service_add_message_notification */
2557 kern_return_t
is_io_service_add_interest_notification(
2558 io_object_t service
,
2559 io_name_t type_of_interest
,
2561 io_async_ref_t reference
,
2562 mach_msg_type_number_t referenceCnt
,
2563 io_object_t
* notification
)
2565 return (internal_io_service_add_interest_notification(service
, type_of_interest
,
2566 port
, &reference
[0], sizeof(io_async_ref_t
), false, notification
));
2569 /* Routine io_service_add_interest_notification_64 */
2570 kern_return_t
is_io_service_add_interest_notification_64(
2571 io_object_t service
,
2572 io_name_t type_of_interest
,
2573 mach_port_t wake_port
,
2574 io_async_ref64_t reference
,
2575 mach_msg_type_number_t referenceCnt
,
2576 io_object_t
*notification
)
2578 return (internal_io_service_add_interest_notification(service
, type_of_interest
,
2579 wake_port
, &reference
[0], sizeof(io_async_ref64_t
), true, notification
));
2583 /* Routine io_service_acknowledge_notification */
2584 kern_return_t
is_io_service_acknowledge_notification(
2585 io_object_t _service
,
2586 natural_t notify_ref
,
2587 natural_t response
)
2589 CHECK( IOService
, _service
, service
);
2591 return( service
->acknowledgeNotification( (IONotificationRef
)(uintptr_t) notify_ref
,
2592 (IOOptionBits
) response
));
2596 /* Routine io_connect_get_semaphore */
2597 kern_return_t
is_io_connect_get_notification_semaphore(
2598 io_connect_t connection
,
2599 natural_t notification_type
,
2600 semaphore_t
*semaphore
)
2602 CHECK( IOUserClient
, connection
, client
);
2604 IOStatisticsClientCall();
2605 return( client
->getNotificationSemaphore( (UInt32
) notification_type
,
2609 /* Routine io_registry_get_root_entry */
2610 kern_return_t
is_io_registry_get_root_entry(
2611 mach_port_t master_port
,
2614 IORegistryEntry
* entry
;
2616 if( master_port
!= master_device_port
)
2617 return( kIOReturnNotPrivileged
);
2619 entry
= IORegistryEntry::getRegistryRoot();
2624 return( kIOReturnSuccess
);
2627 /* Routine io_registry_create_iterator */
2628 kern_return_t
is_io_registry_create_iterator(
2629 mach_port_t master_port
,
2632 io_object_t
*iterator
)
2634 if( master_port
!= master_device_port
)
2635 return( kIOReturnNotPrivileged
);
2637 *iterator
= IOUserIterator::withIterator(
2638 IORegistryIterator::iterateOver(
2639 IORegistryEntry::getPlane( plane
), options
));
2641 return( *iterator
? kIOReturnSuccess
: kIOReturnBadArgument
);
2644 /* Routine io_registry_entry_create_iterator */
2645 kern_return_t
is_io_registry_entry_create_iterator(
2646 io_object_t registry_entry
,
2649 io_object_t
*iterator
)
2651 CHECK( IORegistryEntry
, registry_entry
, entry
);
2653 *iterator
= IOUserIterator::withIterator(
2654 IORegistryIterator::iterateOver( entry
,
2655 IORegistryEntry::getPlane( plane
), options
));
2657 return( *iterator
? kIOReturnSuccess
: kIOReturnBadArgument
);
2660 /* Routine io_registry_iterator_enter */
2661 kern_return_t
is_io_registry_iterator_enter_entry(
2662 io_object_t iterator
)
2664 CHECKLOCKED( IORegistryIterator
, iterator
, iter
);
2666 IOLockLock(oIter
->lock
);
2668 IOLockUnlock(oIter
->lock
);
2670 return( kIOReturnSuccess
);
2673 /* Routine io_registry_iterator_exit */
2674 kern_return_t
is_io_registry_iterator_exit_entry(
2675 io_object_t iterator
)
2679 CHECKLOCKED( IORegistryIterator
, iterator
, iter
);
2681 IOLockLock(oIter
->lock
);
2682 didIt
= iter
->exitEntry();
2683 IOLockUnlock(oIter
->lock
);
2685 return( didIt
? kIOReturnSuccess
: kIOReturnNoDevice
);
2688 /* Routine io_registry_entry_from_path */
2689 kern_return_t
is_io_registry_entry_from_path(
2690 mach_port_t master_port
,
2692 io_object_t
*registry_entry
)
2694 IORegistryEntry
* entry
;
2696 if( master_port
!= master_device_port
)
2697 return( kIOReturnNotPrivileged
);
2699 entry
= IORegistryEntry::fromPath( path
);
2701 *registry_entry
= entry
;
2703 return( kIOReturnSuccess
);
2707 /* Routine io_registry_entry_from_path */
2708 kern_return_t
is_io_registry_entry_from_path_ool(
2709 mach_port_t master_port
,
2710 io_string_inband_t path
,
2711 io_buf_ptr_t path_ool
,
2712 mach_msg_type_number_t path_oolCnt
,
2713 kern_return_t
*result
,
2714 io_object_t
*registry_entry
)
2716 IORegistryEntry
* entry
;
2717 vm_map_offset_t map_data
;
2722 if (master_port
!= master_device_port
) return(kIOReturnNotPrivileged
);
2726 res
= err
= KERN_SUCCESS
;
2727 if (path
[0]) cpath
= path
;
2730 if (!path_oolCnt
) return(kIOReturnBadArgument
);
2731 if (path_oolCnt
> (sizeof(io_struct_inband_t
) * 1024)) return(kIOReturnMessageTooLarge
);
2733 err
= vm_map_copyout(kernel_map
, &map_data
, (vm_map_copy_t
) path_ool
);
2734 if (KERN_SUCCESS
== err
)
2736 // must return success to mig after vm_map_copyout() succeeds, so result is actual
2737 cpath
= CAST_DOWN(const char *, map_data
);
2738 if (cpath
[path_oolCnt
- 1]) res
= kIOReturnBadArgument
;
2742 if ((KERN_SUCCESS
== err
) && (KERN_SUCCESS
== res
))
2744 entry
= IORegistryEntry::fromPath(cpath
);
2745 res
= entry
? kIOReturnSuccess
: kIOReturnNotFound
;
2748 if (map_data
) vm_deallocate(kernel_map
, map_data
, path_oolCnt
);
2750 if (KERN_SUCCESS
!= err
) res
= err
;
2751 *registry_entry
= entry
;
2758 /* Routine io_registry_entry_in_plane */
2759 kern_return_t
is_io_registry_entry_in_plane(
2760 io_object_t registry_entry
,
2762 boolean_t
*inPlane
)
2764 CHECK( IORegistryEntry
, registry_entry
, entry
);
2766 *inPlane
= entry
->inPlane( IORegistryEntry::getPlane( plane
));
2768 return( kIOReturnSuccess
);
2772 /* Routine io_registry_entry_get_path */
2773 kern_return_t
is_io_registry_entry_get_path(
2774 io_object_t registry_entry
,
2779 CHECK( IORegistryEntry
, registry_entry
, entry
);
2781 length
= sizeof( io_string_t
);
2782 if( entry
->getPath( path
, &length
, IORegistryEntry::getPlane( plane
)))
2783 return( kIOReturnSuccess
);
2785 return( kIOReturnBadArgument
);
2788 /* Routine io_registry_entry_get_path */
2789 kern_return_t
is_io_registry_entry_get_path_ool(
2790 io_object_t registry_entry
,
2792 io_string_inband_t path
,
2793 io_buf_ptr_t
*path_ool
,
2794 mach_msg_type_number_t
*path_oolCnt
)
2796 enum { kMaxPath
= 16384 };
2801 CHECK( IORegistryEntry
, registry_entry
, entry
);
2805 length
= sizeof(io_string_inband_t
);
2806 if (entry
->getPath(path
, &length
, IORegistryEntry::getPlane(plane
))) err
= kIOReturnSuccess
;
2810 buf
= IONew(char, length
);
2811 if (!buf
) err
= kIOReturnNoMemory
;
2812 else if (!entry
->getPath(buf
, &length
, IORegistryEntry::getPlane(plane
))) err
= kIOReturnError
;
2815 *path_oolCnt
= length
;
2816 err
= copyoutkdata(buf
, length
, path_ool
);
2818 if (buf
) IODelete(buf
, char, kMaxPath
);
2825 /* Routine io_registry_entry_get_name */
2826 kern_return_t
is_io_registry_entry_get_name(
2827 io_object_t registry_entry
,
2830 CHECK( IORegistryEntry
, registry_entry
, entry
);
2832 strncpy( name
, entry
->getName(), sizeof( io_name_t
));
2834 return( kIOReturnSuccess
);
2837 /* Routine io_registry_entry_get_name_in_plane */
2838 kern_return_t
is_io_registry_entry_get_name_in_plane(
2839 io_object_t registry_entry
,
2840 io_name_t planeName
,
2843 const IORegistryPlane
* plane
;
2844 CHECK( IORegistryEntry
, registry_entry
, entry
);
2847 plane
= IORegistryEntry::getPlane( planeName
);
2851 strncpy( name
, entry
->getName( plane
), sizeof( io_name_t
));
2853 return( kIOReturnSuccess
);
2856 /* Routine io_registry_entry_get_location_in_plane */
2857 kern_return_t
is_io_registry_entry_get_location_in_plane(
2858 io_object_t registry_entry
,
2859 io_name_t planeName
,
2860 io_name_t location
)
2862 const IORegistryPlane
* plane
;
2863 CHECK( IORegistryEntry
, registry_entry
, entry
);
2866 plane
= IORegistryEntry::getPlane( planeName
);
2870 const char * cstr
= entry
->getLocation( plane
);
2873 strncpy( location
, cstr
, sizeof( io_name_t
));
2874 return( kIOReturnSuccess
);
2876 return( kIOReturnNotFound
);
2879 /* Routine io_registry_entry_get_registry_entry_id */
2880 kern_return_t
is_io_registry_entry_get_registry_entry_id(
2881 io_object_t registry_entry
,
2882 uint64_t *entry_id
)
2884 CHECK( IORegistryEntry
, registry_entry
, entry
);
2886 *entry_id
= entry
->getRegistryEntryID();
2888 return (kIOReturnSuccess
);
2891 /* Routine io_registry_entry_get_property */
2892 kern_return_t
is_io_registry_entry_get_property_bytes(
2893 io_object_t registry_entry
,
2894 io_name_t property_name
,
2895 io_struct_inband_t buf
,
2896 mach_msg_type_number_t
*dataCnt
)
2904 unsigned int len
= 0;
2905 const void * bytes
= 0;
2906 IOReturn ret
= kIOReturnSuccess
;
2908 CHECK( IORegistryEntry
, registry_entry
, entry
);
2911 if (0 != mac_iokit_check_get_property(kauth_cred_get(), entry
, property_name
))
2912 return kIOReturnNotPermitted
;
2915 obj
= entry
->copyProperty(property_name
);
2917 return( kIOReturnNoResources
);
2919 // One day OSData will be a common container base class
2921 if( (data
= OSDynamicCast( OSData
, obj
))) {
2922 len
= data
->getLength();
2923 bytes
= data
->getBytesNoCopy();
2924 if (!data
->isSerializable()) len
= 0;
2926 } else if( (str
= OSDynamicCast( OSString
, obj
))) {
2927 len
= str
->getLength() + 1;
2928 bytes
= str
->getCStringNoCopy();
2930 } else if( (boo
= OSDynamicCast( OSBoolean
, obj
))) {
2931 len
= boo
->isTrue() ? sizeof("Yes") : sizeof("No");
2932 bytes
= boo
->isTrue() ? "Yes" : "No";
2934 } else if( (off
= OSDynamicCast( OSNumber
, obj
))) {
2935 offsetBytes
= off
->unsigned64BitValue();
2936 len
= off
->numberOfBytes();
2937 if (len
> sizeof(offsetBytes
)) len
= sizeof(offsetBytes
);
2938 bytes
= &offsetBytes
;
2939 #ifdef __BIG_ENDIAN__
2940 bytes
= (const void *)
2941 (((UInt32
) bytes
) + (sizeof( UInt64
) - len
));
2945 ret
= kIOReturnBadArgument
;
2949 ret
= kIOReturnIPCError
;
2952 bcopy( bytes
, buf
, len
);
2961 /* Routine io_registry_entry_get_property */
2962 kern_return_t
is_io_registry_entry_get_property(
2963 io_object_t registry_entry
,
2964 io_name_t property_name
,
2965 io_buf_ptr_t
*properties
,
2966 mach_msg_type_number_t
*propertiesCnt
)
2972 CHECK( IORegistryEntry
, registry_entry
, entry
);
2975 if (0 != mac_iokit_check_get_property(kauth_cred_get(), entry
, property_name
))
2976 return kIOReturnNotPermitted
;
2979 obj
= entry
->copyProperty(property_name
);
2981 return( kIOReturnNotFound
);
2983 OSSerialize
* s
= OSSerialize::withCapacity(4096);
2986 return( kIOReturnNoMemory
);
2989 if( obj
->serialize( s
)) {
2990 len
= s
->getLength();
2991 *propertiesCnt
= len
;
2992 err
= copyoutkdata( s
->text(), len
, properties
);
2995 err
= kIOReturnUnsupported
;
3003 /* Routine io_registry_entry_get_property_recursively */
3004 kern_return_t
is_io_registry_entry_get_property_recursively(
3005 io_object_t registry_entry
,
3007 io_name_t property_name
,
3009 io_buf_ptr_t
*properties
,
3010 mach_msg_type_number_t
*propertiesCnt
)
3016 CHECK( IORegistryEntry
, registry_entry
, entry
);
3019 if (0 != mac_iokit_check_get_property(kauth_cred_get(), entry
, property_name
))
3020 return kIOReturnNotPermitted
;
3023 obj
= entry
->copyProperty( property_name
,
3024 IORegistryEntry::getPlane( plane
), options
);
3026 return( kIOReturnNotFound
);
3028 OSSerialize
* s
= OSSerialize::withCapacity(4096);
3031 return( kIOReturnNoMemory
);
3034 if( obj
->serialize( s
)) {
3035 len
= s
->getLength();
3036 *propertiesCnt
= len
;
3037 err
= copyoutkdata( s
->text(), len
, properties
);
3040 err
= kIOReturnUnsupported
;
3048 /* Routine io_registry_entry_get_properties */
3049 kern_return_t
is_io_registry_entry_get_properties(
3050 io_object_t registry_entry
,
3051 io_buf_ptr_t
*properties
,
3052 mach_msg_type_number_t
*propertiesCnt
)
3054 return (kIOReturnUnsupported
);
3059 struct GetPropertiesEditorRef
3062 IORegistryEntry
* entry
;
3063 OSCollection
* root
;
3066 static const OSMetaClassBase
*
3067 GetPropertiesEditor(void * reference
,
3069 OSCollection
* container
,
3070 const OSSymbol
* name
,
3071 const OSMetaClassBase
* value
)
3073 GetPropertiesEditorRef
* ref
= (typeof(ref
)) reference
;
3075 if (!ref
->root
) ref
->root
= container
;
3076 if (ref
->root
== container
)
3078 if (0 != mac_iokit_check_get_property(ref
->cred
, ref
->entry
, name
->getCStringNoCopy()))
3083 if (value
) value
->retain();
3087 #endif /* CONFIG_MACF */
3089 /* Routine io_registry_entry_get_properties */
3090 kern_return_t
is_io_registry_entry_get_properties_bin(
3091 io_object_t registry_entry
,
3092 io_buf_ptr_t
*properties
,
3093 mach_msg_type_number_t
*propertiesCnt
)
3095 kern_return_t err
= kIOReturnSuccess
;
3098 OSSerialize::Editor editor
= 0;
3101 CHECK(IORegistryEntry
, registry_entry
, entry
);
3104 GetPropertiesEditorRef ref
;
3105 if (mac_iokit_check_filter_properties(kauth_cred_get(), entry
))
3107 editor
= &GetPropertiesEditor
;
3109 ref
.cred
= kauth_cred_get();
3115 s
= OSSerialize::binaryWithCapacity(4096, editor
, editRef
);
3116 if (!s
) return (kIOReturnNoMemory
);
3118 if (!entry
->serializeProperties(s
)) err
= kIOReturnUnsupported
;
3120 if (kIOReturnSuccess
== err
)
3122 len
= s
->getLength();
3123 *propertiesCnt
= len
;
3124 err
= copyoutkdata(s
->text(), len
, properties
);
3131 /* Routine io_registry_entry_get_property_bin */
3132 kern_return_t
is_io_registry_entry_get_property_bin(
3133 io_object_t registry_entry
,
3135 io_name_t property_name
,
3137 io_buf_ptr_t
*properties
,
3138 mach_msg_type_number_t
*propertiesCnt
)
3143 const OSSymbol
* sym
;
3145 CHECK( IORegistryEntry
, registry_entry
, entry
);
3148 if (0 != mac_iokit_check_get_property(kauth_cred_get(), entry
, property_name
))
3149 return kIOReturnNotPermitted
;
3152 sym
= OSSymbol::withCString(property_name
);
3153 if (!sym
) return (kIOReturnNoMemory
);
3155 if (gIORegistryEntryPropertyKeysKey
== sym
)
3157 obj
= entry
->copyPropertyKeys();
3161 if ((kIORegistryIterateRecursively
& options
) && plane
[0])
3163 obj
= entry
->copyProperty(property_name
,
3164 IORegistryEntry::getPlane(plane
), options
);
3168 obj
= entry
->copyProperty(property_name
);
3170 if (obj
&& gIORemoveOnReadProperties
->containsObject(sym
)) entry
->removeProperty(sym
);
3174 if (!obj
) return (kIOReturnNotFound
);
3176 OSSerialize
* s
= OSSerialize::binaryWithCapacity(4096);
3179 return( kIOReturnNoMemory
);
3182 if( obj
->serialize( s
)) {
3183 len
= s
->getLength();
3184 *propertiesCnt
= len
;
3185 err
= copyoutkdata( s
->text(), len
, properties
);
3187 } else err
= kIOReturnUnsupported
;
3196 /* Routine io_registry_entry_set_properties */
3197 kern_return_t is_io_registry_entry_set_properties
3199 io_object_t registry_entry
,
3200 io_buf_ptr_t properties
,
3201 mach_msg_type_number_t propertiesCnt
,
3202 kern_return_t
* result
)
3208 vm_map_offset_t map_data
;
3210 CHECK( IORegistryEntry
, registry_entry
, entry
);
3212 if( propertiesCnt
> sizeof(io_struct_inband_t
) * 1024)
3213 return( kIOReturnMessageTooLarge
);
3215 err
= vm_map_copyout( kernel_map
, &map_data
, (vm_map_copy_t
) properties
);
3216 data
= CAST_DOWN(vm_offset_t
, map_data
);
3218 if( KERN_SUCCESS
== err
) {
3220 FAKE_STACK_FRAME(entry
->getMetaClass());
3222 // must return success after vm_map_copyout() succeeds
3223 obj
= OSUnserializeXML( (const char *) data
, propertiesCnt
);
3224 vm_deallocate( kernel_map
, data
, propertiesCnt
);
3227 res
= kIOReturnBadArgument
;
3229 else if (0 != mac_iokit_check_set_properties(kauth_cred_get(),
3230 registry_entry
, obj
))
3232 res
= kIOReturnNotPermitted
;
3237 res
= entry
->setProperties( obj
);
3243 FAKE_STACK_FRAME_END();
3252 /* Routine io_registry_entry_get_child_iterator */
3253 kern_return_t
is_io_registry_entry_get_child_iterator(
3254 io_object_t registry_entry
,
3256 io_object_t
*iterator
)
3258 CHECK( IORegistryEntry
, registry_entry
, entry
);
3260 *iterator
= entry
->getChildIterator(
3261 IORegistryEntry::getPlane( plane
));
3263 return( kIOReturnSuccess
);
3266 /* Routine io_registry_entry_get_parent_iterator */
3267 kern_return_t
is_io_registry_entry_get_parent_iterator(
3268 io_object_t registry_entry
,
3270 io_object_t
*iterator
)
3272 CHECK( IORegistryEntry
, registry_entry
, entry
);
3274 *iterator
= entry
->getParentIterator(
3275 IORegistryEntry::getPlane( plane
));
3277 return( kIOReturnSuccess
);
3280 /* Routine io_service_get_busy_state */
3281 kern_return_t
is_io_service_get_busy_state(
3282 io_object_t _service
,
3283 uint32_t *busyState
)
3285 CHECK( IOService
, _service
, service
);
3287 *busyState
= service
->getBusyState();
3289 return( kIOReturnSuccess
);
3292 /* Routine io_service_get_state */
3293 kern_return_t
is_io_service_get_state(
3294 io_object_t _service
,
3296 uint32_t *busy_state
,
3297 uint64_t *accumulated_busy_time
)
3299 CHECK( IOService
, _service
, service
);
3301 *state
= service
->getState();
3302 *busy_state
= service
->getBusyState();
3303 *accumulated_busy_time
= service
->getAccumulatedBusyTime();
3305 return( kIOReturnSuccess
);
3308 /* Routine io_service_wait_quiet */
3309 kern_return_t
is_io_service_wait_quiet(
3310 io_object_t _service
,
3311 mach_timespec_t wait_time
)
3315 CHECK( IOService
, _service
, service
);
3317 timeoutNS
= wait_time
.tv_sec
;
3318 timeoutNS
*= kSecondScale
;
3319 timeoutNS
+= wait_time
.tv_nsec
;
3321 return( service
->waitQuiet(timeoutNS
) );
3324 /* Routine io_service_request_probe */
3325 kern_return_t
is_io_service_request_probe(
3326 io_object_t _service
,
3329 CHECK( IOService
, _service
, service
);
3331 return( service
->requestProbe( options
));
3334 /* Routine io_service_get_authorization_id */
3335 kern_return_t
is_io_service_get_authorization_id(
3336 io_object_t _service
,
3337 uint64_t *authorization_id
)
3341 CHECK( IOService
, _service
, service
);
3343 kr
= IOUserClient::clientHasPrivilege( (void *) current_task(),
3344 kIOClientPrivilegeAdministrator
);
3345 if( kIOReturnSuccess
!= kr
)
3348 *authorization_id
= service
->getAuthorizationID();
3353 /* Routine io_service_set_authorization_id */
3354 kern_return_t
is_io_service_set_authorization_id(
3355 io_object_t _service
,
3356 uint64_t authorization_id
)
3358 CHECK( IOService
, _service
, service
);
3360 return( service
->setAuthorizationID( authorization_id
) );
3363 /* Routine io_service_open_ndr */
3364 kern_return_t
is_io_service_open_extended(
3365 io_object_t _service
,
3367 uint32_t connect_type
,
3369 io_buf_ptr_t properties
,
3370 mach_msg_type_number_t propertiesCnt
,
3371 kern_return_t
* result
,
3372 io_object_t
*connection
)
3374 IOUserClient
* client
= 0;
3375 kern_return_t err
= KERN_SUCCESS
;
3376 IOReturn res
= kIOReturnSuccess
;
3377 OSDictionary
* propertiesDict
= 0;
3379 bool disallowAccess
;
3381 CHECK( IOService
, _service
, service
);
3383 if (!owningTask
) return (kIOReturnBadArgument
);
3384 assert(owningTask
== current_task());
3385 if (owningTask
!= current_task()) return (kIOReturnBadArgument
);
3389 if (properties
) return (kIOReturnUnsupported
);
3394 vm_map_offset_t map_data
;
3396 if( propertiesCnt
> sizeof(io_struct_inband_t
))
3397 return( kIOReturnMessageTooLarge
);
3399 err
= vm_map_copyout( kernel_map
, &map_data
, (vm_map_copy_t
) properties
);
3401 data
= CAST_DOWN(vm_offset_t
, map_data
);
3402 if (KERN_SUCCESS
== err
)
3404 // must return success after vm_map_copyout() succeeds
3405 obj
= OSUnserializeXML( (const char *) data
, propertiesCnt
);
3406 vm_deallocate( kernel_map
, data
, propertiesCnt
);
3407 propertiesDict
= OSDynamicCast(OSDictionary
, obj
);
3408 if (!propertiesDict
)
3410 res
= kIOReturnBadArgument
;
3415 if (kIOReturnSuccess
!= res
)
3419 crossEndian
= (ndr
.int_rep
!= NDR_record
.int_rep
);
3422 if (!propertiesDict
)
3423 propertiesDict
= OSDictionary::withCapacity(4);
3424 OSData
* data
= OSData::withBytes(&ndr
, sizeof(ndr
));
3428 propertiesDict
->setObject(kIOUserClientCrossEndianKey
, data
);
3433 res
= service
->newUserClient( owningTask
, (void *) owningTask
,
3434 connect_type
, propertiesDict
, &client
);
3437 propertiesDict
->release();
3439 if (res
== kIOReturnSuccess
)
3441 assert( OSDynamicCast(IOUserClient
, client
) );
3443 client
->sharedInstance
= (0 != client
->getProperty(kIOUserClientSharedInstanceKey
));
3444 client
->closed
= false;
3446 disallowAccess
= (crossEndian
3447 && (kOSBooleanTrue
!= service
->getProperty(kIOUserClientCrossEndianCompatibleKey
))
3448 && (kOSBooleanTrue
!= client
->getProperty(kIOUserClientCrossEndianCompatibleKey
)));
3449 if (disallowAccess
) res
= kIOReturnUnsupported
;
3451 else if (0 != mac_iokit_check_open(kauth_cred_get(), client
, connect_type
))
3452 res
= kIOReturnNotPermitted
;
3455 if (kIOReturnSuccess
== res
) res
= client
->registerOwner(owningTask
);
3457 if (kIOReturnSuccess
!= res
)
3459 IOStatisticsClientCall();
3460 client
->clientClose();
3465 OSString
* creatorName
= IOCopyLogNameForPID(proc_selfpid());
3468 client
->setProperty(kIOUserClientCreatorKey
, creatorName
);
3469 creatorName
->release();
3471 client
->setTerminateDefer(service
, false);
3476 *connection
= client
;
3482 /* Routine io_service_close */
3483 kern_return_t
is_io_service_close(
3484 io_object_t connection
)
3487 if ((mappings
= OSDynamicCast(OSSet
, connection
)))
3488 return( kIOReturnSuccess
);
3490 CHECK( IOUserClient
, connection
, client
);
3492 IOStatisticsClientCall();
3494 if (client
->sharedInstance
|| OSCompareAndSwap8(0, 1, &client
->closed
))
3496 client
->clientClose();
3500 IOLog("ignored is_io_service_close(0x%qx,%s)\n",
3501 client
->getRegistryEntryID(), client
->getName());
3504 return( kIOReturnSuccess
);
3507 /* Routine io_connect_get_service */
3508 kern_return_t
is_io_connect_get_service(
3509 io_object_t connection
,
3510 io_object_t
*service
)
3512 IOService
* theService
;
3514 CHECK( IOUserClient
, connection
, client
);
3516 theService
= client
->getService();
3518 theService
->retain();
3520 *service
= theService
;
3522 return( theService
? kIOReturnSuccess
: kIOReturnUnsupported
);
3525 /* Routine io_connect_set_notification_port */
3526 kern_return_t
is_io_connect_set_notification_port(
3527 io_object_t connection
,
3528 uint32_t notification_type
,
3532 CHECK( IOUserClient
, connection
, client
);
3534 IOStatisticsClientCall();
3535 return( client
->registerNotificationPort( port
, notification_type
,
3536 (io_user_reference_t
) reference
));
3539 /* Routine io_connect_set_notification_port */
3540 kern_return_t
is_io_connect_set_notification_port_64(
3541 io_object_t connection
,
3542 uint32_t notification_type
,
3544 io_user_reference_t reference
)
3546 CHECK( IOUserClient
, connection
, client
);
3548 IOStatisticsClientCall();
3549 return( client
->registerNotificationPort( port
, notification_type
,
3553 /* Routine io_connect_map_memory_into_task */
3554 kern_return_t is_io_connect_map_memory_into_task
3556 io_connect_t connection
,
3557 uint32_t memory_type
,
3559 mach_vm_address_t
*address
,
3560 mach_vm_size_t
*size
,
3567 CHECK( IOUserClient
, connection
, client
);
3569 if (!into_task
) return (kIOReturnBadArgument
);
3571 IOStatisticsClientCall();
3572 map
= client
->mapClientMemory64( memory_type
, into_task
, flags
, *address
);
3575 *address
= map
->getAddress();
3577 *size
= map
->getSize();
3579 if( client
->sharedInstance
3580 || (into_task
!= current_task())) {
3581 // push a name out to the task owning the map,
3582 // so we can clean up maps
3583 mach_port_name_t name __unused
=
3584 IOMachPort::makeSendRightForTask(
3585 into_task
, map
, IKOT_IOKIT_OBJECT
);
3588 // keep it with the user client
3589 IOLockLock( gIOObjectPortLock
);
3590 if( 0 == client
->mappings
)
3591 client
->mappings
= OSSet::withCapacity(2);
3592 if( client
->mappings
)
3593 client
->mappings
->setObject( map
);
3594 IOLockUnlock( gIOObjectPortLock
);
3597 err
= kIOReturnSuccess
;
3600 err
= kIOReturnBadArgument
;
3605 /* Routine is_io_connect_map_memory */
3606 kern_return_t
is_io_connect_map_memory(
3607 io_object_t connect
,
3615 mach_vm_address_t address
;
3616 mach_vm_size_t size
;
3618 address
= SCALAR64(*mapAddr
);
3619 size
= SCALAR64(*mapSize
);
3621 err
= is_io_connect_map_memory_into_task(connect
, type
, task
, &address
, &size
, flags
);
3623 *mapAddr
= SCALAR32(address
);
3624 *mapSize
= SCALAR32(size
);
3631 IOMemoryMap
* IOUserClient::removeMappingForDescriptor(IOMemoryDescriptor
* mem
)
3634 IOMemoryMap
* map
= 0;
3636 IOLockLock(gIOObjectPortLock
);
3638 iter
= OSCollectionIterator::withCollection(mappings
);
3641 while ((map
= OSDynamicCast(IOMemoryMap
, iter
->getNextObject())))
3643 if(mem
== map
->getMemoryDescriptor())
3646 mappings
->removeObject(map
);
3653 IOLockUnlock(gIOObjectPortLock
);
3660 /* Routine io_connect_unmap_memory_from_task */
3661 kern_return_t is_io_connect_unmap_memory_from_task
3663 io_connect_t connection
,
3664 uint32_t memory_type
,
3666 mach_vm_address_t address
)
3669 IOOptionBits options
= 0;
3670 IOMemoryDescriptor
* memory
= 0;
3673 CHECK( IOUserClient
, connection
, client
);
3675 if (!from_task
) return (kIOReturnBadArgument
);
3677 IOStatisticsClientCall();
3678 err
= client
->clientMemoryForType( (UInt32
) memory_type
, &options
, &memory
);
3680 if( memory
&& (kIOReturnSuccess
== err
)) {
3682 options
= (options
& ~kIOMapUserOptionsMask
)
3683 | kIOMapAnywhere
| kIOMapReference
;
3685 map
= memory
->createMappingInTask( from_task
, address
, options
);
3689 IOLockLock( gIOObjectPortLock
);
3690 if( client
->mappings
)
3691 client
->mappings
->removeObject( map
);
3692 IOLockUnlock( gIOObjectPortLock
);
3694 mach_port_name_t name
= 0;
3695 if (from_task
!= current_task())
3696 name
= IOMachPort::makeSendRightForTask( from_task
, map
, IKOT_IOKIT_OBJECT
);
3699 map
->userClientUnmap();
3700 err
= iokit_mod_send_right( from_task
, name
, -2 );
3701 err
= kIOReturnSuccess
;
3704 IOMachPort::releasePortForObject( map
, IKOT_IOKIT_OBJECT
);
3705 if (from_task
== current_task())
3709 err
= kIOReturnBadArgument
;
3715 kern_return_t
is_io_connect_unmap_memory(
3716 io_object_t connect
,
3722 mach_vm_address_t address
;
3724 address
= SCALAR64(mapAddr
);
3726 err
= is_io_connect_unmap_memory_from_task(connect
, type
, task
, mapAddr
);
3732 /* Routine io_connect_add_client */
3733 kern_return_t
is_io_connect_add_client(
3734 io_object_t connection
,
3735 io_object_t connect_to
)
3737 CHECK( IOUserClient
, connection
, client
);
3738 CHECK( IOUserClient
, connect_to
, to
);
3740 IOStatisticsClientCall();
3741 return( client
->connectClient( to
) );
3745 /* Routine io_connect_set_properties */
3746 kern_return_t
is_io_connect_set_properties(
3747 io_object_t connection
,
3748 io_buf_ptr_t properties
,
3749 mach_msg_type_number_t propertiesCnt
,
3750 kern_return_t
* result
)
3752 return( is_io_registry_entry_set_properties( connection
, properties
, propertiesCnt
, result
));
3755 /* Routine io_user_client_method */
3756 kern_return_t is_io_connect_method_var_output
3758 io_connect_t connection
,
3760 io_scalar_inband64_t scalar_input
,
3761 mach_msg_type_number_t scalar_inputCnt
,
3762 io_struct_inband_t inband_input
,
3763 mach_msg_type_number_t inband_inputCnt
,
3764 mach_vm_address_t ool_input
,
3765 mach_vm_size_t ool_input_size
,
3766 io_struct_inband_t inband_output
,
3767 mach_msg_type_number_t
*inband_outputCnt
,
3768 io_scalar_inband64_t scalar_output
,
3769 mach_msg_type_number_t
*scalar_outputCnt
,
3770 io_buf_ptr_t
*var_output
,
3771 mach_msg_type_number_t
*var_outputCnt
3774 CHECK( IOUserClient
, connection
, client
);
3776 IOExternalMethodArguments args
;
3778 IOMemoryDescriptor
* inputMD
= 0;
3779 OSObject
* structureVariableOutputData
= 0;
3781 bzero(&args
.__reserved
[0], sizeof(args
.__reserved
));
3782 args
.__reservedA
= 0;
3783 args
.version
= kIOExternalMethodArgumentsCurrentVersion
;
3785 args
.selector
= selector
;
3787 args
.asyncWakePort
= MACH_PORT_NULL
;
3788 args
.asyncReference
= 0;
3789 args
.asyncReferenceCount
= 0;
3790 args
.structureVariableOutputData
= &structureVariableOutputData
;
3792 args
.scalarInput
= scalar_input
;
3793 args
.scalarInputCount
= scalar_inputCnt
;
3794 args
.structureInput
= inband_input
;
3795 args
.structureInputSize
= inband_inputCnt
;
3797 if (ool_input
&& (ool_input_size
<= sizeof(io_struct_inband_t
))) return (kIOReturnIPCError
);
3800 inputMD
= IOMemoryDescriptor::withAddressRange(ool_input
, ool_input_size
,
3801 kIODirectionOut
| kIOMemoryMapCopyOnWrite
,
3804 args
.structureInputDescriptor
= inputMD
;
3806 args
.scalarOutput
= scalar_output
;
3807 args
.scalarOutputCount
= *scalar_outputCnt
;
3808 bzero(&scalar_output
[0], *scalar_outputCnt
* sizeof(scalar_output
[0]));
3809 args
.structureOutput
= inband_output
;
3810 args
.structureOutputSize
= *inband_outputCnt
;
3811 args
.structureOutputDescriptor
= NULL
;
3812 args
.structureOutputDescriptorSize
= 0;
3814 IOStatisticsClientCall();
3815 ret
= client
->externalMethod( selector
, &args
);
3817 *scalar_outputCnt
= args
.scalarOutputCount
;
3818 *inband_outputCnt
= args
.structureOutputSize
;
3820 if (var_outputCnt
&& var_output
&& (kIOReturnSuccess
== ret
))
3822 OSSerialize
* serialize
;
3826 if ((serialize
= OSDynamicCast(OSSerialize
, structureVariableOutputData
)))
3828 len
= serialize
->getLength();
3829 *var_outputCnt
= len
;
3830 ret
= copyoutkdata(serialize
->text(), len
, var_output
);
3832 else if ((data
= OSDynamicCast(OSData
, structureVariableOutputData
)))
3834 len
= data
->getLength();
3835 *var_outputCnt
= len
;
3836 ret
= copyoutkdata(data
->getBytesNoCopy(), len
, var_output
);
3840 ret
= kIOReturnUnderrun
;
3846 if (structureVariableOutputData
)
3847 structureVariableOutputData
->release();
3852 /* Routine io_user_client_method */
3853 kern_return_t is_io_connect_method
3855 io_connect_t connection
,
3857 io_scalar_inband64_t scalar_input
,
3858 mach_msg_type_number_t scalar_inputCnt
,
3859 io_struct_inband_t inband_input
,
3860 mach_msg_type_number_t inband_inputCnt
,
3861 mach_vm_address_t ool_input
,
3862 mach_vm_size_t ool_input_size
,
3863 io_struct_inband_t inband_output
,
3864 mach_msg_type_number_t
*inband_outputCnt
,
3865 io_scalar_inband64_t scalar_output
,
3866 mach_msg_type_number_t
*scalar_outputCnt
,
3867 mach_vm_address_t ool_output
,
3868 mach_vm_size_t
*ool_output_size
3871 CHECK( IOUserClient
, connection
, client
);
3873 IOExternalMethodArguments args
;
3875 IOMemoryDescriptor
* inputMD
= 0;
3876 IOMemoryDescriptor
* outputMD
= 0;
3878 bzero(&args
.__reserved
[0], sizeof(args
.__reserved
));
3879 args
.__reservedA
= 0;
3880 args
.version
= kIOExternalMethodArgumentsCurrentVersion
;
3882 args
.selector
= selector
;
3884 args
.asyncWakePort
= MACH_PORT_NULL
;
3885 args
.asyncReference
= 0;
3886 args
.asyncReferenceCount
= 0;
3887 args
.structureVariableOutputData
= 0;
3889 args
.scalarInput
= scalar_input
;
3890 args
.scalarInputCount
= scalar_inputCnt
;
3891 args
.structureInput
= inband_input
;
3892 args
.structureInputSize
= inband_inputCnt
;
3894 if (ool_input
&& (ool_input_size
<= sizeof(io_struct_inband_t
))) return (kIOReturnIPCError
);
3895 if (ool_output
&& (*ool_output_size
<= sizeof(io_struct_inband_t
))) return (kIOReturnIPCError
);
3898 inputMD
= IOMemoryDescriptor::withAddressRange(ool_input
, ool_input_size
,
3899 kIODirectionOut
| kIOMemoryMapCopyOnWrite
,
3902 args
.structureInputDescriptor
= inputMD
;
3904 args
.scalarOutput
= scalar_output
;
3905 args
.scalarOutputCount
= *scalar_outputCnt
;
3906 bzero(&scalar_output
[0], *scalar_outputCnt
* sizeof(scalar_output
[0]));
3907 args
.structureOutput
= inband_output
;
3908 args
.structureOutputSize
= *inband_outputCnt
;
3910 if (ool_output
&& ool_output_size
)
3912 outputMD
= IOMemoryDescriptor::withAddressRange(ool_output
, *ool_output_size
,
3913 kIODirectionIn
, current_task());
3916 args
.structureOutputDescriptor
= outputMD
;
3917 args
.structureOutputDescriptorSize
= ool_output_size
? *ool_output_size
: 0;
3919 IOStatisticsClientCall();
3920 ret
= client
->externalMethod( selector
, &args
);
3922 *scalar_outputCnt
= args
.scalarOutputCount
;
3923 *inband_outputCnt
= args
.structureOutputSize
;
3924 *ool_output_size
= args
.structureOutputDescriptorSize
;
3929 outputMD
->release();
3934 /* Routine io_async_user_client_method */
3935 kern_return_t is_io_connect_async_method
3937 io_connect_t connection
,
3938 mach_port_t wake_port
,
3939 io_async_ref64_t reference
,
3940 mach_msg_type_number_t referenceCnt
,
3942 io_scalar_inband64_t scalar_input
,
3943 mach_msg_type_number_t scalar_inputCnt
,
3944 io_struct_inband_t inband_input
,
3945 mach_msg_type_number_t inband_inputCnt
,
3946 mach_vm_address_t ool_input
,
3947 mach_vm_size_t ool_input_size
,
3948 io_struct_inband_t inband_output
,
3949 mach_msg_type_number_t
*inband_outputCnt
,
3950 io_scalar_inband64_t scalar_output
,
3951 mach_msg_type_number_t
*scalar_outputCnt
,
3952 mach_vm_address_t ool_output
,
3953 mach_vm_size_t
* ool_output_size
3956 CHECK( IOUserClient
, connection
, client
);
3958 IOExternalMethodArguments args
;
3960 IOMemoryDescriptor
* inputMD
= 0;
3961 IOMemoryDescriptor
* outputMD
= 0;
3963 bzero(&args
.__reserved
[0], sizeof(args
.__reserved
));
3964 args
.__reservedA
= 0;
3965 args
.version
= kIOExternalMethodArgumentsCurrentVersion
;
3967 reference
[0] = (io_user_reference_t
) wake_port
;
3968 if (vm_map_is_64bit(get_task_map(current_task())))
3969 reference
[0] |= kIOUCAsync64Flag
;
3971 args
.selector
= selector
;
3973 args
.asyncWakePort
= wake_port
;
3974 args
.asyncReference
= reference
;
3975 args
.asyncReferenceCount
= referenceCnt
;
3977 args
.scalarInput
= scalar_input
;
3978 args
.scalarInputCount
= scalar_inputCnt
;
3979 args
.structureInput
= inband_input
;
3980 args
.structureInputSize
= inband_inputCnt
;
3982 if (ool_input
&& (ool_input_size
<= sizeof(io_struct_inband_t
))) return (kIOReturnIPCError
);
3983 if (ool_output
&& (*ool_output_size
<= sizeof(io_struct_inband_t
))) return (kIOReturnIPCError
);
3986 inputMD
= IOMemoryDescriptor::withAddressRange(ool_input
, ool_input_size
,
3987 kIODirectionOut
| kIOMemoryMapCopyOnWrite
,
3990 args
.structureInputDescriptor
= inputMD
;
3992 args
.scalarOutput
= scalar_output
;
3993 args
.scalarOutputCount
= *scalar_outputCnt
;
3994 bzero(&scalar_output
[0], *scalar_outputCnt
* sizeof(scalar_output
[0]));
3995 args
.structureOutput
= inband_output
;
3996 args
.structureOutputSize
= *inband_outputCnt
;
4000 outputMD
= IOMemoryDescriptor::withAddressRange(ool_output
, *ool_output_size
,
4001 kIODirectionIn
, current_task());
4004 args
.structureOutputDescriptor
= outputMD
;
4005 args
.structureOutputDescriptorSize
= *ool_output_size
;
4007 IOStatisticsClientCall();
4008 ret
= client
->externalMethod( selector
, &args
);
4010 *inband_outputCnt
= args
.structureOutputSize
;
4011 *ool_output_size
= args
.structureOutputDescriptorSize
;
4016 outputMD
->release();
4021 /* Routine io_connect_method_scalarI_scalarO */
4022 kern_return_t
is_io_connect_method_scalarI_scalarO(
4023 io_object_t connect
,
4025 io_scalar_inband_t input
,
4026 mach_msg_type_number_t inputCount
,
4027 io_scalar_inband_t output
,
4028 mach_msg_type_number_t
* outputCount
)
4032 io_scalar_inband64_t _input
;
4033 io_scalar_inband64_t _output
;
4035 mach_msg_type_number_t struct_outputCnt
= 0;
4036 mach_vm_size_t ool_output_size
= 0;
4038 bzero(&_output
[0], sizeof(_output
));
4039 for (i
= 0; i
< inputCount
; i
++)
4040 _input
[i
] = SCALAR64(input
[i
]);
4042 err
= is_io_connect_method(connect
, index
,
4046 NULL
, &struct_outputCnt
,
4047 _output
, outputCount
,
4048 0, &ool_output_size
);
4050 for (i
= 0; i
< *outputCount
; i
++)
4051 output
[i
] = SCALAR32(_output
[i
]);
4056 kern_return_t
shim_io_connect_method_scalarI_scalarO(
4057 IOExternalMethod
* method
,
4059 const io_user_scalar_t
* input
,
4060 mach_msg_type_number_t inputCount
,
4061 io_user_scalar_t
* output
,
4062 mach_msg_type_number_t
* outputCount
)
4065 io_scalar_inband_t _output
;
4067 err
= kIOReturnBadArgument
;
4069 bzero(&_output
[0], sizeof(_output
));
4072 if( inputCount
!= method
->count0
)
4074 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
);
4075 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4078 if( *outputCount
!= method
->count1
)
4080 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)*outputCount
, (uint64_t)method
->count1
);
4081 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)*outputCount
, uint64_t, (uint64_t)method
->count1
);
4085 func
= method
->func
;
4087 switch( inputCount
) {
4090 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4091 ARG32(input
[3]), ARG32(input
[4]), ARG32(input
[5]) );
4094 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4095 ARG32(input
[3]), ARG32(input
[4]),
4099 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4101 &_output
[0], &_output
[1] );
4104 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4105 &_output
[0], &_output
[1], &_output
[2] );
4108 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]),
4109 &_output
[0], &_output
[1], &_output
[2],
4113 err
= (object
->*func
)( ARG32(input
[0]),
4114 &_output
[0], &_output
[1], &_output
[2],
4115 &_output
[3], &_output
[4] );
4118 err
= (object
->*func
)( &_output
[0], &_output
[1], &_output
[2],
4119 &_output
[3], &_output
[4], &_output
[5] );
4123 IOLog("%s: Bad method table\n", object
->getName());
4129 for (i
= 0; i
< *outputCount
; i
++)
4130 output
[i
] = SCALAR32(_output
[i
]);
4135 /* Routine io_async_method_scalarI_scalarO */
4136 kern_return_t
is_io_async_method_scalarI_scalarO(
4137 io_object_t connect
,
4138 mach_port_t wake_port
,
4139 io_async_ref_t reference
,
4140 mach_msg_type_number_t referenceCnt
,
4142 io_scalar_inband_t input
,
4143 mach_msg_type_number_t inputCount
,
4144 io_scalar_inband_t output
,
4145 mach_msg_type_number_t
* outputCount
)
4149 io_scalar_inband64_t _input
;
4150 io_scalar_inband64_t _output
;
4151 io_async_ref64_t _reference
;
4153 bzero(&_output
[0], sizeof(_output
));
4154 for (i
= 0; i
< referenceCnt
; i
++)
4155 _reference
[i
] = REF64(reference
[i
]);
4157 mach_msg_type_number_t struct_outputCnt
= 0;
4158 mach_vm_size_t ool_output_size
= 0;
4160 for (i
= 0; i
< inputCount
; i
++)
4161 _input
[i
] = SCALAR64(input
[i
]);
4163 err
= is_io_connect_async_method(connect
,
4164 wake_port
, _reference
, referenceCnt
,
4169 NULL
, &struct_outputCnt
,
4170 _output
, outputCount
,
4171 0, &ool_output_size
);
4173 for (i
= 0; i
< *outputCount
; i
++)
4174 output
[i
] = SCALAR32(_output
[i
]);
4178 /* Routine io_async_method_scalarI_structureO */
4179 kern_return_t
is_io_async_method_scalarI_structureO(
4180 io_object_t connect
,
4181 mach_port_t wake_port
,
4182 io_async_ref_t reference
,
4183 mach_msg_type_number_t referenceCnt
,
4185 io_scalar_inband_t input
,
4186 mach_msg_type_number_t inputCount
,
4187 io_struct_inband_t output
,
4188 mach_msg_type_number_t
* outputCount
)
4191 io_scalar_inband64_t _input
;
4192 io_async_ref64_t _reference
;
4194 for (i
= 0; i
< referenceCnt
; i
++)
4195 _reference
[i
] = REF64(reference
[i
]);
4197 mach_msg_type_number_t scalar_outputCnt
= 0;
4198 mach_vm_size_t ool_output_size
= 0;
4200 for (i
= 0; i
< inputCount
; i
++)
4201 _input
[i
] = SCALAR64(input
[i
]);
4203 return (is_io_connect_async_method(connect
,
4204 wake_port
, _reference
, referenceCnt
,
4209 output
, outputCount
,
4210 NULL
, &scalar_outputCnt
,
4211 0, &ool_output_size
));
4214 /* Routine io_async_method_scalarI_structureI */
4215 kern_return_t
is_io_async_method_scalarI_structureI(
4216 io_connect_t connect
,
4217 mach_port_t wake_port
,
4218 io_async_ref_t reference
,
4219 mach_msg_type_number_t referenceCnt
,
4221 io_scalar_inband_t input
,
4222 mach_msg_type_number_t inputCount
,
4223 io_struct_inband_t inputStruct
,
4224 mach_msg_type_number_t inputStructCount
)
4227 io_scalar_inband64_t _input
;
4228 io_async_ref64_t _reference
;
4230 for (i
= 0; i
< referenceCnt
; i
++)
4231 _reference
[i
] = REF64(reference
[i
]);
4233 mach_msg_type_number_t scalar_outputCnt
= 0;
4234 mach_msg_type_number_t inband_outputCnt
= 0;
4235 mach_vm_size_t ool_output_size
= 0;
4237 for (i
= 0; i
< inputCount
; i
++)
4238 _input
[i
] = SCALAR64(input
[i
]);
4240 return (is_io_connect_async_method(connect
,
4241 wake_port
, _reference
, referenceCnt
,
4244 inputStruct
, inputStructCount
,
4246 NULL
, &inband_outputCnt
,
4247 NULL
, &scalar_outputCnt
,
4248 0, &ool_output_size
));
4251 /* Routine io_async_method_structureI_structureO */
4252 kern_return_t
is_io_async_method_structureI_structureO(
4253 io_object_t connect
,
4254 mach_port_t wake_port
,
4255 io_async_ref_t reference
,
4256 mach_msg_type_number_t referenceCnt
,
4258 io_struct_inband_t input
,
4259 mach_msg_type_number_t inputCount
,
4260 io_struct_inband_t output
,
4261 mach_msg_type_number_t
* outputCount
)
4264 mach_msg_type_number_t scalar_outputCnt
= 0;
4265 mach_vm_size_t ool_output_size
= 0;
4266 io_async_ref64_t _reference
;
4268 for (i
= 0; i
< referenceCnt
; i
++)
4269 _reference
[i
] = REF64(reference
[i
]);
4271 return (is_io_connect_async_method(connect
,
4272 wake_port
, _reference
, referenceCnt
,
4277 output
, outputCount
,
4278 NULL
, &scalar_outputCnt
,
4279 0, &ool_output_size
));
4283 kern_return_t
shim_io_async_method_scalarI_scalarO(
4284 IOExternalAsyncMethod
* method
,
4286 mach_port_t asyncWakePort
,
4287 io_user_reference_t
* asyncReference
,
4288 uint32_t asyncReferenceCount
,
4289 const io_user_scalar_t
* input
,
4290 mach_msg_type_number_t inputCount
,
4291 io_user_scalar_t
* output
,
4292 mach_msg_type_number_t
* outputCount
)
4296 io_scalar_inband_t _output
;
4298 io_async_ref_t reference
;
4300 bzero(&_output
[0], sizeof(_output
));
4301 for (i
= 0; i
< asyncReferenceCount
; i
++)
4302 reference
[i
] = REF32(asyncReference
[i
]);
4304 err
= kIOReturnBadArgument
;
4308 if( inputCount
!= method
->count0
)
4310 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
);
4311 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4314 if( *outputCount
!= method
->count1
)
4316 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)*outputCount
, (uint64_t)method
->count1
);
4317 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)*outputCount
, uint64_t, (uint64_t)method
->count1
);
4321 func
= method
->func
;
4323 switch( inputCount
) {
4326 err
= (object
->*func
)( reference
,
4327 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4328 ARG32(input
[3]), ARG32(input
[4]), ARG32(input
[5]) );
4331 err
= (object
->*func
)( reference
,
4332 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4333 ARG32(input
[3]), ARG32(input
[4]),
4337 err
= (object
->*func
)( reference
,
4338 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4340 &_output
[0], &_output
[1] );
4343 err
= (object
->*func
)( reference
,
4344 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4345 &_output
[0], &_output
[1], &_output
[2] );
4348 err
= (object
->*func
)( reference
,
4349 ARG32(input
[0]), ARG32(input
[1]),
4350 &_output
[0], &_output
[1], &_output
[2],
4354 err
= (object
->*func
)( reference
,
4356 &_output
[0], &_output
[1], &_output
[2],
4357 &_output
[3], &_output
[4] );
4360 err
= (object
->*func
)( reference
,
4361 &_output
[0], &_output
[1], &_output
[2],
4362 &_output
[3], &_output
[4], &_output
[5] );
4366 IOLog("%s: Bad method table\n", object
->getName());
4371 for (i
= 0; i
< *outputCount
; i
++)
4372 output
[i
] = SCALAR32(_output
[i
]);
4378 /* Routine io_connect_method_scalarI_structureO */
4379 kern_return_t
is_io_connect_method_scalarI_structureO(
4380 io_object_t connect
,
4382 io_scalar_inband_t input
,
4383 mach_msg_type_number_t inputCount
,
4384 io_struct_inband_t output
,
4385 mach_msg_type_number_t
* outputCount
)
4388 io_scalar_inband64_t _input
;
4390 mach_msg_type_number_t scalar_outputCnt
= 0;
4391 mach_vm_size_t ool_output_size
= 0;
4393 for (i
= 0; i
< inputCount
; i
++)
4394 _input
[i
] = SCALAR64(input
[i
]);
4396 return (is_io_connect_method(connect
, index
,
4400 output
, outputCount
,
4401 NULL
, &scalar_outputCnt
,
4402 0, &ool_output_size
));
4405 kern_return_t
shim_io_connect_method_scalarI_structureO(
4407 IOExternalMethod
* method
,
4409 const io_user_scalar_t
* input
,
4410 mach_msg_type_number_t inputCount
,
4411 io_struct_inband_t output
,
4412 IOByteCount
* outputCount
)
4417 err
= kIOReturnBadArgument
;
4420 if( inputCount
!= method
->count0
)
4422 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
);
4423 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4426 if( (kIOUCVariableStructureSize
!= method
->count1
)
4427 && (*outputCount
!= method
->count1
))
4429 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)*outputCount
, (uint64_t)method
->count1
, (uint64_t)kIOUCVariableStructureSize
);
4430 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)*outputCount
, uint64_t, (uint64_t)method
->count1
);
4434 func
= method
->func
;
4436 switch( inputCount
) {
4439 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4440 ARG32(input
[3]), ARG32(input
[4]),
4444 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4446 output
, (void *)outputCount
);
4449 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4450 output
, (void *)outputCount
, 0 );
4453 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]),
4454 output
, (void *)outputCount
, 0, 0 );
4457 err
= (object
->*func
)( ARG32(input
[0]),
4458 output
, (void *)outputCount
, 0, 0, 0 );
4461 err
= (object
->*func
)( output
, (void *)outputCount
, 0, 0, 0, 0 );
4465 IOLog("%s: Bad method table\n", object
->getName());
4474 kern_return_t
shim_io_async_method_scalarI_structureO(
4475 IOExternalAsyncMethod
* method
,
4477 mach_port_t asyncWakePort
,
4478 io_user_reference_t
* asyncReference
,
4479 uint32_t asyncReferenceCount
,
4480 const io_user_scalar_t
* input
,
4481 mach_msg_type_number_t inputCount
,
4482 io_struct_inband_t output
,
4483 mach_msg_type_number_t
* outputCount
)
4488 io_async_ref_t reference
;
4490 for (i
= 0; i
< asyncReferenceCount
; i
++)
4491 reference
[i
] = REF32(asyncReference
[i
]);
4493 err
= kIOReturnBadArgument
;
4495 if( inputCount
!= method
->count0
)
4497 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
);
4498 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4501 if( (kIOUCVariableStructureSize
!= method
->count1
)
4502 && (*outputCount
!= method
->count1
))
4504 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)*outputCount
, (uint64_t)method
->count1
, (uint64_t)kIOUCVariableStructureSize
);
4505 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)*outputCount
, uint64_t, (uint64_t)method
->count1
);
4509 func
= method
->func
;
4511 switch( inputCount
) {
4514 err
= (object
->*func
)( reference
,
4515 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4516 ARG32(input
[3]), ARG32(input
[4]),
4520 err
= (object
->*func
)( reference
,
4521 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4523 output
, (void *)outputCount
);
4526 err
= (object
->*func
)( reference
,
4527 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4528 output
, (void *)outputCount
, 0 );
4531 err
= (object
->*func
)( reference
,
4532 ARG32(input
[0]), ARG32(input
[1]),
4533 output
, (void *)outputCount
, 0, 0 );
4536 err
= (object
->*func
)( reference
,
4538 output
, (void *)outputCount
, 0, 0, 0 );
4541 err
= (object
->*func
)( reference
,
4542 output
, (void *)outputCount
, 0, 0, 0, 0 );
4546 IOLog("%s: Bad method table\n", object
->getName());
4554 /* Routine io_connect_method_scalarI_structureI */
4555 kern_return_t
is_io_connect_method_scalarI_structureI(
4556 io_connect_t connect
,
4558 io_scalar_inband_t input
,
4559 mach_msg_type_number_t inputCount
,
4560 io_struct_inband_t inputStruct
,
4561 mach_msg_type_number_t inputStructCount
)
4564 io_scalar_inband64_t _input
;
4566 mach_msg_type_number_t scalar_outputCnt
= 0;
4567 mach_msg_type_number_t inband_outputCnt
= 0;
4568 mach_vm_size_t ool_output_size
= 0;
4570 for (i
= 0; i
< inputCount
; i
++)
4571 _input
[i
] = SCALAR64(input
[i
]);
4573 return (is_io_connect_method(connect
, index
,
4575 inputStruct
, inputStructCount
,
4577 NULL
, &inband_outputCnt
,
4578 NULL
, &scalar_outputCnt
,
4579 0, &ool_output_size
));
4582 kern_return_t
shim_io_connect_method_scalarI_structureI(
4583 IOExternalMethod
* method
,
4585 const io_user_scalar_t
* input
,
4586 mach_msg_type_number_t inputCount
,
4587 io_struct_inband_t inputStruct
,
4588 mach_msg_type_number_t inputStructCount
)
4591 IOReturn err
= kIOReturnBadArgument
;
4595 if (inputCount
!= method
->count0
)
4597 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
);
4598 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4601 if( (kIOUCVariableStructureSize
!= method
->count1
)
4602 && (inputStructCount
!= method
->count1
))
4604 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputStructCount
, (uint64_t)method
->count1
, (uint64_t)kIOUCVariableStructureSize
);
4605 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputStructCount
, uint64_t, (uint64_t)method
->count1
);
4609 func
= method
->func
;
4611 switch( inputCount
) {
4614 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4615 ARG32(input
[3]), ARG32(input
[4]),
4619 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), (void *) input
[2],
4621 inputStruct
, (void *)(uintptr_t)inputStructCount
);
4624 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4625 inputStruct
, (void *)(uintptr_t)inputStructCount
,
4629 err
= (object
->*func
)( ARG32(input
[0]), ARG32(input
[1]),
4630 inputStruct
, (void *)(uintptr_t)inputStructCount
,
4634 err
= (object
->*func
)( ARG32(input
[0]),
4635 inputStruct
, (void *)(uintptr_t)inputStructCount
,
4639 err
= (object
->*func
)( inputStruct
, (void *)(uintptr_t)inputStructCount
,
4644 IOLog("%s: Bad method table\n", object
->getName());
4652 kern_return_t
shim_io_async_method_scalarI_structureI(
4653 IOExternalAsyncMethod
* method
,
4655 mach_port_t asyncWakePort
,
4656 io_user_reference_t
* asyncReference
,
4657 uint32_t asyncReferenceCount
,
4658 const io_user_scalar_t
* input
,
4659 mach_msg_type_number_t inputCount
,
4660 io_struct_inband_t inputStruct
,
4661 mach_msg_type_number_t inputStructCount
)
4665 IOReturn err
= kIOReturnBadArgument
;
4666 io_async_ref_t reference
;
4668 for (i
= 0; i
< asyncReferenceCount
; i
++)
4669 reference
[i
] = REF32(asyncReference
[i
]);
4673 if (inputCount
!= method
->count0
)
4675 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
);
4676 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4679 if( (kIOUCVariableStructureSize
!= method
->count1
)
4680 && (inputStructCount
!= method
->count1
))
4682 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputStructCount
, (uint64_t)method
->count1
, (uint64_t)kIOUCVariableStructureSize
);
4683 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputStructCount
, uint64_t, (uint64_t)method
->count1
);
4687 func
= method
->func
;
4689 switch( inputCount
) {
4692 err
= (object
->*func
)( reference
,
4693 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4694 ARG32(input
[3]), ARG32(input
[4]),
4698 err
= (object
->*func
)( reference
,
4699 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4701 inputStruct
, (void *)(uintptr_t)inputStructCount
);
4704 err
= (object
->*func
)( reference
,
4705 ARG32(input
[0]), ARG32(input
[1]), ARG32(input
[2]),
4706 inputStruct
, (void *)(uintptr_t)inputStructCount
,
4710 err
= (object
->*func
)( reference
,
4711 ARG32(input
[0]), ARG32(input
[1]),
4712 inputStruct
, (void *)(uintptr_t)inputStructCount
,
4716 err
= (object
->*func
)( reference
,
4718 inputStruct
, (void *)(uintptr_t)inputStructCount
,
4722 err
= (object
->*func
)( reference
,
4723 inputStruct
, (void *)(uintptr_t)inputStructCount
,
4728 IOLog("%s: Bad method table\n", object
->getName());
4736 /* Routine io_connect_method_structureI_structureO */
4737 kern_return_t
is_io_connect_method_structureI_structureO(
4738 io_object_t connect
,
4740 io_struct_inband_t input
,
4741 mach_msg_type_number_t inputCount
,
4742 io_struct_inband_t output
,
4743 mach_msg_type_number_t
* outputCount
)
4745 mach_msg_type_number_t scalar_outputCnt
= 0;
4746 mach_vm_size_t ool_output_size
= 0;
4748 return (is_io_connect_method(connect
, index
,
4752 output
, outputCount
,
4753 NULL
, &scalar_outputCnt
,
4754 0, &ool_output_size
));
4757 kern_return_t
shim_io_connect_method_structureI_structureO(
4758 IOExternalMethod
* method
,
4760 io_struct_inband_t input
,
4761 mach_msg_type_number_t inputCount
,
4762 io_struct_inband_t output
,
4763 IOByteCount
* outputCount
)
4766 IOReturn err
= kIOReturnBadArgument
;
4770 if( (kIOUCVariableStructureSize
!= method
->count0
)
4771 && (inputCount
!= method
->count0
))
4773 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
, (uint64_t)kIOUCVariableStructureSize
);
4774 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4777 if( (kIOUCVariableStructureSize
!= method
->count1
)
4778 && (*outputCount
!= method
->count1
))
4780 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)*outputCount
, (uint64_t)method
->count1
, (uint64_t)kIOUCVariableStructureSize
);
4781 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)*outputCount
, uint64_t, (uint64_t)method
->count1
);
4785 func
= method
->func
;
4787 if( method
->count1
) {
4788 if( method
->count0
) {
4789 err
= (object
->*func
)( input
, output
,
4790 (void *)(uintptr_t)inputCount
, outputCount
, 0, 0 );
4792 err
= (object
->*func
)( output
, outputCount
, 0, 0, 0, 0 );
4795 err
= (object
->*func
)( input
, (void *)(uintptr_t)inputCount
, 0, 0, 0, 0 );
4804 kern_return_t
shim_io_async_method_structureI_structureO(
4805 IOExternalAsyncMethod
* method
,
4807 mach_port_t asyncWakePort
,
4808 io_user_reference_t
* asyncReference
,
4809 uint32_t asyncReferenceCount
,
4810 io_struct_inband_t input
,
4811 mach_msg_type_number_t inputCount
,
4812 io_struct_inband_t output
,
4813 mach_msg_type_number_t
* outputCount
)
4818 io_async_ref_t reference
;
4820 for (i
= 0; i
< asyncReferenceCount
; i
++)
4821 reference
[i
] = REF32(asyncReference
[i
]);
4823 err
= kIOReturnBadArgument
;
4826 if( (kIOUCVariableStructureSize
!= method
->count0
)
4827 && (inputCount
!= method
->count0
))
4829 IOLog("%s:%d %s: IOUserClient inputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)inputCount
, (uint64_t)method
->count0
, (uint64_t)kIOUCVariableStructureSize
);
4830 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)inputCount
, uint64_t, (uint64_t)method
->count0
);
4833 if( (kIOUCVariableStructureSize
!= method
->count1
)
4834 && (*outputCount
!= method
->count1
))
4836 IOLog("%s:%d %s: IOUserClient outputCount count mismatch 0x%llx 0x%llx 0x%llx\n", __FUNCTION__
, __LINE__
, object
->getName(), (uint64_t)*outputCount
, (uint64_t)method
->count1
, (uint64_t)kIOUCVariableStructureSize
);
4837 DTRACE_IO2(iokit_count_mismatch
, uint64_t, (uint64_t)*outputCount
, uint64_t, (uint64_t)method
->count1
);
4841 func
= method
->func
;
4843 if( method
->count1
) {
4844 if( method
->count0
) {
4845 err
= (object
->*func
)( reference
,
4847 (void *)(uintptr_t)inputCount
, outputCount
, 0, 0 );
4849 err
= (object
->*func
)( reference
,
4850 output
, outputCount
, 0, 0, 0, 0 );
4853 err
= (object
->*func
)( reference
,
4854 input
, (void *)(uintptr_t)inputCount
, 0, 0, 0, 0 );
4863 bool gIOKextdClearedBusy
= false;
4866 /* Routine io_catalog_send_data */
4867 kern_return_t
is_io_catalog_send_data(
4868 mach_port_t master_port
,
4870 io_buf_ptr_t inData
,
4871 mach_msg_type_number_t inDataCount
,
4872 kern_return_t
* result
)
4875 return kIOReturnNotPrivileged
;
4876 #else /* NO_KEXTD */
4879 kern_return_t kr
= kIOReturnError
;
4881 //printf("io_catalog_send_data called. flag: %d\n", flag);
4883 if( master_port
!= master_device_port
)
4884 return kIOReturnNotPrivileged
;
4886 if( (flag
!= kIOCatalogRemoveKernelLinker
&&
4887 flag
!= kIOCatalogKextdActive
&&
4888 flag
!= kIOCatalogKextdFinishedLaunching
) &&
4889 ( !inData
|| !inDataCount
) )
4891 return kIOReturnBadArgument
;
4894 if (!IOTaskHasEntitlement(current_task(), "com.apple.rootless.kext-management"))
4896 OSString
* taskName
= IOCopyLogNameForPID(proc_selfpid());
4897 IOLog("IOCatalogueSendData(%s): Not entitled\n", taskName
? taskName
->getCStringNoCopy() : "");
4898 OSSafeReleaseNULL(taskName
);
4899 // For now, fake success to not break applications relying on this function succeeding.
4900 // See <rdar://problem/32554970> for more details.
4901 return kIOReturnSuccess
;
4905 vm_map_offset_t map_data
;
4907 if( inDataCount
> sizeof(io_struct_inband_t
) * 1024)
4908 return( kIOReturnMessageTooLarge
);
4910 kr
= vm_map_copyout( kernel_map
, &map_data
, (vm_map_copy_t
)inData
);
4911 data
= CAST_DOWN(vm_offset_t
, map_data
);
4913 if( kr
!= KERN_SUCCESS
)
4916 // must return success after vm_map_copyout() succeeds
4919 obj
= (OSObject
*)OSUnserializeXML((const char *)data
, inDataCount
);
4920 vm_deallocate( kernel_map
, data
, inDataCount
);
4922 *result
= kIOReturnNoMemory
;
4923 return( KERN_SUCCESS
);
4929 case kIOCatalogResetDrivers
:
4930 case kIOCatalogResetDriversNoMatch
: {
4933 array
= OSDynamicCast(OSArray
, obj
);
4935 if ( !gIOCatalogue
->resetAndAddDrivers(array
,
4936 flag
== kIOCatalogResetDrivers
) ) {
4938 kr
= kIOReturnError
;
4941 kr
= kIOReturnBadArgument
;
4946 case kIOCatalogAddDrivers
:
4947 case kIOCatalogAddDriversNoMatch
: {
4950 array
= OSDynamicCast(OSArray
, obj
);
4952 if ( !gIOCatalogue
->addDrivers( array
,
4953 flag
== kIOCatalogAddDrivers
) ) {
4954 kr
= kIOReturnError
;
4958 kr
= kIOReturnBadArgument
;
4963 case kIOCatalogRemoveDrivers
:
4964 case kIOCatalogRemoveDriversNoMatch
: {
4965 OSDictionary
* dict
;
4967 dict
= OSDynamicCast(OSDictionary
, obj
);
4969 if ( !gIOCatalogue
->removeDrivers( dict
,
4970 flag
== kIOCatalogRemoveDrivers
) ) {
4971 kr
= kIOReturnError
;
4975 kr
= kIOReturnBadArgument
;
4980 case kIOCatalogStartMatching
: {
4981 OSDictionary
* dict
;
4983 dict
= OSDynamicCast(OSDictionary
, obj
);
4985 if ( !gIOCatalogue
->startMatching( dict
) ) {
4986 kr
= kIOReturnError
;
4990 kr
= kIOReturnBadArgument
;
4995 case kIOCatalogRemoveKernelLinker
:
4996 kr
= KERN_NOT_SUPPORTED
;
4999 case kIOCatalogKextdActive
:
5001 IOServiceTrace(IOSERVICE_KEXTD_ALIVE
, 0, 0, 0, 0);
5002 OSKext::setKextdActive();
5004 /* Dump all nonloaded startup extensions; kextd will now send them
5007 OSKext::flushNonloadedKexts( /* flushPrelinkedKexts */ false);
5009 kr
= kIOReturnSuccess
;
5012 case kIOCatalogKextdFinishedLaunching
: {
5014 if (!gIOKextdClearedBusy
) {
5015 IOService
* serviceRoot
= IOService::getServiceRoot();
5017 IOServiceTrace(IOSERVICE_KEXTD_READY
, 0, 0, 0, 0);
5018 serviceRoot
->adjustBusy(-1);
5019 gIOKextdClearedBusy
= true;
5023 kr
= kIOReturnSuccess
;
5028 kr
= kIOReturnBadArgument
;
5032 if (obj
) obj
->release();
5035 return( KERN_SUCCESS
);
5036 #endif /* NO_KEXTD */
5039 /* Routine io_catalog_terminate */
5040 kern_return_t
is_io_catalog_terminate(
5041 mach_port_t master_port
,
5047 if( master_port
!= master_device_port
)
5048 return kIOReturnNotPrivileged
;
5050 kr
= IOUserClient::clientHasPrivilege( (void *) current_task(),
5051 kIOClientPrivilegeAdministrator
);
5052 if( kIOReturnSuccess
!= kr
)
5056 #if !defined(SECURE_KERNEL)
5057 case kIOCatalogServiceTerminate
:
5059 IOService
* service
;
5061 iter
= IORegistryIterator::iterateOver(gIOServicePlane
,
5062 kIORegistryIterateRecursively
);
5064 return kIOReturnNoMemory
;
5068 while( (service
= (IOService
*)iter
->getNextObject()) ) {
5069 if( service
->metaCast(name
)) {
5070 if ( !service
->terminate( kIOServiceRequired
5071 | kIOServiceSynchronous
) ) {
5072 kr
= kIOReturnUnsupported
;
5077 } while( !service
&& !iter
->isValid());
5081 case kIOCatalogModuleUnload
:
5082 case kIOCatalogModuleTerminate
:
5083 kr
= gIOCatalogue
->terminateDriversForModule(name
,
5084 flag
== kIOCatalogModuleUnload
);
5089 kr
= kIOReturnBadArgument
;
5096 /* Routine io_catalog_get_data */
5097 kern_return_t
is_io_catalog_get_data(
5098 mach_port_t master_port
,
5100 io_buf_ptr_t
*outData
,
5101 mach_msg_type_number_t
*outDataCount
)
5103 kern_return_t kr
= kIOReturnSuccess
;
5106 if( master_port
!= master_device_port
)
5107 return kIOReturnNotPrivileged
;
5109 //printf("io_catalog_get_data called. flag: %d\n", flag);
5111 s
= OSSerialize::withCapacity(4096);
5113 return kIOReturnNoMemory
;
5115 kr
= gIOCatalogue
->serializeData(flag
, s
);
5117 if ( kr
== kIOReturnSuccess
) {
5122 size
= s
->getLength();
5123 kr
= vm_allocate(kernel_map
, &data
, size
, VM_FLAGS_ANYWHERE
);
5124 if ( kr
== kIOReturnSuccess
) {
5125 bcopy(s
->text(), (void *)data
, size
);
5126 kr
= vm_map_copyin(kernel_map
, (vm_map_address_t
)data
,
5127 (vm_map_size_t
)size
, true, ©
);
5128 *outData
= (char *)copy
;
5129 *outDataCount
= size
;
5138 /* Routine io_catalog_get_gen_count */
5139 kern_return_t
is_io_catalog_get_gen_count(
5140 mach_port_t master_port
,
5143 if( master_port
!= master_device_port
)
5144 return kIOReturnNotPrivileged
;
5146 //printf("io_catalog_get_gen_count called.\n");
5149 return kIOReturnBadArgument
;
5151 *genCount
= gIOCatalogue
->getGenerationCount();
5153 return kIOReturnSuccess
;
5156 /* Routine io_catalog_module_loaded.
5157 * Is invoked from IOKitLib's IOCatalogueModuleLoaded(). Doesn't seem to be used.
5159 kern_return_t
is_io_catalog_module_loaded(
5160 mach_port_t master_port
,
5163 if( master_port
!= master_device_port
)
5164 return kIOReturnNotPrivileged
;
5166 //printf("io_catalog_module_loaded called. name %s\n", name);
5169 return kIOReturnBadArgument
;
5171 gIOCatalogue
->moduleHasLoaded(name
);
5173 return kIOReturnSuccess
;
5176 kern_return_t
is_io_catalog_reset(
5177 mach_port_t master_port
,
5180 if( master_port
!= master_device_port
)
5181 return kIOReturnNotPrivileged
;
5184 case kIOCatalogResetDefault
:
5185 gIOCatalogue
->reset();
5189 return kIOReturnBadArgument
;
5192 return kIOReturnSuccess
;
5195 kern_return_t
iokit_user_client_trap(struct iokit_user_client_trap_args
*args
)
5197 kern_return_t result
= kIOReturnBadArgument
;
5198 IOUserClient
*userClient
;
5200 if ((userClient
= OSDynamicCast(IOUserClient
,
5201 iokit_lookup_connect_ref_current_task((OSObject
*)(args
->userClientRef
))))) {
5202 IOExternalTrap
*trap
;
5203 IOService
*target
= NULL
;
5205 trap
= userClient
->getTargetAndTrapForIndex(&target
, args
->index
);
5207 if (trap
&& target
) {
5213 result
= (target
->*func
)(args
->p1
, args
->p2
, args
->p3
, args
->p4
, args
->p5
, args
->p6
);
5217 iokit_remove_connect_reference(userClient
);
5225 IOReturn
IOUserClient::externalMethod( uint32_t selector
, IOExternalMethodArguments
* args
,
5226 IOExternalMethodDispatch
* dispatch
, OSObject
* target
, void * reference
)
5230 IOByteCount structureOutputSize
;
5235 count
= dispatch
->checkScalarInputCount
;
5236 if ((kIOUCVariableStructureSize
!= count
) && (count
!= args
->scalarInputCount
))
5238 return (kIOReturnBadArgument
);
5241 count
= dispatch
->checkStructureInputSize
;
5242 if ((kIOUCVariableStructureSize
!= count
)
5243 && (count
!= ((args
->structureInputDescriptor
)
5244 ? args
->structureInputDescriptor
->getLength() : args
->structureInputSize
)))
5246 return (kIOReturnBadArgument
);
5249 count
= dispatch
->checkScalarOutputCount
;
5250 if ((kIOUCVariableStructureSize
!= count
) && (count
!= args
->scalarOutputCount
))
5252 return (kIOReturnBadArgument
);
5255 count
= dispatch
->checkStructureOutputSize
;
5256 if ((kIOUCVariableStructureSize
!= count
)
5257 && (count
!= ((args
->structureOutputDescriptor
)
5258 ? args
->structureOutputDescriptor
->getLength() : args
->structureOutputSize
)))
5260 return (kIOReturnBadArgument
);
5263 if (dispatch
->function
)
5264 err
= (*dispatch
->function
)(target
, reference
, args
);
5266 err
= kIOReturnNoCompletion
; /* implementator can dispatch */
5272 // pre-Leopard API's don't do ool structs
5273 if (args
->structureInputDescriptor
|| args
->structureOutputDescriptor
)
5275 err
= kIOReturnIPCError
;
5279 structureOutputSize
= args
->structureOutputSize
;
5281 if (args
->asyncWakePort
)
5283 IOExternalAsyncMethod
* method
;
5285 if( !(method
= getAsyncTargetAndMethodForIndex(&object
, selector
)) || !object
)
5286 return (kIOReturnUnsupported
);
5288 if (kIOUCForegroundOnly
& method
->flags
)
5290 if (task_is_gpu_denied(current_task()))
5291 return (kIOReturnNotPermitted
);
5294 switch (method
->flags
& kIOUCTypeMask
)
5296 case kIOUCScalarIStructI
:
5297 err
= shim_io_async_method_scalarI_structureI( method
, object
,
5298 args
->asyncWakePort
, args
->asyncReference
, args
->asyncReferenceCount
,
5299 args
->scalarInput
, args
->scalarInputCount
,
5300 (char *)args
->structureInput
, args
->structureInputSize
);
5303 case kIOUCScalarIScalarO
:
5304 err
= shim_io_async_method_scalarI_scalarO( method
, object
,
5305 args
->asyncWakePort
, args
->asyncReference
, args
->asyncReferenceCount
,
5306 args
->scalarInput
, args
->scalarInputCount
,
5307 args
->scalarOutput
, &args
->scalarOutputCount
);
5310 case kIOUCScalarIStructO
:
5311 err
= shim_io_async_method_scalarI_structureO( method
, object
,
5312 args
->asyncWakePort
, args
->asyncReference
, args
->asyncReferenceCount
,
5313 args
->scalarInput
, args
->scalarInputCount
,
5314 (char *) args
->structureOutput
, &args
->structureOutputSize
);
5318 case kIOUCStructIStructO
:
5319 err
= shim_io_async_method_structureI_structureO( method
, object
,
5320 args
->asyncWakePort
, args
->asyncReference
, args
->asyncReferenceCount
,
5321 (char *)args
->structureInput
, args
->structureInputSize
,
5322 (char *) args
->structureOutput
, &args
->structureOutputSize
);
5326 err
= kIOReturnBadArgument
;
5332 IOExternalMethod
* method
;
5334 if( !(method
= getTargetAndMethodForIndex(&object
, selector
)) || !object
)
5335 return (kIOReturnUnsupported
);
5337 if (kIOUCForegroundOnly
& method
->flags
)
5339 if (task_is_gpu_denied(current_task()))
5340 return (kIOReturnNotPermitted
);
5343 switch (method
->flags
& kIOUCTypeMask
)
5345 case kIOUCScalarIStructI
:
5346 err
= shim_io_connect_method_scalarI_structureI( method
, object
,
5347 args
->scalarInput
, args
->scalarInputCount
,
5348 (char *) args
->structureInput
, args
->structureInputSize
);
5351 case kIOUCScalarIScalarO
:
5352 err
= shim_io_connect_method_scalarI_scalarO( method
, object
,
5353 args
->scalarInput
, args
->scalarInputCount
,
5354 args
->scalarOutput
, &args
->scalarOutputCount
);
5357 case kIOUCScalarIStructO
:
5358 err
= shim_io_connect_method_scalarI_structureO( method
, object
,
5359 args
->scalarInput
, args
->scalarInputCount
,
5360 (char *) args
->structureOutput
, &structureOutputSize
);
5364 case kIOUCStructIStructO
:
5365 err
= shim_io_connect_method_structureI_structureO( method
, object
,
5366 (char *) args
->structureInput
, args
->structureInputSize
,
5367 (char *) args
->structureOutput
, &structureOutputSize
);
5371 err
= kIOReturnBadArgument
;
5376 args
->structureOutputSize
= structureOutputSize
;
5382 OSMetaClassDefineReservedUnused(IOUserClient
, 0);
5383 OSMetaClassDefineReservedUnused(IOUserClient
, 1);
5385 OSMetaClassDefineReservedUsed(IOUserClient
, 0);
5386 OSMetaClassDefineReservedUsed(IOUserClient
, 1);
5388 OSMetaClassDefineReservedUnused(IOUserClient
, 2);
5389 OSMetaClassDefineReservedUnused(IOUserClient
, 3);
5390 OSMetaClassDefineReservedUnused(IOUserClient
, 4);
5391 OSMetaClassDefineReservedUnused(IOUserClient
, 5);
5392 OSMetaClassDefineReservedUnused(IOUserClient
, 6);
5393 OSMetaClassDefineReservedUnused(IOUserClient
, 7);
5394 OSMetaClassDefineReservedUnused(IOUserClient
, 8);
5395 OSMetaClassDefineReservedUnused(IOUserClient
, 9);
5396 OSMetaClassDefineReservedUnused(IOUserClient
, 10);
5397 OSMetaClassDefineReservedUnused(IOUserClient
, 11);
5398 OSMetaClassDefineReservedUnused(IOUserClient
, 12);
5399 OSMetaClassDefineReservedUnused(IOUserClient
, 13);
5400 OSMetaClassDefineReservedUnused(IOUserClient
, 14);
5401 OSMetaClassDefineReservedUnused(IOUserClient
, 15);