android4.3原机拨号,Android——4.2 - 3G移植之路之 reference-ril .pppd 拨号上网 (三)

本文主要是介绍android4.3原机拨号,Android——4.2 - 3G移植之路之 reference-ril .pppd 拨号上网 (三),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Android的RIL机制中的 reference-ril.c 即为厂商提供的驱动接口,这个驱动源码各个厂商都是有提供的,网上也有下载,我现在用的就是huawei wcdma的,最后编译成

libreference-ril.so,关于这个接口驱动在RIL中所扮演的角色可参考

Android——RIL 机制源码分析

android 4.2自带pppd源码在/external/ppp/pppd中.同样,kernel中也是需要打开对point-to-point 的支持,在network support里面.

一.requestSetupDataCall:

这个就是在reference-ril.c 中的数据流量的request,上层的接口通过onRequest 的RIL_REQUEST_SETUP_DATA_CALL请求。这些在上面说到的源码分析里有详细分析,这里就只从拨号连接分析。

staticvoidrequestSetupDataCall(voidvoid*data,size_tdatalen, RIL_Token t)

{

constcharchar*apn;

charchar*cmd;

interr;

ATResponse*p_response =NULL;

charppp_dnses[(PROPERTY_VALUE_MAX*2) +3] = {'\0'};//初始化属性数组,用于临时存储拨号的属性变量

charppp_local_ip[PROPERTY_VALUE_MAX] = {'\0'};

charril_pppd_tty[PROPERTY_VALUE_MAX] = {'\0'};

charppp_dns1[PROPERTY_VALUE_MAX] = {'\0'};

charppp_dns2[PROPERTY_VALUE_MAX] = {'\0'};

charppp_gw[PROPERTY_VALUE_MAX] = {'\0'};

charexit_code[PROPERTY_VALUE_MAX] = {'\0'};

intn =1;

RIL_Data_Call_Response_v6*responses = alloca(n*sizeof(RIL_Data_Call_Response_v6));//使用的是ipv6

apn = ((constcharchar**)data)[2];//取传进来的接入点 apn

/*    ALOGD("jscese display in reference  APN == '%s' \n",apn);

apn ="3gnet";*/

ALOGD("[%s] jscese display in reference  APN '%s' ", __func__, apn);

#ifdef USE_TI_COMMANDS

// Config for multislot class 10 (probably default anyway eh?)

err = at_send_command("AT%CPRIM=\"GMM\",\"CONFIG MULTISLOT_CLASS=<10>\"",

NULL);

err = at_send_command("AT%DATA=2,\"UART\",1,,\"SER\",\"UART\",0",NULL);

#endif /* USE_TI_COMMANDS */

intfd, qmistatus;

size_tcur =0;

size_tlen;

ssize_t written, rlen;

charstatus[32] = {0};

intretry =10;

constcharchar*pdp_type;

ALOGD("requesting data connection to APN '%s'", apn);

pdp_type = "IP";// jscese add for dial

asprintf(&cmd, "AT+CGDCONT=1,\"%s\",\"%s\",,0,0", pdp_type, apn);

//FIXME check for error here

err = at_send_command(cmd, NULL);//发送接入点的AT指令

free(cmd);

#if 0

// Set required QoS params to default

err = at_send_command("AT+CGQREQ=1",NULL);

// Set minimum QoS params to default

err = at_send_command("AT+CGQMIN=1",NULL);

// packet-domain event reporting

err = at_send_command("AT+CGEREP=1,0",NULL);

// Hangup anything that's happening there now

err = at_send_command("AT+CGACT=1,0",NULL);

#endif

// Start data on PDP context 1

if(strcmp(apn,"3gnet") ==0)

{

ALOGD("jscese display in reference  is 3gnet \n");

err = at_send_command("ATD*99***1#", &p_response);//这个就是联通3G上网需要拨的号码 ATD*99***1# 获取连接

}

elseif(strcmp(apn,"ctnet") ==0)

{

ALOGD("jscese display in reference  is ctnet \n");

err = at_send_command("ATD#777", &p_response);

}

if(err <0|| p_response->success ==0)

{

gotoerror;

}

at_response_free(p_response);

sleep(1);//Wait for the modem to finish

property_set("net.ppp1.local-ip","");

property_set("net.gprs.ppp-exit","");

property_set("ctl.start","pppd_gprs");//如果上面的拨号AT指令成功返回,这里就启用之前定义好的一个pppd service 调用脚本去拨号获取IP 等网络参数

// Dialup

sleep(3);

/*jscese add try 5 times to get ip*/

intiRetry =5;

while(iRetry >0)

{

property_get("net.gprs.ppp-exit", exit_code,"");

if(strcmp(exit_code,"0") !=0)

{

ALOGE("PPPd exit with code %s", exit_code);

iRetry = 0;

break;

}

ALOGI("Waiting For Property");

if(wait_for_property("net.ppp1.local-ip",NULL,10) <0)//监测 ip 地址的属性值,这个值我放在 pppd拨号脚本里面来进行设置,如果成功这里就能监测到

{

ALOGE("[%s]: wait for IP from ppp link at %d\n", __func__, iRetry);

}

else

{

ALOGI("[%s]: got IP from ppp link\r\n", __func__);

break;

}

iRetry--;

}

if(iRetry <=0)

{

ALOGE("[%s]: fail to get IP\r\n", __func__);

gotoerror;

}

/*    if (wait_for_property("net.ppp1.local-ip", NULL, 10) 

ALOGE("Timeout waiting net.ppp1.local-ip - giving up!\n");

goto error;

}*/

property_get("net.ppp1.local-ip", ppp_local_ip,NULL);

property_get("net.ppp1.dns1", ppp_dns1,NULL);

property_get("net.ppp1.dns2", ppp_dns2,NULL);

property_get("net.ppp1.gw", ppp_gw,NULL);

sprintf(ppp_dnses, "%s %s", ppp_dns1, ppp_dns2);

ALOGI("Got net.ppp1.local-ip: %s\n", ppp_local_ip);

responses[0].status=0;

responses[0].suggestedRetryTime= -1;

responses[0].cid=1;

responses[0].active=2;

responses[0].type="PPP";

responses[0].ifname= PPP_TTY_PATH;

responses[0].addresses= ppp_local_ip;

responses[0].dnses= ppp_dnses;

responses[0].gateways= ppp_gw;//通过获取到的 网络属性 设置这个responses 提供给上层

RIL_onRequestComplete(t, RIL_E_SUCCESS, responses, n*sizeof(RIL_Data_Call_Response_v6));

return;

error: RIL_onRequestComplete(t, RIL_E_GENERIC_FAILURE,NULL,0);

at_response_free(p_response);

}

