WIFI模块开发教程之W600网络篇2:AP模式下TCP Server通信

前言

本文研究如何在AP模式下进行TCP Server通信,所谓AP模式是说模块起来一个softAP热点,可以供其他WIFI设备连接,当其他设备连接成功后,另WIFI模块作为服务端,等待局域网中其他客户端连接后通信。

一、 理论基础

本节要处理的有两个问题,其一是如何利用RT_Thread连接路由器,其二是如何使用Socket套接字编程搞定TCP Server程序编写。

1.连接路由器

模块需要开启station,并且连接到一个路由器,RT_Thread中只需要调用wlan.mgnt.h中的函数即可。

rt_wlan_set_mode(RT_WLAN_DEVICE_AP_NAME, RT_WLAN_AP);
rt_wlan_start_ap("sand", "12345678");

2.Socket套接字实现TCP Server

服务端套接字使用流程以及和客户端交互框图如下所示:

二、使用实例

1.程序

w60x/applications/4-ap_tcp_server/main.c

#include <rtthread.h>
#include <rtdevice.h>
#include <sys/socket.h> //使用BSD socket需要包含此头文件

#define TCP_SERVER_ADDR "192.168.169.1"
#define TCP_SERVER_PORT 8089

static void tcp_client_thread_entry(void *args)
{
    int ret = 0;
    int fd = (int) args;
    int len = 0;
    fd_set readfds;
    char buf[512] = { 0x00 };
    struct timeval t;

    t.tv_sec = 5;
    t.tv_usec = 0;

    while (1)
    {
        FD_ZERO( &readfds );
        FD_SET( fd, &readfds );
        ret = select( fd+1, &readfds, NULL, NULL, &t);
        if ((ret > 0) && FD_ISSET(fd, &readfds))
        {
            len = recv(fd, buf, sizeof(buf), 0);
            if (len > 0)
            {
                buf[len] = 0x00;
                rt_kprintf("receive data:%s\r\n", buf);
            }else
            {
                rt_kprintf("receive data from tcp server error!\r\n");
                goto exit;
            }

            if (-1 != fd)
            {
                rt_sprintf(buf, "%s\r\n", buf);
                ret = send(fd, buf, strlen(buf), 0);
                if (ret < 0)
                {
                    rt_kprintf("send error, closee socket");
                    goto exit;
                }
            }
        }
    }

    exit:
    if (-1 != fd)
    {
        closesocket( &fd );
        fd = -1;
    }
}

static void tcp_server_thread_entry(void *args)
{
    int ret = 0;
    int fd = -1, client_fd = -1;
    struct sockaddr_in server_addr, client_addr;
    struct  timeval t;
    fd_set readfds;
    socklen_t sockaddr_t_size;

    char buf[512] = { 0x00 };
    int len = 0;

reconnect:
    fd = socket(AF_INET, SOCK_STREAM, 0);
    if (-1 == fd)
    {
        rt_kprintf("create socket error!!!\r\n");
        goto exit;
    }
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(TCP_SERVER_PORT);
    server_addr.sin_addr.s_addr = INADDR_ANY;
    rt_memset(&server_addr.sin_zero, 0x00, sizeof(server_addr.sin_zero));

    ret = bind(fd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr ));
    if (0 == ret)
    {
        rt_kprintf("bind success\r\n");
    }else
    {
        rt_kprintf("connect error!!!\r\n");
        goto try_reconnect;
    }

    ret = listen(fd, 5);
    if (-1 == ret)
    {
        rt_kprintf("Listen error\r\n");
        goto try_reconnect;
    }

    t.tv_sec = 2;
    t.tv_usec = 0;

    while (1)
    {
        FD_ZERO(&readfds);
        FD_SET(fd, &readfds);

        ret = select(fd + 1, &readfds, 0, 0, &t);
        if (-1 == ret)
        {
            rt_kprintf("select() error!\r\n");
            goto try_reconnect;
        }
        // else if(0 == ret)
        // {
        //     rt_kprintf("select() timeout!\r\n");
        // }
        else if(ret > 0)
        {
            if (FD_ISSET(fd, &readfds));
            {

                client_fd = accept(fd, (struct sockaddr *)&client_addr, &sockaddr_t_size);
                if (client_fd < 0)
                {
                    rt_kprintf("accept connection failed! errno = %d\n", errno);
                    continue;
                }
                rt_kprintf("client connected, ip:%s, port:%d", inet_ntoa(client_addr), ntohs(client_addr.sin_port));

                //create client thread
                rt_thread_t uart_thread = rt_thread_create("clients", tcp_client_thread_entry, client_fd, 4*1024, 25, 10);
                if (uart_thread != NULL)
                {
                    rt_thread_startup(uart_thread);
                }else
                {
                    ret = RT_ERROR;
                    rt_kprintf("create tcp client error!!!");
                    closesocket(fd);
                }
            }
        }
    }
