printf("%02X ",buf);
printf("n");
}
char *get_serial_dev_name(char *serial_name) /* 获取串口名称 */
{
if (strcasecmp(serial_name, "COM2") == 0) /* 传入参函数是COM2(不区分大小写),则返回串口对应的设备节点/dev/ttymxc1 */
return "/dev/ttymxc1";
else if (strcasecmp(serial_name, "COM3") == 0)
return "/dev/ttymxc2";
else if (strcasecmp(serial_name, "COM4") == 0)
return "/dev/ttymxc3";
else if (strcasecmp(serial_name, "COM5") == 0)
return "/dev/ttymxc4 ";
else
//return NULL;
return serial_name;
}
static int termios_init(struct termios *tios,int baud,int parity,int data_bits,int stop_bits) //串口 传入波特率,校验位,数据位,停止位。
{
speed_t baud_rate;
if (tios == NULL)
return -1;
tios->c_line = 0;
tios->c_cc[VMIN ] = 0;
tios->c_cc[VTIME] = 0;
/* configure the input modes... */
tios->c_iflag = IGNBRK | IGNPAR | INPCK;
/* configure the output modes... */
tios->c_oflag = 0; /* enable implementation-defined output processing */
/* configure the control modes... */
tios->c_cflag = CREAD | CLOCAL;
if (data_bits == 5) //对传入的数据位进行判断
tios->c_cflag |= CS5;
else if (data_bits == 6)
tios->c_cflag |= CS6;
else if (data_bits == 7)
tios->c_cflag |= CS7;
else if (data_bits == 8)
tios->c_cflag |= CS8;
else
return -1;
if (stop_bits == 1) //对传入的停止位进行判断
tios->c_cflag &=~ CSTOPB;
else if (stop_bits == 2)
tios->c_cflag |= CSTOPB;
else
return -1;
if(parity == 0) //校验位: 0:无校验 1:奇校验 2:偶校验 3:mark parity 4:space parity
{ /* none */
tios->c_cflag &=~ PARENB;
tios->c_cflag &=~ PARODD;
}
else if(parity == 2)
{ /* even */
tios->c_cflag |= PARENB;
tios->c_cflag &=~ PARODD;
}
else if(parity == 1)
{ /* odd */
tios->c_cflag |= PARENB;
tios->c_cflag |= PARODD;
}
else if (parity == 3)/* 2008.12.17 add for support mark and space parity */
{
/* mark */
tios->c_cflag |= PARENB;
tios->c_cflag |= CMSPAR;
tios->c_cflag |= PARODD;
}
else if (parity == 4)
{
/* space */
tios->c_cflag |= PARENB;
tios->c_cflag |= CMSPAR;
}
else
return -1;
/* configure the local modes... */
tios->c_lflag = 0; /* enable implementation-defined input processing */
/* Set the baud rate */
switch(baud) //对传入的波特率进行判断
{
case 110:
baud_rate = B110;
break;
case 300:
baud_rate = B300;
break;
case 600:
baud_rate = B600;
break;
case 1200:
baud_rate = B1200;
break;
case 2400:
baud_rate = B2400;
break;
case 4800:
baud_rate = B4800;
break;
case 9600:
baud_rate = B9600;
break;
case 19200:
baud_rate = B19200;
break;
case 38400:
baud_rate = B38400;
break;
case 57600:
baud_rate = B57600;
break;
case 115200:
baud_rate = B115200;
break;
case 230400:
baud_rate = B230400;
break;
case 460800:
baud_rate = B460800;
break;
case 576000:
baud_rate = B576000;
break;
case 921600:
baud_rate = B921600;
break;
default:
return -1;
}
if ((cfsetispeed(tios, baud_rate) < 0) ||(cfsetospeed(tios, baud_rate) < 0))
return -1;
return 0;
}
//return serial fd, error return -1
int serial_open(char *serial_name, int baud,int parity,int data_bits,int stop_bits,int timeout) //打开串口
{
struct termios settings;
int fd;
char *serial_dev_name;
if (serial_fd >= 0)
return 0;
if (termios_init(&settings, baud,parity,data_bits,stop_bits) < 0)
return 0;
serial_dev_name = get_serial_dev_name(serial_name);
if (serial_dev_name == NULL)
{
fprintf(stderr, "invalid serial name:%sn", serial_name);
return 0;
}
if((fd = open(serial_dev_name, O_RDWR | O_NOCTTY | O_NDELAY))< 0)
return 0;
if(tcgetattr(fd, &old_tios) < 0)
{
close(fd);
return 0;
}
if(tcsetattr(fd, TCSANOW, &settings) < 0)
{
close(fd);
return 0;
}
serial_fd = fd;
return 1;
}
int serial_close() //关闭串口
{
if (serial_fd < 0)
return 0;
tcsetattr(serial_fd, TCSANOW, &(old_tios));
close(serial_fd);
serial_fd = -1;
return 1;
}
int serial_flush(int flag)
{
if (serial_fd < 0)
return 0;
if (flag == SERIAL_FLUSH_TX)
tcflush(serial_fd,TCOFLUSH);
else if (flag == SERIAL_FLUSH_RX)
tcflush(serial_fd,TCIFLUSH);
else if (flag == (SERIAL_FLUSH_RX|SERIAL_FLUSH_TX) )
tcflush(serial_fd,TCIOFLUSH);
return 1;
}
//return 1: poll ok, 0: timeout, -1: error
int serial_poll(int timeout)
{
int fd = serial_fd;
fd_set rfds;
struct timeval tv;
int sel_res;
FD_ZERO(&rfds);
FD_SET(fd, &rfds);
tv.tv_sec = timeout/1000;
tv.tv_usec = (timeout % 1000)*1000;
return select(fd + 1, &rfds, NULL,NULL,&tv);
}
int uart_read(uint8_t* buffer,uint32_t len,uint32_t timeout)
{
while(serial_poll(timeout)>0) //当另一端传入数据时会接收到数据
{
return read(serial_fd,buffer,len);
}
}
int uart_write(uint8_t* buffer,uint32_t len)
{
write(serial_fd,buffer,len);
}
int main(int argc, char *argv[])
{
int fd;
uint8_t recv_buffer[128];
int i, ret;
if (argc != 3) //需要传入串口对应的设备名
{
fprintf(stderr, "usage:serialtest com_port baudn");
return -1;
}
ret = serial_open(argv[1], atoi(argv[2]), SERIAL_PARITY_NO, 8, SERIAL_STOPBIT_ONE, 1000); //调用上面封装好的打开串口函数 传入设备名,波特率,校验位,数据位,停止位以及超时时间
if (ret == 0) //打开失败
{
fprintf(stderr, "open serial failedn");
return -1;
}
while(1)
{
int len = 0;
if((len = uart_read(recv_buffer,sizeof(recv_buffer),100)) > 0)
{
uart_write(recv_buffer,len);
}
}
serial_close();
return 0;
}
编译
aarch64-linux-gnu-gcc uart_test.c -o uarttest
添加可执行权限chmod +x uarttest
运行
./uarttest /dev/ttySC1 115200
收发测试使用上位机自动发送,看是否收发完全一致。
测试OK
波特率误差测试实测115200波特率为115385
误差为0.16%,误差比较小。
最大波特率测试
且只能是46800~921600等倍数,不能设置为500000.
最大只能配置到921600
root@g2uliot:~# ./uarttest /dev/ttySC1 921600
^C
root@g2uliot:~# ./uarttest /dev/ttySC1 1000000
open serial failed
root@g2uliot:~#
设置为460800通讯OK
设置为921600时通讯失败,不确定是不是杜邦线接的太长的原因。后面换短点再试。
总结以上进行了串口的收发测试,和波特率准确度测试都OK。最大波特率只能配置到921600,这里不知道是驱动限制还是芯片的限制。这个还是有点低的,很多MCU的串口都可以跑到2Mbps甚至3Mbps。波特率不能设置为任意值这一点也不是很友好。