1 /*
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 
17 package android.net.wifi.p2p;
18 
19 import android.annotation.IntDef;
20 import android.annotation.IntRange;
21 import android.annotation.NonNull;
22 import android.annotation.Nullable;
23 import android.compat.annotation.UnsupportedAppUsage;
24 import android.os.Build;
25 import android.os.Parcel;
26 import android.os.Parcelable;
27 
28 import androidx.annotation.RequiresApi;
29 
30 import com.android.modules.utils.build.SdkLevel;
31 
32 import java.lang.annotation.Retention;
33 import java.lang.annotation.RetentionPolicy;
34 import java.util.Locale;
35 
36 /**
37  * A class representing Wifi Display information for a device.
38  *
39  * See Wifi Display technical specification v1.0.0, section 5.1.2.
40  * See Wifi Display technical specification v2.0.0, section 5.1.12 for Wifi Display R2.
41  */
42 public final class WifiP2pWfdInfo implements Parcelable {
43 
44     private boolean mEnabled;
45 
46     /** Device information bitmap */
47     private int mDeviceInfo;
48 
49     /** R2 Device information bitmap */
50     private int mR2DeviceInfo = -1;
51 
52     /** @hide */
53     @Retention(RetentionPolicy.SOURCE)
54     @IntDef(prefix = { "DEVICE_TYPE_" }, value = {
55             DEVICE_TYPE_WFD_SOURCE,
56             DEVICE_TYPE_PRIMARY_SINK,
57             DEVICE_TYPE_SECONDARY_SINK,
58             DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK})
59     public @interface DeviceType {}
60 
61     /** The device is a Wifi Display Source. */
62     public static final int DEVICE_TYPE_WFD_SOURCE = 0;
63     /** The device is a primary sink. */
64     public static final int DEVICE_TYPE_PRIMARY_SINK = 1;
65     /** The device is a secondary sink. This type is only supported by R1. */
66     public static final int DEVICE_TYPE_SECONDARY_SINK = 2;
67     /** The device is dual-role capable i.e. either a WFD source or a primary sink. */
68     public static final int DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK = 3;
69 
70     /** @hide */
71     @Retention(RetentionPolicy.SOURCE)
72     @IntDef(prefix = { "PREFERRED_CONNECTIVITY_" }, value = {
73             PREFERRED_CONNECTIVITY_P2P,
74             PREFERRED_CONNECTIVITY_TDLS})
75     public @interface PreferredConnectivity {}
76 
77     /** Wifi Display (WFD) preferred connectivity is Wifi Direct (P2P). */
78     public static final int PREFERRED_CONNECTIVITY_P2P = 0;
79     /** Wifi Display (WFD) preferred connectivity is TDLS. */
80     public static final int PREFERRED_CONNECTIVITY_TDLS = 1;
81 
82     /** @hide */
83     @Retention(RetentionPolicy.SOURCE)
84     @IntDef(flag = true, prefix = {"DEVICE_INFO_"}, value = {
85             DEVICE_INFO_DEVICE_TYPE_MASK,
86             DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SOURCE,
87             DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SINK,
88             DEVICE_INFO_SESSION_AVAILABLE_MASK,
89             DEVICE_INFO_WFD_SERVICE_DISCOVERY_SUPPORT,
90             DEVICE_INFO_PREFERRED_CONNECTIVITY_MASK,
91             DEVICE_INFO_CONTENT_PROTECTION_SUPPORT,
92             DEVICE_INFO_TIME_SYNCHRONIZATION_SUPPORT,
93             DEVICE_INFO_AUDIO_UNSUPPORTED_AT_PRIMARY_SINK,
94             DEVICE_INFO_AUDIO_ONLY_SUPPORT_AT_SOURCE,
95             DEVICE_INFO_TDLS_PERSISTENT_GROUP,
96             DEVICE_INFO_TDLS_PERSISTENT_GROUP_REINVOKE})
97     public @interface DeviceInfoMask {}
98 
99     /** @hide */
100     @Retention(RetentionPolicy.SOURCE)
101     @IntDef(flag = true, prefix = {"DEVICE_INFO_"}, value = {DEVICE_INFO_DEVICE_TYPE_MASK})
102     public @interface R2DeviceInfoMask {}
103 
104     /**
105      * {@link #getDeviceInfo()} & {@link #DEVICE_INFO_DEVICE_TYPE_MASK} is one of
106      * {@link #DEVICE_TYPE_WFD_SOURCE}, {@link #DEVICE_TYPE_PRIMARY_SINK},
107      * {@link #DEVICE_TYPE_SECONDARY_SINK} or {@link #DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK}.
108      *
109      * The bit definition is listed in 5.1.2 WFD Device Information Subelement and
110      * 5.1.12 WFD R2 Device Information Subelement in Wifi Display Technical Specification.
111      */
112     public static final int DEVICE_INFO_DEVICE_TYPE_MASK = 1 << 1 | 1 << 0;
113     /**
114      * Bit field for {@link #getDeviceInfo()}, indicates that coupled sink is supported at source.
115      *
116      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
117      * Wifi Display Technical Specification.
118      */
119     public static final int DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SOURCE = 1 << 2;
120     /**
121      * Bit field for {@link #getDeviceInfo()}, indicates that coupled sink is supporeted at sink.
122      *
123      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
124      * Wifi Display Technical Specification.
125      */
126     public static final int DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SINK = 1 << 3;
127     private static final int SESSION_AVAILABLE_BIT1 = 1 << 4;
128     private static final int SESSION_AVAILABLE_BIT2 = 1 << 5;
129     /**
130      * Bit field for {@link #getDeviceInfo()}, indicates that Wifi Display session is available.
131      *
132      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
133      * Wifi Display Technical Specification.
134      */
135     public static final int DEVICE_INFO_SESSION_AVAILABLE_MASK =
136             SESSION_AVAILABLE_BIT2 | SESSION_AVAILABLE_BIT1;
137     /**
138      * Bit field for {@link #getDeviceInfo()}, indicates that Wifi Display discovery is supported.
139      *
140      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
141      * Wifi Display Technical Specification.
142      */
143     public static final int DEVICE_INFO_WFD_SERVICE_DISCOVERY_SUPPORT = 1 << 6;
144     /**
145      * Bit field for {@link #getDeviceInfo()}, indicate the preferred connectifity for Wifi Display.
146      *
147      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
148      * Wifi Display Technical Specification.
149      * The value is one of {@link #PREFERRED_CONNECTIVITY_P2P} or
150      * {@link #PREFERRED_CONNECTIVITY_TDLS}.
151      */
152     public static final int DEVICE_INFO_PREFERRED_CONNECTIVITY_MASK = 1 << 7;
153     /**
154      * Bit field for {@link #getDeviceInfo()}, indicate the support of Content Protection
155      * using the HDCP system 2.0/2.1.
156      *
157      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
158      * Wifi Display Technical Specification.
159      */
160     public static final int DEVICE_INFO_CONTENT_PROTECTION_SUPPORT = 1 << 8;
161     /**
162      * Bit field for {@link #getDeviceInfo()}, indicate time synchronization
163      * using 802.1AS is supported.
164      *
165      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
166      * Wifi Display Technical Specification.
167      */
168     public static final int DEVICE_INFO_TIME_SYNCHRONIZATION_SUPPORT = 1 << 9;
169     /**
170      * Bit field for {@link #getDeviceInfo()}, indicate audio is not supported at primary sink.
171      *
172      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
173      * Wifi Display Technical Specification.
174      */
175     public static final int DEVICE_INFO_AUDIO_UNSUPPORTED_AT_PRIMARY_SINK = 1 << 10;
176     /**
177      * Bit field for {@link #getDeviceInfo()}, indicate audo is only supported at source.
178      *
179      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
180      * Wifi Display Technical Specification.
181      */
182     public static final int DEVICE_INFO_AUDIO_ONLY_SUPPORT_AT_SOURCE = 1 << 11;
183     /** Bit field for {@link #getDeviceInfo()}, indicate that TDLS persistent group is intended.
184      *
185      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
186      * Wifi Display Technical Specification.
187      */
188     public static final int DEVICE_INFO_TDLS_PERSISTENT_GROUP = 1 << 12;
189     /** Bit field for {@link #getDeviceInfo()}, indicate that the request is for
190      * re-invocation of TDLS persistent group.
191      *
192      * The bit definition is listed in 5.1.2 WFD Device Information Subelement in
193      * Wifi Display Technical Specification.
194      */
195     public static final int DEVICE_INFO_TDLS_PERSISTENT_GROUP_REINVOKE = 1 << 13;
196 
197     private int mCtrlPort;
198 
199     private int mMaxThroughput;
200 
201     /** Default constructor. */
WifiP2pWfdInfo()202     public WifiP2pWfdInfo() {}
203 
204     /** @hide */
205     @UnsupportedAppUsage
WifiP2pWfdInfo(int devInfo, int ctrlPort, int maxTput)206     public WifiP2pWfdInfo(int devInfo, int ctrlPort, int maxTput) {
207         mEnabled = true;
208         mDeviceInfo = devInfo;
209         mCtrlPort = ctrlPort;
210         mMaxThroughput = maxTput;
211         mR2DeviceInfo = -1;
212     }
213 
214     /**
215      * Return R1 raw device info, See
216      * Wifi Display technical specification v1.0.0, section 5.1.2.
217      * Access bit fields by DEVICE_INFO_* constants.
218      */
219     @DeviceInfoMask
getDeviceInfo()220     public int getDeviceInfo() {
221         return mDeviceInfo;
222     }
223 
224     /**
225      * Set Wifi Display R2 raw device info, see
226      * Wifi Display technical specification v2.0.0, section 5.1.12.
227      * Access bit fields by {@link #DEVICE_INFO_DEVICE_TYPE_MASK}.
228      *
229      * @param r2DeviceInfo the raw data of R2 device info.
230      * @hide
231      */
setR2DeviceInfo(int r2DeviceInfo)232     public void setR2DeviceInfo(int r2DeviceInfo) {
233         mR2DeviceInfo = r2DeviceInfo;
234     }
235 
236     /**
237      * Return R2 raw device info, See
238      * Wifi Display technical specification v2.0.0, section 5.1.12.
239      * Access bit fields by {@link #DEVICE_INFO_DEVICE_TYPE_MASK}.
240      */
241     @R2DeviceInfoMask
getR2DeviceInfo()242     public int getR2DeviceInfo() {
243         return mR2DeviceInfo;
244     }
245 
246     /** Returns true is Wifi Display is enabled, false otherwise. */
isEnabled()247     public boolean isEnabled() {
248         return mEnabled;
249     }
250 
251     /** Returns true is Wifi Display R2 is enabled, false otherwise. */
isR2Supported()252     public boolean isR2Supported() {
253         return mR2DeviceInfo >= 0;
254     }
255 
256     /**
257      * Sets whether Wifi Display should be enabled.
258      *
259      * @param enabled true to enable Wifi Display, false to disable
260      */
setEnabled(boolean enabled)261     public void setEnabled(boolean enabled) {
262         mEnabled = enabled;
263     }
264 
265     /**
266      * Sets the type of the Wifi Display R2 device.
267      * See Wifi Display technical specification v2.0.0, section 5.1.12 for Wifi Display R2.
268      * Before calling this API, call {@link WifiManager#isWifiDisplayR2Supported()
269      * to know whether Wifi Display R2 is supported or not.
270      * If R2 info was filled without Wifi Display R2 support,
271      * {@link WifiP2pManager#setWfdInfo(Channel, WifiP2pWfdInfo, ActionListener)
272      * would fail.
273      *
274      * @param deviceType One of {@link #DEVICE_TYPE_WFD_SOURCE}, {@link #DEVICE_TYPE_PRIMARY_SINK},
275      * {@link #DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK}
276      * @return true if the device type was successfully set, false otherwise
277      */
278     @RequiresApi(Build.VERSION_CODES.S)
setR2DeviceType(@eviceType int deviceType)279     public boolean setR2DeviceType(@DeviceType int deviceType) {
280         if (!SdkLevel.isAtLeastS()) {
281             throw new UnsupportedOperationException();
282         }
283         if (DEVICE_TYPE_WFD_SOURCE != deviceType
284                 && DEVICE_TYPE_PRIMARY_SINK != deviceType
285                 && DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK != deviceType) {
286             return false;
287         }
288         if (!isR2Supported()) mR2DeviceInfo = 0;
289         mR2DeviceInfo &= ~DEVICE_INFO_DEVICE_TYPE_MASK;
290         mR2DeviceInfo |= deviceType;
291         return true;
292     }
293 
294     /**
295      * Get the type of the device.
296      * One of {@link #DEVICE_TYPE_WFD_SOURCE}, {@link #DEVICE_TYPE_PRIMARY_SINK},
297      * {@link #DEVICE_TYPE_SECONDARY_SINK}, {@link #DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK}
298      */
299     @DeviceType
getDeviceType()300     public int getDeviceType() {
301         return mDeviceInfo & DEVICE_INFO_DEVICE_TYPE_MASK;
302     }
303 
304     /**
305      * Get the type of the R2 device.
306      * One of {@link #DEVICE_TYPE_WFD_SOURCE}, {@link #DEVICE_TYPE_PRIMARY_SINK},
307      * or {@link #DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK}
308      */
309     @DeviceType
getR2DeviceType()310     public int getR2DeviceType() {
311         return mR2DeviceInfo & DEVICE_INFO_DEVICE_TYPE_MASK;
312     }
313 
314     /**
315      * Sets the type of the device.
316      *
317      * @param deviceType One of {@link #DEVICE_TYPE_WFD_SOURCE}, {@link #DEVICE_TYPE_PRIMARY_SINK},
318      * {@link #DEVICE_TYPE_SECONDARY_SINK}, {@link #DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK}
319      * @return true if the device type was successfully set, false otherwise
320      */
setDeviceType(@eviceType int deviceType)321     public boolean setDeviceType(@DeviceType int deviceType) {
322         if (DEVICE_TYPE_WFD_SOURCE <= deviceType
323                 && deviceType <= DEVICE_TYPE_SOURCE_OR_PRIMARY_SINK) {
324             mDeviceInfo &= ~DEVICE_INFO_DEVICE_TYPE_MASK;
325             mDeviceInfo |= deviceType;
326             return true;
327         }
328         return false;
329     }
330 
331     /** Returns true if a session is available, false otherwise. */
isSessionAvailable()332     public boolean isSessionAvailable() {
333         return (mDeviceInfo & DEVICE_INFO_SESSION_AVAILABLE_MASK) != 0;
334     }
335 
336     /**
337      * Sets whether a session is available.
338      *
339      * @param enabled true to indicate that a session is available, false otherwise.
340      */
setSessionAvailable(boolean enabled)341     public void setSessionAvailable(boolean enabled) {
342         if (enabled) {
343             mDeviceInfo |= SESSION_AVAILABLE_BIT1;
344             mDeviceInfo &= ~SESSION_AVAILABLE_BIT2;
345         } else {
346             mDeviceInfo &= ~DEVICE_INFO_SESSION_AVAILABLE_MASK;
347         }
348     }
349 
350     /**
351      * @return true if Content Protection using the HDCP system 2.0/2.1 is supported.
352      */
isContentProtectionSupported()353     public boolean isContentProtectionSupported() {
354         return (mDeviceInfo & DEVICE_INFO_CONTENT_PROTECTION_SUPPORT) != 0;
355     }
356 
357     /**
358      * Sets whether Content Protection using the HDCP system 2.0/2.1 is supported.
359      *
360      * @param enabled true to indicate that Content Protection is supported, false otherwise.
361      */
setContentProtectionSupported(boolean enabled)362     public void setContentProtectionSupported(boolean enabled) {
363         if (enabled) {
364             mDeviceInfo |= DEVICE_INFO_CONTENT_PROTECTION_SUPPORT;
365         } else {
366             mDeviceInfo &= ~DEVICE_INFO_CONTENT_PROTECTION_SUPPORT;
367         }
368     }
369 
370     /**
371      * Returns true if Coupled Sink is supported by WFD Source.
372      * See Wifi Display technical specification v1.0.0, section 4.9.
373      */
isCoupledSinkSupportedAtSource()374     public boolean isCoupledSinkSupportedAtSource() {
375         return (mDeviceInfo & DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SOURCE) != 0;
376     }
377 
378     /**
379      * Sets whether Coupled Sink feature is supported by WFD Source.
380      * See Wifi Display technical specification v1.0.0, section 4.9.
381      *
382      * @param enabled true to indicate support for coupled sink, false otherwise.
383      */
setCoupledSinkSupportAtSource(boolean enabled)384     public void setCoupledSinkSupportAtSource(boolean enabled) {
385         if (enabled) {
386             mDeviceInfo |= DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SOURCE;
387         } else {
388             mDeviceInfo &= ~DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SOURCE;
389         }
390     }
391 
392     /**
393      * Returns true if Coupled Sink is supported by WFD Sink.
394      * See Wifi Display technical specification v1.0.0, section 4.9.
395      */
isCoupledSinkSupportedAtSink()396     public boolean isCoupledSinkSupportedAtSink() {
397         return (mDeviceInfo & DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SINK) != 0;
398     }
399 
400     /**
401      * Sets whether Coupled Sink feature is supported by WFD Sink.
402      * See Wifi Display technical specification v1.0.0, section 4.9.
403      *
404      * @param enabled true to indicate support for coupled sink, false otherwise.
405      */
setCoupledSinkSupportAtSink(boolean enabled)406     public void setCoupledSinkSupportAtSink(boolean enabled) {
407         if (enabled) {
408             mDeviceInfo |= DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SINK;
409         } else {
410             mDeviceInfo &= ~DEVICE_INFO_COUPLED_SINK_SUPPORT_AT_SINK;
411         }
412     }
413 
414     /** Returns the TCP port at which the WFD Device listens for RTSP messages. */
getControlPort()415     public int getControlPort() {
416         return mCtrlPort;
417     }
418 
419     /** Sets the TCP port at which the WFD Device listens for RTSP messages. */
setControlPort(@ntRangefrom = 0) int port)420     public void setControlPort(@IntRange(from = 0) int port) {
421         mCtrlPort = port;
422     }
423 
424     /** Sets the maximum average throughput capability of the WFD Device, in megabits/second. */
setMaxThroughput(@ntRangefrom = 0) int maxThroughput)425     public void setMaxThroughput(@IntRange(from = 0) int maxThroughput) {
426         mMaxThroughput = maxThroughput;
427     }
428 
429     /** Returns the maximum average throughput capability of the WFD Device, in megabits/second. */
getMaxThroughput()430     public int getMaxThroughput() {
431         return mMaxThroughput;
432     }
433 
434     /** @hide */
getDeviceInfoHex()435     public String getDeviceInfoHex() {
436         return String.format(
437                 Locale.US, "%04x%04x%04x", mDeviceInfo, mCtrlPort, mMaxThroughput);
438     }
439 
440     /** @hide */
getR2DeviceInfoHex()441     public String getR2DeviceInfoHex() {
442         return String.format(Locale.US, "%04x%04x", 2, mR2DeviceInfo);
443     }
444 
445     @Override
toString()446     public String toString() {
447         StringBuffer sbuf = new StringBuffer();
448         sbuf.append("WFD enabled: ").append(mEnabled);
449         sbuf.append("\n WFD DeviceInfo: ").append(mDeviceInfo);
450         sbuf.append("\n WFD CtrlPort: ").append(mCtrlPort);
451         sbuf.append("\n WFD MaxThroughput: ").append(mMaxThroughput);
452         sbuf.append("\n WFD R2 DeviceInfo: ").append(mR2DeviceInfo);
453         return sbuf.toString();
454     }
455 
456     /** Implement the Parcelable interface */
describeContents()457     public int describeContents() {
458         return 0;
459     }
460 
461     /** Copy constructor. */
WifiP2pWfdInfo(@ullable WifiP2pWfdInfo source)462     public WifiP2pWfdInfo(@Nullable WifiP2pWfdInfo source) {
463         if (source != null) {
464             mEnabled = source.mEnabled;
465             mDeviceInfo = source.mDeviceInfo;
466             mCtrlPort = source.mCtrlPort;
467             mMaxThroughput = source.mMaxThroughput;
468             mR2DeviceInfo = source.mR2DeviceInfo;
469         }
470     }
471 
472     /** Implement the Parcelable interface */
473     @Override
writeToParcel(@onNull Parcel dest, int flags)474     public void writeToParcel(@NonNull Parcel dest, int flags) {
475         dest.writeInt(mEnabled ? 1 : 0);
476         dest.writeInt(mDeviceInfo);
477         dest.writeInt(mCtrlPort);
478         dest.writeInt(mMaxThroughput);
479         dest.writeInt(mR2DeviceInfo);
480     }
481 
readFromParcel(Parcel in)482     private void readFromParcel(Parcel in) {
483         mEnabled = (in.readInt() == 1);
484         mDeviceInfo = in.readInt();
485         mCtrlPort = in.readInt();
486         mMaxThroughput = in.readInt();
487         mR2DeviceInfo = in.readInt();
488     }
489 
490     /** Implement the Parcelable interface */
491     public static final @NonNull Creator<WifiP2pWfdInfo> CREATOR =
492         new Creator<WifiP2pWfdInfo>() {
493             public WifiP2pWfdInfo createFromParcel(Parcel in) {
494                 WifiP2pWfdInfo device = new WifiP2pWfdInfo();
495                 device.readFromParcel(in);
496                 return device;
497             }
498 
499             public WifiP2pWfdInfo[] newArray(int size) {
500                 return new WifiP2pWfdInfo[size];
501             }
502         };
503 }
504