Android U 匹配不到APN,无法发起数据建立的问题分析

2023-11-22 19:04

本文主要是介绍Android U 匹配不到APN,无法发起数据建立的问题分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问题

打开数据开关后,没有data PDN请求发起,因此无法上网。

根据日志确定是没有找到合适的data profile,原因一般有:

1、APN 没有配置

2、APN 类型/网络能力不满足——APN type或bearer

3、APN 配置了但被disable了——APN proofile的carrierEnable值

日志分析

fail log:

APN 被disable了,所以无法匹配。

//preferred=true说明当前选中的是此条APN

11-20 15:01:51.409140  2316  3480 D DPM-0   : getApnSettingForNetworkRequest: networkType=13 isMtkImsDataSupport=true hasEimsCapability=false mPreferredDataProfile=[DataProfile=[ApnSetting]....] dataProfilesCount=0 dataProfiles=[] mAllDataProfiles=[[DataProfile=[ApnSetting] INTERNET, 3080, xxxxxx, internet, , null, , null, null, 0, supl | hipri | default, IPV4V6, IP, false, 0, true, 0, 0, 0, 0, 0, gid, xxxxx000000000000, false, UNKNOWN, UNKNOWN, 0, -1, -1, false, 961, TrafficDescriptor={mDnn=internet, null}, preferred=true]...]
11-20 15:01:51.410224  2316  3480 D DPM-0   : Can't find any data profile that can satisfy [NetworkRequest [ NONE id=0, [ Capabilities: DUN&TRUSTED&NOT_VPN&NOT_VCN_MANAGED Uid: 1001 UnderlyingNetworks: Null] ], mPriority=30, state=UNSATISFIED, mAttachedDataNetwork=null, isMetered=true, created time=15:01:51.407, evaluation result=null]
11-20 15:01:51.411224  1366  1366 D ImsManager: [1] Registration callback removed.
11-20 15:01:51.413463  2316  3480 D DPM-0   : Unable to find a data profile for [NetworkRequest [ NONE id=0, [ Capabilities: DUN&TRUSTED&NOT_VPN&NOT_VCN_MANAGED Uid: 1001 UnderlyingNetworks: Null] ], mPriority=30, state=UNSATISFIED, mAttachedDataNetwork=null, isMetered=true, created time=15:01:51.408, evaluation result=null]

正常日志:

dataProfilesCount 指明网络能力匹配的APN数

