2 * Copyright (c) 2000, 2001, 2004, 2006, 2009, 2010 Apple Inc. All rights reserved.
4 * @APPLE_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. Please obtain a copy of the License at
10 * http://www.opensource.apple.com/apsl/ and read it before using this
13 * The Original Code and all software distributed under the License are
14 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
15 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
16 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
18 * Please see the License for the specific language governing rights and
19 * limitations under the License.
21 * @APPLE_LICENSE_HEADER_END@
25 * Modification History
27 * June 1, 2001 Allan Nathanson <ajn@apple.com>
28 * - public API conversion
30 * March 31, 2000 Allan Nathanson <ajn@apple.com>
34 #include <mach/mach.h>
35 #include <mach/mach_error.h>
37 #include <SystemConfiguration/SystemConfiguration.h>
38 #include <SystemConfiguration/SCPrivate.h>
39 #include "SCDynamicStoreInternal.h"
40 #include "config.h" /* MiG generated file */
43 waitForMachMessage(mach_port_t port
)
46 u_int8_t buf
[sizeof(mach_msg_empty_t
) + MAX_TRAILER_SIZE
];
47 mach_msg_empty_rcv_t msg
;
51 status
= mach_msg(¬ify_msg
.msg
.header
, /* msg */
52 MACH_RCV_MSG
, /* options */
54 sizeof(notify_msg
), /* rcv_size */
56 MACH_MSG_TIMEOUT_NONE
, /* timeout */
57 MACH_PORT_NULL
); /* notify */
58 if (status
!= KERN_SUCCESS
) {
59 SCLog(TRUE
, LOG_DEBUG
, CFSTR("waitForMachMessage mach_msg(): %s"), mach_error_string(status
));
63 return notify_msg
.msg
.header
.msgh_id
;
68 SCDynamicStoreNotifyWait(SCDynamicStoreRef store
)
70 SCDynamicStorePrivateRef storePrivate
= (SCDynamicStorePrivateRef
)store
;
73 mach_port_t oldNotify
;
78 /* sorry, you must provide a session */
79 _SCErrorSet(kSCStatusNoStoreSession
);
83 if (storePrivate
->server
== MACH_PORT_NULL
) {
84 /* sorry, you must have an open session to play */
85 _SCErrorSet(kSCStatusNoStoreServer
);
89 if (storePrivate
->notifyStatus
!= NotifierNotRegistered
) {
90 /* sorry, you can only have one notification registered at once */
91 _SCErrorSet(kSCStatusNotifierActive
);
95 /* Allocating port (for server response) */
96 status
= mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE
, &port
);
97 if (status
!= KERN_SUCCESS
) {
98 SCLog(TRUE
, LOG_ERR
, CFSTR("SCDynamicStoreNotifyWait mach_port_allocate(): %s"), mach_error_string(status
));
103 status
= mach_port_insert_right(mach_task_self(),
106 MACH_MSG_TYPE_MAKE_SEND
);
107 if (status
!= KERN_SUCCESS
) {
109 * We can't insert a send right into our own port! This should
110 * only happen if someone stomped on OUR port (so let's leave
113 SCLog(TRUE
, LOG_ERR
, CFSTR("SCDynamicStoreNotifyWait mach_port_insert_right(): %s"), mach_error_string(status
));
118 /* Request a notification when/if the server dies */
119 status
= mach_port_request_notification(mach_task_self(),
121 MACH_NOTIFY_NO_SENDERS
,
124 MACH_MSG_TYPE_MAKE_SEND_ONCE
,
126 if (status
!= KERN_SUCCESS
) {
128 * We can't request a notification for our own port! This should
129 * only happen if someone stomped on OUR port (so let's leave
132 SCLog(TRUE
, LOG_ERR
, CFSTR("SCDynamicStoreNotifyWait mach_port_request_notification(): %s"), mach_error_string(status
));
137 if (oldNotify
!= MACH_PORT_NULL
) {
138 SCLog(TRUE
, LOG_ERR
, CFSTR("SCDynamicStoreNotifyWait(): oldNotify != MACH_PORT_NULL"));
143 status
= notifyviaport(storePrivate
->server
,
148 if (status
!= KERN_SUCCESS
) {
149 if ((status
== MACH_SEND_INVALID_DEST
) || (status
== MIG_SERVER_DIED
)) {
150 /* the server's gone and our session port's dead, remove the dead name right */
151 (void) mach_port_deallocate(mach_task_self(), storePrivate
->server
);
153 /* we got an unexpected error, leave the [session] port alone */
154 SCLog(TRUE
, LOG_ERR
, CFSTR("SCDynamicStoreNotifyWait notifyviaport(): %s"), mach_error_string(status
));
156 storePrivate
->server
= MACH_PORT_NULL
;
157 if ((status
== MACH_SEND_INVALID_DEST
) || (status
== MIG_SERVER_DIED
)) {
158 if (__SCDynamicStoreReconnect(store
)) {
161 /* remove the send right that we tried (but failed) to pass to the server */
162 (void) mach_port_deallocate(mach_task_self(), port
);
165 /* remove our receive right */
166 (void) mach_port_mod_refs(mach_task_self(), port
, MACH_PORT_RIGHT_RECEIVE
, -1);
170 if (sc_status
!= kSCStatusOK
) {
171 _SCErrorSet(sc_status
);
175 /* set notifier active */
176 storePrivate
->notifyStatus
= Using_NotifierWait
;
178 msgid
= waitForMachMessage(port
);
180 /* set notifier inactive */
181 storePrivate
->notifyStatus
= NotifierNotRegistered
;
183 if (msgid
== MACH_NOTIFY_NO_SENDERS
) {
184 /* the server closed the notifier port */
186 SCLog(_sc_verbose
, LOG_DEBUG
, CFSTR("SCDynamicStoreNotifyWait notifier port closed, destroying port %d"), port
);
188 _SCErrorSet(kSCStatusNoStoreServer
);
193 /* one of the mach routines returned an error */
195 SCLog(_sc_verbose
, LOG_DEBUG
, CFSTR("SCDynamicStoreNotifyWait communication with server failed, destroying port %d"), port
);
197 (void) mach_port_mod_refs(mach_task_self(), port
, MACH_PORT_RIGHT_RECEIVE
, -1);
198 _SCErrorSet(kSCStatusNoStoreServer
);
202 // something changed, cancelling notification request
203 status
= notifycancel(storePrivate
->server
,
206 if (status
!= KERN_SUCCESS
) {
207 if ((status
== MACH_SEND_INVALID_DEST
) || (status
== MIG_SERVER_DIED
)) {
208 /* the server's gone and our session port's dead, remove the dead name right */
209 (void) mach_port_deallocate(mach_task_self(), storePrivate
->server
);
211 /* we got an unexpected error, leave the [session] port alone */
212 SCLog(TRUE
, LOG_ERR
, CFSTR("SCDynamicStoreNotifyWait notifycancel(): %s"), mach_error_string(status
));
214 storePrivate
->server
= MACH_PORT_NULL
;
215 if (((status
== MACH_SEND_INVALID_DEST
) || (status
== MIG_SERVER_DIED
)) &&
216 __SCDynamicStoreReconnect(store
)) {
217 sc_status
= kSCStatusOK
;
223 /* remove our receive right */
224 (void) mach_port_mod_refs(mach_task_self(), port
, MACH_PORT_RIGHT_RECEIVE
, -1);
226 if (sc_status
!= kSCStatusOK
) {
227 _SCErrorSet(sc_status
);