目录:

一、总体思路

二、启动Daemon

三、作者文章合集

书承上回(Hi3516如何连接Wifi(一)),上一篇聊了一下怎样在Hi3516中用wpa_supplicant连接到Wifi热点,本文讲一下如何通过编程实现。

一、总体思路

首先我们需要搞清楚Hi3516中Wifi的相关模块,以及他们之间的关系,其实和linux是很相似的。首先,我们需要运行一个的Daemon,也就是上文提到的wpa_supplicant,负责对网卡的硬件调用,比如连接wifi、断开wifi、启动热点等等。这个Daemon开放一个socket端口,外部程序可以通过本地连接向其发送指令实现间接对wifi的调用,这无疑是给我们提供了很大的便利,不用从底层重新造轮子了。

鸿蒙OS代码中,有一个示例,在//applications/sample/camera/communication/wpa_cli,实现了连接Daemon、扫描热点、连接热点等功能。

现在方案就很明确了,第一启动Daemon,第二向Daemon发送命令。下面我们就来详细分析如何实现。

二、启动Daemon

查看代码//applications/sample/camera/communication/wpa_supplicant/src/wpa_sample.c,找到main函数,发现它只做了一件事情,那就是调用pthread_create创建了一个线程,线程执行的函数是ThreadMain。而ThreadMain也只做了一件事情,那就是加载/usr/lib/libwpa.so,然后执行了其中的wpa_main函数,同时把命令行参数传递了进去。而wpa_main函数具体调用网卡就是通过hdf框架向内核态发送消息了,这里就不再赘述。

static void* ThreadMain()
{
printf("[WpaSample]init wpa_supplicant.\n"); void *handleLibWpa = dlopen("/usr/lib/libwpa.so", RTLD_NOW | RTLD_LOCAL);
if (handleLibWpa == NULL) {
printf("[WpaSample]dlopen libwpa failed.\n");
return NULL;
}
int (*func)(int, char **) = NULL;
func = dlsym(handleLibWpa, "wpa_main");
if (func == NULL) {
dlclose(handleLibWpa);
printf("[WpaSample]dlsym wpa_main failed.\n");
return NULL;
}
int ret = func(g_wpaArgc, g_wpaArg); printf("[WpaSample]run wpa_main failed, ret:%d.\n", ret);
for (int i = 0; i < g_wpaArgc; i++) {
printf("[WpaSample]arg %d:%s.\n", i, g_wpaArg[i]);
} if (dlclose(handleLibWpa) != 0) {
printf("[WpaSample]dlclose libwpa failed.\n");
return NULL;
}
return NULL;
} int main(int argc, char *argv[])
{
g_wpaArgc = argc;
for (int i = 0; i < g_wpaArgc; i++) {
g_wpaArg[i] = argv[i];
} int ret = pthread_create(&g_wpaThread, NULL, ThreadMain, NULL);
if (ret != 0) {
printf("[WpaSample]create thread failed error:%s.\n", strerror(ret));
return 1;
}
pthread_join(g_wpaThread, NULL);
return 0;
}

我们要做的就是仿照main函数写自己的代码,把参数固定就可以了。我们的参数是这样的:g_wpaArg[0]="",g_wpaArg[1]="-iwlan0",g_wpaArg[2]="-c/etc/wpa_supplicant.conf",其中第0个参数是可执行文件的名称,这里可以随意填或者直接留空。

如果只是想启动Daemon,不连接到任何Wifi热点,那这里第二个参数-c指向的.conf文件中,不应该包含ssid和psk,也就是直接使用系统自带的默认conf就可以。我们在上一篇文章中修改了wpa_supplicant.conf,加入了ssid和psk。这里做一个改进,原wpa_supplicant.conf保持不变,新增一个wpa_supplicant_(你的热点名称).conf,加入ssid和psk,然后修改//applications/sample/camera/communication/wpa_supplicant/BUILD.gn,添加需要copy的文件:

copy("config2") {
sources = [
"config/wpa_supplicant_(xxx).conf"
]
outputs = [
"$root_out_dir/etc/wpa_supplicant_(xxx).conf"
]
}

