想要实现4G上网有两种方式,要么加多一个4G路由器,再通过优先接入;要么通过增加4G模块(可为USB或PCIE等多种接口),直接进行拨号上网。尝试在一款J1900工控机上(该工控机自带SIM插槽),通过增加PCIE接口的美格4G模块SLM750
,进行拨号上网。Windows系统下已测试过,直接安装厂家提供驱动,可以正常上网,说明硬件方面是完全支持的。本文参照厂家提供的嵌入式方案,进行驱动编译安装,并编译拨号软件,最终实现工控机4G上网功能。
系统为Cento 7.6 64bit,基本环境为Basic Web Server安装(理论上与安装环境模式无关,最小安装也可以)。需要下载内核源码,Centos 7.6的内核版本为3.10.0-957
,源码可在此链接下载。另外还需要厂家提供的GobiNet
网卡拨号的驱动及拨号工具源码,一张能4G上网的手机卡或物联网卡,接好模块天线。
将下载好的源码,解压到看到linux-3.10.0-957.21.3.el7.tar.xz
文件,将其放到/usr/src/kernels
文件夹下,并执行如下命令:
$ tar xvf linux-3.10.0-957.21.3.el7.tar.xz // 解压内核源码文件
$ mv linux-3.10.0-957.21.3.el7 3.10.0-957.el7.x86_64 // 重命名文件夹
之所以要更改文件夹名称,是因为厂家的GobiNet
驱动源码,Makefile
文件中:
obj-m := GobiNet.o
GobiNet-objs := GobiUSBNet.o QMIDevice.o QMI.o
PWD := $(shell pwd)
OUTPUTDIR=/lib/modules/`uname -r`/kernel/drivers/net/usb/
#ifeq ($(ARCH),)
#EARCH := $(shell uname -m)
#endif
#ifeq ($(CROSS_COMPILE),)
#CROSS_COMPILE :=
#endif
#ifeq ($(KDIR),)
KDIR := /lib/modules/$(shell uname -r)/build # 这里通过uname -r 获取了内核名称
#endif
default:
# ln -sf makefile Makefile
#$(MAKE) ARCH=${ARCH} CROSS_COMPILE=${CROSS_COMPILE} -C $(KDIR) M=$(PWD) modules
$(MAKE) CROSS_COMPILE=${CROSS_COMPILE} -C $(KDIR) M=$(PWD) modules
install: default
mkdir -p $(OUTPUTDIR)
cp -f GobiNet.ko $(OUTPUTDIR)
depmod
modprobe -r GobiNet
modprobe GobiNet
clean:
# rm -rf Makefile # 这里这段代码去掉,否则执行make clean会把Makefile文件也删除了
rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions Module.* modules.order
如果为其他版本的系统,将文件夹对应修改为uname -r
得到的名称即可。
打开内核源码文件 /3.10.0-957.el7.x86_64/drivers/usb/serial/option.c
,在/* Vendor and product IDs */
下增加宏定义:
/* Vendor and product IDs */
#define MEIG_VENDOR_ID 0x05C6
#define MEIG_PRODUCT_730 0xF601
#define MEIG_VENDOR_ID_720 0x2dee
#define MEIG_PRODUCT_720 0x4d07
#define MEIG_PRODUCT_720_ECM 0x4d02
在option_ids
结构体数组增加4G模块的VID
和PID
:
static const struct usb_device_id option_ids[] = {
{
USB_DEVICE(MEIG_VENDOR_ID,MEIG_PRODUCT_730) },
{
USB_DEVICE(MEIG_VENDOR_ID_720,MEIG_PRODUCT_720) },
{
USB_DEVICE(MEIG_VENDOR_ID_720,MEIG_PRODUCT_720_ECM) },
使用option
驱动,修改 /3.10.0-957.el7.x86_64/driver/usb/serial/option.c
,在option_probe
函数添加如下代码:
static int option_probe(struct usb_serial *serial,
const struct usb_device_id *id)
{
struct usb_interface_descriptor *iface_desc =
&serial->interface->cur_altsetting->desc;
struct usb_device_descriptor *dev_desc = &serial->dev->descriptor;
const struct option_blacklist_info *blacklist;
/* Never bind to the CD-Rom emulation interface */
if (iface_desc->bInterfaceClass == 0x08)
return -ENODEV;
/*
* Don't bind reserved interfaces (like network ones) which often have
* the same class/subclass/protocol as the serial interfaces. Look at
* the Windows driver .INF files for reserved interface numbers.
*/
blacklist = (void *)id->driver_info;
if (blacklist && test_bit(iface_desc->bInterfaceNumber,
&blacklist->reserved))
return -ENODEV;
// struct usb_wwan_intf_private *data; 文档中的这个语句其实没有
// 开始添加代码
if (serial->dev->descriptor.idVendor == MEIG_VENDOR_ID &&
(serial->dev->descriptor.idProduct == MEIG_PRODUCT_730)&&
serial->interface->cur_altsetting->desc.bInterfaceNumber >= 4)
return -ENODEV;
if (serial->dev->descriptor.idVendor == MEIG_VENDOR_ID_720&&
(serial->dev->descriptor.idProduct == MEIG_PRODUCT_720)&&
serial->interface->cur_altsetting->desc.bInterfaceNumber >= 4)
return -ENODEV;
// 完成添加代码
使用usb-serial
驱动,/3.10.0-957.el7.x86_64/driver/usb/serial/usb-serial.c
,在usb_serial_probe
函数添加如下代码:
serial = create_serial (dev, interface, type);
if (!serial) {
unlock_kernel();
dev_err(&interface->dev, "%s - out of memory\n", __FUNCTION__);
return -ENOMEM;
}
//开始添加代码 厂家文档写的是宏定义,在该文件中无法找到会报错,这里直接改成了对应值
if ( (serial->dev->descriptor.idVendor == 0x50C6 &&
(serial->dev->descriptor.idProduct == 0xF601) )&&
serial->interface->cur_altsetting->desc.bInterfaceNumber >=4 )
return -ENOMEM;
if (serial->dev->descriptor.idVendor == 0x2dee &&
(serial->dev->descriptor.idProduct == 0x4d07)&&
serial->interface->cur_altsetting->desc.bInterfaceNumber >= 4)
return -ENODEV;
//完成添加代码
### 配置编译参数
$ cd /usr/src/kernels/3.10.0-957.el7.x86_64 # 切换到内核源码所在路径
$ cp /boot/config-3.10.0-957.el7.x86_64 ./.config # 拷贝当前内核的编译配置
$ make oldconfig # 在已有内核基础上进行配置
$ yum install gcc gdb make elfutils-libelf-devel
需要说明的是,Centos 7.6 默认就开启了device drivers->usb support->usb serial converter support->USB driver for GSM and CDMA modems
和device drivers->Network device support->usb Network Adapters->Multi-purpose USB Networking Framework
这两个组件,所以拷贝原有内核的编译配置即可使用。
执行如下命令开始编译源码,对应的线程数字按照实际机器进行配置,这个过程会比较慢。
$ make -j 8
$ make modules_install
如果有其他错误提示,则安装对应的软件包依赖即可,这里编译后不进行安装,因为内核是一样的,编译内核只是为了编译驱动时能找到一些依赖。
这里采用的是单独编译的方式,所以上一个步骤没有和内核一块打包编译,主要是为了在不动原来内核的情况下使用,以防上面的其他软件运行受影响。
到驱动目录下,执行如下命令:
$ make # 编译驱动
$ make install # 安装驱动
正常编译安装的话,不会有其他的警告或者错误,驱动成功后,可以看到新的网卡。一般是ethX
这种格式,但还没有IP地址,需要使用拨号软件。
在厂家提供的源码中,由于是嵌入式的方案,默认获取IP地址的是busybox
中的udhcpc
命令,在udhcpc.c
文件中,可以注释掉这样代码,以及这行代码上面两行的寻找默认配置文件的语句。本文管理网卡的工具是NetworkManager
,Gobinet拨号后,会自动检测网卡状态,进行获取IP地址操作。其它系统根据实际需要,进行修改。本文做出的修改如下:
if (profile->ipv4.Address) {
#ifdef USE_DHCLIENT
snprintf(udhcpc_cmd, sizeof(udhcpc_cmd), "dhclient -4 -d --no-pid %s", ifname);
dhclient_alive++;
#else
// 注释掉获取默认配置文件
//if (access("/usr/share/udhcpc/default.script", X_OK)) {
// dbg_time("Fail to access /usr/share/udhcpc/default.script, errno: %d (%s)", errno, strerror(errno));
//}
//-f,--foreground Run in foreground
//-b,--background Background if lease is not obtained
//-n,--now Exit if lease is not obtained
//-q,--quit Exit after obtaining lease
//-t,--retries N Send up to N discover packets (default 3)
// 注释定义的获取IP命令
//snprintf(udhcpc_cmd, sizeof(udhcpc_cmd), "busybox udhcpc -f -n -q -t 5 -i %s", ifname);
#endif
// 注释掉命令线程
//pthread_create(&udhcpc_thread_id, &udhcpc_thread_attr, udhcpc_thread_function, (void*)strdup(udhcpc_cmd));
sleep(1);
}
执行如下命令编译拨号工具
$ make # 编译
$ ./MeiG-CM & # 后台执行拨号工具
如果拨号成功,可以看到对应的网卡会获取到IP地址,并可以正常上网。
可以使用systemctl
管理拨号工具,新建一个文件MeiG-CM.service
,并写入如下内容:
[Unit]
Description=MeiG-CM Service
After=network.target
[Service]
Type=simple
User=root
Restart=always
RestartSec=5s
ExecStart=/home/MeiG-CM/MeiG-CM # 这里更改为对应的可执行文件所在路径
[Install]
WantedBy=multi-user.target
执行以下命令可配置服务并设置开机自启动:
$ cp MeiG-CM.service /usr/lib/systemd/system/ # 拷贝服务文件到系统目录
$ systemctl daemon-reload # 重新检测加载服务,使其被系统识别到
$ systemctl start MeiG-CM.service # 手动启动服务
$ systemctl enable MeiG-CM.service # 配置开机自启动
至此,在Centos 7.6上就可以自动配置MeiG SLM50 4G模块上网。如果卡被停用后再启用,也不需要重新启动机器,会自动重新拨号。