blob: 86bd5028266bca8e037bcce57fee6777106c83bf [file] [log] [blame]
Irfan Sheriff7d024d32012-03-22 17:01:39 -07001/*
2 * Copyright (C) 2012 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.net.nsd;
18
Irfan Sheriff3ef889b2012-04-17 23:15:29 -070019import android.annotation.SdkConstant;
20import android.annotation.SdkConstant.SdkConstantType;
Irfan Sheriff7d024d32012-03-22 17:01:39 -070021import android.content.Context;
Irfan Sheriff7d024d32012-03-22 17:01:39 -070022import android.os.Handler;
Irfan Sheriff22af38c2012-05-03 16:44:27 -070023import android.os.HandlerThread;
Irfan Sheriff7d024d32012-03-22 17:01:39 -070024import android.os.Looper;
25import android.os.Message;
26import android.os.RemoteException;
27import android.os.Messenger;
Irfan Sheriff92784672012-04-13 12:15:41 -070028import android.text.TextUtils;
Irfan Sheriff7d024d32012-03-22 17:01:39 -070029import android.util.Log;
Irfan Sheriff22af38c2012-05-03 16:44:27 -070030import android.util.SparseArray;
31
32import java.util.concurrent.CountDownLatch;
Irfan Sheriff7d024d32012-03-22 17:01:39 -070033
34import com.android.internal.util.AsyncChannel;
35import com.android.internal.util.Protocol;
36
37/**
Irfan Sheriff92784672012-04-13 12:15:41 -070038 * The Network Service Discovery Manager class provides the API to discover services
39 * on a network. As an example, if device A and device B are connected over a Wi-Fi
40 * network, a game registered on device A can be discovered by a game on device
41 * B. Another example use case is an application discovering printers on the network.
42 *
43 * <p> The API currently supports DNS based service discovery and discovery is currently
Irfan Sheriff22af38c2012-05-03 16:44:27 -070044 * limited to a local network over Multicast DNS. DNS service discovery is described at
45 * http://files.dns-sd.org/draft-cheshire-dnsext-dns-sd.txt
Irfan Sheriff7d024d32012-03-22 17:01:39 -070046 *
47 * <p> The API is asynchronous and responses to requests from an application are on listener
Irfan Sheriff22af38c2012-05-03 16:44:27 -070048 * callbacks on a seperate thread.
Irfan Sheriff7d024d32012-03-22 17:01:39 -070049 *
Irfan Sheriff92784672012-04-13 12:15:41 -070050 * <p> There are three main operations the API supports - registration, discovery and resolution.
51 * <pre>
52 * Application start
53 * |
Irfan Sheriff22af38c2012-05-03 16:44:27 -070054 * |
55 * | onServiceRegistered()
56 * Register any local services /
57 * to be advertised with \
58 * registerService() onRegistrationFailed()
59 * |
60 * |
61 * discoverServices()
62 * |
63 * Maintain a list to track
64 * discovered services
65 * |
66 * |--------->
67 * | |
68 * | onServiceFound()
69 * | |
70 * | add service to list
71 * | |
72 * |<----------
73 * |
74 * |--------->
75 * | |
76 * | onServiceLost()
77 * | |
78 * | remove service from list
79 * | |
80 * |<----------
81 * |
82 * |
83 * | Connect to a service
84 * | from list ?
85 * |
86 * resolveService()
87 * |
88 * onServiceResolved()
89 * |
90 * Establish connection to service
91 * with the host and port information
Irfan Sheriff92784672012-04-13 12:15:41 -070092 *
93 * </pre>
94 * An application that needs to advertise itself over a network for other applications to
95 * discover it can do so with a call to {@link #registerService}. If Example is a http based
96 * application that can provide HTML data to peer services, it can register a name "Example"
97 * with service type "_http._tcp". A successful registration is notified with a callback to
Irfan Sheriff22af38c2012-05-03 16:44:27 -070098 * {@link RegistrationListener#onServiceRegistered} and a failure to register is notified
99 * over {@link RegistrationListener#onRegistrationFailed}
Irfan Sheriff92784672012-04-13 12:15:41 -0700100 *
101 * <p> A peer application looking for http services can initiate a discovery for "_http._tcp"
102 * with a call to {@link #discoverServices}. A service found is notified with a callback
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700103 * to {@link DiscoveryListener#onServiceFound} and a service lost is notified on
104 * {@link DiscoveryListener#onServiceLost}.
Irfan Sheriff92784672012-04-13 12:15:41 -0700105 *
106 * <p> Once the peer application discovers the "Example" http srevice, and needs to receive data
107 * from the "Example" application, it can initiate a resolve with {@link #resolveService} to
108 * resolve the host and port details for the purpose of establishing a connection. A successful
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700109 * resolve is notified on {@link ResolveListener#onServiceResolved} and a failure is notified
110 * on {@link ResolveListener#onResolveFailed}.
Irfan Sheriff92784672012-04-13 12:15:41 -0700111 *
112 * Applications can reserve for a service type at
113 * http://www.iana.org/form/ports-service. Existing services can be found at
114 * http://www.iana.org/assignments/service-names-port-numbers/service-names-port-numbers.xml
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700115 *
116 * Get an instance of this class by calling {@link android.content.Context#getSystemService(String)
117 * Context.getSystemService(Context.NSD_SERVICE)}.
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700118 *
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700119 * {@see NsdServiceInfo}
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700120 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700121public final class NsdManager {
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700122 private static final String TAG = "NsdManager";
123 INsdManager mService;
124
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700125 /**
126 * Broadcast intent action to indicate whether network service discovery is
127 * enabled or disabled. An extra {@link #EXTRA_NSD_STATE} provides the state
128 * information as int.
129 *
130 * @see #EXTRA_NSD_STATE
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700131 */
132 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION)
Irfan Sheriff6c07ba82012-04-17 23:23:42 -0700133 public static final String ACTION_NSD_STATE_CHANGED =
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700134 "android.net.nsd.STATE_CHANGED";
135
136 /**
137 * The lookup key for an int that indicates whether network service discovery is enabled
138 * or disabled. Retrieve it with {@link android.content.Intent#getIntExtra(String,int)}.
139 *
140 * @see #NSD_STATE_DISABLED
141 * @see #NSD_STATE_ENABLED
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700142 */
143 public static final String EXTRA_NSD_STATE = "nsd_state";
144
145 /**
146 * Network service discovery is disabled
147 *
Irfan Sheriff54ac7a52012-04-19 10:26:34 -0700148 * @see #ACTION_NSD_STATE_CHANGED
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700149 */
150 public static final int NSD_STATE_DISABLED = 1;
151
152 /**
153 * Network service discovery is enabled
154 *
Irfan Sheriff54ac7a52012-04-19 10:26:34 -0700155 * @see #ACTION_NSD_STATE_CHANGED
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700156 */
157 public static final int NSD_STATE_ENABLED = 2;
158
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700159 private static final int BASE = Protocol.BASE_NSD_MANAGER;
160
161 /** @hide */
162 public static final int DISCOVER_SERVICES = BASE + 1;
163 /** @hide */
164 public static final int DISCOVER_SERVICES_STARTED = BASE + 2;
165 /** @hide */
166 public static final int DISCOVER_SERVICES_FAILED = BASE + 3;
167 /** @hide */
168 public static final int SERVICE_FOUND = BASE + 4;
169 /** @hide */
170 public static final int SERVICE_LOST = BASE + 5;
171
172 /** @hide */
173 public static final int STOP_DISCOVERY = BASE + 6;
174 /** @hide */
175 public static final int STOP_DISCOVERY_FAILED = BASE + 7;
176 /** @hide */
177 public static final int STOP_DISCOVERY_SUCCEEDED = BASE + 8;
178
179 /** @hide */
180 public static final int REGISTER_SERVICE = BASE + 9;
181 /** @hide */
182 public static final int REGISTER_SERVICE_FAILED = BASE + 10;
183 /** @hide */
184 public static final int REGISTER_SERVICE_SUCCEEDED = BASE + 11;
185
186 /** @hide */
Irfan Sheriff92784672012-04-13 12:15:41 -0700187 public static final int UNREGISTER_SERVICE = BASE + 12;
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700188 /** @hide */
Irfan Sheriff92784672012-04-13 12:15:41 -0700189 public static final int UNREGISTER_SERVICE_FAILED = BASE + 13;
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700190 /** @hide */
Irfan Sheriff92784672012-04-13 12:15:41 -0700191 public static final int UNREGISTER_SERVICE_SUCCEEDED = BASE + 14;
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700192
193 /** @hide */
Irfan Sheriff92784672012-04-13 12:15:41 -0700194 public static final int RESOLVE_SERVICE = BASE + 18;
Irfan Sheriff817388e2012-04-11 14:52:19 -0700195 /** @hide */
Irfan Sheriff92784672012-04-13 12:15:41 -0700196 public static final int RESOLVE_SERVICE_FAILED = BASE + 19;
Irfan Sheriff817388e2012-04-11 14:52:19 -0700197 /** @hide */
Irfan Sheriff92784672012-04-13 12:15:41 -0700198 public static final int RESOLVE_SERVICE_SUCCEEDED = BASE + 20;
Irfan Sheriff817388e2012-04-11 14:52:19 -0700199
Irfan Sheriff92784672012-04-13 12:15:41 -0700200 /** @hide */
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700201 public static final int ENABLE = BASE + 24;
202 /** @hide */
203 public static final int DISABLE = BASE + 25;
204
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700205 /** @hide */
206 public static final int NATIVE_DAEMON_EVENT = BASE + 26;
207
208 /** Dns based service discovery protocol */
209 public static final int PROTOCOL_DNS_SD = 0x0001;
210
211 private Context mContext;
212
213 private static final int INVALID_LISTENER_KEY = 0;
Dave Platte7369bd2014-03-28 13:33:04 -0700214 private static final int BUSY_LISTENER_KEY = -1;
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700215 private int mListenerKey = 1;
216 private final SparseArray mListenerMap = new SparseArray();
217 private final SparseArray<NsdServiceInfo> mServiceMap = new SparseArray<NsdServiceInfo>();
218 private final Object mMapLock = new Object();
219
220 private final AsyncChannel mAsyncChannel = new AsyncChannel();
221 private ServiceHandler mHandler;
222 private final CountDownLatch mConnected = new CountDownLatch(1);
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700223
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700224 /**
225 * Create a new Nsd instance. Applications use
226 * {@link android.content.Context#getSystemService Context.getSystemService()} to retrieve
227 * {@link android.content.Context#NSD_SERVICE Context.NSD_SERVICE}.
228 * @param service the Binder interface
229 * @hide - hide this because it takes in a parameter of type INsdManager, which
230 * is a system private class.
231 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700232 public NsdManager(Context context, INsdManager service) {
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700233 mService = service;
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700234 mContext = context;
235 init();
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700236 }
237
238 /**
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700239 * Failures are passed with {@link RegistrationListener#onRegistrationFailed},
240 * {@link RegistrationListener#onUnregistrationFailed},
241 * {@link DiscoveryListener#onStartDiscoveryFailed},
242 * {@link DiscoveryListener#onStopDiscoveryFailed} or {@link ResolveListener#onResolveFailed}.
243 *
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700244 * Indicates that the operation failed due to an internal error.
245 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700246 public static final int FAILURE_INTERNAL_ERROR = 0;
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700247
248 /**
Irfan Sheriff817388e2012-04-11 14:52:19 -0700249 * Indicates that the operation failed because it is already active.
250 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700251 public static final int FAILURE_ALREADY_ACTIVE = 3;
Irfan Sheriff817388e2012-04-11 14:52:19 -0700252
253 /**
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700254 * Indicates that the operation failed because the maximum outstanding
255 * requests from the applications have reached.
Irfan Sheriff817388e2012-04-11 14:52:19 -0700256 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700257 public static final int FAILURE_MAX_LIMIT = 4;
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700258
Irfan Sheriff92784672012-04-13 12:15:41 -0700259 /** Interface for callback invocation for service discovery */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700260 public interface DiscoveryListener {
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700261
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700262 public void onStartDiscoveryFailed(String serviceType, int errorCode);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700263
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700264 public void onStopDiscoveryFailed(String serviceType, int errorCode);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700265
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700266 public void onDiscoveryStarted(String serviceType);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700267
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700268 public void onDiscoveryStopped(String serviceType);
269
270 public void onServiceFound(NsdServiceInfo serviceInfo);
271
272 public void onServiceLost(NsdServiceInfo serviceInfo);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700273
274 }
275
Irfan Sheriff92784672012-04-13 12:15:41 -0700276 /** Interface for callback invocation for service registration */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700277 public interface RegistrationListener {
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700278
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700279 public void onRegistrationFailed(NsdServiceInfo serviceInfo, int errorCode);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700280
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700281 public void onUnregistrationFailed(NsdServiceInfo serviceInfo, int errorCode);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700282
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700283 public void onServiceRegistered(NsdServiceInfo serviceInfo);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700284
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700285 public void onServiceUnregistered(NsdServiceInfo serviceInfo);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700286 }
287
Irfan Sheriff92784672012-04-13 12:15:41 -0700288 /** Interface for callback invocation for service resolution */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700289 public interface ResolveListener {
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700290
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700291 public void onResolveFailed(NsdServiceInfo serviceInfo, int errorCode);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700292
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700293 public void onServiceResolved(NsdServiceInfo serviceInfo);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700294 }
295
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700296 private class ServiceHandler extends Handler {
297 ServiceHandler(Looper looper) {
298 super(looper);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700299 }
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700300
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700301 @Override
302 public void handleMessage(Message message) {
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700303 switch (message.what) {
304 case AsyncChannel.CMD_CHANNEL_HALF_CONNECTED:
305 mAsyncChannel.sendMessage(AsyncChannel.CMD_CHANNEL_FULL_CONNECTION);
Dave Platt3fc376b2014-02-27 16:16:20 -0800306 return;
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700307 case AsyncChannel.CMD_CHANNEL_FULLY_CONNECTED:
Robert Greenwaltaf2eefb2013-05-02 15:45:32 -0700308 mConnected.countDown();
Dave Platt3fc376b2014-02-27 16:16:20 -0800309 return;
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700310 case AsyncChannel.CMD_CHANNEL_DISCONNECTED:
311 Log.e(TAG, "Channel lost");
Dave Platt3fc376b2014-02-27 16:16:20 -0800312 return;
313 default:
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700314 break;
Dave Platt3fc376b2014-02-27 16:16:20 -0800315 }
316 Object listener = getListener(message.arg2);
317 if (listener == null) {
318 Log.d(TAG, "Stale key " + message.arg2);
319 return;
320 }
Dave Platt3fc376b2014-02-27 16:16:20 -0800321 NsdServiceInfo ns = getNsdService(message.arg2);
322 switch (message.what) {
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700323 case DISCOVER_SERVICES_STARTED:
Dave Platt3fc376b2014-02-27 16:16:20 -0800324 String s = getNsdServiceInfoType((NsdServiceInfo) message.obj);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700325 ((DiscoveryListener) listener).onDiscoveryStarted(s);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700326 break;
327 case DISCOVER_SERVICES_FAILED:
Dave Platte7369bd2014-03-28 13:33:04 -0700328 removeListener(message.arg2);
Dave Platt3fc376b2014-02-27 16:16:20 -0800329 ((DiscoveryListener) listener).onStartDiscoveryFailed(getNsdServiceInfoType(ns),
330 message.arg1);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700331 break;
332 case SERVICE_FOUND:
333 ((DiscoveryListener) listener).onServiceFound((NsdServiceInfo) message.obj);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700334 break;
335 case SERVICE_LOST:
336 ((DiscoveryListener) listener).onServiceLost((NsdServiceInfo) message.obj);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700337 break;
338 case STOP_DISCOVERY_FAILED:
Dave Platte7369bd2014-03-28 13:33:04 -0700339 removeListener(message.arg2);
Dave Platt3fc376b2014-02-27 16:16:20 -0800340 ((DiscoveryListener) listener).onStopDiscoveryFailed(getNsdServiceInfoType(ns),
341 message.arg1);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700342 break;
343 case STOP_DISCOVERY_SUCCEEDED:
Dave Platte7369bd2014-03-28 13:33:04 -0700344 removeListener(message.arg2);
Dave Platt3fc376b2014-02-27 16:16:20 -0800345 ((DiscoveryListener) listener).onDiscoveryStopped(getNsdServiceInfoType(ns));
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700346 break;
347 case REGISTER_SERVICE_FAILED:
Dave Platte7369bd2014-03-28 13:33:04 -0700348 removeListener(message.arg2);
Dave Platt3fc376b2014-02-27 16:16:20 -0800349 ((RegistrationListener) listener).onRegistrationFailed(ns, message.arg1);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700350 break;
351 case REGISTER_SERVICE_SUCCEEDED:
352 ((RegistrationListener) listener).onServiceRegistered(
353 (NsdServiceInfo) message.obj);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700354 break;
355 case UNREGISTER_SERVICE_FAILED:
Dave Platte7369bd2014-03-28 13:33:04 -0700356 removeListener(message.arg2);
Dave Platt3fc376b2014-02-27 16:16:20 -0800357 ((RegistrationListener) listener).onUnregistrationFailed(ns, message.arg1);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700358 break;
359 case UNREGISTER_SERVICE_SUCCEEDED:
Dave Platte7369bd2014-03-28 13:33:04 -0700360 removeListener(message.arg2);
Dave Platt3fc376b2014-02-27 16:16:20 -0800361 ((RegistrationListener) listener).onServiceUnregistered(ns);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700362 break;
363 case RESOLVE_SERVICE_FAILED:
Dave Platte7369bd2014-03-28 13:33:04 -0700364 removeListener(message.arg2);
Dave Platt3fc376b2014-02-27 16:16:20 -0800365 ((ResolveListener) listener).onResolveFailed(ns, message.arg1);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700366 break;
367 case RESOLVE_SERVICE_SUCCEEDED:
Dave Platte7369bd2014-03-28 13:33:04 -0700368 removeListener(message.arg2);
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700369 ((ResolveListener) listener).onServiceResolved((NsdServiceInfo) message.obj);
370 break;
371 default:
372 Log.d(TAG, "Ignored " + message);
373 break;
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700374 }
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700375 }
Irfan Sheriff92784672012-04-13 12:15:41 -0700376 }
377
Dave Platte7369bd2014-03-28 13:33:04 -0700378 // if the listener is already in the map, reject it. Otherwise, add it and
379 // return its key.
380
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700381 private int putListener(Object listener, NsdServiceInfo s) {
382 if (listener == null) return INVALID_LISTENER_KEY;
383 int key;
384 synchronized (mMapLock) {
Dave Platte7369bd2014-03-28 13:33:04 -0700385 int valueIndex = mListenerMap.indexOfValue(listener);
386 if (valueIndex != -1) {
387 return BUSY_LISTENER_KEY;
388 }
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700389 do {
390 key = mListenerKey++;
391 } while (key == INVALID_LISTENER_KEY);
392 mListenerMap.put(key, listener);
393 mServiceMap.put(key, s);
394 }
395 return key;
396 }
397
398 private Object getListener(int key) {
399 if (key == INVALID_LISTENER_KEY) return null;
400 synchronized (mMapLock) {
401 return mListenerMap.get(key);
402 }
403 }
404
405 private NsdServiceInfo getNsdService(int key) {
406 synchronized (mMapLock) {
407 return mServiceMap.get(key);
408 }
409 }
410
411 private void removeListener(int key) {
412 if (key == INVALID_LISTENER_KEY) return;
413 synchronized (mMapLock) {
414 mListenerMap.remove(key);
415 mServiceMap.remove(key);
416 }
417 }
418
419 private int getListenerKey(Object listener) {
420 synchronized (mMapLock) {
421 int valueIndex = mListenerMap.indexOfValue(listener);
422 if (valueIndex != -1) {
423 return mListenerMap.keyAt(valueIndex);
424 }
425 }
426 return INVALID_LISTENER_KEY;
427 }
428
Dave Platt3fc376b2014-02-27 16:16:20 -0800429 private String getNsdServiceInfoType(NsdServiceInfo s) {
430 if (s == null) return "?";
431 return s.getServiceType();
432 }
433
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700434 /**
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700435 * Initialize AsyncChannel
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700436 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700437 private void init() {
438 final Messenger messenger = getMessenger();
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700439 if (messenger == null) throw new RuntimeException("Failed to initialize");
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700440 HandlerThread t = new HandlerThread("NsdManager");
441 t.start();
442 mHandler = new ServiceHandler(t.getLooper());
443 mAsyncChannel.connect(mContext, mHandler, messenger);
444 try {
445 mConnected.await();
446 } catch (InterruptedException e) {
447 Log.e(TAG, "interrupted wait at init");
448 }
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700449 }
450
451 /**
Irfan Sheriff92784672012-04-13 12:15:41 -0700452 * Register a service to be discovered by other services.
453 *
454 * <p> The function call immediately returns after sending a request to register service
Dave Platte7369bd2014-03-28 13:33:04 -0700455 * to the framework. The application is notified of a successful registration
456 * through the callback {@link RegistrationListener#onServiceRegistered} or a failure
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700457 * through {@link RegistrationListener#onRegistrationFailed}.
Irfan Sheriff92784672012-04-13 12:15:41 -0700458 *
Dave Platte7369bd2014-03-28 13:33:04 -0700459 * <p> The application should call {@link #unregisterService} when the service
460 * registration is no longer required, and/or whenever the application is stopped.
461 *
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700462 * @param serviceInfo The service being registered
463 * @param protocolType The service discovery protocol
464 * @param listener The listener notifies of a successful registration and is used to
465 * unregister this service through a call on {@link #unregisterService}. Cannot be null.
Dave Platte7369bd2014-03-28 13:33:04 -0700466 * Cannot be in use for an active service registration.
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700467 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700468 public void registerService(NsdServiceInfo serviceInfo, int protocolType,
469 RegistrationListener listener) {
470 if (TextUtils.isEmpty(serviceInfo.getServiceName()) ||
471 TextUtils.isEmpty(serviceInfo.getServiceType())) {
Irfan Sheriff92784672012-04-13 12:15:41 -0700472 throw new IllegalArgumentException("Service name or type cannot be empty");
473 }
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700474 if (serviceInfo.getPort() <= 0) {
Irfan Sheriff92784672012-04-13 12:15:41 -0700475 throw new IllegalArgumentException("Invalid port number");
476 }
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700477 if (listener == null) {
478 throw new IllegalArgumentException("listener cannot be null");
479 }
480 if (protocolType != PROTOCOL_DNS_SD) {
481 throw new IllegalArgumentException("Unsupported protocol");
482 }
Dave Platte7369bd2014-03-28 13:33:04 -0700483 int key = putListener(listener, serviceInfo);
484 if (key == BUSY_LISTENER_KEY) {
485 throw new IllegalArgumentException("listener already in use");
486 }
487 mAsyncChannel.sendMessage(REGISTER_SERVICE, 0, key, serviceInfo);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700488 }
489
Irfan Sheriff92784672012-04-13 12:15:41 -0700490 /**
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700491 * Unregister a service registered through {@link #registerService}. A successful
492 * unregister is notified to the application with a call to
493 * {@link RegistrationListener#onServiceUnregistered}.
494 *
495 * @param listener This should be the listener object that was passed to
496 * {@link #registerService}. It identifies the service that should be unregistered
Dave Platte7369bd2014-03-28 13:33:04 -0700497 * and notifies of a successful or unsuccessful unregistration via the listener
498 * callbacks. In API versions 20 and above, the listener object may be used for
499 * another service registration once the callback has been called. In API versions <= 19,
500 * there is no entirely reliable way to know when a listener may be re-used, and a new
501 * listener should be created for each service registration request.
Irfan Sheriff92784672012-04-13 12:15:41 -0700502 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700503 public void unregisterService(RegistrationListener listener) {
504 int id = getListenerKey(listener);
505 if (id == INVALID_LISTENER_KEY) {
506 throw new IllegalArgumentException("listener not registered");
507 }
508 if (listener == null) {
509 throw new IllegalArgumentException("listener cannot be null");
510 }
511 mAsyncChannel.sendMessage(UNREGISTER_SERVICE, 0, id);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700512 }
513
Irfan Sheriff92784672012-04-13 12:15:41 -0700514 /**
515 * Initiate service discovery to browse for instances of a service type. Service discovery
516 * consumes network bandwidth and will continue until the application calls
517 * {@link #stopServiceDiscovery}.
518 *
519 * <p> The function call immediately returns after sending a request to start service
520 * discovery to the framework. The application is notified of a success to initiate
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700521 * discovery through the callback {@link DiscoveryListener#onDiscoveryStarted} or a failure
522 * through {@link DiscoveryListener#onStartDiscoveryFailed}.
Irfan Sheriff92784672012-04-13 12:15:41 -0700523 *
524 * <p> Upon successful start, application is notified when a service is found with
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700525 * {@link DiscoveryListener#onServiceFound} or when a service is lost with
526 * {@link DiscoveryListener#onServiceLost}.
Irfan Sheriff92784672012-04-13 12:15:41 -0700527 *
528 * <p> Upon failure to start, service discovery is not active and application does
529 * not need to invoke {@link #stopServiceDiscovery}
530 *
Dave Platte7369bd2014-03-28 13:33:04 -0700531 * <p> The application should call {@link #stopServiceDiscovery} when discovery of this
532 * service type is no longer required, and/or whenever the application is paused or
533 * stopped.
534 *
Irfan Sheriff92784672012-04-13 12:15:41 -0700535 * @param serviceType The service type being discovered. Examples include "_http._tcp" for
536 * http services or "_ipp._tcp" for printers
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700537 * @param protocolType The service discovery protocol
538 * @param listener The listener notifies of a successful discovery and is used
539 * to stop discovery on this serviceType through a call on {@link #stopServiceDiscovery}.
Dave Platte7369bd2014-03-28 13:33:04 -0700540 * Cannot be null. Cannot be in use for an active service discovery.
Irfan Sheriff92784672012-04-13 12:15:41 -0700541 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700542 public void discoverServices(String serviceType, int protocolType, DiscoveryListener listener) {
Irfan Sheriff92784672012-04-13 12:15:41 -0700543 if (listener == null) {
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700544 throw new IllegalArgumentException("listener cannot be null");
Irfan Sheriff92784672012-04-13 12:15:41 -0700545 }
546 if (TextUtils.isEmpty(serviceType)) {
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700547 throw new IllegalArgumentException("Service type cannot be empty");
Irfan Sheriff92784672012-04-13 12:15:41 -0700548 }
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700549
550 if (protocolType != PROTOCOL_DNS_SD) {
551 throw new IllegalArgumentException("Unsupported protocol");
552 }
553
554 NsdServiceInfo s = new NsdServiceInfo();
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700555 s.setServiceType(serviceType);
Dave Platte7369bd2014-03-28 13:33:04 -0700556
557 int key = putListener(listener, s);
558 if (key == BUSY_LISTENER_KEY) {
559 throw new IllegalArgumentException("listener already in use");
560 }
561
562 mAsyncChannel.sendMessage(DISCOVER_SERVICES, 0, key, s);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700563 }
564
Irfan Sheriff92784672012-04-13 12:15:41 -0700565 /**
Dave Platte7369bd2014-03-28 13:33:04 -0700566 * Stop service discovery initiated with {@link #discoverServices}. An active service
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700567 * discovery is notified to the application with {@link DiscoveryListener#onDiscoveryStarted}
568 * and it stays active until the application invokes a stop service discovery. A successful
569 * stop is notified to with a call to {@link DiscoveryListener#onDiscoveryStopped}.
Irfan Sheriff92784672012-04-13 12:15:41 -0700570 *
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700571 * <p> Upon failure to stop service discovery, application is notified through
572 * {@link DiscoveryListener#onStopDiscoveryFailed}.
Irfan Sheriff92784672012-04-13 12:15:41 -0700573 *
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700574 * @param listener This should be the listener object that was passed to {@link #discoverServices}.
Dave Platte7369bd2014-03-28 13:33:04 -0700575 * It identifies the discovery that should be stopped and notifies of a successful or
576 * unsuccessful stop. In API versions 20 and above, the listener object may be used for
577 * another service discovery once the callback has been called. In API versions <= 19,
578 * there is no entirely reliable way to know when a listener may be re-used, and a new
579 * listener should be created for each service discovery request.
Irfan Sheriff92784672012-04-13 12:15:41 -0700580 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700581 public void stopServiceDiscovery(DiscoveryListener listener) {
582 int id = getListenerKey(listener);
583 if (id == INVALID_LISTENER_KEY) {
584 throw new IllegalArgumentException("service discovery not active on listener");
585 }
586 if (listener == null) {
587 throw new IllegalArgumentException("listener cannot be null");
588 }
589 mAsyncChannel.sendMessage(STOP_DISCOVERY, 0, id);
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700590 }
591
Irfan Sheriff92784672012-04-13 12:15:41 -0700592 /**
593 * Resolve a discovered service. An application can resolve a service right before
594 * establishing a connection to fetch the IP and port details on which to setup
595 * the connection.
596 *
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700597 * @param serviceInfo service to be resolved
Irfan Sheriff92784672012-04-13 12:15:41 -0700598 * @param listener to receive callback upon success or failure. Cannot be null.
Dave Platte7369bd2014-03-28 13:33:04 -0700599 * Cannot be in use for an active service resolution.
Irfan Sheriff92784672012-04-13 12:15:41 -0700600 */
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700601 public void resolveService(NsdServiceInfo serviceInfo, ResolveListener listener) {
602 if (TextUtils.isEmpty(serviceInfo.getServiceName()) ||
603 TextUtils.isEmpty(serviceInfo.getServiceType())) {
Irfan Sheriff92784672012-04-13 12:15:41 -0700604 throw new IllegalArgumentException("Service name or type cannot be empty");
605 }
Irfan Sheriff22af38c2012-05-03 16:44:27 -0700606 if (listener == null) {
607 throw new IllegalArgumentException("listener cannot be null");
608 }
Dave Platte7369bd2014-03-28 13:33:04 -0700609
610 int key = putListener(listener, serviceInfo);
611
612 if (key == BUSY_LISTENER_KEY) {
613 throw new IllegalArgumentException("listener already in use");
614 }
615 mAsyncChannel.sendMessage(RESOLVE_SERVICE, 0, key, serviceInfo);
Irfan Sheriff817388e2012-04-11 14:52:19 -0700616 }
617
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700618 /** Internal use only @hide */
619 public void setEnabled(boolean enabled) {
620 try {
621 mService.setEnabled(enabled);
Jeff Sharkeyc53962d2016-03-01 19:27:23 -0700622 } catch (RemoteException e) {
623 throw e.rethrowFromSystemServer();
624 }
Irfan Sheriff3ef889b2012-04-17 23:15:29 -0700625 }
626
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700627 /**
628 * Get a reference to NetworkService handler. This is used to establish
629 * an AsyncChannel communication with the service
630 *
631 * @return Messenger pointing to the NetworkService handler
632 */
633 private Messenger getMessenger() {
634 try {
635 return mService.getMessenger();
636 } catch (RemoteException e) {
Jeff Sharkeyc53962d2016-03-01 19:27:23 -0700637 throw e.rethrowFromSystemServer();
Irfan Sheriff7d024d32012-03-22 17:01:39 -0700638 }
639 }
640}