wifi连接上后是怎么提供网络的?

2024-03-09 20:20

本文主要是介绍wifi连接上后是怎么提供网络的?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

干了六个月的网络协议栈,又回到了wifi老本行,所以我最近又开始研读 Android wifi fwk的源码了
之前还在干wifi的时候就思考过一个问题,wifi区别于蓝牙的一个很明显的点是,wifi可以提供 access to Internet
所以我想看看wifi连接成功后是怎么给上层app提供网络的

前言:研读过程中发现整个机制中有大量的类,且它们的命名非常符合自身的功能,这是我觉得oop语言很有趣也是很容易上手的原因
与之前解决 WNS 选网过程一样,解读的过程中我会尝试给出一个大概的通俗语言描述的模型

注册 NetworkOffer

WifiService(SystemService的子类)在完全开机后,会执行 handleBootCompleted() 方法,注册自己魔改的几个NetworkFactory
我们只关注 WifiNetworkFactory

//WifiServiceImpl.java
public void handleBootCompleted() {mWifiThreadRunner.post(() -> {Log.d(TAG, "Handle boot completed");//...mWifiInjector.getWifiNetworkFactory().register();mWifiInjector.getUntrustedWifiNetworkFactory().register();mWifiInjector.getOemWifiNetworkFactory().register();});
}

NetworkFactory register动作主要做的是

  1. ConnectivityManager.registerNetworkProvider(new NetworkProvider)
  2. handleOfferNetwork -> ConnectivityManager.offerNetwork
//NetworkFactoryImpl.java
private void register(final String logTag, final boolean listenToAllRequests) {mProvider = new NetworkProvider(mContext, NetworkFactoryImpl.this.getLooper(), logTag) {@Overridepublic void onNetworkRequested(@NonNull NetworkRequest request, int score,int servingProviderId) {handleAddRequest(request);}@Overridepublic void onNetworkRequestWithdrawn(@NonNull NetworkRequest request) {handleRemoveRequest(request);}};((ConnectivityManager) mContext.getSystemService(Context.CONNECTIVITY_SERVICE)).registerNetworkProvider(mProvider);if (listenToAllRequests) {sendMessage(obtainMessage(CMD_LISTEN_TO_ALL_REQUESTS));} else {sendMessage(obtainMessage(CMD_OFFER_NETWORK));}
}//handle CMD_OFFER_NETWORK msg 的逻辑
//此处 mScore == 0
private void handleOfferNetwork(@NonNull final NetworkScore score) {mProvider.registerNetworkOffer(score, mCapabilityFilter, mExecutor, mRequestCallback);
}
//NetworkProvider.java
public void registerNetworkOffer(@NonNull final NetworkScore score,@NonNull final NetworkCapabilities caps, @NonNull final Executor executor,@NonNull final NetworkOfferCallback callback) {mContext.getSystemService(ConnectivityManager.class).offerNetwork(providerId, score, caps, proxy);
}

简单总结:到这里,wifi fwk已经成功地向 CS 注册了一个 NetworkProvider,并且经由这个 NetworkProvider 给出了一个 NetworkOffer
这个 NetworkOffer 就是用来去 satisfy 上层app发起的 NetworkRequest 的,当 CS match 某NetworkOffer 和 某NetworkRequest 的时候,就意味着这个app发起的网络请求将由匹配的网络来处理
不过,wifi fwk给出的这个 NetworkOffer尚处于不能 handle network request 的状态,因为它的 score 是 0

创建NetworkAgent

wifi fwk STA mode的状态机进入 L2ConnectedState 后(L2表示关联上了,之后会进行DHCP,拿到ip后进入L3ConnectedState),会创建 NetworkAgent

//ClientModeImpl.java
//class L2ConnectedState的EA
mNetworkAgent = mWifiInjector.makeWifiNetworkAgent(nc, mLinkProperties, naConfig,mNetworkFactory.getProvider(), new WifiNetworkAgentCallback());
mWifiScoreReport.setNetworkAgent(mNetworkAgent);

如上代码所示,WifiScoreReport 持 WifiNetworkAgent ,它会根据wifi链路状态实时地更新wifi网络在 CS 中的评分(通过调用 sendNetworkScore 方法)

创建 NetworkAgent 触发的一系列动作如下
最主要的是 创建NetworkMonitor

//WifiNetworkAgent.java
public WifiNetworkAgent(@NonNull Context context,@NonNull Looper looper,@NonNull NetworkCapabilities nc,@NonNull LinkProperties lp,@NonNull NetworkAgentConfig config,@Nullable NetworkProvider provider,@NonNull Callback wifiNetworkAgentCallback) {super(context, looper, TAG, nc, lp, ConnectedScore.WIFI_INITIAL_SCORE, config, provider);register();
}//NetworkAgent.java
public Network register() {synchronized (mRegisterLock) {final ConnectivityManager cm = (ConnectivityManager) mInitialConfiguration.context.getSystemService(Context.CONNECTIVITY_SERVICE);mNetwork = cm.registerNetworkAgent(new NetworkAgentBinder(mHandler),new NetworkInfo(mInitialConfiguration.info),mInitialConfiguration.properties, mInitialConfiguration.capabilities,mInitialConfiguration.score, mInitialConfiguration.config, providerId);}return mNetwork;
}//ConnectivityService.java
private Network registerNetworkAgentInternal(INetworkAgent na, NetworkInfo networkInfo,LinkProperties linkProperties, NetworkCapabilities networkCapabilities,NetworkScore currentScore, NetworkAgentConfig networkAgentConfig, int providerId,int uid) {LinkProperties lp = new LinkProperties(linkProperties);final NetworkCapabilities nc = new NetworkCapabilities(networkCapabilities);final NetworkAgentInfo nai = new NetworkAgentInfo(na,new Network(mNetIdManager.reserveNetId()), new NetworkInfo(networkInfo), lp, nc,currentScore, mContext, mTrackerHandler, new NetworkAgentConfig(networkAgentConfig),this, mNetd, mDnsResolver, providerId, uid, mLingerDelayMs,mQosCallbackTracker, mDeps);mDeps.getNetworkStack().makeNetworkMonitor(nai.network, name, new NetworkMonitorCallbacks(nai));return nai.network;
}

NetworkMonitor创建成功后,CS 会触发 onNetworkMonitorCreated 回调,处理逻辑是 mNetworkMonitor.start()
我们在 main log中经常可以看到 NetworkMonitor http/https/dns probe fail,这说明 NetworkMonitor 的功能是受控去 probe 指定url,根据返回的status code判断当前链接的状态

实时更新 Network

wifi fwk STA mode处于 L2ConnectedState,接收到 CMD_IP_CONFIGURATION_SUCCESSFUL msg(意味着DHCP完成,这个网络有了自己的ip)

//ClientModeImpl.java
case CMD_IP_CONFIGURATION_SUCCESSFUL: {if (getConnectedWifiConfigurationInternal() == null || mNetworkAgent == null) {mWifiNative.disconnect(mInterfaceName);} else {handleSuccessfulIpConfiguration();sendConnectedState();transitionTo(mL3ConnectedState);}break;
}private void sendConnectedState() {mNetworkAgent.markConnected();sendNetworkChangeBroadcast(DetailedState.CONNECTED);
}//NetworkAgent.java
public void markConnected() {mNetworkInfo.setDetailedState(NetworkInfo.DetailedState.CONNECTED, null /* reason */,mNetworkInfo.getExtraInfo());queueOrSendNetworkInfo(mNetworkInfo);
}

markConnected 之后,NetworkAgent 会把更新了状态的 NetworkInfo 同步给 CS
同步过程如下

//NetworkAgentInfo.java
public void sendNetworkInfo(@NonNull NetworkInfo info) {mHandler.obtainMessage(NetworkAgent.EVENT_NETWORK_INFO_CHANGED,new Pair<>(NetworkAgentInfo.this, info)).sendToTarget();
}//ConnectivityService.java
//NetworkStateTrackerHandler handle this msg
case NetworkAgent.EVENT_NETWORK_INFO_CHANGED: {NetworkInfo info = (NetworkInfo) arg.second;updateNetworkInfo(nai, info);break;
}private void updateNetworkInfo(NetworkAgentInfo networkAgent, NetworkInfo info) {if (!networkAgent.everConnected && state == NetworkInfo.State.CONNECTED) {rematchAllNetworksAndRequests();} else if (state == NetworkInfo.State.DISCONNECTED) {networkAgent.disconnect();}
}

rematchAllNetworksAndRequests 做的事情便是重新匹配 Network 和 NetworkRequest
我们在main log中经常可以见到 CS NetworkRequest reassign 的打印,这代表原来由某网络负责的NetworkRequest被分配给另一网络处理了
任何一个网络的状态稍有改变都会触发这个动作

WifiScoreReport sendNetworkScore 后触发的流程与上面类似,只不过 msg 换成了 EVENT_NETWORK_SCORE_CHANGED

//ConnectivityService.java
private void updateNetworkScore(@NonNull final NetworkAgentInfo nai, final NetworkScore score) {nai.setScore(score);rematchAllNetworksAndRequests();
}

总结

WifiService 在开机完成后就像 CS 注册了一个 NetworkProvider,并经由这个NetworkProvider注册了一个 NetworkOffer,只不过处于一个无法向外提供服务的状态。
当连接上某wifi(完成了数据链路层的连接)后,wifi 创建一个 NetworkAgent 用于管理这个网络,同时CS创建一个 NetworkMonitor 用于监管这个网络。
当DHCP完成,wifi网络理论上可以与外界通信后,wifi 的NetworkAgent实时地告知 CS 其NetworkInfo 的改动, CS 会重新调整 NetworkRequest 与 Network 的匹配关系。

后续研读计划

这个方面实际要研读的东西还有很多,一开始是想要搞懂wifi连接上后如何提供网络的

看和写的过程中意识到还应该系统地弄清楚这些问题

  • wifi 改动网络评分的细节(WifiNetworkAgent + WifiScoreReport)
  • CS rematch 的细节(显然CS绝对不是唯评分论的单一匹配机制)
  • macth后一个NetworkRequest是怎么被satisfy的 (数据通路是怎样的)

后记:理解了这个模型,大概可以搞清wifi连接上后是怎么提供网络的了,但是搞清楚这个对于日常工作做trouble shooting有什么帮助吗?
貌似也没有多大帮助
哈哈哈哈,图一乐

这篇关于wifi连接上后是怎么提供网络的?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/791844

相关文章

Debian 13升级后网络转发等功能异常怎么办? 并非错误而是管理机制变更

《Debian13升级后网络转发等功能异常怎么办?并非错误而是管理机制变更》很多朋友反馈,更新到Debian13后网络转发等功能异常,这并非BUG而是Debian13Trixie调整... 日前 Debian 13 Trixie 发布后已经有众多网友升级到新版本,只不过升级后发现某些功能存在异常,例如网络转

Mac电脑如何通过 IntelliJ IDEA 远程连接 MySQL

《Mac电脑如何通过IntelliJIDEA远程连接MySQL》本文详解Mac通过IntelliJIDEA远程连接MySQL的步骤,本文通过图文并茂的形式给大家介绍的非常详细,感兴趣的朋友跟... 目录MAC电脑通过 IntelliJ IDEA 远程连接 mysql 的详细教程一、前缀条件确认二、打开 ID

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

python连接sqlite3简单用法完整例子

《python连接sqlite3简单用法完整例子》SQLite3是一个内置的Python模块,可以通过Python的标准库轻松地使用,无需进行额外安装和配置,:本文主要介绍python连接sqli... 目录1. 连接到数据库2. 创建游标对象3. 创建表4. 插入数据5. 查询数据6. 更新数据7. 删除

99%的人都选错了! 路由器WiFi双频合一还是分开好的专业解析与适用场景探讨

《99%的人都选错了!路由器WiFi双频合一还是分开好的专业解析与适用场景探讨》关于双频路由器的“双频合一”与“分开使用”两种模式,用户往往存在诸多疑问,本文将从多个维度深入探讨这两种模式的优缺点,... 在如今“没有WiFi就等于与世隔绝”的时代,越来越多家庭、办公室都开始配置双频无线路由器。但你有没有注

Python开发简易网络服务器的示例详解(新手入门)

《Python开发简易网络服务器的示例详解(新手入门)》网络服务器是互联网基础设施的核心组件,它本质上是一个持续运行的程序,负责监听特定端口,本文将使用Python开发一个简单的网络服务器,感兴趣的小... 目录网络服务器基础概念python内置服务器模块1. HTTP服务器模块2. Socket服务器模块

Go语言网络故障诊断与调试技巧

《Go语言网络故障诊断与调试技巧》在分布式系统和微服务架构的浪潮中,网络编程成为系统性能和可靠性的核心支柱,从高并发的API服务到实时通信应用,网络的稳定性直接影响用户体验,本文面向熟悉Go基本语法和... 目录1. 引言2. Go 语言网络编程的优势与特色2.1 简洁高效的标准库2.2 强大的并发模型2.

在 Spring Boot 中连接 MySQL 数据库的详细步骤

《在SpringBoot中连接MySQL数据库的详细步骤》本文介绍了SpringBoot连接MySQL数据库的流程,添加依赖、配置连接信息、创建实体类与仓库接口,通过自动配置实现数据库操作,... 目录一、添加依赖二、配置数据库连接三、创建实体类四、创建仓库接口五、创建服务类六、创建控制器七、运行应用程序八

解决hive启动时java.net.ConnectException:拒绝连接的问题

《解决hive启动时java.net.ConnectException:拒绝连接的问题》Hadoop集群连接被拒,需检查集群是否启动、关闭防火墙/SELinux、确认安全模式退出,若问题仍存,查看日志... 目录错误发生原因解决方式1.关闭防火墙2.关闭selinux3.启动集群4.检查集群是否正常启动5.

在Linux系统上连接GitHub的方法步骤(适用2025年)

《在Linux系统上连接GitHub的方法步骤(适用2025年)》在2025年,使用Linux系统连接GitHub的推荐方式是通过SSH(SecureShell)协议进行身份验证,这种方式不仅安全,还... 目录步骤一:检查并安装 Git步骤二:生成 SSH 密钥步骤三:将 SSH 公钥添加到 github