2829|2

9

帖子

0

TA的资源

一粒金砂(中级)

楼主
 

合:进阶操作IO之USB-VCP USART透传 [复制链接]

0x00 前言

串口算是最常用的芯片接口之一了,无论对调试还是实际的通信,都有及其广泛的使用。而USB算是现在事实上的消费者最常见的接口,笔者观察到demo单板存在一个USB串口转换的接口与一个USB的接口,这里就对于USB转换串口进行了透传的测试。

0x10 测试环境

  • demo单板
  • 电脑一台不必多讲
  • IAR 9+,主要是编译器支持会好些

这里需要注意的是,当前的串口是用CH340转换为USB COM进行发送的,所以是有一定对于芯片的部分限制的。所以这里特别使用了工业常用的115200波特率进行驱动。

0x20 测试代码与测试步骤

这里笔者图省事,使用了官方固件库的ACM库,这个在官方的固件库内就有,这里也就不在赘述了。

但是官方的ACM库很尴尬,而且网络流传的官方固件库其实使用了不一样的串口定义,所以官方的固件库需要修改关于eval的相关宏定义。

需要修改的地方如下:

/* definition for COM, connected to USART1 
#define EVAL_COM                         USART1
#define EVAL_COM_CLK                     RCU_USART1

#define EVAL_COM_TX_PIN                  GPIO_PIN_2
#define EVAL_COM_RX_PIN                  GPIO_PIN_3

#define EVAL_COM_GPIO_PORT               GPIOA
#define EVAL_COM_GPIO_CLK                RCU_GPIOA
#define EVAL_COM_AF                      GPIO_AF_7*/

//需要修改的位置
/* definition for COM, connected to USART1 */
#define EVAL_COM                         USART0
#define EVAL_COM_CLK                     RCU_USART0

#define EVAL_COM_TX_PIN                  GPIO_PIN_9
#define EVAL_COM_RX_PIN                  GPIO_PIN_10

#define EVAL_COM_GPIO_PORT               GPIOA
#define EVAL_COM_GPIO_CLK                RCU_GPIOA
#define EVAL_COM_AF                      GPIO_AF_7

 

这样就可以直接使用官方的初始化库函数。

实现之后,官方的ACM对于串口的接受函数还是有些不实用的,于是笔者自己重新封装了一下。

void usb_write(char * buf,unsigned int len)
{
	usb_cdc_handler *cdc = usbd_cdc.class_data[CDC_COM_INTERFACE];
	unsigned int size = len;
	
	if (len <64)
		return;
	size = len;
	for (int i = 0; i < len; ++i)
	{
		buf[i] = cdc->data[i];
	}
	cdc_acm_data_send(&usbd_cdc);

}

unsigned int usb_read(char * buf ,unsigned int len)
{
	usb_cdc_handler *cdc = usbd_cdc.class_data[CDC_COM_INTERFACE];
	unsigned int size = 0;
	if(len == 0)
	{
		return 0;
	}
	
        cdc_acm_data_receive(&usbd_cdc);
	for (int i = 0; i < cdc->receive_length; ++i)
	{
		buf[i] = cdc->data[i];
	}
        size = cdc->receive_length;//这个必须要清空,否则会一直更新
        cdc->receive_length = 0;
	return size;
}


void usart_write(char * buf,unsigned int len)
{
	if(len != 0)
	{
		for (int i = 0; i < len; ++i)
		{
			usart_data_transmit (EVAL_COM, buf[i]);
			while (SET != usart_flag_get (EVAL_COM, USART_FLAG_TBE));
		}
	}
}

unsigned int usart_read(char * buf ,unsigned int len,unsigned int timeout)
{
	if(len != 0)
	{
		for (int i = 0; i < len; ++i)
		{
			while (SET != usart_flag_get (EVAL_COM, USART_FLAG_RBNE));
			buf[i] = usart_data_receive(EVAL_COM);
		}
	}
}

0x30 测试结果与总结

最后效果还是可以的,但是可能是笔者一个项目挂载过度,导致当前的接受出现了一定的丢包,但是应该也是笔者自己的代码问题(代码内部过多的栓塞等待)。但是本身也是作为评测,这里也就不再详细完善了,再详细完善可以使用DMA事件或者是使用类操作系统的挂起——切换操作。

左侧是接收端,右侧是发送端,可以看到,每次接受的包都是一样的字符。

此帖出自GD32 MCU论坛

最新回复

各种频率都试了一下吗?还有就是接收与发送是不是都用的DMA?  详情 回复 发表于 2022-2-6 08:49
点赞 关注
 

回复
举报

6993

帖子

11

TA的资源

版主

沙发
 
各种频率都试了一下吗?还有就是接收与发送是不是都用的DMA?
此帖出自GD32 MCU论坛

点评

1. 这里只测试了115200 2. 示例代码并没有使用DMA传输  详情 回复 发表于 2022-2-6 14:35
 
 
 

回复

9

帖子

0

TA的资源

一粒金砂(中级)

板凳
 
lugl4313820 发表于 2022-2-6 08:49 各种频率都试了一下吗?还有就是接收与发送是不是都用的DMA?

1. 这里只测试了115200

2. 示例代码并没有使用DMA传输

此帖出自GD32 MCU论坛
 
 
 

回复
您需要登录后才可以回帖 登录 | 注册

随便看看
查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/8 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表