try_reconnect:
    if (-1 != fd)
    {
        closesocket(fd);
    }
    rt_thread_sleep(1);
    goto reconnect;

exit:
    if (-1 != fd)
    {
        closesocket(fd);
    }
    rt_kprintf("thread server exit!\r\n");
}

int main(void)
{
    rt_err_t ret = RT_EOK;
    char str[] = "hello world!\r\n";

    // create ap
    rt_wlan_set_mode(RT_WLAN_DEVICE_AP_NAME, RT_WLAN_AP);
    rt_wlan_start_ap("sand", "12345678");

    //create server
    rt_thread_t uart_thread = rt_thread_create("server", tcp_server_thread_entry, RT_NULL, 4*1024, 25, 10);
    if (uart_thread != NULL)
    {
        rt_thread_startup(uart_thread);
    }else
    {
        ret = RT_ERROR;
        rt_kprintf("create tcp client error!!!");
    }

exit:
    return ret;
}

2.目标文件配置

前面几个事例都是修改了RT_Thread默认路径下w60x/applications/main.c下的文件,然后使用scons编译,考虑到代码的开源,之后会统一放到github上,调整代码为如下所示:

在main.c外增加一个文件夹,直接输入scons会提示编译错误,此时需要修改aplications/SConscript脚本。

我们先看下默认的SConscript脚本内容:

Import(‘RTT_ROOT‘)
Import(‘rtconfig‘)
from building import *

cwd = GetCurrentDir()
src  = Glob(‘*.c‘)
CPPPATH = [cwd]

group = DefineGroup(‘Applications‘, src, depend = [‘‘], CPPPATH = CPPPATH)

Return(‘group‘)

现在要编译4-ap_tcp_server中的main.c,需修改SConscript中第6行代码为src = Glob(‘4-ap_tcp_server/main.c‘),之前为*.c 表示applications文件目录下所有.c文件,修改后编译的时候仅编译4-ap_tcp_server/main.c,不编译其他文件内容,修改后SConscript内容如下所示:

Import(‘RTT_ROOT‘)
Import(‘rtconfig‘)
from building import *

cwd = GetCurrentDir()
src  = Glob(‘4-ap_tcp_server/main.c‘)
CPPPATH = [cwd]

group = DefineGroup(‘Applications‘, src, depend = [‘‘], CPPPATH = CPPPATH)

Return(‘group‘)

三、下载运行

在ENV控制台,输入scons命令,在build/Bin目录下生成rtthread_1M.FLS,

烧录运行后,电脑连接模块起来的热点,然后打开电脑网络调试助手,开启两个个TCP client,指定TCP Server的IP均为192.168.169.1,端口为8089,通过网络助手发送数据给模块,,模块收到后,会加上"\r\n",然后返回给网络助手。

网络助手界面如下:

模块调试串口信息如下:

上午图片表示,网络助手1发送device1 say hello给模块,网络助手2发送device2 say hello给模块,模块分别回应两者对应的消息。

四、结语

本节完,实际操作过程中需要注意的地方有如下几点:

(1) 注意模块每次accept到一个客户端连接后,都会创建一个相同的线程,处理和当前客户端之后的数据交互问题,所以本demo支持一个模块的TCP Server和多个TCP Client进行数据交互

(2) 设备调试串口信息中有一条:receive data from tcp server error!,这个是由于手动关闭了一个TCP Client,随后重新开启,模块日志检测到client connected,说明操作正常

(3)资料获取

如您在使用过程中有任何问题,请加QQ群进一步交流

QQ交流群:906015840 (备注:物联网项目交流)

获取源码:关注下方公众号,回复w600即可

一叶孤沙出品:一沙一世界,一叶一菩提

原文地址:https://blog.51cto.com/14616151/2484655

时间: 2024-10-28 23:52:03

WIFI模块开发教程之W600网络篇2:AP模式下TCP Server通信的相关文章

WIFI模块开发教程之W600网络篇1:AP模式下TCP Client通信

前言 本文研究如何在AP模式下进行TCP Client通信,所谓AP模式是说模块起来一个softAP热点,可以供其他WIFI设备连接,当其他设备连接成功后,另WIFI模块作为客户端,局域网中其他设备作为服务端进行TCP数据通信. 一.理论基础 本节要处理的有两个问题,其一是如何利用RT_Thread起来一个softAP,其二是如何使用Socket套接字编程搞定TCP Client程序编写. 1.模块开启SoftAP 模块需要起来一个名字为sand,密码为12345678的热点,RT_Thread

# IT明星不是梦 # WIFI模块开发教程之W600连云篇1:onenet三色灯项目小程序篇②

