目次
0.概览
1.App-->Framework状态机的流程
1.0 StateMachine底子知识
1.1 WifiEnabler
1.2 WifiManager
1.3 WifiServiceImpl
1.4 ActiveModeWarden
WifiController状态机
1.5 ConcreteClientModeManager
ClientModeStateMachine状态机
IdleState的处理惩罚
StartedState的处理惩罚
ConnectModeState
1.6 ClientModeImpl
2.WifiNative-->Hal流程
2.1 setupInterfaceForClientInScanMode 初始化interface
2.1.1 加载驱动和初始化vendor HAL接口
【流程概览】
【Java层的处理惩罚】
【Hal层的处理惩罚】
2.1.2 创建sta interface
【流程概览】
【Java层的处理惩罚】
【Hal层的处理惩罚】
2.1.3 wificond配置 interface
2.1.4 小结
2.2 switchClientInterfaceToConnectivityMode 启动supplicant
2.2.1 WifiNative
2.2.2 SupplicantStaIfaceHalAidlImpl.startDaemon()流程
【Java层流程】
【JNI层】
【Native层流程】
参考链接:
0.概览
本篇介绍Android14 WiFi 的 Enable流程。
我们可以在 Settings界面上举行 开/关 Wi-Fi 的操作,打开Wi-Fi涉及的流程比力多,这里先给出大概的流程有个整体的印象,后文会结合源码举行分析。
① WiFi Enable 过程中,Framework层涉及3个状态机:
② 抛开Framework层各种状态的切换,Enable WiFi的操作终极是由 WifiNative 的两个方法触发的:
1.setupInterfaceForClientInScanMode()启动扫描模式:加载驱动和vendor hal、wificond配置interface等;
2.switchClientInterfaceToConnectivityMode()切换到连接模式:启动supplicant服务。
1.App-->Framework状态机的流程
下面分析wifi打开过程中,从 App 到 Framework状态机的流程,下图给出了涉及的关键类:
1.0 StateMachine底子知识
Android状态机的重要工作是消息处理惩罚和状态切换。它通过界说一组状态和状态之间的切换,来控制相关运动。
StateMachine路径为:
frameworks/libs/modules-utils/java/com/android/internal/util/StateMachine.java
①State基本结构:
enter():在状态机转入这个状态时调用的方法;
exit():在状态机转出这个状态时调用的方法。
processMessage():用于处理惩罚当前状态的消息。如果当前状态处理惩罚不了,则将消息交给父状态处理惩罚。若根状态依然无法处理惩罚,则将变乱抛弃并通知体系。
②StateMachine重要API:
addState():添加状态
setInitialState():设置初始状态
start():启动状态机
transitionTo():用于切换状态。在切换过程中,会自动调用旧状态的exit()方法和新状态的enter()方法。
deferMessage():将消息缓存,直到下一次状态切换完成后处理惩罚。
...
1.1 WifiEnabler
应用层通过 Context.getSystemService 来获取 WifiManager对象,并通过它调用wifi服务的接口:
- //packages/apps/Settings/src/com/android/settings/wifi/WifiEnabler.java
- public class WifiEnabler implements SwitchWidgetController.OnSwitchChangeListener {
- ...
- private final WifiManager mWifiManager;
- WifiEnabler(Context context, SwitchWidgetController switchWidget,
- MetricsFeatureProvider metricsFeatureProvider,
- ConnectivityManager connectivityManager) {
- mContext = context;
- ...
- //1.获取 WifiManager 实例
- mWifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
- ...
- }
-
- public boolean onSwitchToggled(boolean isChecked) {
- ...
- //2.调用 WifiManager的 setWifiEnabled 函数,实现开关wifi
- if (!mWifiManager.setWifiEnabled(isChecked)) {
- // Error
- ...
- }
- return true;
- }
复制代码 1.2 WifiManager
WifiManager是袒露给应用的api接口类:
- //packages/modules/Wifi/framework/java/android/net/wifi/WifiManager.java
- package android.net.wifi;
- ...
- @SystemService(Context.WIFI_SERVICE)
- public class WifiManager {
- ...
- IWifiManager mService;
- ...
- public boolean setWifiEnabled(boolean enabled) {
- try {
- //通过AIDL调用 WifiServiceImpl 的方法
- return mService.setWifiEnabled(mContext.getOpPackageName(), enabled);
- } catch (RemoteException e) {
- throw e.rethrowFromSystemServer();
- }
- }
复制代码 1.3 WifiServiceImpl
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiServiceImpl.java
- /**
- * see {@link android.net.wifi.WifiManager#setWifiEnabled(boolean)}
- * @param enable {@code true} to enable, {@code false} to disable.
- * @return {@code true} if the enable/disable operation was
- * started or is already in the queue.
- */
- @Override
- public synchronized boolean setWifiEnabled(String packageName, boolean enable) {
- ...
- setWifiEnabledInternal(packageName, enable, callingUid, callingPid, isPrivileged);
- return true;
- }
- private void setWifiEnabledInternal(String packageName, boolean enable,
- int callingUid, int callingPid, boolean isPrivileged) {
- //打印关键信息:
- mLog.info("setWifiEnabled package=% uid=% enable=% isPrivileged=%").c(packageName)
- .c(callingUid).c(enable).c(isPrivileged).flush();
- ...
- //关注点:调用 ActiveModeWarden 的方法
- mActiveModeWarden.wifiToggled(new WorkSource(callingUid, packageName));
- ...
- }
复制代码 1.4 ActiveModeWarden
- //packages/modules/Wifi/service/java/com/android/server/wifi/ActiveModeWarden.java
- /**
- * This class provides the implementation for different WiFi operating modes.
- */
- public class ActiveModeWarden {
- private static final String TAG = "WifiActiveModeWarden";
- ...
- /** Wifi has been toggled. */
- public void wifiToggled(WorkSource requestorWs) {
- //1.向 WifiController 状态机发送 CMD_WIFI_TOGGLED 消息:
- mWifiController.sendMessage(WifiController.CMD_WIFI_TOGGLED, requestorWs);
- }
复制代码 WifiController状态机
WifiController 是ActiveModeWarden的内部类,是一个状态机,来看下它的构造函数:
- /**
- * WifiController is the class used to manage wifi state for various operating
- * modes (normal, airplane, wifi hotspot, etc.).
- */
- private class WifiController extends StateMachine {
- ...
- // DefaultState
- // / \
- // DisabledState(初始状态) EnabledState
- // 在 WifiController.start方法中,会设置初始状态为 DisabledState
- // wifi打开过程中的状态切换:DisabledState --> EnabledState
- WifiController() {
- super(TAG, mLooper);
- final int threshold = mContext.getResources().getInteger(
- R.integer.config_wifiConfigurationWifiRunnerThresholdInMs);
- DefaultState defaultState = new DefaultState(threshold);
- mEnabledState = new EnabledState(threshold);
- mDisabledState = new DisabledState(threshold);
- addState(defaultState); {
- addState(mDisabledState, defaultState);
- addState(mEnabledState, defaultState);
- }
- ...
- }
复制代码 当前处于DisabledState状态, 来看下 CMD_WIFI_TOGGLED 消息的处理惩罚:
- class DisabledState extends BaseState {
- ...
- @Override
- public boolean processMessageFiltered(Message msg) {
- switch (msg.what) {
- case CMD_WIFI_TOGGLED:
- case CMD_SCAN_ALWAYS_MODE_CHANGED:
- handleStaToggleChangeInDisabledState((WorkSource) msg.obj);
- break;
- ...
- default:
- return NOT_HANDLED;
- }
- return HANDLED;
- }
- ...
- }
- private void handleStaToggleChangeInDisabledState(WorkSource requestorWs) {
- if (shouldEnableSta()) {
- //1.会触发启动wifi流程
- startPrimaryOrScanOnlyClientModeManager(requestorWs);
- //2.状态切换到EnabledState
- transitionTo(mEnabledState);
- }
- }
- /**
- * Method to enable a new primary client mode manager.
- */
- private boolean startPrimaryOrScanOnlyClientModeManager(WorkSource requestorWs) {
- ActiveModeManager.ClientRole role = getRoleForPrimaryOrScanOnlyClientModeManager();
- if (role == ROLE_CLIENT_PRIMARY) {//走这个分支
- return startPrimaryClientModeManager(requestorWs);
- } else if (role == ROLE_CLIENT_SCAN_ONLY) {
- return startScanOnlyClientModeManager(requestorWs);
- } else {
- return false;
- }
- }
- /**
- * Method to enable a new primary client mode manager in connect mode.
- */
- private boolean startPrimaryClientModeManager(WorkSource requestorWs) {
- ...
- Log.d(TAG, "Starting primary ClientModeManager in connect mode");
- //创建 ConcreteClientModeManager 对象(会去启动wifi)
- ConcreteClientModeManager manager = mWifiInjector.makeClientModeManager(
- new ClientListener(), requestorWs, ROLE_CLIENT_PRIMARY, mVerboseLoggingEnabled);
- mClientModeManagers.add(manager);
- mLastPrimaryClientModeManagerRequestorWs = requestorWs;
- return true;
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiInjector.java
- /**
- * Create a ClientModeManager
- *
- * @param listener listener for ClientModeManager state changes
- * @return a new instance of ClientModeManager
- */
- public ConcreteClientModeManager makeClientModeManager(
- @NonNull ClientModeManager.Listener<ConcreteClientModeManager> listener,
- @NonNull WorkSource requestorWs,
- @NonNull ActiveModeManager.ClientRole role,
- boolean verboseLoggingEnabled) {
- return new ConcreteClientModeManager(
- mContext, mWifiHandlerThread.getLooper(), mClock,
- mWifiNative, listener, mWifiMetrics, mWakeupController,
- this, mSelfRecovery, mWifiGlobals, mDefaultClientModeManager,
- mClock.getElapsedSinceBootMillis(), requestorWs, role, mBroadcastQueue,
- verboseLoggingEnabled);
- }
复制代码
1.5 ConcreteClientModeManager
来看下ConcreteClientModeManager 的构造函数:
- //packages/modules/Wifi/service/java/com/android/server/wifi/ConcreteClientModeManager.java
- ConcreteClientModeManager(
- Context context, @NonNull Looper looper, Clock clock,
- WifiNative wifiNative, @NonNull Listener<ConcreteClientModeManager> listener,
- WifiMetrics wifiMetrics,
- WakeupController wakeupController, WifiInjector wifiInjector,
- SelfRecovery selfRecovery, WifiGlobals wifiGlobals,
- DefaultClientModeManager defaultClientModeManager, long id,
- @NonNull WorkSource requestorWs, @NonNull ClientRole role,
- @NonNull ClientModeManagerBroadcastQueue broadcastQueue,
- boolean verboseLoggingEnabled) {
- ...
- //1.创建并启动 ClientModeStateMachine 状态机
- mStateMachine = new ClientModeStateMachine(looper);
- ...
- //2.向状态机发送CMD_START的消息
- mStateMachine.sendMessage(ClientModeStateMachine.CMD_START, mTargetRoleChangeInfo);
- }
复制代码 ClientModeStateMachine状态机
ClientModeStateMachine 是 ConcreteClientModeManager的内部类,来看下它的构造函数:
- //packages/modules/Wifi/service/java/com/android/server/wifi/ConcreteClientModeManager.java
- // IdleState(初始状态)
- // |
- // StartedState
- // / \
- // ScanOnlyModeState ConnectModeState
- //Wifi打开过程中的状态切换:IdleState-->StartedState-->ConnectModeState
- ClientModeStateMachine(Looper looper) {
- super(TAG, looper);
- final int threshold = mContext.getResources().getInteger(
- R.integer.config_wifiConfigurationWifiRunnerThresholdInMs);
- mIdleState = new IdleState(threshold);
- mStartedState = new StartedState(threshold);
- mScanOnlyModeState = new ScanOnlyModeState(threshold);
- mConnectModeState = new ConnectModeState(threshold);
- addState(mIdleState);
- addState(mStartedState, mIdleState);
- addState(mScanOnlyModeState, mStartedState);
- addState(mConnectModeState, mStartedState);
- //设置初始状态
- setInitialState(mIdleState);
- start();
- }
复制代码 IdleState的处理惩罚
当前处于 IdleState 状态, 来看下 CMD_START消息的处理惩罚:
- //packages/modules/Wifi/service/java/com/android/server/wifi/ConcreteClientModeManager.java
- private class IdleState extends RunnerState {
- ...
- @Override
- public boolean processMessageImpl(Message message) {
- if (mVerboseLoggingEnabled) {
- Log.d(getTag(),
- getName() + " cmd = " + getWhatToString(message.what) + " "
- + message.toString());
- }
- switch (message.what) {
- case CMD_START:
- // Always start in scan mode first.
- RoleChangeInfo roleChangeInfo = (RoleChangeInfo) message.obj;
- //1.调用WifiNative方法去创建wlan接口(会加载驱动和vendor hal)
- mClientInterfaceName = mWifiNative.setupInterfaceForClientInScanMode(
- mWifiNativeInterfaceCallback, roleChangeInfo.requestorWs,
- ConcreteClientModeManager.this);
- ...
- mWifiNative.setWifiNativeInterfaceEventCallback(
- mWifiNativeInterfaceEventCallback);
- if (roleChangeInfo.role instanceof ClientConnectivityRole) {//进入这个分支
- //2.发送 CMD_SWITCH_TO_CONNECT_MODE,会触发supplicant的启动
- sendMessage(CMD_SWITCH_TO_CONNECT_MODE, roleChangeInfo);
- // 切换到 StartedState 状态
- transitionTo(mStartedState);
- } else {
- mScanRoleChangeInfoToSetOnTransition = roleChangeInfo;
- transitionTo(mScanOnlyModeState);
- }
- break;
- ...
- }
- return HANDLED;
- }
复制代码 从上面看出,IdleState 处理惩罚CMD_START消息过程中:
1.会调用 WifiNative.setupInterfaceForClientInScanMode()去创建 wlan 接口进入扫描模式,具体流程会放到下一节举行分析。
2.会切换到 StartedState 状态,来看下 CMD_SWITCH_TO_CONNECT_MODE 消息的处理惩罚:
StartedState的处理惩罚
- //packages/modules/Wifi/service/java/com/android/server/wifi/ConcreteClientModeManager.java
- private class StartedState extends RunnerState {
- ...
- @Override
- public boolean processMessageImpl(Message message) {
- if (mVerboseLoggingEnabled) {
- Log.d(getTag(),
- getName() + " cmd = " + getWhatToString(message.what) + " "
- + message.toString());
- }
- switch (message.what) {
- ...
- case CMD_SWITCH_TO_CONNECT_MODE: {
- RoleChangeInfo roleChangeInfo = (RoleChangeInfo) message.obj;
- updateConnectModeState(roleChangeInfo.role,
- WifiManager.WIFI_STATE_ENABLING,
- WifiManager.WIFI_STATE_DISABLED);
- //1.调用WifiNative方法,将接口切换到连接模式。会触发supplicant的启动
- if (!mWifiNative.switchClientInterfaceToConnectivityMode(
- mClientInterfaceName, roleChangeInfo.requestorWs)) {
- ...
- break;
- }
- // Role set in the enter of ConnectModeState.
- mConnectRoleChangeInfoToSetOnTransition = roleChangeInfo;
- //2.切换状态
- transitionTo(mConnectModeState);
- break;
- }
复制代码 从上面看出,StartedState 处理惩罚CMD_SWITCH_TO_CONNECT_MODE消息过程中:
1.会调用 WifiNative.switchClientInterfaceToConnectivityMode() 将设备接口切换到连接模式,从而触发 supplicant服务的启动。具体流程会在下一节分析;
2.会切换到 ConnectModeState 状态:
ConnectModeState
- //packages/modules/Wifi/service/java/com/android/server/wifi/ConcreteClientModeManager.java
- private class ConnectModeState extends RunnerState {
- ...
- @Override
- public void enterImpl() {
- Log.d(getTag(), "entering ConnectModeState, starting ClientModeImpl");
- if (mClientInterfaceName == null) {
- Log.e(getTag(), "Supposed to start ClientModeImpl, but iface is null!");
- } else {
- if (mClientModeImpl != null) {
- Log.e(getTag(), "ConnectModeState.enter(): mClientModeImpl is already "
- + "instantiated?!");
- }
- //实例化 ClientModeImpl 状态机(ClientModeImpl构造函数中会启动状态机)
- mClientModeImpl = mWifiInjector.makeClientModeImpl(
- mClientInterfaceName, ConcreteClientModeManager.this,
- mVerboseLoggingEnabled);
- ...
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiInjector.java
- /**
- * Create a ClientModeImpl
- * @param ifaceName interface name for the ClientModeImpl
- * @param clientModeManager ClientModeManager that will own the ClientModeImpl
- */
- public ClientModeImpl makeClientModeImpl(
- @NonNull String ifaceName,
- @NonNull ConcreteClientModeManager clientModeManager,
- boolean verboseLoggingEnabled) {
- ExtendedWifiInfo wifiInfo = new ExtendedWifiInfo(mWifiGlobals, ifaceName);
- //1.创建 SupplicantStateTracker 状态机;
- SupplicantStateTracker supplicantStateTracker = new SupplicantStateTracker(...);
- //2.创建并返回 ClientModeImpl 状态机
- return new ClientModeImpl(mContext, ..., supplicantStateTracker, ...);
- }
复制代码 1.6 ClientModeImpl
来看下 ClientModeImpl 的构造函数:
- //packages/modules/Wifi/service/java/com/android/server/wifi/ClientModeImpl.java
- /**
- * Implementation of ClientMode. Event handling for Client mode logic is done here,
- * and all changes in connectivity state are initiated here.
- *
- * Note: No external modules should be calling into {@link ClientModeImpl}. Please plumb it via
- * {@link ClientModeManager} until b/160014176 is fixed.
- */
- public class ClientModeImpl extends StateMachine implements ClientMode {
- ...
- /** Note that this constructor will also start() the StateMachine. */
- public ClientModeImpl(
- @NonNull WifiContext context,
- ...
- @NonNull WifiNotificationManager wifiNotificationManager) {
- //...
- //1.构建状态树:
- // ConnectableState
- // / \
- // ConnectingOrConnectedState DisconnectedState(初始状态)
- // / \
- // L2ConnectingState L2ConnectedState
- // / | \ \
- // WaitBeforeL3ProvisioningState L3ProvisioningState L3ConnectedState RoamingState
- addState(mConnectableState); {
- addState(mConnectingOrConnectedState, mConnectableState); {
- addState(mL2ConnectingState, mConnectingOrConnectedState);
- addState(mL2ConnectedState, mConnectingOrConnectedState); {
- addState(mWaitBeforeL3ProvisioningState, mL2ConnectedState);
- addState(mL3ProvisioningState, mL2ConnectedState);
- addState(mL3ConnectedState, mL2ConnectedState);
- addState(mRoamingState, mL2ConnectedState);
- }
- }
- addState(mDisconnectedState, mConnectableState);
- }
-
- //2.设置初始状态为 DisconnectedState
- setInitialState(mDisconnectedState);
-
- setLogOnlyTransitions(false);
-
- //3.启动状态机
- start();
- ...
- }
复制代码
至此,wifi打开过程中,Framework层涉及的3个重要状态机已初步介绍完毕:
状态机 | 描述 | 状态切换 | ActiveModeWarden$WifiController
| 记录上层的WiFi开关状态
| DisabledState -> EnabledState
| ConcreteClientModeManager$ClientModeStateMachine
| WiFi的设备状态
| IdleState -> StartedState -> ConnectModeState
| ClientModeImpl
| 管理WiFi的连接状态
| DisconnectedState
|
接下来将介绍 WifiNative 的两个重要方法:
1.setupInterfaceForClientInScanMode()启动扫描模式:加载驱动和vendor hal、wificond配置interface;
2.switchClientInterfaceToConnectivityMode()切换到连接模式:会触发supplicant服务的启动
2.WifiNative-->Hal流程
2.1 setupInterfaceForClientInScanMode 初始化interface
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiNative.java
- /**
- * Setup an interface for client mode (for scan) operations.
- *
- * This method configures an interface in STA mode in the native daemons
- * (wificond, vendor HAL).
- *
- * @param interfaceCallback Associated callback for notifying status changes for the iface.
- * @param requestorWs Requestor worksource.
- * @param concreteClientModeManager ConcreteClientModeManager requesting the interface.
- * @return Returns the name of the allocated interface, will be null on failure.
- */
- public String setupInterfaceForClientInScanMode(
- @NonNull InterfaceCallback interfaceCallback, @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- synchronized (mLock) {
- //1.加载驱动和vendor hal
- if (!startHal()) {
- ...
- }
- //2.创建并初始化 Iface
- Iface iface = mIfaceMgr.allocateIface(Iface.IFACE_TYPE_STA_FOR_SCAN);
- ...
- iface.externalListener = interfaceCallback;
- iface.name = createStaIface(iface, requestorWs, concreteClientModeManager);
- ...
- //3.wificond 对 iface 进行配置
- if (!mWifiCondManager.setupInterfaceForClientMode(iface.name, Runnable::run,
- new NormalScanEventCallback(iface.name),
- new PnoScanEventCallback(iface.name))) {
- ...
- }
- //4.监听interface的 up/down
- registerInterfaceObserver();
- iface.networkObserver = new NetworkObserverInternal(iface.id);
- if (!registerNetworkObserver(iface.networkObserver)) {
- ...
- }
- //5.开启supplicant监听(此时supplicant还未启动)
- mWifiMonitor.startMonitoring(iface.name);
- // Just to avoid any race conditions with interface state change callbacks,
- // update the interface state before we exit.
- onInterfaceStateChanged(iface, isInterfaceUp(iface.name));
- mWifiVendorHal.enableLinkLayerStats(iface.name);
- Log.i(TAG, "Successfully setup " + iface);
- //6.获取芯片支持的wifi特征
- iface.featureSet = getSupportedFeatureSetInternal(iface.name);
- updateSupportedBandForStaInternal(iface);
- mWifiVendorHal.enableStaChannelForPeerNetwork(mContext.getResources().getBoolean(
- R.bool.config_wifiEnableStaIndoorChannelForPeerNetwork),
- mContext.getResources().getBoolean(
- R.bool.config_wifiEnableStaDfsChannelForPeerNetwork));
- return iface.name;
- }
- }
复制代码 2.1.1 加载驱动和初始化vendor HAL接口
【流程概览】
【Java层的处理惩罚】
① WifiNative.startHal() --> WifiVendorHal.startVendorHal() --> HalDeviceManager.start() --> startWifi()
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiNative.java
- /** Helper method invoked to start supplicant if there were no ifaces */
- private boolean startHal() {
- synchronized (mLock) {
- if (!mIfaceMgr.hasAnyIface()) {//如果没有Iface
- if (mWifiVendorHal.isVendorHalSupported()) {//如果支持vendor hal
- if (!mWifiVendorHal.startVendorHal()) {//启动vendor hal
- Log.e(TAG, "Failed to start vendor HAL");
- return false;
- }
- ...
- } ...
- }
- registerWificondListenerIfNecessary();
- return true;
- }
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiVendorHal.java
- /**
- * Bring up the Vendor HAL.
- * @return true on success, false otherwise.
- */
- public boolean startVendorHal() {
- synchronized (sLock) {
- if (!mHalDeviceManager.start()) {//关键
- mLog.err("Failed to start vendor HAL").flush();
- return false;
- }
- //打印信息:
- mLog.info("Vendor Hal started successfully").flush();
- return true;
- }
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/HalDeviceManager.java
- /**
- * Attempts to start Wi-Fi. Returns the success (true) or failure (false) or
- * the start operation. Will also dispatch any registered ManagerStatusCallback.onStart() on
- * success.
- */
- public boolean start() {
- return startWifi();
- }
- private boolean startWifi() {
- if (VDBG) Log.d(TAG, "startWifi");
- //初始化 Wi-Fi HAL service(其实在SystemServer启动WifiService时,会初始化Wi-Fi HAL service,所以这里是为了确保)
- initializeInternal();
- synchronized (mLock) {
- int triedCount = 0;
- while (triedCount <= START_HAL_RETRY_TIMES) {//最多尝试启动三次
- //1.调用wifi hal的start方法去启动
- int status = mWifiHal.start();
- if (status == WifiHal.WIFI_STATUS_SUCCESS) {
- ...
- //2.获取所有芯片信息
- WifiChipInfo[] wifiChipInfos = getAllChipInfo();
- ...
- return true;
- } ...
- }
- Log.e(TAG, "Cannot start IWifi after trying " + triedCount + " times");
- return false;
- }
- }
复制代码 ② WifiHal.start() --> WifiHalAidlImpl.start()
- //************************** 来看下 WifiHal 的start 方法 *************************
- //packages/modules/Wifi/service/java/com/android/server/wifi/hal/WifiHal.java
- /**
- * See comments for {@link IWifiHal#start()}
- */
- public @WifiStatusCode int start() {
- return validateAndCall("start", WIFI_STATUS_ERROR_UNKNOWN,
- () -> mWifiHal.start());
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/hal/WifiHalAidlImpl.java
- public class WifiHalAidlImpl implements IWifiHal {
- private static final String TAG = "WifiHalAidlImpl";
- private static final String HAL_INSTANCE_NAME =
- android.hardware.wifi.IWifi.DESCRIPTOR + "/default";
- private android.hardware.wifi.IWifi mWifi;
- /**
- * See comments for {@link IWifiHal#start()}
- */
- @Override
- public @WifiHal.WifiStatusCode int start() {
- final String methodStr = "start";
- synchronized (mLock) {
- try {
- if (!checkWifiAndLogFailure(methodStr)) return WifiHal.WIFI_STATUS_ERROR_UNKNOWN;
- //调用 wifi hal 服务的 start 方法
- mWifi.start();
- return WifiHal.WIFI_STATUS_SUCCESS;
- } ...
- }
- }
复制代码 【Hal层的处理惩罚】
- //**************************** HAL层的处理 ****************************
- //hardware/interfaces/wifi/aidl/default/wifi.cpp
- ndk::ScopedAStatus Wifi::start() {
- return validateAndCall(this, WifiStatusCode::ERROR_UNKNOWN, &Wifi::startInternal);
- }
- ndk::ScopedAStatus Wifi::startInternal() {
- ...
- //关注点1:加载驱动和初始化vendor hal
- ndk::ScopedAStatus wifi_status = initializeModeControllerAndLegacyHal();
- if (wifi_status.isOk()) {
- // Register the callback for subsystem restart
- ...
- //关注点2:当HAL启动后,创建chip实例
- // Create the chip instance once the HAL is started.
- int32_t chipId = kPrimaryChipId;
- for (auto& hal : legacy_hals_) {
- chips_.push_back(
- WifiChip::create(chipId, chipId == kPrimaryChipId, hal, mode_controller_,
- std::make_shared<iface_util::WifiIfaceUtil>(iface_tool_, hal),
- feature_flags_, on_subsystem_restart_callback, false));
- chipId++;
- }
- ...
- LOG(INFO) << "Wifi HAL started";
- } ...
- return wifi_status;
- }
复制代码 继承跟踪驱动的加载、vendor hal 接口的初始化流程:
- //hardware/interfaces/wifi/aidl/default/wifi.cpp
- ndk::ScopedAStatus Wifi::initializeModeControllerAndLegacyHal() {
- //1.1.加载驱动
- if (!mode_controller_->initialize()) {
- ...
- }
- //1.2.获取vendor hal
- legacy_hals_ = legacy_hal_factory_->getHals();
- if (legacy_hals_.empty()) return createWifiStatus(WifiStatusCode::ERROR_UNKNOWN);
- ...
- for (auto& hal : legacy_hals_) {
- //1.3.初始化legacy_hal(目前没做什么)
- legacy_hal::wifi_error legacy_status = hal->initialize();
- ...
- }
- return ndk::ScopedAStatus::ok();
- }
- //1.1加载驱动流程:
- //hardware/interfaces/wifi/aidl/default/wifi_mode_controller.cpp
- bool WifiModeController::initialize() {
- if (!driver_tool_->LoadDriver()) {
- LOG(ERROR) << "Failed to load WiFi driver";
- return false;
- }
- return true;
- }
- //1.2 获取vendor hal流程:
- //hardware/interfaces/wifi/aidl/default/wifi_legacy_hal_factory.cpp
- std::vector<std::shared_ptr<WifiLegacyHal>> WifiLegacyHalFactory::getHals() {
- if (legacy_hals_.empty()) {
- //1.2.1 先从已链接的so库中初始化vendor hal(对函数指针进行赋值);
- //1.2.2 如果失败,说明是多wifi芯片的设备(Android12开始支持多wifi芯片),需要从 xml 文件中解析vendor hal信息并进行初始化
- if (!initVendorHalDescriptorFromLinked()) initVendorHalsDescriptorList();
- for (auto& desc : descs_) {
- std::shared_ptr<WifiLegacyHal> hal =
- std::make_shared<WifiLegacyHal>(iface_tool_, desc.fn, desc.primary);
- legacy_hals_.push_back(hal);
- }
- }
- return legacy_hals_;
- }
- //1.2.1 流程跟踪:
- bool WifiLegacyHalFactory::initVendorHalDescriptorFromLinked() {
- wifi_hal_lib_desc desc;
- if (!initLinkedHalFunctionTable(&desc.fn)) return false;
- ...
- return true;
- }
- bool WifiLegacyHalFactory::initLinkedHalFunctionTable(wifi_hal_fn* hal_fn) {
- init_wifi_vendor_hal_func_table_t initfn;
- //在已经链接的库中查找 init_wifi_vendor_hal_func_table 方法
- initfn = (init_wifi_vendor_hal_func_table_t)dlsym(RTLD_DEFAULT,
- "init_wifi_vendor_hal_func_table");
- if (!initfn) {
- LOG(INFO) << "no vendor HAL library linked, will try dynamic load";
- return false;
- }
- //先给 wifi_hal_fn 的所有函数指针赋一个空实现,否则会是null,后面上层调用时会报错
- if (!initHalFuncTableWithStubs(hal_fn)) {
- LOG(ERROR) << "Can not initialize the basic function pointer table";
- return false;
- }
- //调用 init_wifi_vendor_hal_func_table 方法,具体是由vendor提供的legacy_hal实现的,一般只是初始化函数指针。
- if (initfn(hal_fn) != WIFI_SUCCESS) {
- LOG(ERROR) << "Can not initialize the vendor function pointer table";
- return false;
- }
- return true;
- }
复制代码 我们可以从 libwifi-hal 中的Android.bp 看下库的配置:
文件路径:frameworks/opt/net/wifi/libwifi_hal/Android.bp(选取部分内容)
- soong_config_module_type {
- name: "wifi_cc_defaults",
- module_type: "cc_defaults",
- config_namespace: "wifi",
- bool_variables: [
- "multiple_vendor_hals", // WIFI_MULTIPLE_VENDOR_HALS
- "google_wifi_config_lib", // $(wildcard vendor/google/libraries/GoogleWifiConfigLib)
- ],
- value_variables: [
- "driver_module_path", // WIFI_DRIVER_MODULE_PATH
- "driver_module_arg", // WIFI_DRIVER_MODULE_ARG
- "driver_module_name", // WIFI_DRIVER_MODULE_NAME
- "driver_fw_path_sta", // WIFI_DRIVER_FW_PATH_STA
- ...
- "driver_state_off", // WIFI_DRIVER_STATE_OFF
- ],
- variables: [
- "board_wlan_device", // BOARD_WLAN_DEVICE
- ],
- ...
- }
- soong_config_string_variable {
- name: "board_wlan_device",
- values: [
- "bcmdhd",
- "synadhd",
- "qcwcn",
- ...
- "realtek",
- "emulator",
- "rtl",
- "slsi",
- "wlan0",
- ],
- }
- wifi_cc_defaults {
- name: "libwifi_hal_vendor_impl_defaults",
- soong_config_variables: {
- board_wlan_device: {
- bcmdhd: {
- whole_static_libs: ["libwifi-hal-bcm"],
- },
- ...
- },
- },
- }
- cc_library_shared {
- name: "libwifi-hal",
- proprietary: true,
- compile_multilib: "first",
- defaults: [
- "libwifi_hal_cflags_defaults",
- "libwifi_hal_vendor_deps_defaults", // shared_libs used by libwifi-hal-<vendor>
- "libwifi_hal_vendor_impl_defaults",
- ],
- ...
- srcs: [
- "driver_tool.cpp",
- "hal_tool.cpp",
- ],
- whole_static_libs: ["libwifi-hal-common"],
- // b/242259479 remove this
- sanitize: {
- cfi: true,
- integer_overflow: true,
- },
- }
复制代码 一般我们会在 device/xxx/BoardConfig.mk 文件中配置我们使用的硬件信息,可以通过配置 BOARD_WLAN_DEVICE 这个宏来声明我们使用的 wifi 芯片型号。
下面以单wifi芯片为例,看下厂商库相关初始化流程。以使用博通的芯片为例,会配置: BOARD_WLAN_DEVICE := bcmdhd,libwifi_hal 会链接静态库 libwifi-hal-bcm 。libwifi-hal-bcm 库的代码路径为:hardware/broadcom/wlan/bcmdhd/wifi_hal/。
来看下它的 init_wifi_vendor_hal_func_table 方法:
- //hardware/broadcom/wlan/bcmdhd/wifi_hal/wifi_hal.cpp
- /*initialize function pointer table with Broadcom HHAL API*/
- wifi_error init_wifi_vendor_hal_func_table(wifi_hal_fn *fn)
- {
- if (fn == NULL) {
- return WIFI_ERROR_UNKNOWN;
- }
- fn->wifi_initialize = wifi_initialize;
- fn->wifi_wait_for_driver_ready = wifi_wait_for_driver_ready;
- fn->wifi_cleanup = wifi_cleanup;
- fn->wifi_event_loop = wifi_event_loop;
- fn->wifi_get_supported_feature_set = wifi_get_supported_feature_set;
- fn->wifi_get_concurrency_matrix = wifi_get_concurrency_matrix;
- ...
- return WIFI_SUCCESS;
- }
复制代码 可以看到,只是为函数指针赋值。
2.1.2 创建sta interface
【流程概览】
【Java层的处理惩罚】
重要调用链如下:
WifiNative.createStaIface() --> WifiVendorHal.createStaIface() --> HalDeviceManager.createStaIface() --> createStaIface() -->createIface():
①HalDeviceManager.getAllChipInfo()
-->WifiHal.getChip()
-->IWifiHal.getChip()
-->WifiHalAidlImpl.getChip()
--> wifi.cpp::getChip() //通过 hal 接口获取chip信息
--> new WifiChip() //创建WifiChip对象
②HalDeviceManager.createIfaceIfPossible()
-->getBestIfaceCreationProposal();
-->executeChipReconfiguration()
-->WifiChip.configureChip() //通过hal接口配置chip
-->WifiChip.createStaIface() //通过hal接口创建sta interface
下面是源码分析:
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiNative.java
- public String setupInterfaceForClientInScanMode(
- @NonNull InterfaceCallback interfaceCallback, @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- synchronized (mLock) {
- ...
- //2.创建并初始化 Iface
- // 创建一个Iface对象
- Iface iface = mIfaceMgr.allocateIface(Iface.IFACE_TYPE_STA_FOR_SCAN);
- if (iface == null) {
- Log.e(TAG, "Failed to allocate new STA iface");
- return null;
- }
- iface.externalListener = interfaceCallback;
- //关注点:创建sta iface
- iface.name = createStaIface(iface, requestorWs, concreteClientModeManager);
- ...
- }
- /**
- * Helper function to handle creation of STA iface.
- * For devices which do not the support the HAL, this will bypass HalDeviceManager &
- * teardown any existing iface.
- */
- private String createStaIface(@NonNull Iface iface, @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- synchronized (mLock) {
- if (mWifiVendorHal.isVendorHalSupported()) {
- //调用 WifiVendorHal的方法
- return mWifiVendorHal.createStaIface(
- new InterfaceDestoyedListenerInternal(iface.id), requestorWs,
- concreteClientModeManager);
- } ...
- }
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiVendorHal.java
- public String createStaIface(@Nullable InterfaceDestroyedListener destroyedListener,
- @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- synchronized (sLock) {
- //关注点:调用 HalDeviceManager.createStaIface 方法
- WifiStaIface iface = mHalDeviceManager.createStaIface(
- new StaInterfaceDestroyedListenerInternal(destroyedListener), mHalEventHandler,
- requestorWs, concreteClientModeManager);
- ...
- //注册回调方法
- if (!registerStaIfaceCallback(iface)) {
- mLog.err("Failed to register STA iface callback").flush();
- return null;
- }
- ...
- return ifaceName;
- }
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/HalDeviceManager.java
- public WifiStaIface createStaIface(
- @Nullable InterfaceDestroyedListener destroyedListener, @Nullable Handler handler,
- @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- return createStaIface(CHIP_CAPABILITY_ANY, destroyedListener, handler, requestorWs,
- concreteClientModeManager);
- }
- public WifiStaIface createStaIface(
- long requiredChipCapabilities,
- @Nullable InterfaceDestroyedListener destroyedListener, @Nullable Handler handler,
- @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- ...
- WifiStaIface staIface = (WifiStaIface) createIface(HDM_CREATE_IFACE_STA,
- requiredChipCapabilities, destroyedListener, handler, requestorWs);
- ...
- return staIface;
- }
- private WifiHal.WifiInterface createIface(@HdmIfaceTypeForCreation int createIfaceType,
- long requiredChipCapabilities, InterfaceDestroyedListener destroyedListener,
- Handler handler, WorkSource requestorWs) {
- ...
- synchronized (mLock) {
- //获取所有芯片信息
- WifiChipInfo[] chipInfos = getAllChipInfo();
- ...
- //关注点:createIfaceIfPossible
- return createIfaceIfPossible(
- chipInfos, createIfaceType, requiredChipCapabilities,
- destroyedListener, handler, requestorWs);
- }
- }
- private WifiHal.WifiInterface createIfaceIfPossible(
- WifiChipInfo[] chipInfos, @HdmIfaceTypeForCreation int createIfaceType,
- long requiredChipCapabilities, InterfaceDestroyedListener destroyedListener,
- Handler handler, WorkSource requestorWs) {
- int targetHalIfaceType = HAL_IFACE_MAP.get(createIfaceType);
- if (VDBG) {
- Log.d(TAG, "createIfaceIfPossible: chipInfos=" + Arrays.deepToString(chipInfos)
- + ", createIfaceType=" + createIfaceType
- + ", targetHalIfaceType=" + targetHalIfaceType
- + ", requiredChipCapabilities=" + requiredChipCapabilities
- + ", requestorWs=" + requestorWs);
- }
- synchronized (mLock) {
- //获取最合适的iface(暂不分析,一般都是单wifi芯片)
- IfaceCreationData bestIfaceCreationProposal = getBestIfaceCreationProposal(chipInfos,
- createIfaceType, requiredChipCapabilities, requestorWs);
- if (bestIfaceCreationProposal != null) {
- //关注点:对芯片进行重新配置并返回新建的interface。这里会调用 wifi hal的方法
- WifiHal.WifiInterface iface = executeChipReconfiguration(bestIfaceCreationProposal,
- createIfaceType);
- ...
- } else {
- ...
- }
- ...
- return null;
- }
- /**
- * Performs chip reconfiguration per the input:
- * - Removes the specified interfaces
- * - Reconfigures the chip to the new chip mode (if necessary)
- * - Creates the new interface
- *
- * Returns the newly created interface or a null on any error.
- */
- private WifiHal.WifiInterface executeChipReconfiguration(IfaceCreationData ifaceCreationData,
- @HdmIfaceTypeForCreation int createIfaceType) {
- if (mDbg) {
- Log.d(TAG, "executeChipReconfiguration: ifaceCreationData=" + ifaceCreationData
- + ", createIfaceType=" + createIfaceType);
- }
- ...
- synchronized (mLock) {
- // is this a mode change?
- boolean isModeConfigNeeded = !ifaceCreationData.chipInfo.currentModeIdValid
- || ifaceCreationData.chipInfo.currentModeId != ifaceCreationData.chipModeId;
- if (mDbg) Log.d(TAG, "isModeConfigNeeded=" + isModeConfigNeeded);
- Log.i(TAG, "currentModeId=" + ifaceCreationData.chipInfo.currentModeId
- + ", requestModeId=" + ifaceCreationData.chipModeId
- + ", currentModeIdValid=" + ifaceCreationData.chipInfo.currentModeIdValid);
-
- // first delete interfaces/change modes
- if (isModeConfigNeeded) {
- ...
- //通过wifi hal接口配置chip
- boolean success = ifaceCreationData.chipInfo.chip.configureChip(
- ifaceCreationData.chipModeId);
- ...
- }
- ...
- WifiHal.WifiInterface iface = null;
- switch (createIfaceType) {
- case HDM_CREATE_IFACE_STA:
- //通过 wifi hal 接口创建 sta interface
- iface = ifaceCreationData.chipInfo.chip.createStaIface();
- break;
- ...
- }
- ...
- return iface;
- }
-
- //packages/modules/Wifi/service/java/com/android/server/wifi/hal/WifiChip.java
- private IWifiChip mWifiChip;
- /**
- * See comments for {@link IWifiChip#createStaIface()}
- */
- @Nullable
- public WifiStaIface createStaIface() {
- //调用 hal 层接口
- return validateAndCall("createStaIface", null,
- () -> mWifiChip.createStaIface());
- }//packages/modules/Wifi/service/java/com/android/server/wifi/WifiNative.java
- public String setupInterfaceForClientInScanMode(
- ...
- //2.创建并初始化 Iface
- // 创建一个Iface对象
- Iface iface = mIfaceMgr.allocateIface(Iface.IFACE_TYPE_STA_FOR_SCAN);
- if (iface == null) {
- Log.e(TAG, "Failed to allocate new STA iface");
- return null;
- }
- iface.externalListener = interfaceCallback;
- //关注点:创建sta iface
- iface.name = createStaIface(iface, requestorWs, concreteClientModeManager);
- ...
- }
- /**
- * Helper function to handle creation of STA iface.
- * For devices which do not the support the HAL, this will bypass HalDeviceManager &
- * teardown any existing iface.
- */
- private String createStaIface(@NonNull Iface iface, @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- synchronized (mLock) {
- if (mWifiVendorHal.isVendorHalSupported()) {
- //调用 WifiVendorHal的方法
- return mWifiVendorHal.createStaIface(
- new InterfaceDestoyedListenerInternal(iface.id), requestorWs,
- concreteClientModeManager);
- } ...
- }
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiVendorHal.java
- public String createStaIface(@Nullable InterfaceDestroyedListener destroyedListener,
- @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- synchronized (sLock) {
- //关注点:调用 HalDeviceManager.createStaIface 方法
- WifiStaIface iface = mHalDeviceManager.createStaIface(
- new StaInterfaceDestroyedListenerInternal(destroyedListener), mHalEventHandler,
- requestorWs, concreteClientModeManager);
- ...
- //注册回调方法
- if (!registerStaIfaceCallback(iface)) {
- mLog.err("Failed to register STA iface callback").flush();
- return null;
- }
- ...
- return ifaceName;
- }
- }
- //packages/modules/Wifi/service/java/com/android/server/wifi/HalDeviceManager.java
- public WifiStaIface createStaIface(
- @Nullable InterfaceDestroyedListener destroyedListener, @Nullable Handler handler,
- @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- return createStaIface(CHIP_CAPABILITY_ANY, destroyedListener, handler, requestorWs,
- concreteClientModeManager);
- }
- public WifiStaIface createStaIface(
- long requiredChipCapabilities,
- @Nullable InterfaceDestroyedListener destroyedListener, @Nullable Handler handler,
- @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- ...
- WifiStaIface staIface = (WifiStaIface) createIface(HDM_CREATE_IFACE_STA,
- requiredChipCapabilities, destroyedListener, handler, requestorWs);
- ...
- return staIface;
- }
- private WifiHal.WifiInterface createIface(@HdmIfaceTypeForCreation int createIfaceType,
- long requiredChipCapabilities, InterfaceDestroyedListener destroyedListener,
- Handler handler, WorkSource requestorWs) {
- ...
- synchronized (mLock) {
- //获取所有芯片信息
- WifiChipInfo[] chipInfos = getAllChipInfo();
- ...
- //关注点:createIfaceIfPossible
- return createIfaceIfPossible(
- chipInfos, createIfaceType, requiredChipCapabilities,
- destroyedListener, handler, requestorWs);
- }
- }
- private WifiHal.WifiInterface createIfaceIfPossible(
- WifiChipInfo[] chipInfos, @HdmIfaceTypeForCreation int createIfaceType,
- long requiredChipCapabilities, InterfaceDestroyedListener destroyedListener,
- Handler handler, WorkSource requestorWs) {
- int targetHalIfaceType = HAL_IFACE_MAP.get(createIfaceType);
- if (VDBG) {
- Log.d(TAG, "createIfaceIfPossible: chipInfos=" + Arrays.deepToString(chipInfos)
- + ", createIfaceType=" + createIfaceType
- + ", targetHalIfaceType=" + targetHalIfaceType
- + ", requiredChipCapabilities=" + requiredChipCapabilities
- + ", requestorWs=" + requestorWs);
- }
- synchronized (mLock) {
- //获取最合适的iface(暂不分析,一般都是单wifi芯片)
- IfaceCreationData bestIfaceCreationProposal = getBestIfaceCreationProposal(chipInfos,
- createIfaceType, requiredChipCapabilities, requestorWs);
- if (bestIfaceCreationProposal != null) {
- //关注点:对芯片进行重新配置并返回新建的interface。这里会调用 wifi hal的方法
- WifiHal.WifiInterface iface = executeChipReconfiguration(bestIfaceCreationProposal,
- createIfaceType);
- ...
- } else {
- ...
- }
- ...
- return null;
- }
- /**
- * Performs chip reconfiguration per the input:
- * - Removes the specified interfaces
- * - Reconfigures the chip to the new chip mode (if necessary)
- * - Creates the new interface
- *
- * Returns the newly created interface or a null on any error.
- */
- private WifiHal.WifiInterface executeChipReconfiguration(IfaceCreationData ifaceCreationData,
- @HdmIfaceTypeForCreation int createIfaceType) {
- if (mDbg) {
- Log.d(TAG, "executeChipReconfiguration: ifaceCreationData=" + ifaceCreationData
- + ", createIfaceType=" + createIfaceType);
- }
- ...
- synchronized (mLock) {
- // is this a mode change?
- boolean isModeConfigNeeded = !ifaceCreationData.chipInfo.currentModeIdValid
- || ifaceCreationData.chipInfo.currentModeId != ifaceCreationData.chipModeId;
- if (mDbg) Log.d(TAG, "isModeConfigNeeded=" + isModeConfigNeeded);
- Log.i(TAG, "currentModeId=" + ifaceCreationData.chipInfo.currentModeId
- + ", requestModeId=" + ifaceCreationData.chipModeId
- + ", currentModeIdValid=" + ifaceCreationData.chipInfo.currentModeIdValid);
-
- // first delete interfaces/change modes
- if (isModeConfigNeeded) {
- ...
- //通过 wifi hal 接口配置 chip
- boolean success = ifaceCreationData.chipInfo.chip.configureChip(
- ifaceCreationData.chipModeId);
- ...
- }
- ...
- WifiHal.WifiInterface iface = null;
- switch (createIfaceType) {
- case HDM_CREATE_IFACE_STA:
- //通过 wifi hal 接口创建 sta interface
- iface = ifaceCreationData.chipInfo.chip.createStaIface();
- break;
- ...
- }
- ...
- return iface;
- }
-
- //packages/modules/Wifi/service/java/com/android/server/wifi/hal/WifiChip.java
- private IWifiChip mWifiChip;
- /**
- * See comments for {@link IWifiChip#createStaIface()}
- */
- @Nullable
- public WifiStaIface createStaIface() {
- //调用 hal 层接口
- return validateAndCall("createStaIface", null,
- () -> mWifiChip.createStaIface());
- }
复制代码
【Hal层的处理惩罚】
重要调用链如下:
WifiChip::createStaIface() --> createStaIfaceInternal()
--> allocateStaIfaceName() //1.获取interface的名字
-->allocateApOrStaIfaceName() -->getWlanIfaceNameWithType()
-->WifiLegacyHal::getSupportedIfaceName() //优先通过 vendor hal 接口获取;
-->getWlanIfaceName() //如果vendor hal不支持该接口,则通过属性获取
-->WifiLegacyHal::createVirtualInterface() //2.创建假造interface(调用vendor hal接口)
-->WifiStaIface::create() //3.创建 WifiStaIface 对象
- //*********************** HAL层的处理: ***********************
- //hardware/interfaces/wifi/aidl/default/wifi_chip.cpp
- ndk::ScopedAStatus WifiChip::createStaIface(std::shared_ptr<IWifiStaIface>* _aidl_return) {
- return validateAndCall(this, WifiStatusCode::ERROR_WIFI_CHIP_INVALID,
- &WifiChip::createStaIfaceInternal, _aidl_return);
- }
- std::pair<std::shared_ptr<IWifiStaIface>, ndk::ScopedAStatus> WifiChip::createStaIfaceInternal() {
- ...
- //1.获取interface的名字:
- //优先从 vendor hal 的 wifi_get_supported_iface_name() 接口获取;
- //如果 vendor hal不支持该接口,则通过属性 wifi.interface 获取。可以通过 adb shell getprop wifi.interface 命令查看属性值。
- std::string ifname = allocateStaIfaceName();
- //2.创建虚拟interface:会调用vendor hal 的 wifi_virtual_interface_create(),通过 nl80211 接口创建 interface
- legacy_hal::wifi_error legacy_status = legacy_hal_.lock()->createVirtualInterface(
- ifname, aidl_struct_util::convertAidlIfaceTypeToLegacy(IfaceType::STA));
- if (legacy_status != legacy_hal::WIFI_SUCCESS) {
- LOG(ERROR) << "Failed to add interface: " << ifname << " "
- << legacyErrorToString(legacy_status);
- return {nullptr, createWifiStatusFromLegacyError(legacy_status)};
- }
- //3.创建 WifiStaIface 对象(AIDL interface object used to control a STA Iface instance)
- std::shared_ptr<WifiStaIface> iface = WifiStaIface::create(ifname, legacy_hal_, iface_util_);
- sta_ifaces_.push_back(iface);
- for (const auto& callback : event_cb_handler_.getCallbacks()) {
- if (!callback->onIfaceAdded(IfaceType::STA, ifname).isOk()) {
- LOG(ERROR) << "Failed to invoke onIfaceAdded callback";
- }
- }
- setActiveWlanIfaceNameProperty(getFirstActiveWlanIfaceName());
- return {iface, ndk::ScopedAStatus::ok()};
- }
复制代码 至此,wifi的interface已经创建成功。
2.1.3 wificond配置 interface
前面已经加载驱动和初始化vendor hal接口,且wifi interface也已经创建成功;接下来是通过wificond对interface举行配置,使其具备扫描功能。

- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiNative.java
- /**
- * Setup an interface for client mode (for scan) operations.
- *
- * This method configures an interface in STA mode in the native daemons
- * (wificond, vendor HAL).
- *
- * @param interfaceCallback Associated callback for notifying status changes for the iface.
- * @param requestorWs Requestor worksource.
- * @param concreteClientModeManager ConcreteClientModeManager requesting the interface.
- * @return Returns the name of the allocated interface, will be null on failure.
- */
- public String setupInterfaceForClientInScanMode(
- @NonNull InterfaceCallback interfaceCallback, @NonNull WorkSource requestorWs,
- @NonNull ConcreteClientModeManager concreteClientModeManager) {
- synchronized (mLock) {
- ...
- //3.wificond 对 iface 进行配置
- if (!mWifiCondManager.setupInterfaceForClientMode(iface.name, Runnable::run,
- new NormalScanEventCallback(iface.name),
- new PnoScanEventCallback(iface.name))) {
- Log.e(TAG, "Failed to setup iface in wificond=" + iface.name);
- teardownInterface(iface.name);
- mWifiMetrics.incrementNumSetupClientInterfaceFailureDueToWificond();
- return null;
- }
- //frameworks/base/wifi/java/src/android/net/wifi/nl80211/WifiNl80211Manager.java
- public boolean setupInterfaceForClientMode(@NonNull String ifaceName,
- @NonNull @CallbackExecutor Executor executor,
- @NonNull ScanEventCallback scanCallback, @NonNull ScanEventCallback pnoScanCallback) {
- Log.d(TAG, "Setting up interface for client mode: " + ifaceName);
- //1.获取 wifinl80211 服务,并注册死亡通知
- if (!retrieveWificondAndRegisterForDeath()) {
- return false;
- }
- ...
- IClientInterface clientInterface = null;
- try {
- //2.通过 wifinl80211 服务,创建一个适用于Wifi客户端的网络接口
- //(核心是通过netlink跟内核nl80211模块进行通信)
- clientInterface = mWificond.createClientInterface(ifaceName);
- } catch (RemoteException e1) {
- Log.e(TAG, "Failed to get IClientInterface due to remote exception");
- return false;
- }
- ...
- // Refresh Handlers
- mClientInterfaces.put(ifaceName, clientInterface);
- try {
- //3.获取关联的WifiScanner接口
- IWifiScannerImpl wificondScanner = clientInterface.getWifiScannerImpl();
- ...
- ScanEventHandler scanEventHandler = new ScanEventHandler(executor, scanCallback);
- mScanEventHandlers.put(ifaceName, scanEventHandler);
- //订阅扫描事件
- wificondScanner.subscribeScanEvents(scanEventHandler);
- PnoScanEventHandler pnoScanEventHandler = new PnoScanEventHandler(executor,
- pnoScanCallback);
- ...
- } catch (RemoteException e) {
- Log.e(TAG, "Failed to refresh wificond scanner due to remote exception");
- }
- return true;
- }
- //wificond 通过netlink跟内核nl80211模块进行通信,用于提供wifi相关信息和控制功能。
- //目前对该模块不太熟悉,暂不展开相关代码了。
复制代码 2.1.4 小结
setupInterfaceForClientInScanMode 重要完成三件事:①加载驱动,初始化vendor hal接口;②创建并初始化sta接口;③负责扫描功能的wificond对网卡接口举行相关配置。
为什么叫setupInterfaceForClientInScanMode,由于到目前为止,只能扫描,还不能为上层提供连接功能。
接下来必要启动提供连接功能的wpa_supplicant进程。
2.2 switchClientInterfaceToConnectivityMode 启动supplicant
- // packages/modules/Wifi/service/java/com/android/server/wifi/WifiNative.java
- public boolean switchClientInterfaceToConnectivityMode(@NonNull String ifaceName,
- @NonNull WorkSource requestorWs) {
- ...
- //关注点:启动supplicant
- startSupplicant();
-
- ...
- //supplicant中初始化iface
- mSupplicantStaIfaceHal.setupIface(iface.name);
- ...
- }
复制代码 来看下wpa_supplicant的启动流程:
2.2.1 WifiNative
startSupplicant() --> startAndWaitForSupplicantConnection()
- //packages/modules/Wifi/service/java/com/android/server/wifi/WifiNative.java
- /** Helper method invoked to start supplicant if there were no STA ifaces */
- private boolean startSupplicant() {
- synchronized (mLock) {
- //如果没有用于连接的staIface
- if (!mIfaceMgr.hasAnyStaIfaceForConnectivity()) {
- //关注点:启动并等待与supplicant的连接
- if (!startAndWaitForSupplicantConnection()) {
- Log.e(TAG, "Failed to connect to supplicant");
- return false;
- }
- //注册supplicant死亡通知
- if (!mSupplicantStaIfaceHal.registerDeathHandler(
- new SupplicantDeathHandlerInternal())) {
- Log.e(TAG, "Failed to register supplicant death handler");
- return false;
- }
- }
- return true;
- }
- }
- //来看下 startAndWaitForSupplicantConnection 的逻辑:
- private boolean startAndWaitForSupplicantConnection() {
- // Start initialization if not already started.
- // 1.初始化:SupplicantStaIfaceHal.initialize()
- if (!mSupplicantStaIfaceHal.isInitializationStarted()
- && !mSupplicantStaIfaceHal.initialize()) {
- return false;
- }
- // 2.启动服务:SupplicantStaIfaceHal.startDaemon()
- if (!mSupplicantStaIfaceHal.startDaemon()) {
- Log.e(TAG, "Failed to startup supplicant");
- return false;
- }
- // 3.检查supplicant是否启动成功。每100ms检测一次,最多检测50次。
- boolean connected = false;
- int connectTries = 0;
- while (!connected && connectTries++ < CONNECT_TO_SUPPLICANT_RETRY_TIMES) {
- // Check if the initialization is complete.
- connected = mSupplicantStaIfaceHal.isInitializationComplete();
- if (connected) {
- break;
- }
- try {
- Thread.sleep(CONNECT_TO_SUPPLICANT_RETRY_INTERVAL_MS);
- } catch (InterruptedException ignore) {
- }
- }
- return connected;
- }
复制代码 startAndWaitForSupplicantConnection()这个函数重要逻辑:
1.调用mSupplicantStaIfaceHal.startDaemon()启动supplicant;
2.然后通过mSupplicantStaIfaceHal.isInitializationComplete()检查supplicant是否成功启动。
(每100ms检查一次,最多检查50次,)
继承看supplicant服务的启动流程:
2.2.2 SupplicantStaIfaceHalAidlImpl.startDaemon()流程
调用链:
【Java层流程】
- //packages/modules/Wifi/service/java/com/android/server/wifi/SupplicantStaIfaceHalAidlImpl.java
- /**
- * Start the supplicant daemon.
- *
- * @return true on success, false otherwise.
- */
- public boolean startDaemon() {
- synchronized (mLock) {
- ...
- //关注点:获取supplicant服务(会拉起supplicant服务)
- mISupplicant = getSupplicantMockable();
- ...
- Log.i(TAG, "Obtained ISupplicant binder.");
- Log.i(TAG, "Local Version: " + ISupplicant.VERSION);
- try {
- ...
- IBinder serviceBinder = getServiceBinderMockable();
- ...
- //注册死亡通知
- serviceBinder.linkToDeath(mSupplicantDeathRecipient, /* flags= */ 0);
- ...
- return true;
- } ...
- }
- }
- //重点关注:getSupplicantMockable()
- private static final String HAL_INSTANCE_NAME = ISupplicant.DESCRIPTOR + "/default";
- /**
- * Wrapper functions to access HAL objects, created to be mockable in unit tests
- */
- @VisibleForTesting
- protected ISupplicant getSupplicantMockable() {
- synchronized (mLock) {
- try {
- if (SdkLevel.isAtLeastT()) {
- // 通过servicemanager 启动supplicant服务。
- // supplicant服务的名称为:android.hardware.wifi.supplicant.ISupplicant/default
- return ISupplicant.Stub.asInterface(
- ServiceManager.waitForDeclaredService(HAL_INSTANCE_NAME));
- } else {
- return null;
- }
- } ...
- }
- }
- //frameworks/base/core/java/android/os/ServiceManager.java
- /**
- * Returns the specified service from the service manager, if declared.
- *
- * If the service is not running, servicemanager will attempt to start it, and this function
- * will wait for it to be ready.
- * ...
- */
- @SystemApi(client = SystemApi.Client.MODULE_LIBRARIES)
- @Nullable public static IBinder waitForDeclaredService(@NonNull String name) {
- return isDeclared(name) ? waitForService(name) : null;
- }
- public static IBinder waitForService(@NonNull String name) {
- //调用native方法
- return Binder.allowBlocking(waitForServiceNative(name));
- }
复制代码 【JNI层】
- //****************************** JNI 层 ******************************
- //frameworks/base/core/jni/android_os_ServiceManager.cpp
- // Native because we have a client-side wait in waitForService() and we do not
- // want an unnecessary second copy of it.
- static jobject android_os_ServiceManager_waitForServiceNative(JNIEnv* env, jclass /* clazzObj */,
- jstring serviceNameObj) {
- const jchar* serviceName = env->GetStringCritical(serviceNameObj, nullptr);
- ...
- String16 nameCopy = String16(reinterpret_cast<const char16_t *>(serviceName),
- env->GetStringLength(serviceNameObj));
- env->ReleaseStringCritical(serviceNameObj, serviceName);
- //调用native servicemanager 的方法
- sp<IBinder> service = defaultServiceManager()->waitForService(nameCopy);
- ...
- return javaObjectForIBinder(env, service);
- }
复制代码 【Native层流程】
- //************************** Native 层 **************************************
- //************************ servicemanager 客户端 ************************
- //frameworks/native/libs/binder/IServiceManager.cpp
- sp<IBinder> ServiceManagerShim::waitForService(const String16& name16)
- {
- ...
- sp<IBinder> out;
- //关注点:realGetService
- if (Status status = realGetService(name, &out); !status.isOk()) {
- ...
- return nullptr;
- }
- if (out != nullptr) return out;
- ...
- }
- // From the old libbinder IServiceManager interface to IServiceManager.
- class ServiceManagerShim : public IServiceManager
- {
- public:
- ...
- protected:
- sp<AidlServiceManager> mTheRealServiceManager;
- ...
- virtual Status realGetService(const std::string& name, sp<IBinder>* _aidl_return) {
- return mTheRealServiceManager->getService(name, _aidl_return);
- }
- };
- //************************ servicemanager 服务端 ************************
- //frameworks/native/cmds/servicemanager/ServiceManager.cpp
- Status ServiceManager::getService(const std::string& name, sp<IBinder>* outBinder) {
- *outBinder = tryGetService(name, true);
- // returns ok regardless of result for legacy reasons
- return Status::ok();
- }
- sp<IBinder> ServiceManager::tryGetService(const std::string& name, bool startIfNotFound) {
- auto ctx = mAccess->getCallingContext();
- sp<IBinder> out;
- Service* service = nullptr;
- ...
- if (!out && startIfNotFound) {
- //尝试启动service
- tryStartService(name);
- }
- ...
- }
- void ServiceManager::tryStartService(const std::string& name) {
- //注意这里的打印:
- ALOGI("Since '%s' could not be found, trying to start it as a lazy AIDL service. (if it's not "
- "configured to be a lazy service, it may be stuck starting or still starting).",
- name.c_str());
- std::thread([=] {
- if (!base::SetProperty("ctl.interface_start", "aidl/" + name)) {
- ...
- }
- }).detach();
- }
复制代码 可以看到,终极是通过设置属性 "ctrl.interface_start aidl/xxx" 来触发init进程去拉起 supplicant 服务的。supplicant服务的名称睁开为:android.hardware.wifi.supplicant.ISupplicant/default。
来看下supplicant 服务对应的rc文件,路径为:external/wpa_supplicant_8/wpa_supplicant/aidl/android.hardware.wifi.supplicant-service.rc
- service wpa_supplicant /vendor/bin/hw/wpa_supplicant \
- -O/data/vendor/wifi/wpa/sockets -dd \
- -g@android:wpa_wlan0
- # we will start as root and wpa_supplicant will switch to user wifi
- # after setting up the capabilities required for WEXT
- # user wifi
- # group wifi inet keystore
- interface aidl android.hardware.wifi.supplicant.ISupplicant/default
- class main
- #创建一个socket,wpa_s使用该socket接收framework的命令和向上传递event。
- socket wpa_wlan0 dgram 660 wifi wifi
- disabled
- oneshot
复制代码 关于 supplicant 的初始化流程,会放到后续章节举行介绍,这里就不举行睁开。
以下是关键打印:
- I SupplicantStaIfaceHalAidlImpl: Clearing internal state
- I servicemanager: Found android.hardware.wifi.supplicant.ISupplicant/default in device VINTF manifest.
- I servicemanager: Since 'android.hardware.wifi.supplicant.ISupplicant/default' could not be found, trying to start it as a lazy AIDL service. (if it's not configured to be a lazy service, it may be stuck starting or still starting).
- I init : starting service 'wpa_supplicant'...
- I init : Created socket '/dev/socket/wpa_wlan0', mode 660, user 1010, group 1010
- I init : ... started service 'wpa_supplicant' has pid 2024
- I init : Control message: Processed ctl.interface_start for 'aidl/android.hardware.wifi.supplicant.ISupplicant/default' from pid: 130 (/system/bin/servicemanager)
- I wpa_supplicant: Processing aidl events on FD 3
- I wpa_supplicant: Starting AIDL supplicant
- I wpa_supplicant: Interface version: 2
- I wpa_supplicant: Successfully initialized wpa_supplicant
- I SupplicantStaIfaceHalAidlImpl: Obtained ISupplicant binder.
- I SupplicantStaIfaceHalAidlImpl: Local Version: 2
- I SupplicantStaIfaceHalAidlImpl: Remote Version: 2
复制代码 至此, SupplicantStaIfaceHalAidlImpl已经将supplicnat进程成功拉起,并且与supplicant hal service举行了绑定。后续Frameowork层可通过SupplicantStaIfaceHalAidlImpl调用supplicnat相关接口。
参考链接:
Wi-Fi架构(二)-- Wi-Fi打开流程2.1
Android WiFi 开启流程(基于Android S)
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。 |