这样我们wpa_supplicant就有两种操作了,只启动Daemon,和启动Daemon且连接到指定热点,只需要改变-c指定的conf文件。

补充一下,我曾尝试过使用system函数执行wpa_supplicant的方式启动Daemon,但是失败了,原因是鸿蒙暂时还不支持system函数。具体可以看一下system的代码实现\\third_party\musl\src\process\system.c

int system(const char *cmd)
{
pid_t pid;
sigset_t old, reset;
struct sigaction sa = { .sa_handler = SIG_IGN }, oldint, oldquit;
int status = -1, ret;
posix_spawnattr_t attr; unsupported_api(__FUNCTION__);//不受支持的api
...

另外,尝试了用fork创建线程也是可行的。

还有一个文件权限问题。如果你用上述方法编写一个控制台程序来运行是没有问题的,无非就是重写了一个sample里的wpa_supplicant。但是当你在hap中通过ace调用时就出现了错误:

OHOS # 01-01 00:40:03.661 17 59 I 03900/ACE: InitWifi invoked!
[WpaSample]init wpa_supplicant.
01-01 00:40:03.661 17 59 I 03900/ACE: InitDaemon2
Successfully initialized wpa_supplicant
[HDF:E/hdf_syscall_adapter]Open file node failed: /dev/hdfwifi
[HDF:E/HDF_LOG_TAG]WpaMsgServiceInit: fail to get remote service!

看样子是打开/dev/hdfwifi失败了,这个问题我研究了很久,最后意识到hap的执行用户可能和shell不同,shell是root用户在执行,而hap肯定不是root在执行,这导致了权限不足。我看了一下/dev/hdfwifi的权限

OHOS # ls /dev
Directory /dev:
(略)
-rw-rw-r-- 0 u:0 g:99 hdfwifi
(略)
 

其他用户是r权限,显然我们也需要w权限。执行chmod 0666 /dev/hdfwifi就可以了,但烧写后通过连接shell做这件事很不方便,最好能自动化。

这里我们可以借助鸿蒙系统初始化阶段执行的job来实现我们的目的,在\\base\startup\services\init_lite\src\main.c负责执行系统启动后的任务,包括各种job和service,job分为pre-init,init,post-init三个阶段。具体要执行哪些命令,都写在\\vendor\huawei\camera\init_configs\init_liteos_a_3516dv300.cfg配置文件中,我们要做的就是在job中找到post-init,然后在cmds添加我们的指令chmod 0666 /dev/hdfwifi

{
"jobs" : [{
"name" : "pre-init",
"cmds" : [
"mkdir /storage/data/log",
(略)
]
}, {
"name" : "init",
"cmds" : [
"start shell",
(略)
]
}, {
"name" : "post-init",
"cmds" : [
"chown 0 99 /dev/dev_mgr",
"chown 0 99 /dev/hdfwifi",
"chmod 0666 /dev/hdfwifi",//这里
 

下一篇再将如何连接Daemon,真正实现连接Wifi,以及如何通过ACE在UI界面中操作连接Wifi。

作者:老船夫

想了解更多内容,请访问51CTO和华为合作共建的鸿蒙社区:https://harmonyos.51cto.com

Hi3516如何连接Wifi(二)的更多相关文章

  1. 手机连接wifi自动弹窗的原理及其实现方案

    一.手机连上wifi后会自动弹窗的原理 生活中,有很多需要认证的路由器,手机连接wifi热点后会自动弹出一个网页,让用户输入账号和密码,比如星巴克,肯地基,麦当劳,甚至是火车站和机场的候车室.其实这是 ...

  2. [Windows Phone] 如何撰写连接 Wifi、蓝芽、网路、飞航模式的网路设定功能

    原文:[Windows Phone] 如何撰写连接 Wifi.蓝芽.网路.飞航模式的网路设定功能 前言 为了可以使自己的 APP 具备操作网路的功能,在本文分享研究心得,包含在 Windows Pho ...

  3. 树莓派小用手册(安装系统,配置图形界面,连接WiFi,调用摄像头,安装ffmpeg)

    安装树莓派系统(重装) 准备工作: 安装需要干净的TF卡(最好 8G 以上),如果是重装的话,需要先将其清理后再使用.清理步骤下面会给出,清理需要工具 DiskGenius,下载链接:http://w ...

  4. 树莓派 连接wifi与路由器ip绑定

    先推荐几个手机软件 在酷安网里应该能找到 1.JuiceSSH  橘子ssh软件  手机连上路由器 就可以控制局域网内的树莓派 2.VNC Viewer  远程桌面软件VNC 也是连接局域网的树莓派 ...

  5. 树莓派连接wifi

    使用树莓派,通过无线网卡连接wifi,再通过远程桌面或者ssh的连接树莓派比较方便,本文记录树莓派wifi如何设置. 参考链接: http://www.jianshu.com/p/b42e8d3df4 ...

  6. android开发连接wifi addNetwork 返回-1

    如下代码:(注释掉的是会返回-1的代码,未注释的是能够正常连接wifi的代码) // public WifiConfiguration CreateWifiInfo(String SSID, Stri ...

  7. 为什么手机连接wifi会显示已停用?

    1.通常导致手机连接WiFi显示“已停用”故障的原因是由于无线路由器“安全模式”设置不当造成的,对此我们可以通过以下方法来解决: 2.根据无线路由器背面的信息(包括路由器IP地址,登陆用户名和密码), ...

  8. Android 连接Wifi和创建Wifi热点 demo

    android的热点功能不可见,用了反射的技术搞定之外. Eclipse设置语言为utf-8才能查看中文注释 上代码: MainActivity.java package com.widget.hot ...

  9. C#如何连接wifi和指定IP

    大家好哈,这是我第一次写博客,我也是才大一结束,自学了10多天C#,有不对的欢迎大家指正,最近因为项目的事而被Wifi和IP折磨了很久,后来借用了一下外国人的SimpleWifi.dll 再自己写了一 ...

随机推荐

  1. Iterators & Generators in depth

    Iterators & Generators in depth https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/It ...

  2. how to close macos eject icon from menu bar

    how to close macOS eject icon from the menu bar close eject https://apple.stackexchange.com/question ...

  3. svg-path-to-polygons & svg-path-parser

    svg-path-to-polygons & svg-path-parser svg-path-to-polygons https://www.npmjs.com/package/svg-pa ...

  4. Flutter Demo: PageView横向使用

    video import 'package:flutter/material.dart'; void main() => runApp(MyApp()); class MyApp extends ...

  5. OpenCVE-开源漏洞预警平台

    0x01简介 主程序主要是通过使用NVD提供的JSON数据来更新CVE数据,并在前端进行展示.然后通过邮件进行通知,目前也只支持邮件.这个开源预警平台看上去并不是很完善,因为CVE本身就具有预警滞后性 ...

  6. Java并发包源码学习系列:同步组件CountDownLatch源码解析

    目录 CountDownLatch概述 使用案例与基本思路 类图与基本结构 void await() boolean await(long timeout, TimeUnit unit) void c ...

  7. MySQL 修改数据表

    修改数据表: 创建数据表 更改表明 更改字段数据类型 更改字段名称 更改字段名称和数据类型 为表添加新字段 将字段顺序改为第一位 将字段顺序改为另一个字段之后 删除字段 1 use test; 2 3 ...

  8. 将项目加载到tomcat中的时候报错:Tomcat version 6.0 only supports J2EE 1.2, 1.3, 1.4, and Java EE 5 Web modules

    转自:http://jingwang0523.blog.163.com/blog/static/9090710320113294551497/ 最近在用eclipse做项目,新建项目时什么都贪新,用最 ...

  9. 【Notes_9】现代图形学入门——光线追踪(基本原理)

    跟着闫令琪老师的课程学习,总结自己学习到的知识点 课程网址GAMES101 B站课程地址GAMES101 课程资料百度网盘[提取码:0000] 目录 光线追踪 为什么要光线追踪 soft shadow ...

  10. Java中base64与byte[]转换

    1.base64转byte[] 导包: import java.util.Base64 转换: String base64Str = base64的字符串; byte[] bytes = Base64 ...