tcp与串口透传(select)
介绍
tcp作为服务端,监听端口8888,实现串口透传,这里是使用select监听tcp的receive和串口的read,单工通信
- -p 指定tcp端口
- -s 指定串口
- -b 指定波特率 支持4800 9600 115200
- -? 打印具体用法
程序
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h>
#include <linux/serial.h>
#include <sys/ioctl.h>
#include <sys/select.h>
#include <unistd.h>
#include <termios.h>
void Test_err(int flg)
{
if(flg<0)
{
perror(strerror(errno));
exit(1);
}
}
int main(int argc, char** argv)
{
printf("usage : %s ?\n",argv[0]);
short listenPort=8888;
unsigned char uart_port[100]={"/dev/ttyS2"};
int bps=0;
unsigned int baudRate= B9600;
int opt;
while ((opt=getopt(argc,argv,"p:s:b:?")) != -1)
{
switch (opt)
{
case 'p':
listenPort = atoi(optarg);
break;
case 's':
memset(uart_port,0,sizeof(uart_port));
strncpy(uart_port,optarg,99);
break;
case 'b':
bps = atoi(optarg);
break;
default: /* '?' */
fprintf(stderr, "Usage: %s [-p port -s serial port -b baund] \n",
argv[0]);
exit(EXIT_FAILURE);
}
}
switch(bps)
{
case 4800:
baudRate=B4800;
break;
case 19200:
baudRate=B19200;
break;
case 38400:
baudRate=B38400;
break;
case 57600:
baudRate=B57600;
break;
case 115200:
baudRate=B115200;
break;
default:
baudRate=B9600;
bps=9600;
break;
}
printf("tcp-server...enter\n");
printf("set uartport is %d\n",bps);
printf("set uartspeed is %s\n",uart_port);
printf("set tcpport is %d\n",listenPort);
// uart--------------------------------------------------------
int uart_fd=open(uart_port, O_RDWR | O_NOCTTY | O_NDELAY);
struct termios newtio;
memset(&newtio,0, sizeof( newtio ));
newtio.c_cflag=0;
newtio.c_iflag=0;
newtio.c_cflag |= CLOCAL | CREAD;
newtio.c_cflag &= ~CSIZE;
newtio.c_cflag |= CS8;
newtio.c_cflag &= ~PARENB;
newtio.c_cflag &= ~CSTOPB;
newtio.c_cc[VTIME] = 0;
newtio.c_cc[VMIN] = 0;
cfsetispeed(&newtio, baudRate);
cfsetispeed(&newtio, baudRate);
tcflush(uart_fd,TCIFLUSH);
if((tcsetattr(uart_fd,TCSANOW,&newtio))!=0)
{
perror("uart open error\n");
return -1;
}
int portfd;
#if (__GNUC__ == 4 && __GNUC_MINOR__ == 3)
struct my_serial_rs485 rs485conf;
struct my_serial_rs485 rs485conf_bak;
#else
struct serial_rs485 rs485conf;
struct serial_rs485 rs485conf_bak;
#endif
portfd=uart_fd;
/* Driver-specific ioctls: ...\linux-3.10.x\include\uapi\asm-generic\ioctls.h */
#define TIOCGRS485 0x542E
#define TIOCSRS485 0x542F
if (ioctl (portfd, TIOCGRS485, &rs485conf) < 0)
{
/* Error handling.*/
// Log::Inf("ioctl TIOCGRS485 error.\n");
printf( "[ERR] ioctl TIOCGRS485 error.\n");
}
/* Enable RS485 mode: */
rs485conf.flags |= SER_RS485_ENABLED;
/* Set logical level for RTS pin equal to 1 when sending: */
rs485conf.flags |= SER_RS485_RTS_ON_SEND;
//rs485conf.flags |= SER_RS485_RTS_AFTER_SEND;
/* set logical level for RTS pin equal to 0 after sending: */
rs485conf.flags &= ~(SER_RS485_RTS_AFTER_SEND);
//rs485conf.flags &= ~(SER_RS485_RTS_ON_SEND);
/* Set rts delay after send, if needed: */
rs485conf.delay_rts_after_send = 0x80;
if (ioctl (portfd, TIOCSRS485, &rs485conf) < 0)
{
/* Error handling.*/
printf("ioctl TIOCSRS485 error.\n");
}
if (ioctl (portfd, TIOCGRS485, &rs485conf_bak) < 0)
{
/* Error handling.*/
printf("ioctl TIOCGRS485 error.\n");
}
/*----------------------------------------------------*/
unsigned char test_msg[]="this is send by uart\n";
int uart_sent_cnt = write(uart_fd, test_msg, sizeof(test_msg));
// tcp--------------------------------------------------------
int listen_fd=-1;
int accept_fd=-1;
int flags;
int max_listen_num=10;
unsigned char r_buf[255];
int n;
struct sockaddr_in server_addr;
struct sockaddr_in clinet_addr;
listen_fd=socket(AF_INET, SOCK_STREAM, 0);
Test_err(listen_fd);
memset(&server_addr,0,sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
server_addr.sin_port = htons(listenPort);
int i = 1;/* 允许重复使用本地地址与套接字进行绑定 */
setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &i, sizeof(i));
flags=bind(listen_fd, (struct sockaddr *)&server_addr,sizeof(struct sockaddr));
Test_err(flags);
flags= listen(listen_fd, max_listen_num);
Test_err(flags);
int clinet_addr_len=sizeof(clinet_addr);
__ACCEPT_AGAIN__:
accept_fd= accept(listen_fd, (struct sockaddr *)&clinet_addr, &clinet_addr_len);
// show clinet message
printf("accept_fd is %d \n",accept_fd);
printf("port=%d\n",ntohs(clinet_addr.sin_port));
unsigned char strbuf[20];
inet_ntop(AF_INET,&clinet_addr.sin_addr,strbuf,sizeof(strbuf));
printf("ip=%s\n",strbuf);
memset(r_buf , 0, sizeof(r_buf));
int maxfd;
fd_set rset, allset;
struct timeval tv;
FD_ZERO(&allset);
FD_SET(uart_fd, &allset);
FD_SET(accept_fd, &allset);
memset(&tv, 0, sizeof(tv));
tv.tv_usec = 1000 * 1000;
int len=0;
while (accept_fd)
{
rset=allset;
maxfd=(accept_fd>uart_fd?accept_fd:uart_fd);
flags=select(maxfd + 1, &rset, NULL, NULL, &tv);
Test_err(flags);
// get from tcp>>>>>uart
if (FD_ISSET(accept_fd, &rset))
{
n = recv(accept_fd, r_buf, 100, 0);
Test_err(n);
if(n)
{
//write(0,r_buf,n);
printf("Get from tcp %d\n",n);
write(uart_fd,r_buf,n);
}
else
{
printf("socket is disconnect\n");
goto __ACCEPT_AGAIN__;
}
}
// get from uart>>>>tcp
else if (FD_ISSET(uart_fd, &rset))
{
usleep(100000);//100ms 这里等待接受多一点的数据
int ret = read(uart_fd, (char*)r_buf,sizeof(r_buf));
if (ret < 0)
{
if ((errno == EAGAIN) ||
(errno == EWOULDBLOCK) ||
(errno == EINTR))
{
printf("Serial,read timeout error:%s. ignore ti!!!", strerror(errno));
ret = 0;
}
}
else
{
printf("Get from uart %d\n",ret);
send(accept_fd, r_buf, ret, 0);
}
}
}
printf("tcp-server...exit\n");
return 0;
}
tcp与串口透传(select)的更多相关文章
- CC1101 433无线模块,STM8串口透传
CC1101 433无线模块,STM8串口透传 原理图:http://download.csdn.net/detail/cp1300/7496509 下面是STM8程序 CC1101.C /*** ...
- ESP8266局域网 路由器下作服务器模式串口透传 arduino uno示例 模板参考2
ESP8266服务器模式串口透传编译需要下载8266的库文件后才可以正常 准备工作 下载一个Arduino IDE,下载8266的库文件ESP8266服务器模式串口透传编译 功能说明 1.直接使用路由 ...
- ESP8266局域网智能家居 路由器下作服务器模式串口透传 无线通信控制 arduino uno示例 模板参考
准备工作 下载一个Arduino IDE, 下载8266的库文件 ESP8266服务器模式串口透传编译 功能说明 1.直接使用路由器中转数据 2.手机放热点模式直接传输数据 两者有访问IP地址的差别, ...
- 18-ESP8266 SDK开发基础入门篇--TCP 服务器 RTOS版,串口透传,TCP客户端控制LED
https://www.cnblogs.com/yangfengwu/p/11112015.html 先规定一下协议 aa 55 02 01 F1 4C 控制LED点亮 F1 4C为CRC高位和低位 ...
- 16-ESP8266 SDK开发基础入门篇--TCP 服务器 非RTOS运行版,串口透传(串口回调函数处理版)
https://www.cnblogs.com/yangfengwu/p/11105466.html 其实官方给的RTOS的版本就是在原先非RTOS版本上增加的 https://www.cnblogs ...
- PHP workerMan tcp与webSocket 透传互通
<?php $work_path = dirname(__FILE__); chdir($work_path); use \Workerman\Worker; use \Workerman\Li ...
- wifi实现串口透传
环境: 串口服务器Ip:172.16.1.11 串口客户机ip:172.16.1.12 一.网络连接示意图 二.串口服务器的配置 参考:ser2net的编译及测试 三.串口客户端的配置 实际上这是一个 ...
- python 串口 透传
python正常情况通过串口 serial 传输数据的时候,都是以字符串的形式发送的 str = ‘abcd’ ser.write(str.encode())#直接发送str报错,需要发送byte类 ...
- 蓝牙4.0BLE 手机控制 cc2540 CC2541 的串口透传功能已实现
蓝牙4.0BLE 手机控制 cc2540 CC2541 的串口透传功能已实现 尽管蓝牙4.0 BLE芯片CC2540 是单芯片(即用户能够对它进行芯片级代码编写), 是80 ...
随机推荐
- java+批量下载文件到指定文件夹
需求 导出文件后存留在了服务器中,需要提供下载按钮,点击后可下载到本地:(因为涉及多个文件,下载前先将文件进行压缩,提供下载压缩文件) 效果预览 代码 主要方法 /** * 下载生成的所有在线 ...
- CF917D Stranger Trees【矩阵树定理,高斯消元】
题目链接:洛谷 题目大意:给定一个$n$个节点的树$T$,令$ans_k=\sum_{T'}[|T\cap T'|=k]$,即有$k$条边重合.输出$ans_0,ans_1,\ldots,ans_{n ...
- vue中使用echarts画饼状图
echarts的中文文档地址:https://echarts.baidu.com/tutorial.html#5%20%E5%88%86%E9%92%9F%E4%B8%8A%E6%89%8B%20EC ...
- MySQL数据分析(7)-SQL的两大学习框架
大家好,我是jacky,很高兴继续跟大家分享<MySQL数据分析实战>课程,前面的课程基本上我把MySQL的原理都做了一定的介绍,有好多朋友说学习MySQL是没有逻辑的,其实jacky是非 ...
- Python学习日记(三)——Python基本数据类型(运算符、int、str、tuple、dict、range)
运算符 1.算数运算 2.比较运算 3.赋值运算 4.逻辑运算 5.成员运算 基本数据类型 1.数字 int(整型) 在32位机器上,整数的位数为32位,取值范围为-2**31-2**31-1,即-2 ...
- Linux下使用 Nginx
1. 下载Nginx 下载地址 下载完成后,上传到linux服务器,我上传到了 /opt文件夹下. 2. 安装Nginx前的准备 要想使用Nginx需要安装PCRE库和zlib库,否则直接安装Ngin ...
- 安装windows下安装mysql
参考文档:https://www.cnblogs.com/reyinever/p/8551977.html https://www.jb51.net/article/151213.htm 首先下载m ...
- Oracle中shrink space命令
shrink_clause: http://docs.oracle.com/cd/B19306_01/server.102/b14200/statements_3001.htm#i2192484 ...
- Linux系统下查找最近修改过的文件
Linux的终端上,没有windows的搜索那样好用的图形界面工具,但find命令确是很强大的. 比如按名字查找一个文件,可以用 find / -name targetfilename . 唉,如果只 ...
- LightGBM两种使用方式
原生形式使用lightgbm(import lightgbm as lgb) import lightgbm as lgb from sklearn.metrics import mean_squar ...