前言 本文研究如何使用小程序连接云平台,进而控制上一节中连接到onenet云平台的RGB三色灯设备. 一.理论基础 1.先睹为快 视频地址:https://www.ixigua.com/i6806304062091297292/ 2.功能点简述 小程序开机页面 小程序界面布局 小程序访问onenet接口 小程序3秒刷新数据状态 小程序控制弹出编辑框设置房间号 小程序调用storage接口存取数据 二.使用实例 1.云端创建产品 接下来咱们进入正题,首先需要在云端创建一个产品,步骤如下: 创建产品

Xamarin Anroid开发教程之Anroid开发工具及应用介绍

Xamarin Anroid开发教程之Anroid开发工具及应用介绍 Xamarin开发Anroid应用介绍 如今智能手机已经盛行了好几年,而针对这些智能手机的软件开发也变得异常火热.但是在Android平台下只能使用Java开发,iOS平台下也只能使用Objective-C或Swift开发.对于那些C#程序员却只能眼看着这些火热的平台流口水.在几年前只能重新学习一门新的语言,才可以进入这些智能手机开发行业.现在,我们有了Xamarin.C#程序员不需要了解其它语言,就可以开始编写Android

Tkinter教程之Text(2)篇

本文转载自:http://blog.csdn.net/jcodeer/article/details/1811347 '''Tkinter教程之Text(2)篇''''''6.使用tag来指定文本的属性'''#创建一个指定背景颜色的TAG# -*- coding: cp936 -*-from Tkinter import *root = Tk()t = Text(root)# 创建一个TAG,其前景色为红色t.tag_config('a',foreground = 'red')# 使用TAG '

?Unity 2D游戏开发教程之2D游戏的运行效果

Unity 2D游戏开发教程之2D游戏的运行效果 2D游戏的运行效果 本章前前后后使用了很多节的篇幅,到底实现了怎样的一个游戏运行效果呢?或者说,游戏中的精灵会不会如我们所想的那样运行呢?关于这些疑问,会在本节集中揭晓. (1)单击Unity上方,工具栏里的播放按钮,开始运行当前的游戏,默认精灵当前进入的是Idle动画状态,如图1-34所示. 图1-34  Idle状态 (2)当读者按下键盘上的左.右方向键,或者A.D键的时候,精灵会进入Walking动画状态,并且会向左或者向右移动,如图1-3

AppleWatch开发教程之Watch应用对象新增内容介绍以及编写运行代码

AppleWatch开发教程之Watch应用对象新增内容介绍以及编写运行代码 添加Watch应用对象时新增内容介绍 Watch应用对象添加到创建的项目中后,会包含两个部分:Watch App 和 WatchKit Extension,如图2.18所示.其中,Watch App部分位于用户的iWatch上,它目前为止只允许包含Storyboard文件和Resources文件.在我们的项目里,这一部分不包括任何代码.WatchKit Extension部分位于用户的iPhone安装的对应App上,这

iOS 9应用开发教程之ios9中实现button的响应

iOS 9应用开发教程之ios9中实现button的响应 IOS9实现button的响应 button主要是实现用户交互的.即实现响应.button实现响应的方式能够依据加入button的不同分为两种:一种是编辑界面加入button实现的响应:还有一种是使用代码加入button实现的响应. 1.编辑界面加入button实现的响应 使用编辑界面加入button能够使用拖动的方式来实现button的响应,它也是最简单的一种实现响应的方式. [演示样例2-4]下面将实现轻拍button,改变主视图背景

HealthKit开发教程之HealthKit的辅助数据

HealthKit开发教程之HealthKit的辅助数据 在HealthKit中除了主要数据之外,还有6个辅助数据分别为:体积类型数据.压力类型数据.时间类型数据.温度类型数据.标量类型数据和电导率类型数据.本节将针对这6个辅助数据进行讲解. HealthKit的体积类型数据 体积,或称容量.容积,几何学专业术语,是物件占有多少空间的量.体积单位可以用来测量物体的体积.由体积单位和值组成的数据就是体积类型的数据.体积单位可以分为:国际体积单位.英制体积单位和美制体积单位. 1.国际体积单位 体积

iOS 9应用开发教程之ios9的视图

iOS 9应用开发教程之ios9的视图 了解IOS9的视图 在iPhone或者iPad中,用户看到的和摸到的都是视图.视图是用户界面的重要组成元素.本节将主要讲解ios9视图的添加.删除以及位置和大小的设置等内容. ios9视图库介绍 在视图库中存放了iOS开发中所需的所有视图.开发者可以在创建好iOS应用程序的项目以后,打开画布的设计界面,这时在工具窗口的下半个窗口中,单击Show the Object library图标,就会显示出视图库,如图1.46所示. 图1.46  视图库 在视图库中