日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Wifi模块—源码分析Wifi热点扫描(Android P)

發布時間:2025/3/15 Android 63 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Wifi模块—源码分析Wifi热点扫描(Android P) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一 前言

? ? ? ?這次接著講Wifi工程流程中的Wifi熱點查找過程,也是Wifi啟動的過程延續,Wifi啟動過程中會更新Wifi的狀態,框架層也有相應廣播發出,應用層接收到廣播后開始進行熱點的掃描。可以先看前面Wifi啟動的分析過程。

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?Wifi模塊—源碼分析Wifi啟動1(Android P)

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?Wifi模塊—源碼分析Wifi啟動2(Android P)

二 圖示調用流程

? ? ? 由于在 frameworks/base/packages/SettingsLib/src/com/android/settingslib/wifi/WifiTracker.java直接接收到框架層發出的wifi狀態改變的廣播WIFI_STATE_CHANGED_ACTION(這個在后面有交待),所以這里的圖示調用流程將從WifiTracker.java開始。

三 代碼具體流程

? ? ? ?我們先回顧一下之前wifi啟動過程的相關細節,wifi啟動過程會走到ClientModeStateMachine這個狀態機,具體看 Wifi模塊—源碼分析Wifi啟動1(Android P)。

frameworks/opt/net/wifi/service/java/com/android/server/wifi/ClientModeManager.java?

看ClientModeStateMachine的構造函數。

ClientModeStateMachine(Looper looper) {
? ? super(TAG, looper);
?
? ? addState(mIdleState);
? ? addState(mStartedState);
?
? ? setInitialState(mIdleState);
? ? start();
}
再看start()。

/**
* Start client mode.
*/
public void start() {
? ? mStateMachine.sendMessage(ClientModeStateMachine.CMD_START);
}
發送了一個消息ClientModeStateMachine.CMD_START。

