AXI CANFD MicroBlaze 测试笔记

文章目录

    • 前言
    • 测试用的硬件连接
    • Vivado 配置
    • Vitis MicroBlaze CANFD 代码
    • 测试代码
    • 测试截图
    • Github Link

前言

官网: CAN with Flexible Data Rate (CAN FD) (xilinx.com)

特征:

  • 支持8Mb/s的CANFD
  • 多达 3 个数据位发送器延迟补偿(TDC, transmitter delay compensation)
  • 32-deep TX FIFO, 2 * 64-deep RX FIFO, 32 ID Filter
  • 接收支持 顺序(Sequential)模式 和 邮箱(Mailbox)模式
  • Bus-Off 恢复模式: 自动恢复 或 用户干预的自动恢复

参考:

  • How to configure CAN/CANFD clocks and identify the CAN nodes (xilinx.com)
  • canfd 默认文档和示例路径: C:\Xilinx\Vitis\2023.2\data\embeddedsw\XilinxProcessorIPLib\drivers\canfd_v2_8

本文仅作学习交流使用.

测试用的硬件连接

硬件:

  • FPGA: xc7a35tfgg484-2
  • 时钟: 50MHz, 有源, 单端, 3V3, 引脚Y18
  • 复位: 低电平复位, 3V3, 引脚F20
  • UART: TX G16, RX G15
  • CANFD: TX F16, RX E17

Vivado 配置

如图:

请添加图片描述

时钟:

  • clk_out1, 160MHz, can_clk_x2, 也是其它组件的时钟
  • clk_out2, 80MHz, can_clk, 是CAN和CANFD配置的主时钟
  • 低电平复位

调试串口:

  • 9600 or 115200 bps

时钟的配置截图

在这里插入图片描述

CANFD的配置, 收发FIFO Depth都是32, 接收使用顺序模式, 不用FIFO1

在这里插入图片描述

可适当减少RX FIFO-0 Depth以减少RAM消耗. 含MicroBlaze 64K, 一路CANFD, 没有使用任何外部RAM的消耗如下(xc7a35tfgg484-2):

在这里插入图片描述

Vitis MicroBlaze CANFD 代码

初始化:

  • 查找并初始化已有CANFD配置
  • 进入配置模式, 配置位速率, 采样点, TDC, 滤波器, 关闭BRS
  • 设置收发错误事件回调, 使能中断
  • 进入正常模式
  • 注意在C文件中编译, 不然Cpp中 XSetupInterruptSystem 会报错 error: invalid conversion from 'void (*)(void*)' to 'void*' [-fpermissive]

参考代码如下:

int bsp_canfd_init(XCanFd *InstancePtr, uint32_t BaseAddress, uint32_t BaudRate,float SamplePoint, uint32_t FastBaudRate,float FastSamplePoint) {XCanFd_Config *ConfigPtr = XCanFd_LookupConfig(BaseAddress);if (ConfigPtr == NULL) {bsp_canfd_debug_printf("Error: XCanFd_LookupConfig returned NULL\n");return -1;} else {bsp_canfd_debug_printf("XCanFd_Config:\n");bsp_canfd_debug_printf("  BaseAddress: 0x%08X\n", ConfigPtr->BaseAddress);bsp_canfd_debug_printf("  Rx_Mode: %s\n",ConfigPtr->Rx_Mode ? "Mailbox" : "Sequential");bsp_canfd_debug_printf("  NumofRxMbBuf: %d\n", ConfigPtr->NumofRxMbBuf);bsp_canfd_debug_printf("  NumofTxBuf: %d\n", ConfigPtr->NumofTxBuf);bsp_canfd_debug_printf("  IntrId: %d\n", ConfigPtr->IntrId);bsp_canfd_debug_printf("  IntrParent: 0x%08X\n", ConfigPtr->IntrParent);}int Status =XCanFd_CfgInitialize(InstancePtr, ConfigPtr, ConfigPtr->BaseAddress);if (Status != XST_SUCCESS) {bsp_canfd_debug_printf("Error: XCanFd_CfgInitialize returned %d\n", Status);return -2;} else {bsp_canfd_debug_printf("XCanFd_CfgInitialize: Success\n");}// config modeXCanFd_EnterMode(InstancePtr, XCANFD_MODE_CONFIG);while (XCanFd_GetMode(InstancePtr) != XCANFD_MODE_CONFIG);bsp_canfd_debug_printf("XCanFd_EnterMode: XCANFD_MODE_CONFIG\n");// 80MHz / (3 + 1) = 20MHz, 20MHz / (1 + (30 + 1) + (7 + 1)) = 500KHz// XCanFd_SetBitTiming(InstancePtr, 8, 7, 30);XCanFd_SetBaudRatePrescaler(InstancePtr, 0x3);u16 total_tq = 20000000 / BaudRate;u16 tseg1 = (u16)(SamplePoint * total_tq) - 2;u8 tseg2 = (u8)(total_tq - tseg1 - 3);u8 sjw = tseg2 + 1;XCanFd_SetBitTiming(InstancePtr, sjw, tseg2, tseg1);// 80MHz, 80 / (1 + (30 + 1) + (7 + 1)) = 2MHz// XCanFd_SetFBitTiming(InstancePtr, 8, 7, 30);XCanFd_SetFBaudRatePrescaler(InstancePtr, 0x0);u16 ftotal_tq = 80000000 / FastBaudRate;u8 ftseg1 = (u8)(FastSamplePoint * ftotal_tq) - 2;u8 ftseg2 = (u8)(ftotal_tq - ftseg1 - 3);u8 fsjw = ftseg2 + 1;XCanFd_SetFBitTiming(InstancePtr, fsjw, ftseg2, ftseg1);// TDC, 0~31// XCanFd_Set_Tranceiver_Delay_Compensation(InstancePtr, 0x3);XCanFd_SetBitRateSwitch_DisableNominal(InstancePtr);bsp_canfd_debug_printf("XCanFd: %d@0.%d, %d@0.%d\n", BaudRate,(int)(SamplePoint * 1000), FastBaudRate,(int)(FastSamplePoint * 1000));if (XCANFD_GET_RX_MODE(InstancePtr) == 0) {bsp_canfd_debug_printf("RX_MODE Sequential Filter: XCANFD_AFR_UAF_ALL_MASK\n");XCanFd_AcceptFilterDisable(InstancePtr, XCANFD_AFR_UAF_ALL_MASK);XCanFd_AcceptFilterEnable(InstancePtr, XCANFD_AFR_UAF_ALL_MASK);} else {bsp_canfd_debug_printf("RX_MODE Mailbox Filter: Need to be implemented\n");}XCanFd_SetHandler(InstancePtr, XCANFD_HANDLER_SEND, (void *)SendHandler,(void *)InstancePtr);XCanFd_SetHandler(InstancePtr, XCANFD_HANDLER_RECV, (void *)RecvHandler,(void *)InstancePtr);XCanFd_SetHandler(InstancePtr, XCANFD_HANDLER_ERROR, (void *)ErrorHandler,(void *)InstancePtr);XCanFd_SetHandler(InstancePtr, XCANFD_HANDLER_EVENT, (void *)EventHandler,(void *)InstancePtr);Status =XSetupInterruptSystem(InstancePtr, &XCanFd_IntrHandler, ConfigPtr->IntrId,ConfigPtr->IntrParent, XINTERRUPT_DEFAULT_PRIORITY);if (Status != XST_SUCCESS) {bsp_canfd_debug_printf("Error: XSetupInterruptSystem returned %d\n",Status);return 1;} else {bsp_canfd_debug_printf("XSetupInterruptSystem: Success\n");}XCanFd_InterruptEnable(InstancePtr, XCANFD_IXR_ALL);XCanFd_EnterMode(InstancePtr, XCANFD_MODE_NORMAL);while (XCanFd_GetMode(InstancePtr) != XCANFD_MODE_NORMAL);bsp_canfd_debug_printf("XCanFd_EnterMode: XCANFD_MODE_NORMAL\n");return 0;
}

发送套用Linux SocketCAN接口, 毕竟这个用的人多, 易于理解. 因为发送FIFO深度为32, 所以, 正常情况下连发32帧无需插入延时, 也无需自己处理发送队列, 这已经够大部分情况使用了.