11-20 15:19:20.723854  2316  2316 D DPM-0   : getApnSettingForNetworkRequest: networkType=13 isMtkImsDataSupport=true hasEimsCapability=false mPreferredDataProfile=[DataProfile=[ApnSetting] ....] dataProfilesCount=1 dataProfiles=[[DataProfile=[ApnSetting] 

11-20 15:19:20.725075  2316  2316 V DPM-0   : Satisfied profile: [DataProfile=[ApnSetting]

11-20 15:19:20.725871  2316  2316 D DPM-0   : The final available profile: [DataProfile=[ApnSetting] 

11-20 15:19:20.729504  2316  2316 D DNC-0   : Data evaluation: evaluation reason:NEW_REQUEST, Data allowed reason: NORMAL, candidate profile=[DataProfile=[ApnSetting] 

11-20 15:19:20.730116  2316  2316 D DNC-0   : onSetupDataNetwork: dataProfile=[DataProfile=[ApnSetting] 

11-20 15:19:20.731688  2316  2316 D DNC-0   : Creating data network on WWAN with [DataProfile=[ApnSetting] 

11-20 15:19:20.768286  2316  2316 D DSM-C-0 : setupDataCall

11-20 15:19:20.769505  2316  2316 D DN-114-C: ConnectingState: setupData: accessNetwork=EUTRAN, [DataProfile=[ApnSetting] ...], isModemRoaming=false, allowRoaming=true, PDU session id=0, matchAllRuleAllowed=true
11-20 15:19:20.776007  2316  2682 D RILJ    :

 [0868]> SETUP_DATA_CALL,reason=NORMAL,accessNetworkType=EUTRAN,dataProfile=[DataProfile=[ApnSetting] ..],isRoaming=false,allowRoaming=true,linkProperties=null,pduSessionId=0,sliceInfo=null,trafficDescriptor=TrafficDescriptor={mDnn=VZWADMIN, null},matchAllRuleAllowed=true [PHONE0]

11-20 15:19:21.063568  2316  2674 D RILJ    : [0868]< SETUP_DATA_CALL DataCallResponse: { cause=NONE(0x0) retry=-1 cid=1803 linkStatus=2 protocolType=2 ifname=ccmni3 addresses=[10.128.159.113/32, 2600:100f:f00c:330f:0:56:6fe2:e801/64] dnses=[/2001:4888:53:ff00:524:d::, /2001:4888:52:ff00:528:d::, /198.224.166.135, /198.224.167.135] gateways=[/10.128.159.113, /::] pcscf=[] mtu=1428 mtuV4=1428 mtuV6=1428 handoverFailureMode=unknown pduSessionId=0 defaultQos=EpsQos { qosClassId=8 downlink=Bandwidth { maxBitrateKbps=0 guaranteedBitrateKbps=0} uplink=Bandwidth { maxBitrateKbps=0 guaranteedBitrateKbps=0}} qosBearerSessions=[] sliceInfo=null trafficDescriptors=[]} [PHONE0]

 代码分析

DataProfileManager.java - 【frameworks/opt/telephony】

/*** Get the APN setting for the network request.** @param networkRequest The network request.* @param networkType The current data network type.* @param ignorePermanentFailure {@code true} to ignore {@link ApnSetting#getPermanentFailed()}.* This should be set to true for condition-based retry/setup.* @return The APN setting. {@code null} if can't find any satisfiable data profile.*/
private @Nullable ApnSetting getApnSettingForNetworkRequest(@NonNull TelephonyNetworkRequest networkRequest, @NetworkType int networkType,boolean ignorePermanentFailure) {if (!networkRequest.hasAttribute(TelephonyNetworkRequest.CAPABILITY_ATTRIBUTE_APN_SETTING)) {loge("Network request does not have APN setting attribute.");return null;}//1、优先选择preferred APN,如果满足并且不处于PermanentFailure状态则直接请求// If the preferred data profile can be used, always use it if it can satisfy the network// request with current network type (even though it's been marked as permanent failed.)if (mPreferredDataProfile != null&& networkRequest.canBeSatisfiedBy(mPreferredDataProfile)&& mPreferredDataProfile.getApnSetting() != null&& mPreferredDataProfile.getApnSetting().canSupportNetworkType(networkType)) {if (ignorePermanentFailure || !mPreferredDataProfile.getApnSetting().getPermanentFailed()) {return mPreferredDataProfile.getApnSetting();}log("The preferred data profile is permanently failed. Only condition based retry "+ "can happen.");return null;}//2、如果没有首选数据配置文件,则过滤掉不能满足网络请求的数据配置文件//Note:找不到profile 一般就是这里匹配不到了// Filter out the data profile that can't satisfy the request.// Preferred data profile should be returned in the top of the list.List<DataProfile> dataProfiles = mAllDataProfiles.stream().filter(networkRequest::canBeSatisfiedBy)// The longest time hasn't used data profile will be in the front so all the data// profiles can be tried..sorted(Comparator.comparing(DataProfile::getLastSetupTimestamp))    //并按照它们的最后设置时间进行排序,将最长时间未使用的数据配置文件放在列表的前面.collect(Collectors.toList());for (DataProfile dataProfile : dataProfiles) {    //然后尝试筛选并使用它们来满足网络请求。logv("Satisfied profile: " + dataProfile + ", last setup="+ DataUtils.elapsedTimeToString(dataProfile.getLastSetupTimestamp()));}if (dataProfiles.size() == 0) {log("Can't find any data profile that can satisfy " + networkRequest);return null;}// Check if the remaining data profiles can used in current data network type.dataProfiles = dataProfiles.stream().filter(dp -> dp.getApnSetting() != null&& dp.getApnSetting().canSupportNetworkType(networkType)).collect(Collectors.toList());if (dataProfiles.size() == 0) {log("Can't find any data profile for network type "+ TelephonyManager.getNetworkTypeName(networkType));return null;}// Check if preferred data profile set id matches.dataProfiles = dataProfiles.stream().filter(dp -> dp.getApnSetting() != null&& (dp.getApnSetting().getApnSetId()== Telephony.Carriers.MATCH_ALL_APN_SET_ID|| dp.getApnSetting().getApnSetId() == mPreferredDataProfileSetId)).collect(Collectors.toList());if (dataProfiles.size() == 0) {log("Can't find any data profile has APN set id matched. mPreferredDataProfileSetId="+ mPreferredDataProfileSetId);return null;}// Check if data profiles are permanently failed.dataProfiles = dataProfiles.stream().filter(dp -> ignorePermanentFailure || !dp.getApnSetting().getPermanentFailed()).collect(Collectors.toList());if (dataProfiles.size() == 0) {log("The suitable data profiles are all in permanent failed state.");return null;}return dataProfiles.get(0).getApnSetting();
}
排查接口

canBeSatisfiedBy in TelephonyNetworkRequest.java -【frameworks/opt/telephony】

canHandleType in ApnSetting.java【frameworks/base/telephony】


匹配不到关键就是在网络能力评估的过程,canBeSatisfiedBy会根据APN类型筛选能力匹配的项,在canHandleType检查完APN可用性之后,返回enable且符合网络的APN。

/**
* Check if this network request can be satisfied by a data profile.
*
* @param dataProfile The data profile to check.
* @return {@code true} if this network request can be satisfied by the data profile.
*/
public boolean canBeSatisfiedBy(@NonNull DataProfile dataProfile) {// If the network request can be translated to OS/App id, then check if the data profile's// OS/App id can satisfy it.if (hasAttribute(CAPABILITY_ATTRIBUTE_TRAFFIC_DESCRIPTOR_OS_APP_ID)&& getOsAppId() != null) {// The network request has traffic descriptor type capabilities. Match the traffic// descriptor.if (dataProfile.getTrafficDescriptor() != null && Arrays.equals(getOsAppId().getBytes(),dataProfile.getTrafficDescriptor().getOsAppId())) {return true;}}// If the network request can be translated to APN setting or DNN in traffic descriptor,// then check if the data profile's APN setting can satisfy it.if ((hasAttribute(CAPABILITY_ATTRIBUTE_APN_SETTING)|| hasAttribute(CAPABILITY_ATTRIBUTE_TRAFFIC_DESCRIPTOR_DNN))&& dataProfile.getApnSetting() != null) {// Fallback to the legacy APN type matching.List<Integer> apnTypes = Arrays.stream(getCapabilities()).boxed().map(DataUtils::networkCapabilityToApnType).filter(apnType -> apnType != ApnSetting.TYPE_NONE).collect(Collectors.toList());// In case of enterprise network request, the network request will have internet,// but APN type will not have default type as the enterprise apn should not be used// as default network. Ignore default type of the network request if it// has enterprise type as well. This will make sure the network request with// internet and enterprise will be satisfied with data profile with enterprise at the// same time default network request will not get satisfied with enterprise data// profile.// TODO b/232264746if (apnTypes.contains(ApnSetting.TYPE_ENTERPRISE)) {apnTypes.remove((Integer) ApnSetting.TYPE_DEFAULT);}return apnTypes.stream().allMatch(dataProfile.getApnSetting()::canHandleType);}return false;
}/** @hide */
public boolean canHandleType(@ApnType int type) {if (!mCarrierEnabled) {return false;}// DEFAULT can handle HIPRI.return hasApnType(type);
}

定位原因的方法

查看APNSettings配置是否正确,以上问题就是APN配置了但被disable了。

[DataProfile=[ApnSetting] INTERNET, 3080, xxxxxx, internet, , null, , null, null, 0, supl | hipri | default, IPV4V6, IP, false, 0, true, 0, 0, 0, 0, 0, gid, xxxxx000000000000, false, UNKNOWN, UNKNOWN, 0, -1, -1, false, 961, TrafficDescriptor={mDnn=internet, null}, preferred=true]

根据ApnSettings实体节后打印信息,mCarrierEnabled就是上述日志false的位置。

/*** Returns the string representation of ApnSetting.** This method prints null for unset elements. The output doesn't contain password or user.* @hide*/
public String toString() {StringBuilder sb = new StringBuilder();sb.append("[ApnSetting] ").append(mEntryName).append(", ").append(mId).append(", ").append(mOperatorNumeric).append(", ").append(mApnName).append(", ").append(mProxyAddress).append(", ").append(UriToString(mMmsc)).append(", ").append(mMmsProxyAddress).append(", ").append(portToString(mMmsProxyPort)).append(", ").append(portToString(mProxyPort)).append(", ").append(mAuthType).append(", ");final String[] types = getApnTypesStringFromBitmask(mApnTypeBitmask).split(",");sb.append(TextUtils.join(" | ", types));sb.append(", ").append(PROTOCOL_INT_MAP.get(mProtocol));sb.append(", ").append(PROTOCOL_INT_MAP.get(mRoamingProtocol));sb.append(", ").append(mCarrierEnabled);    //APN启用情况sb.append(", ").append(mProfileId);sb.append(", ").append(mPersistent);sb.append(", ").append(mMaxConns);sb.append(", ").append(mWaitTime);sb.append(", ").append(mMaxConnsTime);sb.append(", ").append(mMtuV4);sb.append(", ").append(mMtuV6);sb.append(", ").append(MVNO_TYPE_INT_MAP.get(mMvnoType));sb.append(", ").append(mMvnoMatchData);sb.append(", ").append(mPermanentFailed);sb.append(", ").append(TelephonyManager.convertNetworkTypeBitmaskToString(mNetworkTypeBitmask));sb.append(", ").append(TelephonyManager.convertNetworkTypeBitmaskToString(mLingeringNetworkTypeBitmask));sb.append(", ").append(mApnSetId);sb.append(", ").append(mCarrierId);sb.append(", ").append(mSkip464Xlat);sb.append(", ").append(mAlwaysOn);sb.append(", ").append(Objects.hash(mUser, mPassword));return sb.toString();
}

这篇关于Android U 匹配不到APN,无法发起数据建立的问题分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)

《使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)》字体设计和矢量图形处理是编程中一个有趣且实用的领域,通过Python的matplotlib库,我们可以轻松将字体轮廓... 目录背景知识字体轮廓的表示实现步骤1. 安装依赖库2. 准备数据3. 解析路径指令4. 绘制图形关键

VS配置好Qt环境之后但无法打开ui界面的问题解决

《VS配置好Qt环境之后但无法打开ui界面的问题解决》本文主要介绍了VS配置好Qt环境之后但无法打开ui界面的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 目UKeLvb录找到Qt安装目录中designer.UKeLvBexe的路径找到vs中的解决方案资源

解决mysql插入数据锁等待超时报错:Lock wait timeout exceeded;try restarting transaction

《解决mysql插入数据锁等待超时报错:Lockwaittimeoutexceeded;tryrestartingtransaction》:本文主要介绍解决mysql插入数据锁等待超时报... 目录报错信息解决办法1、数据库中执行如下sql2、再到 INNODB_TRX 事务表中查看总结报错信息Lock

Apache 高级配置实战之从连接保持到日志分析的完整指南

《Apache高级配置实战之从连接保持到日志分析的完整指南》本文带你从连接保持优化开始,一路走到访问控制和日志管理,最后用AWStats来分析网站数据,对Apache配置日志分析相关知识感兴趣的朋友... 目录Apache 高级配置实战:从连接保持到日志分析的完整指南前言 一、Apache 连接保持 - 性

MySQL启动报错:InnoDB表空间丢失问题及解决方法

《MySQL启动报错:InnoDB表空间丢失问题及解决方法》在启动MySQL时,遇到了InnoDB:Tablespace5975wasnotfound,该错误表明MySQL在启动过程中无法找到指定的s... 目录mysql 启动报错:InnoDB 表空间丢失问题及解决方法错误分析解决方案1. 启用 inno

Java使用MethodHandle来替代反射,提高性能问题

《Java使用MethodHandle来替代反射,提高性能问题》:本文主要介绍Java使用MethodHandle来替代反射,提高性能问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录一、认识MethodHandle1、简介2、使用方式3、与反射的区别二、示例1、基本使用2、(重要)

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

Linux lvm实例之如何创建一个专用于MySQL数据存储的LVM卷组

《Linuxlvm实例之如何创建一个专用于MySQL数据存储的LVM卷组》:本文主要介绍使用Linux创建一个专用于MySQL数据存储的LVM卷组的实例,具有很好的参考价值,希望对大家有所帮助,... 目录在Centos 7上创建卷China编程组并配置mysql数据目录1. 检查现有磁盘2. 创建物理卷3. 创

Nacos日志与Raft的数据清理指南

《Nacos日志与Raft的数据清理指南》随着运行时间的增长,Nacos的日志文件(logs/)和Raft持久化数据(data/protocol/raft/)可能会占用大量磁盘空间,影响系统稳定性,本... 目录引言1. Nacos 日志文件(logs/ 目录)清理1.1 日志文件的作用1.2 是否可以删除

Linux中的more 和 less区别对比分析

《Linux中的more和less区别对比分析》在Linux/Unix系统中,more和less都是用于分页查看文本文件的命令,但less是more的增强版,功能更强大,:本文主要介绍Linu... 目录1. 基础功能对比2. 常用操作对比less 的操作3. 实际使用示例4. 为什么推荐 less?5.