二.pppd

在上面 requestSetupDataCall 中启动一个服务来调用pppd拨号,先在init.rc中添加:

#jscese add for usb_switch service when 3G dongle add for serial

service ril-daemon /system/bin/rild -l /system/lib/libreference-ril.so -- -d /dev/ttyUSB2

class main

socket rild stream 660 root radio

socket rild-debug stream 660 radio system

user root

group radio cache inet misc audio log

service pppd_gprs /system/etc/ppp/init.gprs-pppd /dev/ttyUSB0

user root

group radio cache inet misc

disabled

oneshot

## end

上面的服务是用来开机启动rild的,加载libreference-ril动态库,

并且指定了通信端口为串口 -d /dev/ttyUSB2,另外还有 -s 代表是socket设备 -p 代表回环接口

下面的就是pppd拨号的脚本服务了,用来启动拨号脚本,并且指定数据端口.

这个pppd_gprs 服务需要设置权限,因为我是在reference-ril里面通过属性启动的 在/system/core/init/property_service.c中添加如下:

struct{

constcharchar*service;

unsigned intuid;

unsigned intgid;

} control_perms[] = {

{ "dumpstate",AID_SHELL, AID_LOG },

{ "ril-daemon",AID_RADIO, AID_RADIO },

/*jscese add for pppd 3G*/

{ "pppd_gprs",AID_RADIO, AID_LOG },

/*end*/

{NULL,0,0}

};

struct{

constcharchar*prefix;

unsigned intuid;

unsigned intgid;

} property_perms[] = {

...

/*jscese add  pppd for 3G*/

{ "net.ppp1.",AID_RADIO,0},

/*end*/

{ NULL,0,0}

};

