1. 开发环境搭建
gcc就可以
命令行
gcc -v
hy@ubuntu:~/c-dev$ gcc -v
Using built-in specs.
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/7/lto-wrapper
OFFLOAD_TARGET_NAMES=nvptx-none
OFFLOAD_TARGET_DEFAULT=1
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 7.5.0-3ubuntu1~18.04' --with-bugurl=file:///usr/share/doc/gcc-7/README.Bugs --enable-languages=c,ada,c++,go,brig,d,fortran,objc,obj-c++ --prefix=/usr --with-gcc-major-version-only --program-suffix=-7 --program-prefix=x86_64-linux-gnu- --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --libdir=/usr/lib --enable-nls --enable-bootstrap --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --with-default-libstdcxx-abi=new --enable-gnu-unique-object --disable-vtable-verify --enable-libmpx --enable-plugin --enable-default-pie --with-system-zlib --with-target-system-zlib --enable-objc-gc=auto --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --enable-multilib --with-tune=generic --enable-offload-targets=nvptx-none --without-cuda-driver --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 7.5.0 (Ubuntu 7.5.0-3ubuntu1~18.04)
如果没有显示上述内容,说明gcc没有正确安装
sudo apt install gcc
即可
2. 可以建个最简单的helloworld测试先编译环境
新建开发目录
mkdir c-dev
新建文件
sudo gedit hello.c
编辑,保存为hello.c
#include <stdio.h>
int main(void)
{
printf("hello world!\n");
return 0;
}
编译运行
gcc -Wall hello.c -o hello
可以观察到到目录下生成了可执行文件hello
.hy@ubuntu:~/c-dev$ ./hello
hello world!
当然,以上在vscode中配置好c语言环境插件和coderunner,可以直接运行
3. 为了开发调试方便,可以安装虚拟串口和串口调试工具
ubuntu下好用的串口调试工具cutecom,
sudo apt install cutecom
实际插了usb转串口工具才有用
我们为了调试,还需要虚拟两个串口环回
安装虚拟串口工具socat
下载链接http://www.dest-unreach.org/socat/download/socat-1.7.3.2.tar.gz
放到本地tools目录下,然后解压
tar zxvf socat-1.7.3.2.tar.gz
进入目录,配置编译安装
cd socat-1.7.3.2
./configure --disable-fips
一堆yes后配置成功
checking for var environ... yes
configure: creating ./config.status
config.status: creating Makefile
config.status: creating config.h
编译安装
sudo make & sudo make install
成功
......
mkdir -p /usr/local/share/man/man1
/usr/bin/install -c -m 644 ./doc/socat.1 /usr/local/share/man/man1/
创建换回的两个串口,创建前可以查看下已有设备,这样多出来的两个就是创建的
ls /dev/pts*
/dev/pts:
0 1 2 3 4 5 6 ptmx
创建
hy@ubuntu:~/tools/socat-1.7.3.2$ socat -d -d pty,raw,echo=0 pty,raw,echo=0
2021/06/11 08:34:03 socat[11093] N PTY is /dev/pts/7
2021/06/11 08:34:03 socat[11093] N PTY is /dev/pts/8
2021/06/11 08:34:03 socat[11093] N starting data transfer loop with FDs [5,5] and [7,7]
新建的两个串口是/dev/pts/7 和/dev/pts/8
打开一个terminial终端监听8口
hy@ubuntu:~$ cat < /dev/pts/8
此时再打开一个窗口向7发送上面就会显示发送的内容
4. 好了,环境好了,进入主题,可以编写程序,测试串口调用了
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <termios.h>
#define BAUDRATE B115200 ///Baud rate : 115200
#define DEVICE "/dev/pts/7"//设置你的端口号
int nFd = 0;
struct termios stNew;
struct termios stOld;
//Open Port & Set Port
int SerialInit()
{
nFd = open(DEVICE, O_RDWR|O_NOCTTY|O_NDELAY);
if(-1 == nFd)
{
perror("Open Serial Port Error!\n");
return -1;
}
if( (fcntl(nFd, F_SETFL, 0)) < 0 )
{
perror("Fcntl F_SETFL Error!\n");
return -1;
}
if(tcgetattr(nFd, &stOld) != 0)
{
perror("tcgetattr error!\n");
return -1;
}
stNew = stOld;
cfmakeraw(&stNew);//将终端设置为原始模式,该模式下全部的输入数据以字节为单位被处理
//set speed
cfsetispeed(&stNew, BAUDRATE);//115200
cfsetospeed(&stNew, BAUDRATE);
//set databits
stNew.c_cflag |= (CLOCAL|CREAD);
stNew.c_cflag &= ~CSIZE;
stNew.c_cflag |= CS8;
//set parity
stNew.c_cflag &= ~PARENB;
stNew.c_iflag &= ~INPCK;
//set stopbits
stNew.c_cflag &= ~CSTOPB;
stNew.c_cc[VTIME]=0; //指定所要读取字符的最小数量
stNew.c_cc[VMIN]=1; //指定读取第一个字符的等待时间,时间的单位为n*100ms
//假设设置VTIME=0,则无字符输入时read()操作无限期的堵塞
tcflush(nFd,TCIFLUSH); //清空终端未完毕的输入/输出请求及数据。
if( tcsetattr(nFd,TCSANOW,&stNew) != 0 )
{
perror("tcsetattr Error!\n");
return -1;
}
return nFd;
}
int main(int argc, char **argv)
{ int i;
int nRet = 0;
char *sendmsg="test\r\n";
int SIZE=sizeof(sendmsg);
char buf[SIZE];
if( SerialInit() == -1 )
{
perror("SerialInit Error!\n");
return -1;
}
printf("serial port init ok. \r\n");
bzero(buf, SIZE);
while(1)
{ sleep(1);
write(nFd,sendmsg,sizeof(sendmsg));//向串口发送数据
printf("sent msg. %s\n",sendmsg);
/串口接收部分
nRet = read(nFd, buf, SIZE);
if(-1 == nRet)
{
perror("Read Data Error!\n");
break;
}
if(0 < nRet)
{
buf[nRet] = 0;
printf("Recv Data: %s\n", buf);
}
}
close(nFd);
return 0;
}
注意,其中端口改成上面你在本地创建的端口之一,打开另外一个监听,在我们的例子里,cat窗口A监听8,程序向7发送就可以在监听窗口A看到输出,另外一个终端B向7发送,就可以向程序回送,向8发送就可以测试监听是否正常。
编译运行
gcc -Wall serial.c -o serial
./serial
每当B发送回应时,serial程序延时1s发送测试字符串
测试成功,可以把串口调用作为个函数放到其他应用中去。