int bsp_canfd_send(XCanFd *InstancePtr, struct canfd_frame *frame) {bool is_extended = frame->can_id & CAN_EFF_FLAG ? true : false;bool is_remote = frame->can_id & CAN_RTR_FLAG ? true : false;bool is_fd = frame->flags & CANFD_FDF ? true : false;bool is_brs = frame->flags & CANFD_BRS ? true : false;bool is_esi = frame->flags & CANFD_ESI ? true : false;u32 TxFrame[CANFD_MTU];TxFrame[0] = XCanFd_CreateIdValue(CAN_SFF_MASK & (is_extended ? ((frame->can_id & CAN_EFF_MASK) >> 18): frame->can_id),is_extended ? 1 : (u32)is_remote, (u32)is_extended,(u32)is_extended ? (frame->can_id & 0x3FFFF) : 0,is_extended ? (u32)is_remote : 0);if ((!is_fd) && (!is_brs)) {TxFrame[1] = XCanFd_CreateDlcValue(frame->len);} else {if (is_brs) {TxFrame[1] =XCanFd_Create_CanFD_Dlc_BrsValue(XCanFd_GetLen2Dlc(frame->len));} else {TxFrame[1] = XCanFd_Create_CanFD_DlcValue(XCanFd_GetLen2Dlc(frame->len));}}u8 *FramePtr = (u8 *)(&TxFrame[2]);for (int i = 0; i < frame->len; i++) {FramePtr[i] = frame->data[i];}u32 TxBufferNumber;int status = XCanFd_Send(InstancePtr, TxFrame, &TxBufferNumber);if (status == XST_FIFO_NO_ROOM) {bsp_canfd_debug_printf("Error: XCanFd_Send returned XST_FIFO_NO_ROOM\n");return -1;}if (status != XST_SUCCESS) {bsp_canfd_debug_printf("Error: XCanFd_Send returned %d\n", status);return -2;}return 0;
}

接收的示例, 类似 candump:

  • 标准帧是%03X打印, 扩展帧是%08X打印.
  • 远程帧打R, 数据帧打D
  • CANFD打F且长度为[%02d], Classic CAN打-且数据长度为[%d]
  • 开启了BRS打B, 否则打-
  • ESI 暂不关心
  • 当然这里打印接收只是用于测试, 实际不要在中断中阻塞打印
static void RecvHandler(void *CallBackRef) {XCanFd *CanPtr = (XCanFd *)CallBackRef;int Status;u32 RxFrame[CANFD_MTU];/* Check for the design 1 - MailBox 0 - Sequential */if (XCANFD_GET_RX_MODE(CanPtr) == 1) {Status = XCanFd_Recv_Mailbox(CanPtr, RxFrame);} else {Status = XCanFd_Recv_Sequential(CanPtr, RxFrame);}u32 id1 = (RxFrame[0] >> (u32)XCANFD_IDR_ID1_SHIFT) & (u32)0x7FF;u32 is_extended = (RxFrame[0] >> (u32)XCANFD_IDR_IDE_SHIFT) & (u32)0x1;u32 id2 = (RxFrame[0] >> (u32)XCANFD_IDR_ID2_SHIFT) & (u32)0x3FFFF;u32 is_remote = is_extended? (RxFrame[0] & 0x01): ((RxFrame[0] >> (u32)XCANFD_IDR_SRR_SHIFT) & (u32)0x1);/* Get the Dlc inthe form of bytes */u32 len = XCanFd_GetDlc2len(RxFrame[1] & XCANFD_DLCR_DLC_MASK, EDL_CANFD);if (Status != XST_SUCCESS) {bsp_canfd_debug_printf("Error: XCanFd_Recv returned %d\n", Status);return;}u32 is_brs = RxFrame[1] & XCANFD_DLCR_BRS_MASK ? 1 : 0;u32 is_fdf = RxFrame[1] & XCANFD_DLCR_EDL_MASK ? 1 : 0;// bsp_canfd_debug_printf("%08X ", RxFrame[0]);u8 *FramePtr = (u8 *)(&RxFrame[2]);if (is_extended) {bsp_canfd_debug_printf("%08X ", id1 << 18 | id2);} else {bsp_canfd_debug_printf("%03X ", id1);}if (is_remote) {bsp_canfd_debug_printf("R [%d]", len);} else {bsp_canfd_debug_printf("D ");if (is_fdf) {bsp_canfd_debug_printf("F ");} else {bsp_canfd_debug_printf("- ");}if (is_brs) {bsp_canfd_debug_printf("B ");} else {bsp_canfd_debug_printf("- ");}if ((!is_fdf) && (!is_brs)) {bsp_canfd_debug_printf("[%d] ", len);} else {bsp_canfd_debug_printf("[%02d] ", len);}for (int i = 0; i < len; i++) {bsp_canfd_debug_printf("%02X ", FramePtr[i]);}}bsp_canfd_debug_printf("\n");
}