private class IdleState extends State {
?
? ? @Override
? ? public void enter() {
? ? ? ? Log.d(TAG, "entering IdleState");
? ? ? ? mClientInterfaceName = null;
? ? ? ? mIfaceIsUp = false;
? ? }
?
? ? @Override
? ? public boolean processMessage(Message message) {
? ? ? ? switch (message.what) {
? ? ? ? ? ? case CMD_START:
? ? ? ? ? ? ? ? updateWifiState(WifiManager.WIFI_STATE_ENABLING,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? WifiManager.WIFI_STATE_DISABLED);
?
? ? ? ? ? ? ? ? mClientInterfaceName = mWifiNative.setupInterfaceForClientMode(false /* not low priority */, mWifiNativeInterfaceCallback);
? ? ? ? ? ? ? ? if (TextUtils.isEmpty(mClientInterfaceName)) {
? ? ? ? ? ? ? ? ? ? Log.e(TAG, "Failed to create ClientInterface. Sit in Idle");
? ? ? ? ? ? ? ? ? ? updateWifiState(WifiManager.WIFI_STATE_UNKNOWN,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? WifiManager.WIFI_STATE_ENABLING);
? ? ? ? ? ? ? ? ? ? updateWifiState(WifiManager.WIFI_STATE_DISABLED,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? WifiManager.WIFI_STATE_UNKNOWN);
? ? ? ? ? ? ? ? ? ? break;
? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? sendScanAvailableBroadcast(false);
? ? ? ? ? ? ? ? mScanRequestProxy.enableScanningForHiddenNetworks(false);
? ? ? ? ? ? ? ? mScanRequestProxy.clearScanResults();
? ? ? ? ? ? ? ? transitionTo(mStartedState);
? ? ? ? ? ? ? ? break;
? ? ? ? ? ? default:
? ? ? ? ? ? ? ? Log.d(TAG, "received an invalid message: " + message);
? ? ? ? ? ? ? ? return NOT_HANDLED;
? ? ? ? }
? ? ? ? return HANDLED;
? ? }
}
在IdleState的狀態里接收到消息做相關處理updateWifiState,繼續看這個方法。

/**
? ? ?* Update Wifi state and send the broadcast.
? ? ?* @param newState new Wifi state
? ? ?* @param currentState current wifi state
*/
private void updateWifiState(int newState, int currentState) {
? ? if (!mExpectedStop) {
? ? ? ? mListener.onStateChanged(newState);
? ? } else {
? ? ? ? Log.d(TAG, "expected stop, not triggering callbacks: newState = "
? ? ? ? ? ? + newState);
? ? }
?
? ? // Once we report the mode has stopped/failed any other stop signals are redundant
? ? // note: this can happen in failure modes where we get multiple callbacks as underlying
? ? // components/interface stops or the underlying interface is destroyed in cleanup
? ? if (newState == WifiManager.WIFI_STATE_UNKNOWN
? ? ? ? ? ? ? ? || newState == WifiManager.WIFI_STATE_DISABLED) {
? ? ? ? mExpectedStop = true;
? ? }
?
? ? if (newState == WifiManager.WIFI_STATE_UNKNOWN) {
? ? ? ? // do not need to broadcast failure to system
? ? ? ? return;
? ? }
?
? ? mWifiStateMachine.setWifiStateForApiCalls(newState);
?
? ? final Intent intent = new Intent(WifiManager.WIFI_STATE_CHANGED_ACTION);
? ? intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
? ? intent.putExtra(WifiManager.EXTRA_WIFI_STATE, newState);
? ? intent.putExtra(WifiManager.EXTRA_PREVIOUS_WIFI_STATE, currentState);
? ? mContext.sendStickyBroadcastAsUser(intent, UserHandle.ALL);
}
在這里發送了一個廣播sendStickyBroadcastAsUser,廣播具體是WifiManager.WIFI_STATE_CHANGED_ACTION。

? ? ? ? OK,上面的分析都屬于上次分析wifi啟動的部分,就是wifi在啟動過程中會更新wifi狀態并發送wifi狀態改變的廣播。不過。上次在分析wifi啟動過程中也沒有分析這個更新wifi狀態發送廣播的這個內容,這次作為一個補充也是開啟接下來的wifi熱點掃描的流程分析,接下來的wifi掃描過程是指打開wifi之后自動掃描過程而不是手動去刷新掃描。

1 應用層

1.1 packages/apps/Settings/src/com/android/settings/wifi/WifiSettings.java?

在WifiSettings.java里初始化WifiTracker。

@Override
public void onActivityCreated(Bundle savedInstanceState) {
? ? super.onActivityCreated(savedInstanceState);
?
? ? mWifiTracker = WifiTrackerFactory.create(
? ? ? ? ? ? getActivity(), this, getLifecycle(), true, true);
? ? mWifiManager = mWifiTracker.getManager();
? ? ...
}
? ?

1.2 packages/apps/Settings/src/com/android/settings/wifi/WifiEnabler.java

? ? ? ?應用層在掃描過程中貌似沒做什么事情但暫且提一下WifiEnabler也會接收到WifiManager.WIFI_STATE_CHANGED_ACTION廣播并會相關的處理。

private boolean mStateMachineEvent;
private final IntentFilter mIntentFilter;
private final BroadcastReceiver mReceiver = new BroadcastReceiver() {
? ? @Override
? ? public void onReceive(Context context, Intent intent) {
? ? ? ? String action = intent.getAction();
? ? ? ? if (WifiManager.WIFI_STATE_CHANGED_ACTION.equals(action)) {
? ? ? ? ? ? handleWifiStateChanged(mWifiManager.getWifiState());
? ? ? ? } else if (WifiManager.SUPPLICANT_STATE_CHANGED_ACTION.equals(action)) {
? ? ? ? ? ? if (!mConnected.get()) {
? ? ? ? ? ? ? ? handleStateChanged(WifiInfo.getDetailedStateOf((SupplicantState)
? ? ? ? ? ? ? ? ? ? ? ? intent.getParcelableExtra(WifiManager.EXTRA_NEW_STATE)));
? ? ? ? ? ? }
? ? ? ? } else if (WifiManager.NETWORK_STATE_CHANGED_ACTION.equals(action)) {
? ? ? ? ? ? NetworkInfo info = (NetworkInfo) intent.getParcelableExtra(
? ? ? ? ? ? ? ? ? ? WifiManager.EXTRA_NETWORK_INFO);
? ? ? ? ? ? mConnected.set(info.isConnected());
? ? ? ? ? ? handleStateChanged(info.getDetailedState());
? ? ? ? }
? ? }
};
看handleWifiStateChanged。

private void handleWifiStateChanged(int state) {
? ? // Clear any previous state
? ? mSwitchWidget.setDisabledByAdmin(null);
?
? ? switch (state) {
? ? ? ? case WifiManager.WIFI_STATE_ENABLING:
? ? ? ? ? ? break;
? ? ? ? case WifiManager.WIFI_STATE_ENABLED:
? ? ? ? ? ? setSwitchBarChecked(true);
? ? ? ? ? ? mSwitchWidget.setEnabled(true);
? ? ? ? ? ? break;
? ? ? ? case WifiManager.WIFI_STATE_DISABLING:
? ? ? ? ? ? break;
? ? ? ? case WifiManager.WIFI_STATE_DISABLED:
? ? ? ? ? ? setSwitchBarChecked(false);
? ? ? ? ? ? mSwitchWidget.setEnabled(true);
? ? ? ? ? ? break;
? ? ? ? default:
? ? ? ? ? ? setSwitchBarChecked(false);
? ? ? ? ? ? mSwitchWidget.setEnabled(true);
? ? }
?
? ? if (RestrictedLockUtils.hasBaseUserRestriction(mContext,
? ? ? ? ? ? UserManager.DISALLOW_CONFIG_TETHERING, UserHandle.myUserId())) {
? ? ? ? mSwitchWidget.setEnabled(false);
? ? } else {
? ? ? ? final EnforcedAdmin admin = RestrictedLockUtils.checkIfRestrictionEnforced(mContext,
? ? ? ? ? ? UserManager.DISALLOW_CONFIG_TETHERING, UserHandle.myUserId());
? ? ? ? mSwitchWidget.setDisabledByAdmin(admin);
? ? }
}
根據wifi狀態,switchBar的狀態會變化。

2 java框架層

? ? ?真正的掃描過程從這里開始。framework/base/packages/SettingsLib。SettingsLib放在frameworks/base/packages/SettingsLib,因為SystemUi和開機向導中的藍牙WiFi流程也會用到對應的代碼。

2.1 frameworks/base/packages/SettingsLib/src/com/android/settingslib/wifi/WifiTracker.java

/**
* ?Receiver for handling broadcasts.
*
* ?This receiver is registered on the WorkHandler.
*/
@VisibleForTesting
final BroadcastReceiver mReceiver = new BroadcastReceiver() {
? ? @Override
? ? public void onReceive(Context context, Intent intent) {
? ? ? ? String action = intent.getAction();
?
? ? ? ? if (WifiManager.WIFI_STATE_CHANGED_ACTION.equals(action)) {
? ? ? ? ? ? updateWifiState(intent.getIntExtra(WifiManager.EXTRA_WIFI_STATE,
? ? ? ? ? ? ? ? ? ? ? ? ? ? WifiManager.WIFI_STATE_UNKNOWN));
? ? ? ? } else if (WifiManager.SCAN_RESULTS_AVAILABLE_ACTION.equals(action)) {
? ? ? ? ? ? mStaleScanResults = false;
?
? ? ? ? ? ? fetchScansAndConfigsAndUpdateAccessPoints();
? ? ? ? } else if (WifiManager.CONFIGURED_NETWORKS_CHANGED_ACTION.equals(action)
? ? ? ? ? ? ? ? || WifiManager.LINK_CONFIGURATION_CHANGED_ACTION.equals(action)) {
? ? ? ? ? ? fetchScansAndConfigsAndUpdateAccessPoints();
? ? ? ? } else if (WifiManager.NETWORK_STATE_CHANGED_ACTION.equals(action)) {
? ? ? ? ? ? // TODO(sghuman): Refactor these methods so they cannot result in duplicate
? ? ? ? ? ? // onAccessPointsChanged updates being called from this intent.
? ? ? ? ? ? NetworkInfo info = intent.getParcelableExtra(WifiManager.EXTRA_NETWORK_INFO);
? ? ? ? ? ? updateNetworkInfo(info);
? ? ? ? ? ? fetchScansAndConfigsAndUpdateAccessPoints();
? ? ? ? } else if (WifiManager.RSSI_CHANGED_ACTION.equals(action)) {
? ? ? ? ? ? NetworkInfo info =
? ? ? ? ? ? ? ? ? ? mConnectivityManager.getNetworkInfo(mWifiManager.getCurrentNetwork());
? ? ? ? ? ? updateNetworkInfo(info);
? ? ? ? }
? ? }
};
看updateWifiState。

/**
* Handles updates to WifiState.
*
* <p>If Wifi is not enabled in the enabled state, {@link #mStaleScanResults} will be set to
* true.
*/
private void updateWifiState(int state) {
? ? if (state == WifiManager.WIFI_STATE_ENABLED) {
? ? ? ? if (mScanner != null) {
? ? ? ? ? ? // We only need to resume if mScanner isn't null because
? ? ? ? ? ? // that means we want to be scanning.
? ? ? ? ? ? mScanner.resume();
? ? ? ? }
? ? } else {
? ? ? ? clearAccessPointsAndConditionallyUpdate();
? ? ? ? mLastInfo = null;
? ? ? ? mLastNetworkInfo = null;
? ? ? ? if (mScanner != null) {
? ? ? ? ? ? mScanner.pause();
? ? ? ? }
? ? ? ? mStaleScanResults = true;
? ? }
? ? mListener.onWifiStateChanged(state);
}
看mScanner.resume,其中Scanner是內部類。

@VisibleForTesting
class Scanner extends Handler {
? ? static final int MSG_SCAN = 0;
?
? ? private int mRetry = 0;
?
? ? void resume() {
? ? ? ? if (!hasMessages(MSG_SCAN)) {
? ? ? ? ? ? sendEmptyMessage(MSG_SCAN);
? ? ? ? }
? ? }
?
? ? void pause() {
? ? ? ? mRetry = 0;
? ? ? ? removeMessages(MSG_SCAN);
? ? }
?
? ? @VisibleForTesting
? ? boolean isScanning() {
? ? ? ? return hasMessages(MSG_SCAN);
? ? }
?
? ? @Override
? ? public void handleMessage(Message message) {
? ? ? ? if (message.what != MSG_SCAN) return;
? ? ? ? if (mWifiManager.startScan()) {
? ? ? ? ? ? mRetry = 0;
? ? ? ? } else if (++mRetry >= 3) {
? ? ? ? ? ? mRetry = 0;
? ? ? ? ? ? if (mContext != null) {
? ? ? ? ? ? ? ? Toast.makeText(mContext, R.string.wifi_fail_to_scan, Toast.LENGTH_LONG).show();
? ? ? ? ? ? }
? ? ? ? ? ? return;
? ? ? ? }
? ? ? ? sendEmptyMessageDelayed(MSG_SCAN, WIFI_RESCAN_INTERVAL_MS);
? ? }
}
? ? ? ?在resume方法里面發送消息MSG_SCAN,并接收該消息進行處理,看handleMessage。會調用mWifiManager.startScan,并且每隔10s再次進行掃描。

2.2 frameworks/base/wifi/java/android/net/wifi/WifiManager.java

* <p>
* To initiate a Wi-Fi scan, declare the
* {@link android.Manifest.permission#CHANGE_WIFI_STATE}
* permission in the manifest, and perform these steps:
* </p>
* <ol style="1">
* <li>Invoke the following method:
* {@code ((WifiManager) getSystemService(WIFI_SERVICE)).startScan()}</li>
* <li>
* Register a BroadcastReceiver to listen to
* {@code SCAN_RESULTS_AVAILABLE_ACTION}.</li>
* <li>When a broadcast is received, call:
* {@code ((WifiManager) getSystemService(WIFI_SERVICE)).getScanResults()}</li>
* </ol>
* @return {@code true} if the operation succeeded, i.e., the scan was initiated.
* @deprecated The ability for apps to trigger scan requests will be removed in a future
* release.
*/
@Deprecated
public boolean startScan() {
? ? return startScan(null);
}
?
/** @hide */
@SystemApi
@RequiresPermission(android.Manifest.permission.UPDATE_DEVICE_STATS)
public boolean startScan(WorkSource workSource) {
? ? try {
? ? ? ? String packageName = mContext.getOpPackageName();
? ? ? ? return mService.startScan(packageName);
? ? } catch (RemoteException e) {
? ? ? ? throw e.rethrowFromSystemServer();
? ? }
}
看mService.startScan。這個mService是遠程的服務端WifiService,其實現類為WifiServiceImpl,這個過程也是跨進程調用。

2.3 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/wifiServiceImpl.java

/**
* See {@link android.net.wifi.WifiManager#startScan}
*
* @param packageName Package name of the app that requests wifi scan.
*/
@Override
public boolean startScan(String packageName) {
? ? if (enforceChangePermission(packageName) != MODE_ALLOWED) {
? ? ? ? return false;
? ? }
?
? ? int callingUid = Binder.getCallingUid();
? ? long ident = Binder.clearCallingIdentity();
? ? mLog.info("startScan uid=%").c(callingUid).flush();
? ? synchronized (this) {
? ? ? ? if (mInIdleMode) {
? ? ? ? ? ? // Need to send an immediate scan result broadcast in case the
? ? ? ? ? ? // caller is waiting for a result ..
?
? ? ? ? ? ? // TODO: investigate if the logic to cancel scans when idle can move to
? ? ? ? ? ? // WifiScanningServiceImpl. ?This will 1 - clean up WifiServiceImpl and 2 -
? ? ? ? ? ? // avoid plumbing an awkward path to report a cancelled/failed scan. ?This will
? ? ? ? ? ? // be sent directly until b/31398592 is fixed.
? ? ? ? ? ? sendFailedScanBroadcast();
? ? ? ? ? ? mScanPending = true;
? ? ? ? ? ? return false;
? ? ? ? }
? ? }
? ? try {
? ? ? ? mWifiPermissionsUtil.enforceCanAccessScanResults(packageName, callingUid);
? ? ? ? Mutable<Boolean> scanSuccess = new Mutable<>();
? ? ? ? boolean runWithScissorsSuccess = mWifiInjector.getWifiStateMachineHandler()
? ? ? ? ? ? ? ? .runWithScissors(() -> {
? ? ? ? ? ? ? ? ? ? scanSuccess.value = mScanRequestProxy.startScan(callingUid, packageName);
? ? ? ? ? ? ? ? }, RUN_WITH_SCISSORS_TIMEOUT_MILLIS);
? ? ? ? if (!runWithScissorsSuccess) {
? ? ? ? ? ? Log.e(TAG, "Failed to post runnable to start scan");
? ? ? ? ? ? sendFailedScanBroadcast();
? ? ? ? ? ? return false;
? ? ? ? }
? ? ? ? if (!scanSuccess.value) {
? ? ? ? ? ? Log.e(TAG, "Failed to start scan");
? ? ? ? ? ? return false;
? ? ? ? }
? ? } catch (SecurityException e) {
? ? ? ? return false;
? ? } finally {
? ? ? ? Binder.restoreCallingIdentity(ident);
? ? }
? ? return true;
}
繼續看mScanRequestProxy.startScan。

2.4 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/ScanRequestProxy.java

/**
* Initiate a wifi scan.
*
* @param callingUid The uid initiating the wifi scan. Blame will be given to this uid.
* @return true if the scan request was placed or a scan is already ongoing, false otherwise.
*/
public boolean startScan(int callingUid, String packageName) {
? ? if (!retrieveWifiScannerIfNecessary()) {
? ? ? ? Log.e(TAG, "Failed to retrieve wifiscanner");
? ? ? ? sendScanResultFailureBroadcastToPackage(packageName);
? ? ? ? return false;
? ? }
? ? boolean fromSettingsOrSetupWizard =
? ? ? ? ? ? mWifiPermissionsUtil.checkNetworkSettingsPermission(callingUid)
? ? ? ? ? ? ? ? ? ? || mWifiPermissionsUtil.checkNetworkSetupWizardPermission(callingUid);
? ? // Check and throttle scan request from apps without NETWORK_SETTINGS permission.
? ? if (!fromSettingsOrSetupWizard
? ? ? ? ? ? && shouldScanRequestBeThrottledForApp(callingUid, packageName)) {
? ? ? ? Log.i(TAG, "Scan request from " + packageName + " throttled");
? ? ? ? sendScanResultFailureBroadcastToPackage(packageName);
? ? ? ? return false;
? ? }
? ? // Create a worksource using the caller's UID.
? ? WorkSource workSource = new WorkSource(callingUid);
?
? ? // Create the scan settings.
? ? WifiScanner.ScanSettings settings = new WifiScanner.ScanSettings();
? ? // Scan requests from apps with network settings will be of high accuracy type.
? ? if (fromSettingsOrSetupWizard) {
? ? ? ? settings.type = WifiScanner.TYPE_HIGH_ACCURACY;
? ? }
? ? // always do full scans
? ? settings.band = WifiScanner.WIFI_BAND_BOTH_WITH_DFS;
? ? settings.reportEvents = WifiScanner.REPORT_EVENT_AFTER_EACH_SCAN
? ? ? ? ? ? | WifiScanner.REPORT_EVENT_FULL_SCAN_RESULT;
? ? if (mScanningForHiddenNetworksEnabled) {
? ? ? ? // retrieve the list of hidden network SSIDs to scan for, if enabled.
? ? ? ? List<WifiScanner.ScanSettings.HiddenNetwork> hiddenNetworkList =
? ? ? ? ? ? ? ? mWifiConfigManager.retrieveHiddenNetworkList();
? ? ? ? settings.hiddenNetworks = hiddenNetworkList.toArray(
? ? ? ? ? ? ? ? new WifiScanner.ScanSettings.HiddenNetwork[hiddenNetworkList.size()]);
? ? }
? ? mWifiScanner.startScan(settings, new ScanRequestProxyScanListener(), workSource);
? ? mIsScanProcessingComplete = false;
? ? return true;
}
看mWifiScanner.startScan

2.5 ?frameworks/base/wifi/java/android/net/wifi/WifiScanner.java

/**
* starts a single scan and reports results asynchronously
* @param settings specifies various parameters for the scan; for more information look at
* {@link ScanSettings}
* @param listener specifies the object to report events to. This object is also treated as a
* ? ? ? ? ? ? ? ?key for this scan, and must also be specified to cancel the scan. Multiple
* ? ? ? ? ? scans should also not share this object.
*/
@RequiresPermission(android.Manifest.permission.LOCATION_HARDWARE)
public void startScan(ScanSettings settings, ScanListener listener) {
? ? startScan(settings, listener, null);
}
?
/**
* starts a single scan and reports results asynchronously
* @param settings specifies various parameters for the scan; for more information look at
* {@link ScanSettings}
* @param workSource WorkSource to blame for power usage
* @param listener specifies the object to report events to. This object is also treated as a
* ? ? ? ? ? ? ? ? key for this scan, and must also be specified to cancel the scan. Multiple
* ? ? ? ? ? ? ? ? scans should also not share this object.
*/
@RequiresPermission(android.Manifest.permission.LOCATION_HARDWARE)
public void startScan(ScanSettings settings, ScanListener listener, WorkSource workSource) {
? ? Preconditions.checkNotNull(listener, "listener cannot be null");
? ? int key = addListener(listener);
? ? if (key == INVALID_KEY) return;
? ? validateChannel();
? ? Bundle scanParams = new Bundle();
? ? scanParams.putParcelable(SCAN_PARAMS_SCAN_SETTINGS_KEY, settings);
? ? scanParams.putParcelable(SCAN_PARAMS_WORK_SOURCE_KEY, workSource);
? ? mAsyncChannel.sendMessage(CMD_START_SINGLE_SCAN, 0, key, scanParams);
? ??
}
? ? ? ? Android P和Android O框架層代碼改變不是很大,不過Android O的操作基本都是經過WifiStateMachine狀態機,而Android P很多操作都不再經過WifiStateMachine狀態機,在這里是為了把scan功能獨立分開出來。而和Android O之前的代碼相比則有很大區別,這里接下來使用到了雙向異步通道的方式,mAsyncChannel.sendMessage。AsyncChannel處理兩個handler之間消息異步傳遞的問題,這兩個handler可以在一個進程也可以處于不同的進程。?

2.6 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/scanner/WifiScanningServiceImpl.java

private class ClientHandler extends WifiHandler {
?
? ? ClientHandler(String tag, Looper looper) {
? ? ? ? super(tag, looper);
? ? }
?
? ? @Override
? ? public void handleMessage(Message msg) {
? ? ? ? super.handleMessage(msg);
? ? ? ? ...
? ? ? ? switch (msg.what) {
? ? ? ? ? ? case WifiScanner.CMD_START_BACKGROUND_SCAN:
? ? ? ? ? ? case WifiScanner.CMD_STOP_BACKGROUND_SCAN:
? ? ? ? ? ? ? ? mBackgroundScanStateMachine.sendMessage(Message.obtain(msg));
? ? ? ? ? ? ? ? break;
? ? ? ? ? ? case WifiScanner.CMD_START_PNO_SCAN:
? ? ? ? ? ? case WifiScanner.CMD_STOP_PNO_SCAN:
? ? ? ? ? ? ? ? mPnoScanStateMachine.sendMessage(Message.obtain(msg));
? ? ? ? ? ? ? ? break;
? ? ? ? ? ? case WifiScanner.CMD_START_SINGLE_SCAN:
? ? ? ? ? ? case WifiScanner.CMD_STOP_SINGLE_SCAN:
? ? ? ? ? ? ? ? mSingleScanStateMachine.sendMessage(Message.obtain(msg));
? ? ? ? ? ? ? ? break;
? ? ? ? ? ? ...
? ? ? ? }
? ? }
}
? ?ClientHandler接收到CMD_START_SINGLE_SCAN消息并處理也就是向狀態機發送這個消息,mSingleScanStateMachine.sendMessage(Message.obtain(msg))。SingleScanStateMachine是個處理熱點掃描的狀態機。

/**
* State machine that holds the state of single scans. Scans should only be active in the
* ScanningState. The pending scans and active scans maps are swapped when entering
* ScanningState. Any requests queued while scanning will be placed in the pending queue and
* executed after transitioning back to IdleState.
*/
class WifiSingleScanStateMachine extends StateMachine implements WifiNative.ScanEventHandler {
? ? /**
? ? * Maximum age of results that we return from our cache via
? ? * {@link WifiScanner#getScanResults()}.
? ? * This is currently set to 3 minutes to restore parity with the wpa_supplicant's scan
? ? * result cache expiration policy. (See b/62253332 for details)
? ? */
? ? @VisibleForTesting
? ? public static final int CACHED_SCAN_RESULTS_MAX_AGE_IN_MILLIS = 180 * 1000;
?
? ? private final DefaultState mDefaultState = new DefaultState();
? ? private final DriverStartedState mDriverStartedState = new DriverStartedState();
? ? private final IdleState ?mIdleState ?= new IdleState();
? ? private final ScanningState ?mScanningState ?= new ScanningState();
?
? ? private WifiNative.ScanSettings mActiveScanSettings = null;
? ? private RequestList<ScanSettings> mActiveScans = new RequestList<>();
? ? private RequestList<ScanSettings> mPendingScans = new RequestList<>();
?
? ? // Scan results cached from the last full single scan request.
? ? private final List<ScanResult> mCachedScanResults = new ArrayList<>();
?
? ? WifiSingleScanStateMachine(Looper looper) {
? ? ? ? super("WifiSingleScanStateMachine", looper);
?
? ? ? ? setLogRecSize(128);
? ? ? ? setLogOnlyTransitions(false);
?
? ? ? ? // CHECKSTYLE:OFF IndentationCheck
? ? ? ? addState(mDefaultState);
? ? ? ? ? ? addState(mDriverStartedState, mDefaultState);
? ? ? ? ? ? ? ? addState(mIdleState, mDriverStartedState);
? ? ? ? ? ? ? ? addState(mScanningState, mDriverStartedState);
? ? ? ? ? ? // CHECKSTYLE:ON IndentationCheck
?
? ? ? ? ? ? setInitialState(mDefaultState);
? ? }
? ? ...
}
狀態機有四個狀態,初始狀態為默認狀態。在DriverStarted對CMD_START_SINGLE_SCAN消息進行處理。

/**
* State representing when the driver is running. This state is not meant to be transitioned
* directly, but is instead intended as a parent state of ScanningState and IdleState
* to hold common functionality and handle cleaning up scans when the driver is shut down.
*/
class DriverStartedState extends State {
? ? ...
?
? ? @Override
? ? public boolean processMessage(Message msg) {
? ? ? ? ClientInfo ci = mClients.get(msg.replyTo);
?
? ? ? ? switch (msg.what) {
? ? ? ? ? ? case CMD_DRIVER_LOADED:
? ? ? ? ? ? ? ? // Ignore if we're already in driver loaded state.
? ? ? ? ? ? ? ? return HANDLED;
? ? ? ? ? ? case WifiScanner.CMD_START_SINGLE_SCAN:
? ? ? ? ? ? ? ? mWifiMetrics.incrementOneshotScanCount();
? ? ? ? ? ? ? ? int handler = msg.arg2;
? ? ? ? ? ? ? ? Bundle scanParams = (Bundle) msg.obj;
? ? ? ? ? ? ? ? if (scanParams == null) {
? ? ? ? ? ? ? ? ? ? logCallback("singleScanInvalidRequest", ?ci, handler, "null params");
? ? ? ? ? ? ? ? ? ? replyFailed(msg, WifiScanner.REASON_INVALID_REQUEST, "params null");
? ? ? ? ? ? ? ? ? ? return HANDLED;
? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? scanParams.setDefusable(true);
? ? ? ? ? ? ? ? ScanSettings scanSettings =
? ? ? ? ? ? ? ? ? ? ? ? ? ? scanParams.getParcelable(WifiScanner.SCAN_PARAMS_SCAN_SETTINGS_KEY);
? ? ? ? ? ? ? ? WorkSource workSource =
? ? ? ? ? ? ? ? ? ? ? ? ? ? scanParams.getParcelable(WifiScanner.SCAN_PARAMS_WORK_SOURCE_KEY);
? ? ? ? ? ? ? ? if (validateScanRequest(ci, handler, scanSettings)) {
? ? ? ? ? ? ? ? ? ? ? ? ? ? logScanRequest("addSingleScanRequest", ci, handler, workSource,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? scanSettings, null);
? ? ? ? ? ? ? ? ? ? replySucceeded(msg);
?
? ? ? ? ? ? ? ? ? ? // If there is an active scan that will fulfill the scan request then
? ? ? ? ? ? ? ? ? ? // mark this request as an active scan, otherwise mark it pending.
? ? ? ? ? ? ? ? ? ? // If were not currently scanning then try to start a scan. Otherwise
? ? ? ? ? ? ? ? ? ? // this scan will be scheduled when transitioning back to IdleState
? ? ? ? ? ? ? ? ? ? // after finishing the current scan.
? ? ? ? ? ? ? ? ? ? if (getCurrentState() == mScanningState) {
? ? ? ? ? ? ? ? ? ? ? ? if (activeScanSatisfies(scanSettings)) {
? ? ? ? ? ? ? ? ? ? ? ? ? ? mActiveScans.addRequest(ci, handler, workSource, scanSettings);
? ? ? ? ? ? ? ? ? ? ? ? } else {
? ? ? ? ? ? ? ? ? ? ? ? ? ? mPendingScans.addRequest(ci, handler, workSource, scanSettings);
? ? ? ? ? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? ? ? } else {
? ? ? ? ? ? ? ? ? ? ? ? mPendingScans.addRequest(ci, handler, workSource, scanSettings);
? ? ? ? ? ? ? ? ? ? ? ? tryToStartNewScan();
? ? ? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? } else {
? ? ? ? ? ? ? ? ...
? ? ? ? ? ? ? ? } ? ?
? ? ? ? }
? ? } ??
}
看調用的tryToStartNewScan。

void tryToStartNewScan() {
? ? if (mPendingScans.size() == 0) { // no pending requests
? ? ? ? return;
? ? }
? ? ...
? ? if (mScannerImpl.startSingleScan(settings, this)) {
? ? ? ? // store the active scan settings
? ? ? ? mActiveScanSettings = settings;
? ? ? ? // swap pending and active scan requests
? ? ? ? RequestList<ScanSettings> tmp = mActiveScans;
? ? ? ? mActiveScans = mPendingScans;
? ? ? ? mPendingScans = tmp;
? ? ? ? // make sure that the pending list is clear
? ? ? ? mPendingScans.clear();
? ? ? ? transitionTo(mScanningState);
? ? } else {
? ? ? ? mWifiMetrics.incrementScanReturnEntry(
? ? ? ? ? ? ? ? WifiMetricsProto.WifiLog.SCAN_UNKNOWN, mPendingScans.size());
? ? ? ? // notify and cancel failed scans
? ? ? ? sendOpFailedToAllAndClear(mPendingScans, WifiScanner.REASON_UNSPECIFIED,
? ? ? ? ? ? ? ? "Failed to start single scan");
? ? }
}
看mScannerImpl.startSingleScan。

2.7 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/scanner/WifiScannerImpl.java

/**
* Start a one time scan. This method should only be called when there is no scan going on
* (after a callback indicating that the previous scan succeeded/failed).
* @return if the scan paramaters are valid
* Note this may return true even if the parameters are not accepted by the chip because the
* scan may be scheduled async.
*/
public abstract boolean startSingleScan(WifiNative.ScanSettings settings,
? ? ? ? ? ? ? ? ?WifiNative.ScanEventHandler eventHandler);
?

2.8 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/scanner/WificondScannerImpl.java

WificondScannerImpl.java繼承抽象類WifiScannerImpl.java,所以復寫了startSingleScan。

@Override
public boolean startSingleScan(WifiNative.ScanSettings settings,
? ? ? ? WifiNative.ScanEventHandler eventHandler) {
? ? ...
? ? if (!allFreqs.isEmpty()) {
? ? ? ? freqs = allFreqs.getScanFreqs();
? ? ? ? success = mWifiNative.scan(
? ? ? ? ? ? ? ? ? ? ? ? mIfaceName, settings.scanType, freqs, hiddenNetworkSSIDSet);
? ? ? ? if (!success) {
? ? ? ? ? ? Log.e(TAG, "Failed to start scan, freqs=" + freqs);
? ? ? ? }
? ? } else {
? ? ? ? // There is a scan request but no available channels could be scanned for.
? ? ? ? // We regard it as a scan failure in this case.
? ? ? ? Log.e(TAG, "Failed to start scan because there is no available channel to scan");
? ? }
? ? if (success) {
? ? ? ? if (DBG) {
? ? ? ? ? ? Log.d(TAG, "Starting wifi scan for freqs=" + freqs);
? ? ? ? }
?
? ? ? ? mScanTimeoutListener = new AlarmManager.OnAlarmListener() {
? ? ? ? ? ? @Override public void onAlarm() {
? ? ? ? ? ? ? ? handleScanTimeout();
? ? ? ? ? ? }
? ? ? ? };
?
? ? ? ? mAlarmManager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
? ? ? ? ? ? ? ? ? ? mClock.getElapsedSinceBootMillis() + SCAN_TIMEOUT_MS,
? ? ? ? ? ? ? ? ? ? ? ? TIMEOUT_ALARM_TAG, mScanTimeoutListener, mEventHandler);
? ? ? ? } else {
? ? ? ? ? ? // indicate scan failure async
? ? ? ? ? ? mEventHandler.post(new Runnable() {
? ? ? ? ? ? ? ? @Override public void run() {
? ? ? ? ? ? ? ? ? ? reportScanFailure();
? ? ? ? ? ? ? ? }
? ? ? ? ? ? });
? ? ? ? }
?
? ? ? ? return true;
? ? }
}
看 mWifiNative.scan,java框架層最終還是會調用到native層。

2.9 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/WifiNative.java

/**
* Start a scan using wificond for the given parameters.
* @param ifaceName Name of the interface.
* @param scanType Type of scan to perform. One of {@link ScanSettings#SCAN_TYPE_LOW_LATENCY},
* {@link ScanSettings#SCAN_TYPE_LOW_POWER} or {@link ScanSettings#SCAN_TYPE_HIGH_ACCURACY}.
* @param freqs list of frequencies to scan for, if null scan all supported channels.
* @param hiddenNetworkSSIDs List of hidden networks to be scanned for.
* @return Returns true on success.
*/
public boolean scan(@NonNull String ifaceName, int scanType, Set<Integer> freqs,
? ? ? ? ? ? Set<String> hiddenNetworkSSIDs) {
? ? return mWificondControl.scan(ifaceName, scanType, freqs, hiddenNetworkSSIDs);
}
看mWificondControl.scan。

2.10 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/WificondControl.java

/**
* Start a scan using wificond for the given parameters.
* @param ifaceName Name of the interface.
* @param scanType Type of scan to perform.
* @param freqs list of frequencies to scan for, if null scan all supported channels.
* @param hiddenNetworkSSIDs List of hidden networks to be scanned for.
* @return Returns true on success.
*/
public boolean scan(@NonNull String ifaceName,
? ? ? ? ? ? ? ? ? ? int scanType,
? ? ? ? ? ? ? ? ? ? Set<Integer> freqs,
? ? ? ? ? ? ? ? ? ? Set<String> hiddenNetworkSSIDs) {
? ? IWifiScannerImpl scannerImpl = getScannerImpl(ifaceName);
? ? if (scannerImpl == null) {
? ? ? ? Log.e(TAG, "No valid wificond scanner interface handler");
? ? ? ? return false;
? ? }
? ? SingleScanSettings settings = new SingleScanSettings();
? ? try {
? ? ? ? settings.scanType = getScanType(scanType);
? ? } catch (IllegalArgumentException e) {
? ? ? ? Log.e(TAG, "Invalid scan type ", e);
? ? ? ? return false;
? ? }
? ? settings.channelSettings ?= new ArrayList<>();
? ? settings.hiddenNetworks ?= new ArrayList<>();
?
? ? if (freqs != null) {
? ? ? ? for (Integer freq : freqs) {
? ? ? ? ? ? ChannelSettings channel = new ChannelSettings();
? ? ? ? ? ? channel.frequency = freq;
? ? ? ? ? ? settings.channelSettings.add(channel);
? ? ? ? }
? ? }
? ? if (hiddenNetworkSSIDs != null) {
? ? ? ? for (String ssid : hiddenNetworkSSIDs) {
? ? ? ? ? ? HiddenNetwork network = new HiddenNetwork();
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? network.ssid = NativeUtil.byteArrayFromArrayList(NativeUtil.decodeSsid(ssid));
? ? ? ? ? ? } catch (IllegalArgumentException e) {
? ? ? ? ? ? ? ? Log.e(TAG, "Illegal argument " + ssid, e);
? ? ? ? ? ? ? ? continue;
? ? ? ? ? ? }
? ? ? ? ? ? settings.hiddenNetworks.add(network);
? ? ? ? }
? ? }
?
? ? try {
? ? ? ? return scannerImpl.scan(settings);
? ? } catch (RemoteException e1) {
? ? ? ? Log.e(TAG, "Failed to request scan due to remote exception");
? ? }
? ? return false;
}
又調到scannerImpl.scan了,看getScannerImpl。

/** Helper function to look up the scanner impl handle using name */
private IWifiScannerImpl getScannerImpl(@NonNull String ifaceName) {
? ? return mWificondScanners.get(ifaceName);
}
看mWificondScanners怎么來的,從下面可以看到wificondScanner = clientInterface.getWifiScannerImpl()。

/**
* Setup interface for client mode via wificond.
* @return An IClientInterface as wificond client interface binder handler.
* Returns null on failure.
*/
public IClientInterface setupInterfaceForClientMode(@NonNull String ifaceName) {
? ? Log.d(TAG, "Setting up interface for client mode");
? ? if (!retrieveWificondAndRegisterForDeath()) {
? ? ? ? return null;
? ? }
?
? ? IClientInterface clientInterface = null;
? ? try {
? ? ? ? clientInterface = mWificond.createClientInterface(ifaceName);
? ? } catch (RemoteException e1) {
? ? ? ? Log.e(TAG, "Failed to get IClientInterface due to remote exception");
? ? ? ? return null;
? ? }
?
? ? if (clientInterface == null) {
? ? ? ? Log.e(TAG, "Could not get IClientInterface instance from wificond");
? ? ? ? return null;
? ? }
? ? Binder.allowBlocking(clientInterface.asBinder());
?
? ? // Refresh Handlers
? ? mClientInterfaces.put(ifaceName, clientInterface);
? ? try {
? ? ? ? IWifiScannerImpl wificondScanner = clientInterface.getWifiScannerImpl();
? ? ? ? if (wificondScanner == null) {
? ? ? ? ? ? Log.e(TAG, "Failed to get WificondScannerImpl");
? ? ? ? ? ? return null;
? ? ? ? }
? ? ? ? mWificondScanners.put(ifaceName, wificondScanner);
? ? ? ? Binder.allowBlocking(wificondScanner.asBinder());
? ? ? ? ScanEventHandler scanEventHandler = new ScanEventHandler(ifaceName);
? ? ? ? mScanEventHandlers.put(ifaceName, ?scanEventHandler);
? ? ? ? wificondScanner.subscribeScanEvents(scanEventHandler);
? ? ? ? PnoScanEventHandler pnoScanEventHandler = new PnoScanEventHandler(ifaceName);
? ? ? ? mPnoScanEventHandlers.put(ifaceName, ?pnoScanEventHandler);
? ? ? ? wificondScanner.subscribePnoScanEvents(pnoScanEventHandler);
? ? } catch (RemoteException e) {
? ? ? ? Log.e(TAG, "Failed to refresh wificond scanner due to remote exception");
? ? }
?
? ? return clientInterface;
}
看retrieveWificondAndRegisterForDeath。

/**
* Helper method to retrieve the global wificond handle and register for
* death notifications.
*/
private boolean retrieveWificondAndRegisterForDeath() {
? ? if (mWificond != null) {
? ? ? ? if (mVerboseLoggingEnabled) {
? ? ? ? ? ? Log.d(TAG, "Wificond handle already retrieved");
? ? ? ? }
? ? ? ? // We already have a wificond handle.
? ? ? ? return true;
? ? }
? ? mWificond = mWifiInjector.makeWificond();
? ? if (mWificond == null) {
? ? ? ? Log.e(TAG, "Failed to get reference to wificond");
? ? ? ? return false;
? ? }
? ? try {
? ? ? ? mWificond.asBinder().linkToDeath(this, 0);
? ? } catch (RemoteException e) {
? ? ? ? Log.e(TAG, "Failed to register death notification for wificond");
? ? ? ? // The remote has already died.
? ? ? ? return false;
? ? }
? ? return true;
}
看mWiifInjector.makeWificond。

2.11 ?frameworks/opt/net/wifi/service/java/com/android/server/wifi/WifiInjector.java

public IWificond makeWificond() {
? ? // We depend on being able to refresh our binder in WifiStateMachine, so don't cache it.
? ? IBinder binder = ServiceManager.getService(WIFICOND_SERVICE_NAME);
? ? return IWificond.Stub.asInterface(binder);
}
再往下走就是C++層了。往下的部分以后有機會再繼續深入。

四 總結

?wifi框架層從Android O版本開始變化還是挺大的,Android P是延續Android O 的改變做了一些不怎么大的變化:

? ? ? (1) 與其他Android版本相比,整體架構還好,多了wificond跟底層通信,framework整個修改比較大;

? ? ?(2) Android O ?scan、scan_results命令不再通過wpa_supplicant下發到kernel,而是直接由wificond傳送到kernel,而scan results的結果也是直接由kernel傳給wificond,再由wificond傳送給上層去顯示;

? ? ? (3) wifi direct的p2p find還是通過wpa-supplicant通信 ;

? ? ? (4) wifi的連接等操作,還是通過wpa_supplicant進行。
————————————————
版權聲明:本文為CSDN博主「BrighterLi」的原創文章,遵循CC 4.0 BY-SA版權協議,轉載請附上原文出處鏈接及本聲明。
原文鏈接:https://blog.csdn.net/weixin_42093428/article/details/82682804

與50位技術專家面對面20年技術見證,附贈技術全景圖

總結

以上是生活随笔為你收集整理的Wifi模块—源码分析Wifi热点扫描(Android P)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

欧美在一区| 免费a v在线 | 99热这里只有精品8 久久综合毛片 | av字幕在线| 免费看国产曰批40分钟 | 天天综合久久 | 国产 日韩 在线 亚洲 字幕 中文 | 欧美久久久影院 | 色婷在线 | 国产精品久久久久久久久久久免费 | 久久99免费视频 | www.eeuss影院av撸 | 色妞色视频一区二区三区四区 | 国产亚洲婷婷免费 | 久久天天操 | 最新av网站在线观看 | 91福利视频免费观看 | 国产1区在线 | 精品国产乱码一区二 | 成人avav| 九九免费在线观看 | 免费进去里的视频 | 久久天天躁夜夜躁狠狠85麻豆 | 成人少妇影院yyyy | 亚洲另类人人澡 | 欧美性黄网官网 | 国产精品福利av | 中国一级特黄毛片大片久久 | 亚洲天堂精品视频 | 久草91视频 | 婷婷丁香色 | a√天堂中文在线 | 手机av电影在线 | 黄色av影视 | 国产精品第7页 | 1000部国产精品成人观看 | 97精产国品一二三产区在线 | 成人av一区二区兰花在线播放 | 日日干夜夜骑 | 久久久久免费精品 | 免费高清在线视频一区· | 美女网站在线 | 日本黄色黄网站 | 精品一区二区精品 | 午夜久久视频 | av短片在线观看 | 色就是色综合 | 在线视频 国产 日韩 | 久久久精品视频成人 | 美女国内精品自产拍在线播放 | 久久不射电影院 | 精品久久福利 | 久草在线资源观看 | 国产福利91精品 | 91麻豆视频| 日韩av在线影视 | 日韩精品一区二区三区在线播放 | 日本黄色免费大片 | 久久久久激情视频 | 亚洲91视频 | 欧美了一区在线观看 | 日韩免费一级a毛片在线播放一级 | 日日爽视频 | 97超碰影视| 久久成人精品电影 | 91女神的呻吟细腰翘臀美女 | 99精品国产成人一区二区 | 久久艹国产 | 国产又粗又猛又色又黄网站 | 亚洲欧美国产精品久久久久 | 六月丁香激情综合色啪小说 | 亚洲精品午夜国产va久久成人 | 国产视频久久 | 一级特黄aaa大片在线观看 | 夜夜爽www| 婷婷播播网 | 91精品国产九九九久久久亚洲 | 国产午夜精品免费一区二区三区视频 | 丝袜护士aⅴ在线白丝护士 天天综合精品 | 亚洲精品一区二区在线观看 | av网址aaa | 日韩电影一区二区三区在线观看 | 久久久久久久久久网 | 超碰人人av| 婷婷丁香久久五月婷婷 | 日韩久久久久久久久久久久 | 国内精品久久久久久久久久久 | 国产高清视频色在线www | 天天看天天操 | 亚洲成人av一区二区 | 国产精品精品 | av一区二区三区在线 | 国产小视频在线免费观看视频 | 国内精品久久久久久久久久久 | 精品一区二区在线免费观看 | 欧美一级特黄高清视频 | 久久久91精品国产一区二区三区 | 中文字幕久久亚洲 | 久久久久高清 | 久久国产精品免费看 | 大型av综合网站 | 欧美日韩中文字幕在线视频 | 一本一道波多野毛片中文在线 | 精品一区二三区 | 国产婷婷色 | 99久高清在线观看视频99精品热在线观看视频 | 国产成人一二三 | 日韩欧美一区二区在线播放 | 中文字幕成人在线观看 | 亚洲综合视频网 | 毛片网站免费 | 国产麻豆视频在线观看 | 成人av播放 | 久久久久综合 | 又黄又网站 | 免费观看第二部31集 | 丝袜美女视频网站 | 久久人人爽爽人人爽人人片av | 久久精品爱视频 | 国产成人在线网站 | 黄色大片av| 日韩在线欧美在线 | 久久久久网址 | 激情五月婷婷综合 | 国产成人精品三级 | 麻豆国产精品视频 | 九九免费在线观看视频 | 精品一区精品二区高清 | 男女男视频 | 欧美日韩中文字幕在线视频 | 亚洲精品国偷自产在线91正片 | 欧美精品v国产精品v日韩精品 | 99国内精品久久久久久久 | 999久久久精品视频 日韩高清www | 国产亚洲亚洲 | 国产又黄又爽又猛视频日本 | 99久久精品国产观看 | 日韩精品观看 | 亚洲经典视频 | 久久久www成人免费精品张筱雨 | 国产视频一 | 精品国产一区二区久久 | 日韩精品一区二区在线观看 | zzijzzij日本成熟少妇 | 97成人免费视频 | 国产一级免费观看 | 久久首页 | 天天爽人人爽夜夜爽 | 免费涩涩网站 | 成人国产一区二区 | 91成人在线看 | 国产日本亚洲高清 | 国产区精品视频 | av电影在线播放 | 国产人免费人成免费视频 | 亚洲国产精品va在线看黑人动漫 | 麻豆91在线看 | 国产高清在线观看av | 国产玖玖在线 | 91亚洲欧美 | 久草影视在线 | 久久成人资源 | 久久国产精品久久久 | 99在线观看 | www免费视频com━ | 午夜丁香视频在线观看 | 国产精品一区二区三区久久 | 麻豆精品传媒视频 | 免费黄色a级毛片 | 波多野结衣久久资源 | 婷婷久久久 | 久艹在线观看视频 | www.夜夜草| 在线精品视频免费播放 | 四虎国产视频 | 精品视频专区 | 狠狠色综合欧美激情 | 免费电影一区二区三区 | 日韩午夜视频在线观看 | 丁香六月婷婷综合 | 亚洲最新视频在线播放 | 黄网站免费大全入口 | 五月导航 | 国产精品一区在线观看你懂的 | 人成在线免费视频 | 青草视频在线 | 91桃色在线观看视频 | 国产欧美三级 | 国产精品久久久久久久久久久久午夜 | 一级黄色片在线免费观看 | 一区免费观看 | 91污污视频在线观看 | 免费的国产精品 | 免费男女羞羞的视频网站中文字幕 | 精品视频123区在线观看 | www日| 久久综合亚洲鲁鲁五月久久 | 国产成人在线网站 | 欧美精品一区二区在线播放 | 黄色免费观看网址 | 国产免费观看视频 | 久久丁香网| 99热最新地址 | 国产免费高清视频 | 热久久国产精品 | 美女免费视频一区 | 天天艹 | 久久视频中文字幕 | 三级av黄色| 欧美性精品 | 综合伊人久久 | 丁香婷婷激情网 | 亚洲激情综合网 | 婷婷六月激情 | 久久综合久久综合九色 | www.黄色片.com | 久久99热这里只有精品国产 | 国产精品久久久久久久久毛片 | www.com.日本一级| 欧美日韩在线免费观看 | 婷婷综合导航 | 国产成人久久av977小说 | 在线不卡a | av电影在线免费观看 | 成年人国产在线观看 | 伊人资源站 | 日女人电影| 亚州中文av | 91视频免费视频 | 色婷婷啪啪免费在线电影观看 | 日韩精品免费一区二区三区 | 亚洲午夜久久久久久久久久久 | 日日爱av | 日韩电影在线看 | 亚洲国产人午在线一二区 | 国产一级精品在线观看 | 2018亚洲男人天堂 | 久久综合射| 国产黄色电影 | 91完整版观看 | 日本3级在线观看 | 在线观看中文字幕dvd播放 | 超碰国产人人 | 国产亚洲精品久久网站 | 久插视频 | 欧美无极色| 欧美夫妻性生活电影 | 中文字幕在线播放视频 | 亚洲dvd| 国产精品久久久久久久久久久久冷 | av免费黄色 | 国产一二三四在线视频 | 国产精品乱码久久 | 欧美黄在线 | 狠狠色丁香久久婷婷综合五月 | 日本在线精品视频 | 黄色免费网站 | 亚洲激情 | 黄色一级在线视频 | 国产精品一区二区在线 | 亚洲人成免费 | 久久久黄色免费网站 | 中文字幕亚洲国产 | 亚洲综合黄色 | 国产一级三级 | 91视频电影| 天天综合网 天天综合色 | 91精品国产91久久久久 | 国产美女视频免费 | 久久人人看| 久久免费福利视频 | 岛国大片免费视频 | 国产网站av| 久久久久久久久久电影 | 国产无吗一区二区三区在线欢 | 97超视频在线观看 | 在线三级av| 二区三区视频 | 日韩理论在线播放 | 91av视频在线观看 | 91超碰免费在线 | 亚洲蜜桃在线 | 日产av在线播放 | 色综合久久网 | 免费黄av| jizz欧美性9 国产一区高清在线观看 | 国产精选视频 | 成人午夜在线观看 | 操天天操| 日本h视频在线观看 | 国产成人av一区二区三区在线观看 | 丝袜美腿在线视频 | 五月天六月婷婷 | 成人av资源| 99久久婷婷国产一区二区三区 | 国产三级视频 | 精品国产乱码久久久久久三级人 | 日日摸日日爽 | 精品av网站 | 黄色美女免费网站 | 欧美精品国产综合久久 | 超碰成人网 | www.日日日.com| 97超碰人人澡人人爱 | 国产一区二区三区免费在线观看 | 成人综合婷婷国产精品久久免费 | 日韩一区精品 | 99综合影院在线 | 久久久久国产一区二区三区四区 | 国产精品久久精品 | 欧美一区二区三区在线看 | 久久久久亚洲精品中文字幕 | 97视频人人澡人人爽 | www,黄视频| 嫩草91影院 | 手机成人av在线 | 丁香资源影视免费观看 | 色婷婷精品 | 国产黄在线播放 | 国产手机视频在线 | 久久九九久久九九 | av爱干| 超碰97人人在线 | 亚洲成av| 日韩理论影院 | 国语精品视频 | 国产精品 9999| 亚洲免费成人 | 九九影视理伦片 | 一级黄色在线免费观看 | 亚洲精品白浆高清久久久久久 | 欧美视频在线观看免费网址 | 麻豆果冻剧传媒在线播放 | 国产日韩精品在线观看 | 永久免费视频国产 | 精品美女久久久久久免费 | 一区二区三区韩国免费中文网站 | 天天综合网久久综合网 | 中文字幕视频播放 | 99福利片 | 国产99在线 | 国产日本亚洲高清 | 99精品国产aⅴ | 伊人天天狠天天添日日拍 | 国产精品一区二区三区久久 | 色综合久久久久久久久五月 | 欧美一级片在线播放 | 99久久婷婷国产精品综合 | 国产亚洲精品女人久久久久久 | 97超碰在线播放 | 欧美日韩三区二区 | av不卡免费在线观看 | 欧美日韩中文字幕在线视频 | 国产精品久久久视频 | 草久热| 99视频精品 | 黄色软件视频大全免费下载 | 狠狠的操狠狠的干 | 亚洲精品久久久久999中文字幕 | 久久久久久国产精品久久 | 日韩一区二区三区免费视频 | 欧美日韩在线播放一区 | 91完整版在线观看 | 五月综合久久 | 久久99电影 | 国产日韩欧美在线观看 | 欧美日韩在线播放 | 97电影院网 | 国产精品一区二区三区在线播放 | 成人羞羞视频在线观看免费 | 综合色爱 | 国产精品乱码久久久久久1区2区 | 欧美性做爰猛烈叫床潮 | 日韩精品三区四区 | 这里有精品在线视频 | 91夜夜夜| 五月婷婷丁香在线观看 | 久草新在线| 五月综合在线观看 | 精品亚洲网 | 91成版人在线观看入口 | 欧美久久成人 | 国产乱对白刺激视频在线观看女王 | 国产一级电影免费观看 | 九九精品毛片 | 欧美精品久久久久久 | 免费的成人av | 国产一区二区三区在线免费观看 | 成人日批视频 | 狠狠色噜噜狠狠狠狠2021天天 | 在线日韩精品视频 | 在线超碰av | 一区三区视频在线观看 | 9999在线| 亚洲国产成人精品久久 | 狠狠网站| 美女黄色网在线播放 | 成人九九视频 | 久久精品视频3 | 探花视频在线观看 | 欧美一区二区伦理片 | 精品中文字幕在线观看 | www.夜夜干.com | 国产黄大片 | 伊人干综合 | www.天天成人国产电影 | 国产精品6999成人免费视频 | 亚洲最大av在线播放 | 99久久婷婷国产 | 亚洲日韩中文字幕 | 国产精品ssss在线亚洲 | 亚洲国内精品在线 | 久久成人高清 | av一区二区在线观看中文字幕 | 精品美女久久久久久免费 | 成人黄色片免费 | 久久久综合电影 | 中文字幕一区av | 99视频播放 | 天天躁日日躁狠狠躁 | 日韩在线理论 | 91九色视频观看 | 伊人五月婷 | 奇米四色影狠狠爱7777 | 精品久久久99 | 亚洲精品一区二区在线观看 | 亚洲午夜精品久久久 | 激情www| 国产 在线 高清 精品 | 久久er99热精品一区二区三区 | 国产很黄很色的视频 | 国内精品久久天天躁人人爽 | 欧美一级xxxx | 91精品视频在线 | 天天插天天狠天天透 | 亚洲国产wwwccc36天堂 | 亚洲午夜激情网 | 久久艹综合 | 中文字幕在线观看你懂的 | 欧美夫妻性生活电影 | 黄av在线 | 麻豆视频在线播放 | 国产精品免费在线 | 天天操夜| 99精品国产在热久久 | 91在线观看高清 | 黄色一区三区 | 人人搞人人干 | 久久在线影院 | 操操操人人| 亚洲午夜不卡 | 亚洲欧美日韩精品久久奇米一区 | 欧美人人爱 | 一级c片 | 精品无人国产偷自产在线 | 欧美另类网站 | 91九色蝌蚪国产 | 香蕉一区| 五月天激情综合网 | 韩国av在线 | 九色91在线| 国产专区在线 | 亚洲人av免费网站 | 国产精品国产精品 | 久草国产视频 | 久久99免费| 六月丁香激情综合 | 久久综合婷婷国产二区高清 | 国产v在线观看 | 最新av网址大全 | 99久热在线精品视频成人一区 | 97久久精品午夜一区二区 | 国产日韩视频在线播放 | 91精品久久久久 | 精品国产乱码久久久久久1区二区 | 久久久久久久久久久久久国产精品 | 中文字幕在线看视频国产中文版 | 444av| av一区在线 | 超碰999| 大荫蒂欧美视频另类xxxx | 草久中文字幕 | 久久久久99精品成人片三人毛片 | 中文一区二区三区在线观看 | av高清免费在线 | 四虎影视欧美 | 婷婷伊人网 | 六月色 | 亚洲精品99久久久久中文字幕 | 丁香五月缴情综合网 | 亚洲成人资源在线 | 91人人插| 伊人久久精品久久亚洲一区 | 在线免费观看av网站 | 精品国产a | 中文字幕日韩一区二区三区不卡 | 亚洲成人av一区 | 国产精品mm | 97国产情侣爱久久免费观看 | 欧美视频99 | 精品一二 | 亚洲狠狠丁香婷婷综合久久久 | 午夜av在线播放 | 国产精品久久久久久一二三四五 | 色在线中文字幕 | 久久五月网| 免费看成人片 | 99精品区 | 久久伊人91 | av视屏在线 | 国产黄色片在线 | 亚洲,播放| av专区在线 | 色综合天天综合 | 国产伦精品一区二区三区免费 | 国产精品久久一卡二卡 | 亚洲精品国产精品99久久 | 99精品视频一区二区 | 国产亚洲精品久久久久久网站 | 婷婷丁香久久五月婷婷 | 色婷婷中文| 国产成人精品日本亚洲999 | 91精品国产综合久久婷婷香蕉 | 麻豆91在线播放 | 久久成 | 国产香蕉97碰碰久久人人 | 中文字幕国内精品 | 免费观看的黄色 | 国产精品入口麻豆 | 国产资源精品在线观看 | 91麻豆国产福利在线观看 | 国产激情免费 | 月下香电影 | 国产精品久久9 | 美州a亚洲一视本频v色道 | 久久久久久久久久影视 | 五月婷婷综合在线 | 中文字幕在线网 | 精品久久久影院 | 中文字幕在线视频免费播放 | av福利在线播放 | 婷婷色中文网 | 亚洲精品在线观看免费 | 日韩视频免费播放 | 国产精品麻豆91 | 超碰午夜 | 午夜精品一区二区三区在线播放 | 丰满少妇在线观看资源站 | av电影不卡在线 | 亚洲精品在线观看中文字幕 | 欧美精品久久久久久久久久丰满 | 91视频91自拍 | 一本一道久久a久久综合蜜桃 | 国产精品激情在线观看 | 最新午夜 | 精品在线一区二区 | 久久激情久久 | 精品自拍av| 成人久久影院 | 97超碰免费 | 超碰久热 | 欧美日韩一二三四区 | 日本久久久久久久久久 | 18岁免费看片 | 99av在线视频 | 超碰在线公开 | 毛片在线播放网址 | 成人免费看片98欧美 | 2019av在线视频 | 国产精品婷婷 | 日韩欧美一区二区不卡 | 免费黄色小网站 | 五月天久久精品 | 91在线看免费 | 一本到视频在线观看 | 超碰人人99 | 亚洲第一区在线播放 | 久久久在线视频 | 在线视频国产区 | 国产精品不卡在线播放 | 国产精品久久久久久久久蜜臀 | 婷婷去俺也去六月色 | 三级午夜片 | 日本高清xxxx | 99久久精品国产毛片 | av丁香| 久久国产精品99久久久久久进口 | 免费在线观看av网址 | 国产成人三级一区二区在线观看一 | 成人理论在线观看 | 美女国产精品 | 久久黄色a级片 | 日韩精品一区二区免费 | 在线免费观看国产精品 | 91理论电影| 久草在线观看资源 | 99国产视频 | 婷婷丁香九月 | 午夜12点 | 成人av在线影视 | 91在线视频免费观看 | 日日草视频 | 日韩精品久久一区二区三区 | 国产精品第72页 | 中文字幕在线免费看线人 | 国产中文欧美日韩在线 | 区一区二在线 | 又爽又黄又无遮挡网站动态图 | 天天色天天射综合网 | 久久久久亚洲精品国产 | 免费亚洲视频 | 久久免费高清 | 午夜少妇一区二区三区 | 四虎4hu永久免费 | 99久久久成人国产精品 | 免费观看视频黄 | 一区在线电影 | 久久免费视频这里只有精品 | 久久精品国产免费看久久精品 | 日日夜夜综合网 | 日韩电影在线视频 | 亚洲乱码国产乱码精品天美传媒 | 亚洲春色综合另类校园电影 | 六月婷婷久香在线视频 | 深爱五月激情五月 | 国产精品1区2区3区在线观看 | 欧亚日韩精品一区二区在线 | 国产一区二区三区免费视频 | 特级毛片在线观看 | 天天操天天射天天舔 | 日日操日日 | 日b视频在线观看网址 | 日韩精品一区在线观看 | 在线观看 国产 | 黄网在线免费观看 | 国产精品99精品 | 亚洲国产大片 | 久久麻豆视频 | 亚洲精品国偷拍自产在线观看蜜桃 | 欧美成人猛片 | 国产99久久久国产精品 | 亚洲 精品在线视频 | 青青看片| 久久久久久国产精品免费 | 丁香花在线观看免费完整版视频 | 国产不卡精品视频 | 丰满少妇对白在线偷拍 | 久久在草 | 成在人线av | 亚洲国产日韩一区 | 超级碰99| 日韩黄色中文字幕 | 日韩综合在线观看 | 精品久久免费看 | 特级a毛片| 天天干天天操人体 | 91精品成人| 一区二区三区久久精品 | 日韩二区三区在线 | 日本韩国在线不卡 | 在线观看播放av | 国产视频一区二区在线观看 | 欧美一级片免费在线观看 | 国产中文欧美日韩在线 | 久草精品视频在线观看 | 亚洲 欧美日韩 国产 中文 | 国产精品久久久久久久久久直播 | 狠狠色丁香久久婷婷综合五月 | 午夜视频黄 | www.av免费| 日本动漫做毛片一区二区 | 在线a亚洲视频播放在线观看 | 亚洲成年人免费网站 | 日韩精品免费在线观看视频 | 国产 亚洲 欧美 在线 | 久久久久 | 超碰av在线播放 | 在线你懂 | 久久综合中文字幕 | 91久草视频 | 最新国产精品久久精品 | 97超碰人人网 | 久久久伊人网 | 欧美日韩一区二区三区在线免费观看 | 69久久99精品久久久久婷婷 | 精品一区二区在线免费观看 | 亚洲视频axxx | 夜夜操狠狠操 | 99精彩视频在线观看免费 | 超碰夜夜 | 日本三级全黄少妇三2023 | 国产a网站 | 91在线看视频 | 日韩在线国产精品 | 国产日韩精品久久 | 国产精品白浆 | 91热视频在线观看 | 久久婷婷色 | 天天干天天综合 | 欧洲亚洲国产视频 | 日韩精品亚洲专区在线观看 | 国产一级黄色电影 | 色欧美综合 | 在线婷婷 | 天天射天天爱天天干 | 激情亚洲综合在线 | 日韩在线视频线视频免费网站 | 国产精久久久久久久 | 欧美日韩在线视频免费 | 欧美成人a在线 | 日韩电影在线观看一区二区 | www.人人草| 国产精品毛片一区视频播 | 久久草草影视免费网 | 国产精品欧美日韩在线观看 | 免费网站污| 高清久久久久久 | 国产精品永久在线观看 | 98精品国产自产在线观看 | 成人黄大片| 天堂在线成人 | 日韩电影在线一区 | 91热爆视频 | 久久99精品久久久久蜜臀 | 日日干天天插 | 中文字幕在线观看资源 | 免费高清看电视网站 | 久热色超碰 | 九九九九九九精品任你躁 | 中文字幕在线视频一区二区 | 狠狠色噜噜狠狠狠合久 | 亚洲高清精品在线 | 午夜影院日本 | 久久久久久久久久久影院 | 成人av在线看 | 国产在线播放一区 | 91传媒在线看 | 狠狠狠综合 | 欧洲在线免费视频 | 国产精品1区2区 | 亚洲精品午夜一区人人爽 | 黄色片网站av | 国产精品成人一区二区 | 日本久久中文 | 91精品啪在线观看国产81旧版 | 久久成年视频 | 黄色a一级片 | 国产视频一区在线 | 在线v片| 人人讲下载 | 97狠狠干 | 亚色视频在线观看 | 在线免费黄网站 | 不卡的av电影在线观看 | 天天精品视频 | 九九热re| 国产小视频在线播放 | av官网在线 | 日韩黄色一区 | 麻豆传媒在线免费看 | 亚洲高清免费在线 | 狠狠色丁香婷婷综合久小说久 | 美女黄网站视频免费 | 久久这里只有精品视频首页 | 天天操天天操天天操天天操天天操天天操 | 色视频在线看 | 一区二区三区福利 | 91看片淫黄大片91 | 91精品播放| 成人午夜剧场在线观看 | 黄色三级久久 | 四虎免费在线观看 | 欧美成人手机版 | 色www.| 久久亚洲视频 | 日韩久久精品一区二区三区下载 | 成人免费在线视频 | 成人中文字幕+乱码+中文字幕 | 久久婷婷五月综合色丁香 | 97色在线观看免费视频 | 人人射人人爽 | 久久免费成人精品视频 | a级一a一级在线观看 | 啪啪激情网 | 精品人妖videos欧美人妖 | 国产精品久久久久久久电影 | 日日婷婷夜日日天干 | 18性欧美xxxⅹ性满足 | 在线免费视频 你懂得 | 亚洲天堂在线观看完整版 | 免费观看性生交 | 亚洲97在线 | av女优中文字幕在线观看 | 久久av一区二区三区亚洲 | 91精品啪在线观看国产81旧版 | 成人午夜电影网 | 久久成人国产精品入口 | 亚洲欧美日韩国产一区二区三区 | 天天做日日爱夜夜爽 | 福利网址在线观看 | 在线观看精品国产 | 久久国产精品久久国产精品 | 99色亚洲| 成人黄在线 | 四虎影视4hu4虎成人 | 久久久久久久久久久高潮一区二区 | 日日夜夜精品 | 婷婷久久综合网 | 欧美在线观看视频 | 久久av中文字幕片 | 色综合天天爱 | 免费www视频| 精品一区二区日韩 | 欧美日韩免费观看一区=区三区 | 色综合网| 亚洲日本在线视频观看 | 国产成人精品一区一区一区 | 国产成a人亚洲精v品在线观看 | 成年人在线观看 | 狠色在线| 久久精品91久久久久久再现 | 欧美日韩伦理在线 | 久草在线手机观看 | 日韩免费在线视频观看 | 国产不卡在线观看视频 | 香蕉在线播放 | 五月天开心| 久久成人免费电影 | 亚洲永久精品一区 | 91视频 - 114av | 九九精品久久 | 操操操操网 | 去看片| 中文字幕一区二区三区久久 | 色偷偷88888欧美精品久久久 | 波多野结衣电影一区 | 免费看v片网站 | 亚洲日本欧美 | 亚洲电影第一页av | 成人资源站 | 国产精品av在线免费观看 | 欧美一级片免费 | 九九免费在线观看 | 天天玩天天干 | 国产一区二区不卡视频 | 久久国产精品99久久人人澡 | 四虎国产精品免费观看视频优播 | 激情五月婷婷综合网 | 婷婷丁香六月 | 欧美日韩在线视频一区二区 | 亚洲2019精品 | 色午夜影院 | 亚洲一区欧美精品 | 国产综合香蕉五月婷在线 | 27xxoo无遮挡动态视频 | 亚洲国产精久久久久久久 | 国产精品入口传媒 | 亚洲影院一区 | 日韩免费观看一区二区三区 | 国产在线久草 | 国产精品福利午夜在线观看 | a午夜在线 | 国产一区二区手机在线观看 | 精品国产日本 | 亚洲视频,欧洲视频 | 婷婷丁香六月天 | 国产 字幕 制服 中文 在线 | 在线观看成人av | 97精品国产91久久久久久久 | 欧美另类美少妇69xxxx | 久草在线视频免费资源观看 | 黄色avwww| 亚洲天天 | 四虎国产精品成人免费影视 | 精品国产一区二区三区在线 | 久久99精品波多结衣一区 | 成 人 免费 黄 色 视频 | 成 人 黄 色 视频 免费观看 | 91麻豆精品国产91久久久无限制版 | 国产精品久久久久一区二区三区 | 中文字幕一区二区三 | 狠狠干成人综合网 | 不卡的av片 | 精品久久1 | 99精品久久久久久久久久综合 | 91字幕 | 五月婷亚洲 | 五月天天天操 | 亚洲精品国偷拍自产在线观看蜜桃 | 97在线精品国自产拍中文 | 日本最大色倩网站www | 在线观看成人福利 | 亚洲精品午夜国产va久久成人 | 成人资源在线 | 久久久久国产精品午夜一区 | 久久久久欠精品国产毛片国产毛生 | 亚洲成人精品国产 | 国产精品国产三级在线专区 | 国产精品久久久久久久午夜 | 国产免费观看久久 | 免费看毛片在线 | 91完整版在线观看 | 91精品国产九九九久久久亚洲 | 黄色三级av| 在线视频观看你懂的 | 中文字幕高清有码 | 超碰在线日韩 | 免费观看完整版无人区 | 日韩素人在线观看 | 狠狠色丁香婷婷综合欧美 | 99久久精品久久久久久清纯 | 亚洲成年人在线播放 | 日日干激情五月 | 国产精品片 | 在线视频1卡二卡三卡 | 午夜精品一区二区三区免费视频 | 99精品国产一区二区三区不卡 | 婷婷丁香狠狠爱 | 免费91在线观看 | 成人综合婷婷国产精品久久免费 | 日日日爽爽爽 | 亚洲精品综合欧美二区变态 | 亚洲免费观看在线视频 | 91你懂的 | 国产精品久久久久影院 | 97超碰在线资源 | 欧美精品乱码久久久久久按摩 | 亚洲日本va午夜在线影院 | 97国产精品免费 | 国产一二区视频 | 日韩色视频在线观看 | 操操操影院 | 视频在线播放国产 | 天天干天天看 | 99精品系列 | 91色在线观看视频 | 美女福利视频在线 | 欧洲精品亚洲精品 | 中文字幕av免费在线观看 | 免费av在线 | 色99网| 日本xxxx.com | 99久久精品久久亚洲精品 | 日日夜夜免费精品视频 | 91久久黄色| 免费又黄又爽视频 | 久久在线影院 | 国产精品精 | 国产精品久久99精品毛片三a | 日韩中文免费视频 | 国产福利一区在线观看 | 久久这里只有精品9 | 99热超碰| 处女av在线 | 久久视频免费在线观看 | 日韩中文字幕网站 | 亚洲一区二区黄色 | 五月天丁香视频 | 日韩成人高清在线 | 日本黄区免费视频观看 | 国内三级在线观看 | 日韩在线视频网 | 国产精品久久久久三级 | 国产高清视频在线观看 | 亚洲黄色一级视频 | 欧美性成人 | 夜夜骑日日 | 亚洲精品videossex少妇 | 欧美99久久 | 中文字幕一区二区三区在线观看 | 免费毛片一区二区三区久久久 | 欧美日本高清视频 | www麻豆视频 | 久久av中文字幕片 | 午夜一级免费电影 | 日韩免费av片 | 开心激情网五月天 | 免费久久久久久久 | 麻豆影视网| 精品国产免费av | 天天在线视频色 | 最近中文字幕 | 日本久久精品 | 久久不射电影院 | 在线亚洲观看 | 国产五月天婷婷 | 久久99精品国产99久久6尤 | 色婷婷综合久久久久中文字幕1 | 中文字幕在线观看网站 | 最近高清中文字幕在线国语5 | 成人av影视在线 | 亚洲免费色 | 精品美女在线视频 | 日韩在线观看三区 | 九九久久电影 | 成人理论电影 | 国产一区二区三区高清播放 | 狠狠操操|