设置用到的net.ppp1.* 的权限。

init.gprs-pppd 设置执行权限,/system/core/include/private/android_filesystem_config.h中:

staticstructfs_path_config android_files[] = {

...

/*jscese add for pppd */

{ 00777, AID_ROOT,      AID_SHELL,"system/etc/init.gprs-pppd"},

/*    end*/

...

}

init.gprs-pppd:

贴出拨号脚本

#!/system/bin/sh

PPPD_PID=

USER=`/system/bin/getprop net.gprs.user`

PASSWORD=`/system/bin/getprop net.gprs.password`

/system/bin/setprop "net.gprs.ppp-exit"""

/system/bin/pppd $1115200linkname datakey unit1crtscts usepeerdns noauth defaultroute noipdefault ipcp-accept-local ipcp-accept-remote ipcp-max-failure30lcp-echo-interval5lcp-echo-failure30modem dump debug kdebug8

PPPD_EXIT=$?

PPPD_PID=$!

/system/bin/log -t pppd "pppd exited with $PPPD_EXIT"

/system/bin/setprop "net.gprs.ppp-exit""$PPPD_EXIT"

另外在这个拨号脚本同目录下 /system/etc/ppp/ 需要设置和注销ip等参数的脚本,一个在拨号成功时调用,一个在断开网络时调用:

ip-up-datakey:

#!/system/bin/sh

case $1in

ppp1)

/android/bin/iptables --flush;

/android/bin/iptables --table nat --flush;

/android/bin/iptables --delete-chain;

/android/bin/iptables --table nat --append POSTROUTING --out-interface ppp0 -j MASQUERADE;

/android/bin/iptables --append FORWARD --in-interface ppp1 -j ACCEPT;

echo 0> /proc/sys/net/ipv4/ip_forward;

echo 1> /proc/sys/net/ipv4/ip_forward;

;;

ppp0)

/system/bin/setprop "net.interfaces.defaultroute""ppp1";

;;

esac

/system/bin/setprop "net.ppp1.dns1""$DNS1"

/system/bin/setprop "net.ppp1.dns2""$DNS2"

/system/bin/setprop "net.ppp1.local-ip""$IPLOCAL"

/system/bin/setprop "net.ppp1.remote-ip""$IPREMOTE"

/system/bin/setprop "net.ppp1.gw""$IPREMOTE"

/system/bin/setprop "net.ppp1.if""$IFNAME"

ip-down-datakey:

#!/system/bin/sh

case $1in

ppp1)

echo 0> /proc/sys/net/ipv4/ip_forward;

;;

esac

/system/bin/setprop "net.ppp1.dns1"""

/system/bin/setprop "net.ppp1.dns2"""

/system/bin/setprop "net.ppp1.local-ip"""

/system/bin/setprop "net.ppp1.remote-ip"""

/system/bin/setprop "net.ppp1.gw"""

/system/bin/setprop "net.ppp1.if"""

贴张联通apn为 3gnet 的拨号log:

0818b9ca8b590ca3270a3433284dd417.png

这篇关于android4.3原机拨号,Android——4.2 - 3G移植之路之 reference-ril .pppd 拨号上网 (三)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Android使用java实现网络连通性检查详解

《Android使用java实现网络连通性检查详解》这篇文章主要为大家详细介绍了Android使用java实现网络连通性检查的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录NetCheck.Java(可直接拷贝)使用示例(Activity/Fragment 内)权限要求

2025最新版Android Studio安装及组件配置教程(SDK、JDK、Gradle)

《2025最新版AndroidStudio安装及组件配置教程(SDK、JDK、Gradle)》:本文主要介绍2025最新版AndroidStudio安装及组件配置(SDK、JDK、Gradle... 目录原生 android 简介Android Studio必备组件一、Android Studio安装二、A

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

在Android中使用WebView在线查看PDF文件的方法示例

《在Android中使用WebView在线查看PDF文件的方法示例》在Android应用开发中,有时我们需要在客户端展示PDF文件,以便用户可以阅读或交互,:本文主要介绍在Android中使用We... 目录简介:1. WebView组件介绍2. 在androidManifest.XML中添加Interne

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class