3. 编程
PC: open/read/write /dev/ttyACM0
板子: open/read/write /dev/ttyGS0
源码:
#include < stdio.h >
#include < string.h >
#include < sys/types.h >
#include < errno.h >
#include < sys/stat.h >
#include < fcntl.h >
#include < unistd.h >
#include < termios.h >
#include < stdlib.h >
#include < pthread.h >
static struct termios old, current;
/* Initialize new terminal i/o settings */
void initTermios(int echo)
{
tcgetattr(0, &old); /* grab old terminal i/o settings */
current = old; /* make new settings same as old settings */
current.c_lflag &= ~ICANON; /* disable buffered i/o */
if (echo) {
current.c_lflag |= ECHO; /* set echo mode */
} else {
current.c_lflag &= ~ECHO; /* set no echo mode */
}
tcsetattr(0, TCSANOW, ¤t); /* use these new terminal i/o settings now */
}
/* Restore old terminal i/o settings */
void resetTermios(void)
{
tcsetattr(0, TCSANOW, &old);
}
/* set_opt(fd,115200,8,'N',1) */
int set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop)
{
struct termios newtio,oldtio;
if ( tcgetattr( fd,&oldtio) != 0) {
perror("SetupSerial 1");
return -1;
}
bzero( &newtio, sizeof( newtio ) );
newtio.c_cflag |= CLOCAL | CREAD;
newtio.c_cflag &= ~CSIZE;
newtio.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG); /*Input*/
newtio.c_oflag &= ~OPOST; /*Output*/
switch( nBits )
{
case 7:
newtio.c_cflag |= CS7;
break;
case 8:
newtio.c_cflag |= CS8;
break;
}
switch( nEvent )
{
case 'O':
newtio.c_cflag |= PARENB;
newtio.c_cflag |= PARODD;
newtio.c_iflag |= (INPCK | ISTRIP);
break;
case 'E':
newtio.c_iflag |= (INPCK | ISTRIP);
newtio.c_cflag |= PARENB;
newtio.c_cflag &= ~PARODD;
break;
case 'N':
newtio.c_cflag &= ~PARENB;
break;
}
switch( nSpeed )
{
case 2400:
cfsetispeed(&newtio, B2400);
cfsetospeed(&newtio, B2400);
break;
case 4800:
cfsetispeed(&newtio, B4800);
cfsetospeed(&newtio, B4800);
break;
case 9600:
cfsetispeed(&newtio, B9600);
cfsetospeed(&newtio, B9600);
break;
case 115200:
cfsetispeed(&newtio, B115200);
cfsetospeed(&newtio, B115200);
break;
default:
cfsetispeed(&newtio, B9600);
cfsetospeed(&newtio, B9600);
break;
}
if( nStop == 1 )
newtio.c_cflag &= ~CSTOPB;
else if ( nStop == 2 )
newtio.c_cflag |= CSTOPB;
newtio.c_cc[VMIN] = 1; /* 读数据时的最小字节数: 没读到这些数据我就不返回! */
newtio.c_cc[VTIME] = 0; /* 等待第1个数据的时间:
* 比如VMIN设为10表示至少读到10个数据才返回,
* 但是没有数据总不能一直等吧? 可以设置VTIME(单位是10秒)
* 假设VTIME=1,表示:
* 10秒内一个数据都没有的话就返回
* 如果10秒内至少读到了1个字节,那就继续等待,完全读到VMIN个数据再返回
*/
tcflush(fd,TCIFLUSH);
if((tcsetattr(fd,TCSANOW,&newtio))!=0)
{
perror("com set error");
return -1;
}
//printf("set done!n");
return 0;
}
int open_port(char *com)
{
int fd;
//fd = open(com, O_RDWR|O_NOCTTY|O_NDELAY);
fd = open(com, O_RDWR|O_NOCTTY);
if (-1 == fd){
return(-1);
}
if(fcntl(fd, F_SETFL, 0)< 0) /* 设置串口为阻塞状态*/
{
printf("fcntl failed!n");
return -1;
}
return fd;
}
static void *my_read_thread_func(void *data)
{
int fd = (int)data;
int iRet;
char c;
while (1)
{
iRet = read(fd, &c, 1);
printf("%c", c);
fflush(stdout);
}
}
/*
* ./serial_send_recv < dev >
*/
int main(int argc, char **argv)
{
int fd;
int iRet;
char c;
pthread_t tid;
/* 1. open */
/* 2. setup
* 115200,8N1
* RAW mode
* return data immediately
*/
/* 3. write and read */
if (argc != 2)
{
printf("Usage: n");
printf("%s < /dev/ttySAC1 or other >n", argv[0]);
return -1;
}
fd = open_port(argv[1]);
if (fd < 0)
{
printf("open %s err!n", argv[1]);
return -1;
}
iRet = set_opt(fd, 115200, 8, 'N', 1);
if (iRet)
{
printf("set port err!n");
return -1;
}
/* 创建一个读线程 */
iRet = pthread_create(&tid, NULL, my_read_thread_func, (void *)fd);
if (iRet)
{
printf("pthread_create err!n");
return -1;
}
printf("Enter a char: ");
initTermios(1);
// 写线程
while (1)
{
c = getchar();
iRet = write(fd, &c, 1);
if (iRet != 1)
printf("can not write datan");
}
resetTermios();
return 0;
}
4. 上机实验
编译 2 个版本,有两条命令:PC、ARM
gcc -o serial_send_recv_pc serial_send_recv.c -lpthread
arm-buildroot-linux-gnueabihf-gcc -o serial_send_recv_arm serial_send_recv.c -lpthread
使用 USB 线连接板子的 OTG 口、PC 的 USB 口,PC 上监测到 USB 串口后把它连接到 VMWare,确定:
- 开发板上有设备节点:/dev/ttyGS0
- Ubuntu上有设备节点:/dev/ttyACM0
测试:
- 在 Ubuntu 上执行:
sudo ./serial_send_recv_pc /dev/ttyACM0
- 在板子上执行:
sudo ./serial_send_recv_arm /dev/ttyGS0
- 双方即可互发数据
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
嵌入式
+关注
关注
5085文章
19138浏览量
305710 -
Linux
+关注
关注
87文章
11312浏览量
209705 -
编程
+关注
关注
88文章
3619浏览量
93778
发布评论请先 登录
相关推荐
如何配置USB OTG为Gadget serial device?
现需要将OKA40i-C板子上的OTG下载口配置为虚拟串口,实现u***串行通信,利用linux已存在的u***转虚拟串口驱动:USB Gadget Drivers (Serial Gadge
发表于 01-04 06:47
分享一种将RK3399配置成USB gadget设备的方法
1、RK3399的USB type-c接口初始化配置RK3399中提供了USB gadget设备的支持,程序在内核中目录“kernel/drivers/u***/gadget”中,支持
发表于 05-24 10:35
基于Linux内核的Rockchip USB Gadget UAC开发资料分享
1、Rockchip USB Gadget UAC开发步骤本文档提供 Rockchip 平台基于 Linux 内核的 USB Gadget UAC(
发表于 08-10 16:03
RK3288使用USB GADGET实现大容量存储脚本
(){mount -t configfs none /sys/kernel/configmkdir /sys/kernel/config/usb_gadget/rockchip -m
发表于 11-18 16:33
求助,关于USB gadget安装虚拟网口问题求解
1.make linux-menuconfig
USB support ---->
<*>USB Gadget Support ---->
发表于 09-01 06:14
PL2303 USB to Serial Adapter
The PL2303 USB to Serial adapter is your smart and convenient accessory forconnecting RS-23
发表于 10-15 17:58
•111次下载
Universal Serial Bus(USB) Devi
Universal Serial Bus Device Class Definition for Printing Devices
The Universal Serial Bus (USB
发表于 04-11 19:32
•16次下载
PL2303,Edition USB to Serial B
PL2303,Edition USB to Serial Bridge Controller:The PL-2303 operates as a bridge between one USB
发表于 09-19 07:59
•77次下载
[嵌入式linux]将linux板卡虚拟为USB网卡设备(Ethernet Gadget)
kernel menuconfig-> Device Drivers ->USB support -> USB Gadget Support 建议最好选成M,作为内核驱动模块,便于
发表于 11-02 11:36
•12次下载
AMD Xilinx Linux 2022.1 USB Gadget使用
有客户使用Linux中的USB Gadget功能,把MPSoC器件做USB从设备
USB Gadget 应用实例ADB实现
,定义了多个接口描述符,这是 APP 提出的请求。如果 Gadget 设备有足够的端点,那么就会在在 functionfs 跟目录下创建出这些端点,比如 ep1、ep2。 ADB 程序的调用关系如下
USB Gadget zero应用实例程序
1. 编写程序 1.1 编程思路 涉及的程序如下图所示: PC 端基于 libusb 编写应用程序,开发板端直接使用 Linux 自带的 USB Gadget 驱动 zero.c【/drivers
从硬件软件角度理解Gadget框架
2. 从硬件软件角度理解 Gadget 框架 USB 传输的核心是 endpoint,使用 endpoint 可以收发数据。在 endpoint 之上,就可以模拟 USB 串口、USB
评论