hi3516 海思3518C 4G模块移植及运用
海思3518C 4G模块移植及运用 --洋辣椒
联通连接点方式chat-wcdma-connect 内容: ABORT 'NO CARRIER' ABORT 'ERROR' ABORT 'NO DIALTONE' ABORT 'BUSY' ABORT 'NO ANSWER' '' AT OK ATZ OK AT+CGDCONT=1,\"IP\",\"3gnet\",,0,0 OK AT+CFUN=1 OK ATDT*99#
本章程主要介绍LTE模块使用及驱动移植、pppd拨号软件移植,LTE模块主要使用中兴ME3630、韦德S600模块、龙尚U9300C模块、美格SLM730、移远EC20、有方N720;各模块AT指令基本一样,只有几个特定指令不一样(4G信号查询指令、软复位指令)。
1、Pppd移植
本章使用的是ppp-2.4.4版本,开发包为ppp-2.4.4.tar.gz;
需要修改内容:
1.1、修改ppp-2.4.4\pppd\ipcp.c
修改该文件主要是为了在拨号软件获取到网络参数DNS时,工具不会自动设置到嵌入式系统中,需要我们自己进行操作。
在ipcp_up函数中添加:
char cmd1[100] = {0x00};
char cmd2[100] = {0x00};
notice("local IP address %I", go->ouraddr);
notice("remote IP address %I", ho->hisaddr);
//写入配置脚本
if (go->dnsaddr[0])
{
notice("primary DNS address %I", go->dnsaddr[0]);
sprintf(cmd1, "echo nameserver %s > /mnt/mtd/resolv_wirless.conf",
ip_ntoa(go->dnsaddr[0]));
system(cmd1);
}
if (go->dnsaddr[1])
{
notice("secondary DNS address %I", go->dnsaddr[1]);
sprintf(cmd2, "echo nameserver %s >> /mnt/mtd/resolv_wirless.conf",
ip_ntoa(go->dnsaddr[1]));
system(cmd2);
}
//清除默认网关
cifdefaultroute(0,0,0);
//设置新的网关
SetGateway(ifname,ip_ntoa(ho->hisaddr));
设置网关主要是因本章程设计了一个有线\无线自动切换功能,在有线环境下,拨号软件退出(偶尔因信号差退出拨号连接),当需要切换成无线时,重新拨号,这时需要清除原来默认网关,重新设置无线网关。
设置网关函数
static bool
SetGateway(char* _szInterface, char* _szGateway)
{
int s32Ret;
int s32Socketfd = socket(AF_INET, SOCK_DGRAM, 0);
if(s32Socketfd < 0)
{
printf("socket failed by %d", errno);
return -1;
}
struct rtentry rt;
memset(&rt, 0, sizeof(struct rtentry));
memset(&rt.rt_dst, 0, sizeof(rt.rt_dst));
rt.rt_dst.sa_family = AF_INET;
((struct sockaddr_in *)(&rt.rt_dst))->sin_addr.s_addr = inet_addr("0.0.0.0");
memset(&rt.rt_gateway, 0, sizeof(rt.rt_gateway));
rt.rt_gateway.sa_family = AF_INET;
((struct sockaddr_in *)(&rt.rt_gateway))->sin_addr.s_addr = inet_addr(_szGateway);
memset(&rt.rt_genmask, 0, sizeof(rt.rt_genmask));
rt.rt_genmask.sa_family = AF_INET;
((struct sockaddr_in *)(&rt.rt_genmask))->sin_addr.s_addr = 0L;
rt.rt_dev = _szInterface;
rt.rt_flags = RTF_GATEWAY;
s32Ret = ioctl(s32Socketfd, SIOCADDRT, &rt);
if(s32Ret < 0)
{
printf("ioctl SIOCADDRT failed by %d", errno);
close(s32Socketfd);
return -1;
}
close(s32Socketfd);
return 0;
}
以上函数是将拨号获取到的dns写到系统文件缓存中,应用程序需要将该缓存写到系统的dns文件中(linux系统:/etc/resolv.conf,安卓系统:需要通过dnsproxy2工具进行设置该工具将有另外章节介绍);
1.2、判断拨号获取DNS是否正确
因为拨号软件在执行拨号时,当信号非常差的时候,从基站获取到的DNS会错误,所以这里必须在获取DNS时进行过滤。
修改main.c函数:
添加函数
void checkdns __V((char *fmt, ...))
{
va_list pvar;
#if defined(STDC)
va_start(pvar, fmt);
#else
char *fmt;
va_start(pvar);
fmt = va_arg(pvar, char *);
#endif
int n;
char buf[1024];
n = vslprintf(buf, sizeof(buf), fmt, pvar);
va_end(pvar);
//IPCP ConfAck id=0x2 <addr 10.138.113.2> <ms-dns1 210.21.4.130> <ms-dns3 221.5.88.88>
/*
* 该段函数是为了防止拨号后获取到错误的DNS后还继续使用该
* 该参数,解决方案是当获取到配置应答指令后判断是否获取
* 到IP地址和对于的DNS,如果没获取到DNS则推出该拨号程序。
*/
char *pu8ConfAck;
char *pu8addr;
char *pu8msdns;
pu8ConfAck = strstr((char*) buf,"ConfAck");
if(pu8ConfAck != NULL)
if(strncmp(pu8ConfAck,"ConfAck",strlen("ConfAck")) == 0)
{
printf("---ConfAck[%s]\n",pu8ConfAck);
pu8addr = strstr((char*) pu8ConfAck,"addr");
if(pu8addr != NULL)
if(strncmp(pu8addr,"addr",strlen("addr")) == 0)
{
printf("---addr[%s]\n",pu8addr);
pu8msdns = strstr((char*) pu8addr,"ms-dns");
if(pu8msdns == NULL)
exit(1);
printf("---msdns[%s]\n",pu8msdns);
}
}
}
static void
get_input()
{
-----------------
dump_packet("rcvd", p, len);
//-------edw------------------------//
checkdns( "%P", p, len);
//------end-------------------------//
到此基本拨号软件基本修改完成。
注意:android系统移植,安卓sdk自带pppd工具(external\ppp目录下),修改内容和上述基本一样,只有在设置dns时不同,安卓系统使用命令“setprop net.dnsx”进行设置DNS。
1.3、编译程序
生成makefile文件;
编译make CC=arm-hisiv100nptl-linux-gcc
获取到pppd chat
1.4、运营商连接点
联通节点wcdma
移动节点cnmet
电信节点evdo
内容:
debug
nodetach
/dev/ttyUSB2
115200
usepeerdns
noauth
noipdefault
novj
novjccomp
noccp
defaultroute
ipcp-accept-local
ipcp-accept-remote
connect '/system/bin/chat -s -v -f /etc/ppp/peers/chat-wcdma-connect'
disconnect '/system/bin/chat -s -v -f/etc/ppp/peers/chat-disconnect'
三个节点文件内容类似,只有连接点不同:chat-wcdma-connect。
联通连接点方式chat-wcdma-connect
内容:
ABORT 'NO CARRIER'
ABORT 'ERROR'
ABORT 'NO DIALTONE'
ABORT 'BUSY'
ABORT 'NO ANSWER'
'' AT
OK ATZ
OK AT+CGDCONT=1,\"IP\",\"3gnet\",,0,0
OK AT+CFUN=1
OK ATDT*99#
CONNECT ''
移动连接点方式chat-cnmet-connect
ABORT 'NO CARRIER'
ABORT 'ERROR'
ABORT 'NO DIALTONE'
ABORT 'BUSY'
ABORT 'NO ANSWER'
'' AT
OK ATZ
OK AT+CGDCONT=1,\"IP\",\"cmnet\",,0,0
OK AT+CFUN=1
OK ATDT*98*1#
CONNECT ''
电信连接点方式chat-evdo-connect
ABORT 'NO CARRIER'
ABORT 'ERROR'
ABORT 'NO DIALTONE'
ABORT 'BUSY'
ABORT 'NO ANSWER'
'' AT
OK ATZ
OK AT+CGDCONT=1,\"IP\",\"ctlte\",,0,0
OK AT+CFUN=1
OK ATDT*99#
CONNECT ''
1.5、启动拨号
联通运营商:pppd call wcdma&
移动运营商:pppd call cnmet&
电信运营商:pppd call evdo&
2、驱动移植
http://blog.csdn.net/skdkjzz/article/details/22206949
stty -F /dev/ttyUSB0 raw speed 9600 min 0 time 10
使用龙尚4G模块,修改驱动:
- 在内核中添加USB串口驱动和USB网卡驱动,可以选择将其直接编入内核,或者编译为模块待内核启动时加载,总之,要确保Linux内核启动完成后,这两个驱动是运行在内核当中的。
通常,配置内核是通过指令make menuconfig,执行该指令后:
l 添加USB串口驱动:
device drivers-->
usb support-->
usb serial converter support-->
USB driver for GSM and CDMA modems
l 添加USB网卡驱动
devices drivers-->
Network device support-->
usb Network Adapters-->
Mulil-purpose USB Networking Framework
- 注:如果您的内核结构与上面不一致,可能需要在其它的路径下面选择,总之,只要确保源文件中的 option.c 及其相关的部分(USB串口驱动),cdc_ether.c及其相关部分(USB网卡驱动)参与编译即可
- 在内核中添加PPP组件
通常,配置内核是通过指令make menuconfig,执行该指令后:
devices drivers-->
Network device support-->
ppp support-->
ppp filtering
ppp support for async serial ports
ppp support for sync tty ports
ppp deflate compression
ppp BSD-compress compression
- 在内核驱动源文件中添加ZTE模块相关的信息:USB网卡驱动可以自动识别到ZTE模块,因此,其对应的cdc_ether.c文件中不需要添加任何内容。但是USB串口驱动不能自动识别,必须要添加ZTE模块的设备信息到源文件option.c 中。
文件路径:/kernel/drivers/usb/serial/option.c
l 添加USB端口的VID和PID信息,见下面蓝色部分。这里0x0199为ME3860模块和ME3760_V2模块的PID,0x1476为ME3620模块的PID,如果您使用的是其它模块,将其中的PID值更换为相应的值即可。
static const struct usb_device_id option_ids[] = {
+{ USB_DEVICE(0x19d2, 0x1476) }, //add me3630
+{ USB_DEVICE(0x19d2, 0x0016) }, //add me3630
+{ USB_DEVICE(0x1c9e, 0x9b05) }, //add 8300
+{ USB_DEVICE(0x1c9e, 0x9b3c) }, //add 9300
+{ USB_DEVICE(0x05C6, 0x9025) }, //add S600
+{ USB_DEVICE(0x05C6, 0xf601) }, //add slm730
+{ USB_DEVICE(0x2c7c, 0x0125) }, //add yiyuan
+{ USB_DEVICE(0x05C6, 0x9008) }, //add n720……
}
以上ID可以在进入系统中,通过命令“lsusb”,查看模块ID。
l 添加黑名单信息,上面添加模块信息是只添加了设备的VID和PID,没有附加任何额外的端口信息,这样会导致设备的网卡也被加载成为USB串口,下面提供的是一种类似于黑名单的方式,在option_probe函数中,将网卡对应的端口加入黑名单,防止USB网卡被加载成为USB串口。
对于ME3860和ME3760_V2,其网卡对应的端口为0和1,对于ME3620,其网卡对应的端口为3和4。请将以下代码添加到option_probe函数中
printk("idVendor=%x, idProduct=%x, bInterfaceNumber =%d\r\n",
serial->dev->descriptor.idVendor,
serial->dev->descriptor.idProduct,
serial->interface->cur_altsetting->desc. bInterfaceNumber);
if (serial->dev->descriptor.idVendor == 0x19d2 &&
serial->dev->descriptor.idProduct == 0x1476 &&
serial->interface->cur_altsetting->desc. bInterfaceNumber == 3)
return -ENODEV;
if (serial->dev->descriptor.idVendor == 0x19d2 &&
serial->dev->descriptor.idProduct == 0x1476 &&
serial->interface->cur_altsetting->desc. bInterfaceNumber == 4)
return -ENODEV;
if (serial->dev->descriptor.idVendor == 0x19d2 &&
serial->dev->descriptor.idProduct == 0x0199 &&
serial->interface->cur_altsetting->desc. bInterfaceNumber == 0)
return -ENODEV;
if (serial->dev->descriptor.idVendor == 0x19d2 &&
serial->dev->descriptor.idProduct == 0x0199 &&
serial->interface->cur_altsetting->desc. bInterfaceNumber == 1)
return -ENODEV;
if (serial->dev->descriptor.idVendor == 0x1c9e&& //龙尚
serial->dev->descriptor.idProduct == 0x9b05&&
serial->interface->cur_altsetting->desc. bInterfaceNumber == 3)
return -ENODEV;
- 注:第一行的printk是为了方便调试而打印的,虽无实际效果,最好能带上。下面的几个if语句分别判断了需要加入黑名单的端口号,如果您使用的是除ME3860,ME3760_V2和ME3620之外的模块,上面if语句中的判断条件也要做相应修改。
加载驱动后如需加载其他模块,可以通过命令加载:
echo "19d2 1476" >/sys/bus/usb-serial/drivers/option1/new_id
注意:内核补丁会在海思3518C 4G模块移植文件中得到;