测试代码

如下:

  • 位速率 500K@80%, 4M@80%, 如果出现收发不正常, 可能需要微调初始化中的TDC, 检查CAN收发器以及终端电阻
  • 对数据帧远程帧, CAN或CANFD或CANFD BRS都做了测试
  • 连续发送32帧
int main() {xil_printf("============================================\n");XCanFd CanFd0;int Status =bsp_canfd_init(&CanFd0, XPAR_CANFD_0_BASEADDR, 500000, 0.8, 4000000, 0.8);if (Status != 0) {xil_printf("Error: bsp_canfd_init returned %d\n", Status);return -1;}for (int i = 0; i < 4; i++) {struct canfd_frame frame = {.can_id = 0x123,.len = 8,.flags = 0,.__res0 = 0,.__res1 = 0,.data = {0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09,0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13,0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D,0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27,0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F, 0x30, 0x31,0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3A, 0x3B,0x3C, 0x3D, 0x3E, 0x3F},};// std canframe.can_id = 0x123;frame.len = 8;frame.flags = 0;int Status = bsp_canfd_send(&CanFd0, &frame);// std can remoteframe.can_id = 0x124 | CAN_RTR_FLAG;frame.len = 8;frame.flags = 0;Status |= bsp_canfd_send(&CanFd0, &frame);// std can fdframe.can_id = 0x125;frame.len = 64;frame.flags = CANFD_FDF;Status |= bsp_canfd_send(&CanFd0, &frame);// std can fd brsframe.can_id = 0x126;frame.len = 64;frame.flags = CANFD_FDF | CANFD_BRS;Status |= bsp_canfd_send(&CanFd0, &frame);// ext canframe.can_id = 0x12345678 | CAN_EFF_FLAG;frame.len = 8;frame.flags = 0;Status |= bsp_canfd_send(&CanFd0, &frame);// ext can remoteframe.can_id = 0x12345679 | CAN_EFF_FLAG | CAN_RTR_FLAG;frame.len = 8;frame.flags = 0;Status |= bsp_canfd_send(&CanFd0, &frame);// ext can fdframe.can_id = 0x1234567A | CAN_EFF_FLAG;frame.len = 64;frame.flags = CANFD_FDF;Status |= bsp_canfd_send(&CanFd0, &frame);// ext can fd brsframe.can_id = 0x1234567B | CAN_EFF_FLAG;frame.len = 64;frame.flags = CANFD_FDF | CANFD_BRS;Status |= bsp_canfd_send(&CanFd0, &frame);if (Status != 0) {xil_printf("Error: bsp_canfd_send %d returned %d\n", i, Status);return -1;}}while (1) {}return 0;
}

测试截图

设置:

在这里插入图片描述

MCU连续发出的32帧报文:

在这里插入图片描述

逻辑分析仪上可以看出来两帧之间卡着标准的3 ITM来的

在这里插入图片描述

出现BRS时的情况, 左边500K, 右边4M

在这里插入图片描述

CAN分析仪向FPGA发送测试:

在这里插入图片描述

Github Link

domain_controller_orin_x2_tc397/axi_canfd_microblaze at main · weifengdq/domain_controller_orin_x2_tc397 (github.com)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/542738.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

长江存储Nand闪存芯片和SSD解决方案

长江存储科技有限责任公司2016年7月成立于武汉&#xff0c; 是一家专注于3D NAND闪存设计制造一体化的IDM集成电路企业&#xff0c;同时也提供完整的存储器解决方案。长江存储为全球合作伙伴供应3D NAND闪存晶圆及颗粒&#xff0c; 嵌入式存储芯片以及消费级、企业级固态硬盘等…

报Invalid value type for attribute ‘factoryBeanObjectType‘: java.lang.String错误

在springboot中使用Mybatis出现Invalid value type for attribute factoryBeanObjectType: java.lang.String 1、没有使用mybatis 检查pom文件里面的mybatis 可能是缺少这个依赖&#xff0c;或者版本过低 重新导入依赖 <dependency><groupId>org.mybatis.spri…

如何在代理的IP被封后立刻换下一个IP继续任务

目录 前言 1. IP池准备 2. 使用代理IP进行网络请求 3. 处理IP被封的情况 4. 完整代码示例 总结 前言 当进行某些网络操作时&#xff0c;使用代理服务器可以帮助我们隐藏真实IP地址以保护隐私&#xff0c;或者绕过一些限制。然而&#xff0c;经常遇到的问题是代理的IP可能…

深度学习模型部署(十)模型部署配套工具二

上篇blog讲了trtexec和onnx_graphsurgeon两个工具&#xff0c;一个用于将onnx转化为trt模型&#xff0c;另一个用于对onnx模型进行修改。这篇blog讲polygraphy和nsight systems&#xff0c;前者用于进行模型优化以及结果验证&#xff0c;后者用于性能分析。 polygraph polygra…

OJ_数组划分

题干 C实现——复杂度极高的深度优先遍历(容易超时) #include <iostream> #include <vector> using namespace std;//sasb sum //sb-sa diffint sum 0; int diff 0;void DFSFindMinDiff(vector<int> &arr,int pos,int sa) { //sa表示a集合的元素和i…

【SCI论文】“学术丑闻揭露:当AI写作遭遇学术审稿,ChatGPT意外成为论文共作者!“

在最近的学术圈中出现了一篇令人哭笑不得的论文。这篇文章标题为“The three-dimensional porous mesh structure of Cu-base…”发表在《Surfaces and Interfaces》杂志上&#xff0c;竟然包含了ChatGPT的提示语&#xff0c;暴露出了审稿过程中可能的疏忽。 文章讨论了铜基金…

构建部署_Jenkins介绍与安装

构建部署_Jenkins介绍与安装 构建部署_Jenkins介绍与安装Jenkins介绍Jenkins安装 构建部署_Jenkins介绍与安装 Jenkins介绍 Jenkins是一个可扩展的持续集成引擎。 持续集成&#xff0c;就是通常所说的CI&#xff08;Continues Integration&#xff09;&#xff0c;可以说是现…

【PyTorch】基础学习:在Pycharm等IDE中打印或查看Pytorch版本信息

【PyTorch】基础学习&#xff1a;在Pycharm等IDE中打印或查看Pytorch版本信息 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1…

最新潮乎盲盒系统源码,附搭建教程

搭建方法 宝塔创建网站&#xff0c;上传后端程序到根目录&#xff0c;在.env修改数据库账号密码 上传数据库&#xff0c;伪静态thinkphp 运行目录public PHP扩展安装下面的 禁用函数先禁用下面那个&#xff0c;就可以了 前端是uniapp 后台admin 禁用函数putenv、 扩展fileinfo…

【日常记录】【JS】浏览器本身的取色器

文章目录 1、介绍2、打开拾色器并且取样3、代码中止拾色器4、参考连接 1、介绍 EyeDropper 接口表示一个拾色器工具的实例&#xff0c;用户可以打开并使用它从屏幕上选择颜色。 (他还是一个实验性的技术&#xff0c;存在兼容性问题&#xff09; EyeDropper MDN 介绍 2、打开拾…

Android FrameWork 学习路线

目录 前言 学习路线&#xff1a; 1.基础知识 2、AOSP 源码学习 3. AOSP 源码编译系统 4. Hal与硬件服务 5.基础组件 6. Binder 7. 系统启动过程分析 8. 应用层框架​编辑 9. 显示系统 10. Android 输入系统 11. 系统应用 前言 Android Framework 涉及的行业相当广…

热流道融合3D打印技术正在成为模具制造新利器

在模具领域中&#xff0c;3D打印技术与热流道技术联手&#xff0c;能迸发出更耀眼的光芒。两种技术虽然各有特点&#xff0c;但两者结合将形成互补作用&#xff0c;从而实现11&#xff1e;2”的跨越式提升。 将增材制造的灵活思维融入传统模具设计时&#xff0c;不仅能够突破传…