(一) Android O WiFi启动流程梳理

前言:最近又重新拿起来WiFi模块,从WiFi 各个流程梳理开始复习一下。

 

参考博客:https://blog.csdn.net/csdn_of_coder/article/details/51541094  

(里面也不少framework和WiFi的文章,我得过一遍,很对口)

 

aosp: Android O

 

1.wifi启动流程简介

用户可以通过systemUi和设置里的WiFi开关打开WiFi,这时候会调用到wifi framework的相关接口,继而再继续往下启用具体的硬件完成WiFi启动流程,我只对应用到framework层有些简单的了解,本篇也主要注重framework这一块,app层没啥好说的。

 

2.WiFi启动流程梳理

我之前是负责设置模块的,systemUi代码虽然看过,但是不是很熟,所以WiFi打开流程还是从设置的WiFi开关开始梳理吧,不考虑打开飞行模式后打开WiFi的情况=-=

2.1 设置启动WiFi

设置这边说到底其实就是监控WiFi开关的变化,然后根据开关走对应的逻辑处理。

两个比较重要的类:

1)WifiSettings:设置中wifi主界面所对应的代码

2)WifiEnabler:设置中负责wifi开关打开和关闭事件处理的类

/aosp/packages/apps/Settings$ vim ./src/com/android/settings/wifi/WifiSettings.java

    /**
     * @return new WifiEnabler or null (as overridden by WifiSettingsForSetupWizard)
     */
    private WifiEnabler createWifiEnabler() {
        final SettingsActivity activity = (SettingsActivity) getActivity();
        return new WifiEnabler(activity, new SwitchBarController(activity.getSwitchBar()),
            mMetricsFeatureProvider);
    }

/aosp/packages/apps/Settings$ vim ./src/com/android/settings/widget/SwitchBarController.java

public class SwitchBarController extends SwitchWidgetController implements
    SwitchBar.OnSwitchChangeListener {
 
    private final SwitchBar mSwitchBar;
 
    public SwitchBarController(SwitchBar switchBar) {
        mSwitchBar = switchBar;
    }
...
    @Override
    public void startListening() {
        mSwitchBar.addOnSwitchChangeListener(this);
    }
...
    @Override
    public void onSwitchChanged(Switch switchView, boolean isChecked) {
        if (mListener != null) {
            mListener.onSwitchToggled(isChecked);
        }
    }

 

 

2.2 WiFi framework

看下WifiManager的setWifiEabled(boolean)接口

framework/base/wifi/java/android/net/wifi/WifiManager.java

 
  1. /**

  2. * Enable or disable Wi-Fi.

  3. *

  4. * Note: This method will return false if wifi cannot be enabled (e.g., an incompatible mode

  5. * where the user has enabled tethering or Airplane Mode).

  6. *

  7. * Applications need to have the {@link android.Manifest.permission#CHANGE_WIFI_STATE}

  8. * permission to toggle wifi. Callers without the permissions will trigger a

  9. * {@link java.lang.SecurityException}.

  10. *

  11. * @param enabled {@code true} to enable, {@code false} to disable.

  12. * @return {@code true} if the operation succeeds (or if the existing state

  13. * is the same as the requested state). False if wifi cannot be toggled on/off when the

  14. * request is made.

  15. */

  16. public boolean setWifiEnabled(boolean enabled) {

  17. try {

  18. return mService.setWifiEnabled(mContext.getOpPackageName(), enabled);

  19. } catch (RemoteException e) {

  20. throw e.rethrowFromSystemServer();

  21. }

  22. }

2.2.1 WifiService是什么

而mService是啥呢

 
  1. IWifiManager mService;

  2. /**

  3. * Create a new WifiManager instance.

  4. * Applications will almost always want to use

  5. * {@link android.content.Context#getSystemService Context.getSystemService()} to retrieve

  6. * the standard {@link android.content.Context#WIFI_SERVICE Context.WIFI_SERVICE}.

  7. * @param context the application context

  8. * @param service the Binder interface

  9. * @hide - hide this because it takes in a parameter of type IWifiManager, which

  10. * is a system private class.

  11. */

  12. public WifiManager(Context context, IWifiManager service, Looper looper) {

  13. mContext = context;

  14. mService = service;

  15. mLooper = looper;

  16. mTargetSdkVersion = context.getApplicationInfo().targetSdkVersion;

  17. }

应用层的WifiManager都是这么来的

mWifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);

那真正的WifiManager实例是谁来new出来的呢?

有这么一个类:

/framework/base/core/java/android/app/SystemServiceRegistry.java

 
  1. registerService(Context.WIFI_SERVICE, WifiManager.class,

  2. new CachedServiceFetcher() {

  3. @Override

  4. public WifiManager createService(ContextImpl ctx) throws ServiceNotFoundException {

  5. IBinder b = ServiceManager.getServiceOrThrow(Context.WIFI_SERVICE);

  6. IWifiManager service = IWifiManager.Stub.asInterface(b);

  7. return new WifiManager(ctx.getOuterContext(), service,

  8. ConnectivityThread.getInstanceLooper());

  9. }});

