Hi3516如何连接Wifi(二)

系统
文章由鸿蒙社区产出,想要了解更多内容请前往:51CTO和华为官方战略合作共建的鸿蒙技术社区https://harmonyos.51cto.com

[[390308]]

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

书承上回(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框架向内核态发送消息了,这里就不再赘述。

  1. static void* ThreadMain() 
  2.     printf("[WpaSample]init wpa_supplicant.\n"); 
  3.  
  4.     void *handleLibWpa = dlopen("/usr/lib/libwpa.so", RTLD_NOW | RTLD_LOCAL); 
  5.     if (handleLibWpa == NULL) { 
  6.         printf("[WpaSample]dlopen libwpa failed.\n"); 
  7.         return NULL
  8.     } 
  9.     int (*func)(intchar **) = NULL
  10.     func =  dlsym(handleLibWpa, "wpa_main"); 
  11.     if (func == NULL) { 
  12.         dlclose(handleLibWpa); 
  13.         printf("[WpaSample]dlsym wpa_main failed.\n"); 
  14.         return NULL
  15.     } 
  16.     int ret = func(g_wpaArgc, g_wpaArg); 
  17.  
  18.     printf("[WpaSample]run wpa_main failed, ret:%d.\n", ret); 
  19.     for (int i = 0; i < g_wpaArgc; i++) { 
  20.         printf("[WpaSample]arg %d:%s.\n", i, g_wpaArg[i]); 
  21.     } 
  22.  
  23.     if (dlclose(handleLibWpa) != 0) { 
  24.         printf("[WpaSample]dlclose libwpa failed.\n"); 
  25.         return NULL
  26.     } 
  27.     return NULL
  28.  
  29. int main(int argc, char *argv[]) 
  30.     g_wpaArgc = argc; 
  31.     for (int i = 0; i < g_wpaArgc; i++) { 
  32.         g_wpaArg[i] = argv[i]; 
  33.     } 
  34.  
  35.     int ret = pthread_create(&g_wpaThread, NULL, ThreadMain, NULL); 
  36.     if (ret != 0) { 
  37.         printf("[WpaSample]create thread failed error:%s.\n", strerror(ret)); 
  38.         return 1; 
  39.     } 
  40.     pthread_join(g_wpaThread, NULL); 
  41.     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的文件:

  1. copy("config2") { 
  2.     sources = [ 
  3.         "config/wpa_supplicant_(xxx).conf" 
  4.     ] 
  5.     outputs = [ 
  6.         "$root_out_dir/etc/wpa_supplicant_(xxx).conf" 
  7.     ] 

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

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

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

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

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

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

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

  1. OHOS # ls /dev 
  2. Directory /dev: 
  3. (略) 
  4. -rw-rw-r-- 0        u:0     g:99    hdfwifi 
  5. (略) 

其他用户是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

  1.     "jobs" : [{ 
  2.             "name" : "pre-init"
  3.             "cmds" : [ 
  4.                 "mkdir /storage/data/log"
  5.                 (略) 
  6.             ] 
  7.         }, { 
  8.             "name" : "init"
  9.             "cmds" : [ 
  10.                 "start shell"
  11.                 (略) 
  12.             ] 
  13.         }, { 
  14.             "name" : "post-init"
  15.             "cmds" : [ 
  16.                 "chown 0 99 /dev/dev_mgr"
  17.                 "chown 0 99 /dev/hdfwifi"
  18.                 "chmod 0666 /dev/hdfwifi",//这里 

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

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

 

责任编辑:jianghua 来源: 鸿蒙社区
相关推荐

2021-04-09 09:45:21

鸿蒙HarmonyOS应用

2021-03-16 09:49:16

鸿蒙HarmonyOS应用

2021-07-09 14:20:23

鸿蒙HarmonyOS应用

2021-11-09 15:28:41

鸿蒙HarmonyOS应用

2021-05-25 14:47:43

鸿蒙HarmonyOS应用

2022-04-15 14:45:49

Hi3516系统类型烧录鸿蒙

2021-12-03 09:50:39

鸿蒙HarmonyOS应用

2021-06-25 09:28:46

鸿蒙HarmonyOS应用

2021-09-24 10:20:42

鸿蒙HarmonyOS应用

2021-07-21 09:58:50

鸿蒙HarmonyOS应用

2021-03-02 14:30:20

鸿蒙HarmonyOS应用

2021-10-09 10:12:39

鸿蒙HarmonyOS应用

2021-07-19 15:34:05

鸿蒙HarmonyOS应用

2022-02-16 16:01:02

Hi3516开发板鸿蒙

2020-10-16 09:50:37

Hi3861WiFi热点

2021-07-07 09:45:20

鸿蒙HarmonyOS应用

2021-08-06 15:09:22

鸿蒙HarmonyOS应用

2021-07-08 16:16:59

鸿蒙HarmonyOS应用

2022-03-14 15:26:59

Hi3516Ark子系统鸿蒙

2021-07-05 09:35:36

鸿蒙HarmonyOS应用
点赞
收藏

51CTO技术栈公众号