它有个静态代码块,大致如下,负责创建各种manager实例

 
  1. final class SystemServiceRegistry {

  2. ...

  3.  
  4. // Not instantiable.

  5. private SystemServiceRegistry() { }

  6.  
  7. static {

  8. registerService(Context.ACCESSIBILITY_SERVICE, AccessibilityManager.class,

  9. new CachedServiceFetcher() {

  10. @Override

  11. public AccessibilityManager createService(ContextImpl ctx) {

  12. return AccessibilityManager.getInstance(ctx);

  13. }});

  14.  
  15. registerService(Context.CAPTIONING_SERVICE, CaptioningManager.class,

  16. new CachedServiceFetcher() {

  17. @Override

  18. public CaptioningManager createService(ContextImpl ctx) {

  19. return new CaptioningManager(ctx);

  20. }});

  21.  
  22. registerService(Context.ACCOUNT_SERVICE, AccountManager.class,

  23. new CachedServiceFetcher() {

  24. @Override

  25. public AccountManager createService(ContextImpl ctx) throws ServiceNotFoundException {

  26. IBinder b = ServiceManager.getServiceOrThrow(Context.ACCOUNT_SERVICE);

  27. IAccountManager service = IAccountManager.Stub.asInterface(b);

  28. return new AccountManager(ctx, service);

  29. }});

  30. ...

  31. }

而调用呢,则看我们之前说的应用层ContextImpl的getSystemService方法

 
  1. // The system service cache for the system services that are cached per-ContextImpl.

  2. final Object[] mServiceCache = SystemServiceRegistry.createServiceCache();

 
  1. @Override

  2. public Object getSystemService(String name) {

  3. return SystemServiceRegistry.getSystemService(this, name);

  4. }

SystemServiceRegistry

 
  1. // Service registry information.

  2. // This information is never changed once static initialization has completed.

  3. private static final HashMap, String> SYSTEM_SERVICE_NAMES =

  4. new HashMap, String>();

  5. private static final HashMap> SYSTEM_SERVICE_FETCHERS =

  6. new HashMap>();

  7. /**

  8. * Gets a system service from a given context.

  9. */

  10. public static Object getSystemService(ContextImpl ctx, String name) {

  11. ServiceFetcher fetcher = SYSTEM_SERVICE_FETCHERS.get(name);

  12. return fetcher != null ? fetcher.getService(ctx) : null;

  13. }

  14. /**

  15. * Statically registers a system service with the context.

  16. * This method must be called during static initialization only.

  17. */

  18. private static void registerService(String serviceName, Class serviceClass,

  19. ServiceFetcher serviceFetcher) {

  20. SYSTEM_SERVICE_NAMES.put(serviceClass, serviceName);

  21. SYSTEM_SERVICE_FETCHERS.put(serviceName, serviceFetcher);

  22. }

这里主要就弄清了一件事,WifiManager是aidl的客户端,具体逻辑还是要去看服务端的。即看下Service对应代码:

IBinder b = ServiceManager.getServiceOrThrow(Context.WIFI_SERVICE);

 

而之前在(四十一) SystemServer初探有说到WifiService

 

 
  1. // Wifi Service must be started first for wifi-related services.

  2. traceBeginAndSlog("StartWifi");

  3. mSystemServiceManager.startService(WIFI_SERVICE_CLASS);

  4. traceEnd();

 
  1. private static final String WIFI_SERVICE_CLASS =

  2. "com.android.server.wifi.WifiService";

这里感觉是有关的。

ServiceManager是这样的:

 
  1. public final class ServiceManager {

  2. private static final String TAG = "ServiceManager";

  3. /**

  4. * Returns a reference to a service with the given name.

  5. *

  6. * @param name the name of the service to get

  7. * @return a reference to the service, or null if the service doesn't exist

  8. */

  9. public static IBinder getService(String name) {

  10. try {

  11. IBinder service = sCache.get(name);

  12. if (service != null) {

  13. return service;

  14. } else {

  15. return Binder.allowBlocking(getIServiceManager().getService(name));

  16. }

  17. } catch (RemoteException e) {

  18. Log.e(TAG, "error in getService", e);

  19. }

  20. return null;

  21. }

  22.  
  23. /**

  24. * Returns a reference to a service with the given name, or throws

  25. * {@link NullPointerException} if none is found.

  26. *

  27. * @hide

  28. */

  29. public static IBinder getServiceOrThrow(String name) throws ServiceNotFoundException {

  30. final IBinder binder = getService(name);

  31. if (binder != null) {

  32. return binder;

  33. } else {

  34. throw new ServiceNotFoundException(name);

  35. }

  36. }

  37. ...

  38. /**

  39. * This is only intended to be called when the process is first being brought

  40. * up and bound by the activity manager. There is only one thread in the process

  41. * at that time, so no locking is done.

  42. *

  43. * @param cache the cache of service references

  44. * @hide

  45. */

  46. public static void initServiceCache(Map cache) {

  47. if (sCache.size() != 0) {

  48. throw new IllegalStateException("setServiceCache may only be called once");

  49. }

  50. sCache.putAll(cache);

  51. }

可以看到Service的IBinder要么从cache里取出来的,要么getIServiceManager().getService(name)取到的,那就先看下initServiceCache是在哪里调用的?

./base/core/java/android/app/ActivityThread.java

 
  1. public final void bindApplication(String processName, ApplicationInfo appInfo,

  2. List providers, ComponentName instrumentationName,

  3. ProfilerInfo profilerInfo, Bundle instrumentationArgs,

  4. IInstrumentationWatcher instrumentationWatcher,

  5. IUiAutomationConnection instrumentationUiConnection, int debugMode,

  6. boolean enableBinderTracking, boolean trackAllocation,

  7. boolean isRestrictedBackupMode, boolean persistent, Configuration config,

  8. CompatibilityInfo compatInfo, Map services, Bundle coreSettings,

  9. String buildSerial) {

  10.  
  11. if (services != null) {

  12. // Setup the service cache in the ServiceManager

  13. ServiceManager.initServiceCache(services);

  14. }

那bindApplication是在 哪里调用的呢?

AMS:

 
  1. if (app.instr != null) {

  2. thread.bindApplication(processName, appInfo, providers,

  3. app.instr.mClass,

  4. profilerInfo, app.instr.mArguments,

  5. app.instr.mWatcher,

  6. app.instr.mUiAutomationConnection, testMode,

  7. mBinderTransactionTrackingEnabled, enableTrackAllocation,

  8. isRestrictedBackupMode || !normalMode, app.persistent,

  9. new Configuration(getGlobalConfiguration()), app.compat,

  10. getCommonServicesLocked(app.isolated),

  11. mCoreSettingsObserver.getCoreSettingsLocked(),

  12. buildSerial);

  13. } else {

  14. thread.bindApplication(processName, appInfo, providers, null, profilerInfo,

  15. null, null, null, testMode,

  16. mBinderTransactionTrackingEnabled, enableTrackAllocation,

  17. isRestrictedBackupMode || !normalMode, app.persistent,

  18. new Configuration(getGlobalConfiguration()), app.compat,

  19. getCommonServicesLocked(app.isolated),

  20. mCoreSettingsObserver.getCoreSettingsLocked(),

  21. buildSerial);

  22. }

 
  1. /**

  2. * Initialize the application bind args. These are passed to each

  3. * process when the bindApplication() IPC is sent to the process. They're

  4. * lazily setup to make sure the services are running when they're asked for.

  5. */

  6. private HashMap getCommonServicesLocked(boolean isolated) {

  7. // Isolated processes won't get this optimization, so that we don't

  8. // violate the rules about which services they have access to.

  9. if (isolated) {

  10. if (mIsolatedAppBindArgs == null) {

  11. mIsolatedAppBindArgs = new HashMap<>();

  12. mIsolatedAppBindArgs.put("package", ServiceManager.getService("package"));

  13. }

  14. return mIsolatedAppBindArgs;

  15. }

  16.  
  17. if (mAppBindArgs == null) {

  18. mAppBindArgs = new HashMap<>();

  19.  
  20. // Setup the application init args

  21. mAppBindArgs.put("package", ServiceManager.getService("package"));

  22. mAppBindArgs.put("window", ServiceManager.getService("window"));

  23. mAppBindArgs.put(Context.ALARM_SERVICE,

  24. ServiceManager.getService(Context.ALARM_SERVICE));

  25. }

  26. return mAppBindArgs;

  27. }

这里少了wifi相关的Service,所以还是走的

getIServiceManager().getService(name)
 
  1. private static IServiceManager getIServiceManager() {

  2. if (sServiceManager != null) {

  3. return sServiceManager;

  4. }

  5.  
  6. // Find the service manager

  7. sServiceManager = ServiceManagerNative

  8. .asInterface(Binder.allowBlocking(BinderInternal.getContextObject()));

  9. return sServiceManager;

  10. }

这里梳理不下去了=-=尴尬,待续。。。有兴趣的可以看下https://www.jianshu.com/p/93644d6f9e02
 

“从最表层的API过度到JNI层,然后与Lib层通讯”什么鬼,我只是个java工程师=-=

先给个结论,wifiManager调用接口是调用到WifiServiceImpl那边去了。

-------------------------------2018/6/19日更新,native层的正推没办法了,从倒推开始吧----------------------------------------

之前讲过WifiService是在SystemServer启动起来的,流程如下:

SystemServer.java:

 
  1. // Wifi Service must be started first for wifi-related services.

  2. traceBeginAndSlog("StartWifi");

  3. mSystemServiceManager.startService(WIFI_SERVICE_CLASS);

  4. traceEnd();

SystemServiceManager:

简单来说就是调用下WIFI_SERVICE_CLASS("com.android.server.wifi.WifiService")的构造器和onStart方法

 
  1. /**

  2. * Starts a service by class name.

  3. *

  4. * @return The service instance.

  5. */

  6. @SuppressWarnings("unchecked")

  7. public SystemService startService(String className) {

  8. final Class serviceClass;

  9. try {

  10. serviceClass = (Class)Class.forName(className);

  11. } catch (ClassNotFoundException ex) {

  12. Slog.i(TAG, "Starting " + className);

  13. throw new RuntimeException("Failed to create service " + className

  14. + ": service class not found, usually indicates that the caller should "

  15. + "have called PackageManager.hasSystemFeature() to check whether the "

  16. + "feature is available on this device before trying to start the "

  17. + "services that implement it", ex);

  18. }

  19. return startService(serviceClass);

  20. }

  21.  
  22. /**

  23. * Creates and starts a system service. The class must be a subclass of

  24. * {@link com.android.server.SystemService}.

  25. *

  26. * @param serviceClass A Java class that implements the SystemService interface.

  27. * @return The service instance, never null.

  28. * @throws RuntimeException if the service fails to start.

  29. */

  30. @SuppressWarnings("unchecked")

  31. public T startService(Class serviceClass) {

  32. try {

  33. final String name = serviceClass.getName();

  34. Slog.i(TAG, "Starting " + name);

  35. Trace.traceBegin(Trace.TRACE_TAG_SYSTEM_SERVER, "StartService " + name);

  36.  
  37. // Create the service.

  38. if (!SystemService.class.isAssignableFrom(serviceClass)) {

  39. throw new RuntimeException("Failed to create " + name

  40. + ": service must extend " + SystemService.class.getName());

  41. }

  42. final T service;

  43. try {

  44. Constructor constructor = serviceClass.getConstructor(Context.class);

  45. service = constructor.newInstance(mContext);

  46. } catch (InstantiationException ex) {

  47. throw new RuntimeException("Failed to create service " + name

  48. + ": service could not be instantiated", ex);

  49. } catch (IllegalAccessException ex) {

  50. throw new RuntimeException("Failed to create service " + name

  51. + ": service must have a public constructor with a Context argument", ex);

  52. } catch (NoSuchMethodException ex) {

  53. throw new RuntimeException("Failed to create service " + name

  54. + ": service must have a public constructor with a Context argument", ex);

  55. } catch (InvocationTargetException ex) {

  56. throw new RuntimeException("Failed to create service " + name

  57. + ": service constructor threw an exception", ex);

  58. }

  59.  
  60. startService(service);

  61. return service;

  62. } finally {

  63. Trace.traceEnd(Trace.TRACE_TAG_SYSTEM_SERVER);

  64. }

  65. }

  66.  
  67. public void startService(@NonNull final SystemService service) {

  68. // Register it.

  69. mServices.add(service);

  70. // Start it.

  71. long time = SystemClock.elapsedRealtime();

  72. try {

  73. service.onStart();

  74. } catch (RuntimeException ex) {

  75. throw new RuntimeException("Failed to start service " + service.getClass().getName()

  76. + ": onStart threw an exception", ex);

  77. }

  78. warnIfTooLong(SystemClock.elapsedRealtime() - time, service, "onStart");

  79. }

WifiService:

 
  1. public final class WifiService extends SystemService {

  2.  
  3. private static final String TAG = "WifiService";

  4. final WifiServiceImpl mImpl;

  5.  
  6. public WifiService(Context context) {

  7. super(context);

  8. mImpl = new WifiServiceImpl(context, new WifiInjector(context), new WifiAsyncChannel(TAG));

  9. }

  10.  
  11. @Override

  12. public void onStart() {

  13. Log.i(TAG, "Registering " + Context.WIFI_SERVICE);

  14. publishBinderService(Context.WIFI_SERVICE, mImpl);

  15. }

SystemService:

 
  1. protected final void publishBinderService(String name, IBinder service) {

  2. publishBinderService(name, service, false);

  3. }

  4.  
  5. /**

  6. * Publish the service so it is accessible to other services and apps.

  7. */

  8. protected final void publishBinderService(String name, IBinder service,

  9. boolean allowIsolated) {

  10. ServiceManager.addService(name, service, allowIsolated);

  11. }

这边的addService方法和getService方法是对应起来的,一个get,一个add。

所以之前的getService方法就是WifiServiceImpl。

 

IBinder b = ServiceManager.getServiceOrThrow(Context.WIFI_SERVICE);

 

 

 

 

2.2.2 Service的WiFi启动流程

先看下WifiServiceImpl流程:

 
  1. public class WifiServiceImpl extends IWifiManager.Stub {

  2.  
  3. /**

  4. * see {@link android.net.wifi.WifiManager#setWifiEnabled(boolean)}

  5. * @param enable {@code true} to enable, {@code false} to disable.

  6. * @return {@code true} if the enable/disable operation was

  7. * started or is already in the queue.

  8. */

  9. @Override

  10. public synchronized boolean setWifiEnabled(String packageName, boolean enable)

  11. throws RemoteException {

  12. enforceChangePermission();

  13. Slog.d(TAG, "setWifiEnabled: " + enable + " pid=" + Binder.getCallingPid()

  14. + ", uid=" + Binder.getCallingUid() + ", package=" + packageName);

  15. mLog.info("setWifiEnabled package=% uid=% enable=%").c(packageName)

  16. .c(Binder.getCallingUid()).c(enable).flush();

  17.  
  18. boolean isFromSettings = checkNetworkSettingsPermission(

  19. Binder.getCallingPid(), Binder.getCallingUid());

  20.  
  21. // If Airplane mode is enabled, only Settings is allowed to toggle Wifi

  22. if (mSettingsStore.isAirplaneModeOn() && !isFromSettings) {

  23. mLog.info("setWifiEnabled in Airplane mode: only Settings can enable wifi").flush();

  24. return false;

  25. }

  26.  
  27. // If SoftAp is enabled, only Settings is allowed to toggle wifi

  28. boolean apEnabled =

  29. mWifiStateMachine.syncGetWifiApState() != WifiManager.WIFI_AP_STATE_DISABLED;

  30.  
  31. if (apEnabled && !isFromSettings) {

  32. mLog.info("setWifiEnabled SoftAp not disabled: only Settings can enable wifi").flush();

  33. return false;

  34. }

  35.  
  36. /*

  37. * Caller might not have WRITE_SECURE_SETTINGS,

  38. * only CHANGE_WIFI_STATE is enforced

  39. */

  40. long ident = Binder.clearCallingIdentity();

  41. try {

  42. if (! mSettingsStore.handleWifiToggled(enable)) {

  43. // Nothing to do if wifi cannot be toggled

  44. return true;

  45. }

  46. } finally {

  47. Binder.restoreCallingIdentity(ident);

  48. }

  49.  
  50.  
  51. if (mPermissionReviewRequired) {

  52. final int wiFiEnabledState = getWifiEnabledState();

  53. if (enable) {

  54. if (wiFiEnabledState == WifiManager.WIFI_STATE_DISABLING

  55. || wiFiEnabledState == WifiManager.WIFI_STATE_DISABLED) {

  56. if (startConsentUi(packageName, Binder.getCallingUid(),

  57. WifiManager.ACTION_REQUEST_ENABLE)) {

  58. return true;

  59. }

  60. }

  61. } else if (wiFiEnabledState == WifiManager.WIFI_STATE_ENABLING

  62. || wiFiEnabledState == WifiManager.WIFI_STATE_ENABLED) {

  63. if (startConsentUi(packageName, Binder.getCallingUid(),

  64. WifiManager.ACTION_REQUEST_DISABLE)) {

  65. return true;

  66. }

  67. }

  68. }

  69.  
  70. mWifiController.sendMessage(CMD_WIFI_TOGGLED);

  71. return true;

  72. }

mWifiController.sendMessage(CMD_WIFI_TOGGLED); return true; }

mSettingsStore.handleWifiToggled(enable)设置一下SettingsProvider中存储的WIFI_ON的值

 
  1. private void persistWifiState(int state) {

  2. final ContentResolver cr = mContext.getContentResolver();

  3. mPersistWifiState = state;

  4. Settings.Global.putInt(cr, Settings.Global.WIFI_ON, state);

  5. }

 
  1. /* Values tracked in Settings.Global.WIFI_ON */

  2. static final int WIFI_DISABLED = 0;

  3. static final int WIFI_ENABLED = 1;

  4.  
  5. /* Wifi enabled while in airplane mode */

  6. private static final int WIFI_ENABLED_AIRPLANE_OVERRIDE = 2;

  7. /* Wifi disabled due to airplane mode on */

  8. private static final int WIFI_DISABLED_AIRPLANE_ON = 3;

  9. public synchronized boolean handleWifiToggled(boolean wifiEnabled) {

  10. // Can Wi-Fi be toggled in airplane mode ?

  11. if (mAirplaneModeOn && !isAirplaneToggleable()) {

  12. return false;

  13. }

  14.  
  15. if (wifiEnabled) {

  16. if (mAirplaneModeOn) {

  17. persistWifiState(WIFI_ENABLED_AIRPLANE_OVERRIDE);

  18. } else {

  19. persistWifiState(WIFI_ENABLED);

  20. }

  21. } else {

  22. // When wifi state is disabled, we do not care

  23. // if airplane mode is on or not. The scenario of

  24. // wifi being disabled due to airplane mode being turned on

  25. // is handled handleAirplaneModeToggled()

  26. persistWifiState(WIFI_DISABLED);

  27. }

  28. return true;

  29. }

这里接着会调用到WifiController,WifiController是个状态机设计模式,也就是说会根据WiFi的不同状态决定处理WiFi消息的不同逻辑。

初始化逻辑如下:

 
  1. addState(mDefaultState);

  2. addState(mApStaDisabledState, mDefaultState);

  3. addState(mStaEnabledState, mDefaultState);

  4. addState(mDeviceActiveState, mStaEnabledState);

  5. addState(mDeviceInactiveState, mStaEnabledState);

  6. addState(mScanOnlyLockHeldState, mDeviceInactiveState);

  7. addState(mFullLockHeldState, mDeviceInactiveState);

  8. addState(mFullHighPerfLockHeldState, mDeviceInactiveState);

  9. addState(mNoLockHeldState, mDeviceInactiveState);

  10. addState(mStaDisabledWithScanState, mDefaultState);

  11. addState(mApEnabledState, mDefaultState);

  12. addState(mEcmState, mDefaultState);

 

 

 

2.2.3 WifiController

WifiController的状态比较多,而我比较关注从关闭到打开的状态变化,即:

 
  1. class ApStaDisabledState extends State {

  2. private int mDeferredEnableSerialNumber = 0;

  3. private boolean mHaveDeferredEnable = false;

  4. private long mDisabledTimestamp;

  5.  
  6. @Override

  7. public void enter() {

  8. mWifiStateMachine.setSupplicantRunning(false);

  9. // Supplicant can't restart right away, so not the time we switched off

  10. mDisabledTimestamp = SystemClock.elapsedRealtime();

  11. mDeferredEnableSerialNumber++;

  12. mHaveDeferredEnable = false;

  13. mWifiStateMachine.clearANQPCache();

  14. }

  15. @Override

  16. public boolean processMessage(Message msg) {

  17. switch (msg.what) {

  18. case CMD_WIFI_TOGGLED:

  19. case CMD_AIRPLANE_TOGGLED:

  20. if (mSettingsStore.isWifiToggleEnabled()) {

  21. if (doDeferEnable(msg)) {

  22. if (mHaveDeferredEnable) {

  23. // have 2 toggles now, inc serial number an ignore both

  24. mDeferredEnableSerialNumber++;

  25. }

  26. mHaveDeferredEnable = !mHaveDeferredEnable;

  27. break;

  28. }

  29. if (mDeviceIdle == false) {

  30. // wifi is toggled, we need to explicitly tell WifiStateMachine that we

  31. // are headed to connect mode before going to the DeviceActiveState

  32. // since that will start supplicant and WifiStateMachine may not know

  33. // what state to head to (it might go to scan mode).

  34. mWifiStateMachine.setOperationalMode(WifiStateMachine.CONNECT_MODE);

  35. transitionTo(mDeviceActiveState);

  36. } else {

  37. checkLocksAndTransitionWhenDeviceIdle();

  38. }

  39. } else if (mSettingsStore.isScanAlwaysAvailable()) {

  40. transitionTo(mStaDisabledWithScanState);

  41. }

  42. break;

mSettingsStore.isWifiToggleEnabled()) { if (doDeferEnable(msg)) { if (mHaveDeferredEnable) { // have 2 toggles now, inc serial number an ignore both mDeferredEnableSerialNumber++; } mHaveDeferredEnable = !mHaveDeferredEnable; break; } if (mDeviceIdle == false) { // wifi is toggled, we need to explicitly tell WifiStateMachine that we // are headed to connect mode before going to the DeviceActiveState // since that will start supplicant and WifiStateMachine may not know // what state to head to (it might go to scan mode). mWifiStateMachine.setOperationalMode(WifiStateMachine.CONNECT_MODE); transitionTo(mDeviceActiveState); } else { checkLocksAndTransitionWhenDeviceIdle(); } } else if (mSettingsStore.isScanAlwaysAvailable()) { transitionTo(mStaDisabledWithScanState); } break;

现在就上面一串代码挨个解析。

WifiServiceImpl在走到WifiController之前有提及修改了一下SettingsProvider,其实也顺带改了一下WifiSettingsStore的mPersistWifiState值,用来标记wifi状态。

persistWifiState(WIFI_ENABLED);
 
  1. public synchronized boolean isWifiToggleEnabled() {

  2. if (!mCheckSavedStateAtBoot) {

  3. mCheckSavedStateAtBoot = true;

  4. if (testAndClearWifiSavedState()) return true;

  5. }

  6.  
  7. if (mAirplaneModeOn) {

  8. return mPersistWifiState == WIFI_ENABLED_AIRPLANE_OVERRIDE;

  9. } else {

  10. return mPersistWifiState != WIFI_DISABLED;

  11. }

  12. }

收到消息也不是立刻处理的:

 
  1. private boolean doDeferEnable(Message msg) {

  2. long delaySoFar = SystemClock.elapsedRealtime() - mDisabledTimestamp;

  3. if (delaySoFar >= mReEnableDelayMillis) {

  4. return false;

  5. }

  6.  
  7. log("WifiController msg " + msg + " deferred for " +

  8. (mReEnableDelayMillis - delaySoFar) + "ms");

  9.  
  10. // need to defer this action.

  11. Message deferredMsg = obtainMessage(CMD_DEFERRED_TOGGLE);

  12. deferredMsg.obj = Message.obtain(msg);

  13. deferredMsg.arg1 = ++mDeferredEnableSerialNumber;

  14. sendMessageDelayed(deferredMsg, mReEnableDelayMillis - delaySoFar + DEFER_MARGIN_MS);

  15. return true;

  16. }

  17. ...

  18. mReEnableDelayMillis = mFacade.getLongSetting(mContext,

  19. Settings.Global.WIFI_REENABLE_DELAY_MS, DEFAULT_REENABLE_DELAY_MS);

  20. /**

  21. * See {@link Settings.Global#WIFI_REENABLE_DELAY_MS}. This is the default value if a

  22. * Settings.Global value is not present. This is the minimum time after wifi is disabled

  23. * we'll act on an enable. Enable requests received before this delay will be deferred.

  24. */

  25. private static final long DEFAULT_REENABLE_DELAY_MS = 500;

wifi关闭后立刻打开有可能流程还没走完导致问题(压力测试会频繁开关WiFi),所以这边Google应该考虑到这点加了个最短时限500ms,如果短于这时间,就强迫补个差来个500+5ms的延时。

以小于500ms的间隔发送消息会由于mDeferredEnableSerialNumber值自增导致前一个消息失效。

 
  1. case CMD_DEFERRED_TOGGLE:

  2. if (msg.arg1 != mDeferredEnableSerialNumber) {

  3. log("DEFERRED_TOGGLE ignored due to serial mismatch");

  4. break;

  5. }

  6. log("DEFERRED_TOGGLE handled");

  7. sendMessage((Message)(msg.obj));

  8. break;

接着状态机会走到如下逻辑中去,除了WifiStateMachine外,还会将状态改为mDeviceActiveState。

 
  1. // wifi is toggled, we need to explicitly tell WifiStateMachine that we

  2. // are headed to connect mode before going to the DeviceActiveState

  3. // since that will start supplicant and WifiStateMachine may not know

  4. // what state to head to (it might go to scan mode).

  5. mWifiStateMachine.setOperationalMode(WifiStateMachine.CONNECT_MODE);

  6. transitionTo(mDeviceActiveState);

由于是一个类,看下StaEnabledState(DeviceActiveState的父状态)和DeviceActiveState,状态机设计模式状态改变会先走进父状态的enter中。

 
  1. class StaEnabledState extends State {

  2. @Override

  3. public void enter() {

  4. mWifiStateMachine.setSupplicantRunning(true);

  5. }

 

 
  1. /* Parent: StaEnabledState */

  2. class DeviceActiveState extends State {

  3. @Override

  4. public void enter() {

  5. mWifiStateMachine.setOperationalMode(WifiStateMachine.CONNECT_MODE);

  6. mWifiStateMachine.setHighPerfModeEnabled(false);

  7. }

可以看到其实是调用顺序

 
  1. mWifiStateMachine.setOperationalMode(WifiStateMachine.CONNECT_MODE);
  2. mWifiStateMachine.setSupplicantRunning(true);
  3. mWifiStateMachine.setOperationalMode(WifiStateMachine.CONNECT_MODE);
  4. mWifiStateMachine.setHighPerfModeEnabled(false);

2.2.4 WifiStateMachine

wifiStateMachine也是一个状态机:

 
  1. // CHECKSTYLE:OFF IndentationCheck

  2. addState(mDefaultState);

  3. addState(mInitialState, mDefaultState);

  4. addState(mSupplicantStartingState, mDefaultState);

  5. addState(mSupplicantStartedState, mDefaultState);

  6. addState(mScanModeState, mSupplicantStartedState);

  7. addState(mConnectModeState, mSupplicantStartedState);

  8. addState(mL2ConnectedState, mConnectModeState);

  9. addState(mObtainingIpState, mL2ConnectedState);

  10. addState(mConnectedState, mL2ConnectedState);

  11. addState(mRoamingState, mL2ConnectedState);

  12. addState(mDisconnectingState, mConnectModeState);

  13. addState(mDisconnectedState, mConnectModeState);

  14. addState(mWpsRunningState, mConnectModeState);

  15. addState(mWaitForP2pDisableState, mSupplicantStartedState);

  16. addState(mSupplicantStoppingState, mDefaultState);

  17. addState(mSoftApState, mDefaultState);

  18. // CHECKSTYLE:ON IndentationCheck

 

这里看下WifiController下发的4个操作

 

1、3)setOperationalMode:

 
  1. /**

  2. * Track the state of Wifi connectivity. All event handling is done here,

  3. * and all changes in connectivity state are initiated here.

  4. *

  5. * Wi-Fi now supports three modes of operation: Client, SoftAp and p2p

  6. * In the current implementation, we support concurrent wifi p2p and wifi operation.

  7. * The WifiStateMachine handles SoftAp and Client operations while WifiP2pService

  8. * handles p2p operation.

  9. *

  10. * @hide

  11. */

  12. public class WifiStateMachine extends StateMachine implements WifiNative.WifiRssiEventHandler,

  13. WifiMulticastLockManager.FilterController {

 
  1. /**

  2. * TODO: doc

  3. */

  4. public void setOperationalMode(int mode) {

  5. if (mVerboseLoggingEnabled) log("setting operational mode to " + String.valueOf(mode));

  6. sendMessage(CMD_SET_OPERATIONAL_MODE, mode, 0);

  7. })

2)setSupplicantRunning

    /**
 
  1. * TODO: doc

  2. */

  3. public void setSupplicantRunning(boolean enable) {

  4. if (enable) {

  5. sendMessage(CMD_START_SUPPLICANT);

  6. } else {

  7. sendMessage(CMD_STOP_SUPPLICANT);

  8. }

  9. }

4)  setHighPerfModeEnabled

 
  1. /**

  2. * Set high performance mode of operation.

  3. * Enabling would set active power mode and disable suspend optimizations;

  4. * disabling would set auto power mode and enable suspend optimizations

  5. *

  6. * @param enable true if enable, false otherwise

  7. */

  8. public void setHighPerfModeEnabled(boolean enable) {

  9. sendMessage(CMD_SET_HIGH_PERF_MODE, enable ? 1 : 0, 0);

  10. }

 

来看下初始状态对消息的处理

 

 
  1. class InitialState extends State {

  2.  
  3. private void cleanup() {

  4. // Tearing down the client interfaces below is going to stop our supplicant.

  5. mWifiMonitor.stopAllMonitoring();

  6.  
  7. mDeathRecipient.unlinkToDeath();

  8. mWifiNative.tearDown();

  9. }

  10.  
  11. @Override

  12. public void enter() {

  13. mWifiStateTracker.updateState(WifiStateTracker.INVALID);

  14. cleanup();

  15. }

  16.  
  17. @Override

  18. public boolean processMessage(Message message) {

  19. logStateAndMessage(message, this);

  20. switch (message.what) {

  21. case CMD_START_SUPPLICANT:

  22. Pair statusAndInterface =

  23. mWifiNative.setupForClientMode();

  24. if (statusAndInterface.first == WifiNative.SETUP_SUCCESS) {

  25. mClientInterface = statusAndInterface.second;

  26. } else {

  27. incrementMetricsForSetupFailure(statusAndInterface.first);

  28. }

  29. if (mClientInterface == null

  30. || !mDeathRecipient.linkToDeath(mClientInterface.asBinder())) {

  31. setWifiState(WifiManager.WIFI_STATE_UNKNOWN);

  32. cleanup();

  33. break;

  34. }

  35.  
  36. try {

  37. // A runtime crash or shutting down AP mode can leave

  38. // IP addresses configured, and this affects

  39. // connectivity when supplicant starts up.

  40. // Ensure we have no IP addresses before a supplicant start.

  41. mNwService.clearInterfaceAddresses(mInterfaceName);

  42.  
  43. // Set privacy extensions

  44. mNwService.setInterfaceIpv6PrivacyExtensions(mInterfaceName, true);

  45.  
  46. // IPv6 is enabled only as long as access point is connected since:

  47. // - IPv6 addresses and routes stick around after disconnection

  48. // - kernel is unaware when connected and fails to start IPv6 negotiation

  49. // - kernel can start autoconfiguration when 802.1x is not complete

  50. mNwService.disableIpv6(mInterfaceName);

  51. } catch (RemoteException re) {

  52. loge("Unable to change interface settings: " + re);

  53. } catch (IllegalStateException ie) {

  54. loge("Unable to change interface settings: " + ie);

  55. }

  56.  
  57. if (!mWifiNative.enableSupplicant()) {

  58. loge("Failed to start supplicant!");

  59. setWifiState(WifiManager.WIFI_STATE_UNKNOWN);

  60. cleanup();

  61. break;

  62. }

  63. if (mVerboseLoggingEnabled) log("Supplicant start successful");

  64. mWifiMonitor.startMonitoring(mInterfaceName, true);

  65. mWifiInjector.getWifiLastResortWatchdog().clearAllFailureCounts();

  66. setSupplicantLogLevel();

  67. transitionTo(mSupplicantStartingState);

  68. break;

  69. case CMD_START_AP:

  70. transitionTo(mSoftApState);

  71. break;

  72. case CMD_SET_OPERATIONAL_MODE:

  73. mOperationalMode = message.arg1;

  74. if (mOperationalMode != DISABLED_MODE) {

  75. sendMessage(CMD_START_SUPPLICANT);

  76. }

  77. break;

  78. default:

  79. return NOT_HANDLED;

  80. }

  81. return HANDLED;

  82. }

  83. }

在InitialState 中 1、 3 步最后走的逻辑和第2步是一样的,都是发出了一个CMD_START_SUPPLICANT消息,不是很懂这块处理逻辑=-=搜了下真正对CONNECT_MODE有处理的是以下状态,应该是第3步发出的消息会得到ScanModeState的处理:

 
  1. class ScanModeState extends State {

  2. private int mLastOperationMode;

  3. @Override

  4. public void enter() {

  5. mLastOperationMode = mOperationalMode;

  6. mWifiStateTracker.updateState(WifiStateTracker.SCAN_MODE);

  7. }

  8. @Override

  9. public boolean processMessage(Message message) {

  10. logStateAndMessage(message, this);

  11.  
  12. switch(message.what) {

  13. case CMD_SET_OPERATIONAL_MODE:

  14. if (message.arg1 == CONNECT_MODE) {

  15. mOperationalMode = CONNECT_MODE;

  16. setWifiState(WIFI_STATE_ENABLING);

  17. transitionTo(mDisconnectedState);

  18. } else if (message.arg1 == DISABLED_MODE) {

  19. transitionTo(mSupplicantStoppingState);

  20. }

  21. // Nothing to do

  22. break;

  23. // Handle scan. All the connection related commands are

  24. // handled only in ConnectModeState

  25. case CMD_START_SCAN:

  26. handleScanRequest(message);

  27. break;

  28. default:

  29. return NOT_HANDLED;

  30. }

  31. return HANDLED;

  32. }

  33. }

 

先看下接收到该消息进行的关键操作:

 

 

  1. mWifiNative.enableSupplicant()
  2. mWifiMonitor.startMonitoring(mInterfaceName, true);
  3. 切换到SupplicantStartingState状态

PS:

 
  1. /**

  2. * Enable wpa_supplicant via wificond.

  3. * @return Returns true on success.

  4. */

  5. public boolean enableSupplicant() {

  6. return mWificondControl.enableSupplicant();

  7. }

 
  1. /**

  2. * Start Monitoring for wpa_supplicant events.

  3. *

  4. * @param iface Name of iface.

  5. * TODO: Add unit tests for these once we remove the legacy code.

  6. */

  7. public synchronized void startMonitoring(String iface, boolean isStaIface) {

  8. if (ensureConnectedLocked()) {

  9. setMonitoring(iface, true);

  10. broadcastSupplicantConnectionEvent(iface);

  11. } else {

  12. boolean originalMonitoring = isMonitoring(iface);

  13. setMonitoring(iface, true);

  14. broadcastSupplicantDisconnectionEvent(iface);

  15. setMonitoring(iface, originalMonitoring);

  16. Log.e(TAG, "startMonitoring(" + iface + ") failed!");

  17. }

  18. }

 

 

看切换完了再收到这个消息后就是延迟发送了,后面估计是轮到哪个状态就是不同的处理逻辑的,但就我看来这个消息主要就是start supplicant的,start完了应该完成任务了。

 

 

 
  1. case CMD_SET_OPERATIONAL_MODE:

  2. messageHandlingStatus = MESSAGE_HANDLING_STATUS_DEFERRED;

  3. deferMessage(message);

  4. break;

至于第4步

SupplicantStartedState状态会处理:

 
  1. case CMD_SET_HIGH_PERF_MODE:

  2. if (message.arg1 == 1) {

  3. setSuspendOptimizations(SUSPEND_DUE_TO_HIGH_PERF, false);

  4. } else {

  5. setSuspendOptimizations(SUSPEND_DUE_TO_HIGH_PERF, true);

  6. }

  7. break;

都是WifiNative的代码,溜了溜了。

 
  1. private void setSuspendOptimizationsNative(int reason, boolean enabled) {

  2. if (mVerboseLoggingEnabled) {

  3. log("setSuspendOptimizationsNative: " + reason + " " + enabled

  4. + " -want " + mUserWantsSuspendOpt.get()

  5. + " stack:" + Thread.currentThread().getStackTrace()[2].getMethodName()

  6. + " - " + Thread.currentThread().getStackTrace()[3].getMethodName()

  7. + " - " + Thread.currentThread().getStackTrace()[4].getMethodName()

  8. + " - " + Thread.currentThread().getStackTrace()[5].getMethodName());

  9. }

  10. //mWifiNative.setSuspendOptimizations(enabled);

  11.  
  12. if (enabled) {

  13. mSuspendOptNeedsDisabled &= ~reason;

  14. /* None of dhcp, screen or highperf need it disabled and user wants it enabled */

  15. if (mSuspendOptNeedsDisabled == 0 && mUserWantsSuspendOpt.get()) {

  16. if (mVerboseLoggingEnabled) {

  17. log("setSuspendOptimizationsNative do it " + reason + " " + enabled

  18. + " stack:" + Thread.currentThread().getStackTrace()[2].getMethodName()

  19. + " - " + Thread.currentThread().getStackTrace()[3].getMethodName()

  20. + " - " + Thread.currentThread().getStackTrace()[4].getMethodName()

  21. + " - " + Thread.currentThread().getStackTrace()[5].getMethodName());

  22. }

  23. mWifiNative.setSuspendOptimizations(true);

  24. }

  25. } else {

  26. mSuspendOptNeedsDisabled |= reason;

  27. mWifiNative.setSuspendOptimizations(false);

  28. }

  29. }

感觉分析的有点糙,再看看,后期不定时完善。。。

继续梳理见

1.(五十八)Android O WiFi启动流程梳理续——setupForClientMode

2.(五十九)Android O WiFi启动流程梳理续——enableSupplicant

 

3. 总结

转载: https://blog.csdn.net/sinat_20059415/article/details/80716763  

你可能感兴趣的:(wlan,wifi)