MT6835 21位 磁编码器 SPI 平台无关通用驱动框架 STM32

MT6835 21位 磁编码器 SPI 平台无关通用驱动框架 STM32

      • 1. 获取代码:
      • 2. 加入你的项目
        • 2.1 以 STM32 为例:
        • 2.2 以 ESP-IDF 为例:
      • 3. 对接 API
        • 3.1 以 STM32 为例:
      • 4. 更多函数说明
      • 5. 写入 EEPROM 示例

MT6835 Framework

纯C语言实现,跨平台(stm32、ESP32),移植灵活,低耦合高内聚的
MT6835(21位磁编) 驱动框架

1. 获取代码:

请为你的Git账户配置好SSH,然后在你喜欢的项目目录下使用以下命令
或去 项目地址页面 下载

git clone git@github.com:Hotakus/mt6835.git

2. 加入你的项目

2.1 以 STM32 为例:

该项目为标准 CMake 项目,可以很方便的加入你的项目中。
例如,STM32CubeMX 可以自动生成 STM32 的 CMake 工程文件,请在你的根目录的CMakeLists.txt中(不是该项目)加入以下内容:

add_subdirectory(mt6835) # 添加子目录,在 add_executable() 之前# 找到这句话,在上下加入代码
add_executable(${PROJECT_NAME}.elf ${SOURCES} ${LINKER_SCRIPT})# 假设你的项目名为 “your_project”,则添加依赖⬇
# target_link_libraries 一般放在 add_executable() 之后
target_link_libraries(your_project mt6835) 
2.2 以 ESP-IDF 为例:

若你使用其他平台,请跳过该小节
若你使用 ESP-IDF 框架进行 ESP32 开发,首先,你的ESP-IDF项目根目录应该会有一个components文件夹,
若没有,请自己创建一个,这个目录用于添加额外组件(拉取的本项目同样要放在这个目录下)。
接下来,进入mt6835项目根目录,注意到CMakeLists_esp_idf.txt文件,
然后将CMakeLists.txt与该文件名进行交换即可。ESP-IDF 会自动检测


3. 对接 API

程序预留硬件抽象层 API 给用户对接,对于 MT6835,采用 SPI 全双工通信,可以读出21位的角度原始数据,请确保以下:

  • 提前配置好 SPI 全双工通信(例如 STM32 可以使用 STM32CubeMX 配置 SPI1 )
  • SPI 时钟理论不超过 16MHz,
  • CPOL(1), CPHA(1),8bit 数据模式,
  • 软 CS,提前配置好一根 CS 引脚,默认拉高

在该项目的 example 目录,你可以看到预先实现的对应平台的移植文件,可以直接放到你的项目中使用。若你想要自己实现,可以参考下面的示例:

3.1 以 STM32 为例:
#include <math.h>
#include "mt6835.h" // MT6835 驱动头文件
#include "spi.h"    // STM32CubeMX 生成的 SPI 头文件#define SPI_INSTANCE hspi1             // STM32CubeMX 生成的 SPI 句柄
#define SPI_CS       SPI1_CS_Pin       // STM32CubeMX 生成的 CS 引脚
#define SPI_CS_PORT  SPI1_CS_GPIO_Port // STM32CubeMX 生成的 CS 端口// 示例 CS 操作函数,可以根据需要进行修改
static void mt6835_cs_control(mt6835_cs_state_enum_t state) {if (state == MT6835_CS_HIGH) {  // 高电平HAL_GPIO_WritePin(SPI_CS_PORT, SPI_CS, GPIO_PIN_SET);} else {// 低电平HAL_GPIO_WritePin(SPI_CS_PORT, SPI_CS, GPIO_PIN_RESET);}
}// 示例收发函数,可以根据需要进行修改
static void mt6835_spi_send_recv(uint8_t *tx_buf, uint8_t *rx_buf, uint8_t len) {HAL_StatusTypeDef status = HAL_OK;status = HAL_SPI_TransmitReceive_IT(&SPI_INSTANCE, tx_buf, rx_buf, len);if (status != HAL_OK) {printf("spi send_recv failed %d\n\r", status);return;}// wait ITuint32_t tickstart = HAL_GetTick();while (HAL_SPI_GetState(&SPI_INSTANCE) != HAL_SPI_STATE_READY) {if (HAL_GetTick() - tickstart > 1) {printf("spi send_recv timeout\n\r");return;}}
}int main(void) { /* STM32CubeMX 生成的初始化 */HAL_Init();......MX_SPI1_Init();/* 创建 MT6835 对象 */mt6835_t *mt6835 = mt6835_create();/* 链接 SPI 的 CS 操作函数 */mt6835_link_spi_cs_control(mt6835, mt6835_cs_control);/* 链接 SPI 收发函数 */mt6835_link_spi_send_recv(mt6835, mt6835_spi_send_recv);/* 可选 */// mt6835_link_spi_send(mt6835, mt6835_spi_send);// mt6835_link_spi_recv(mt6835, mt6835_spi_recv);/* 是否开启 CRC 校验,使用查表法,通常不会有性能影响* 若开启,则会使用 mt6835_t 结构体中的 crc_res 位存储该次角度读取正确性,* 用户可以根据 crc_res 的值进行判断,*/mt6835_enable_crc_check(mt6835);// mt6835_disable_crc_check(mt6835);    // 禁用 CRC 校验/* 开始读取角度 */uint32_t raw_angle = 0;float radian_angle = 0.0f;while(1) {/* * 读取原始角度数据* 第二个参数为读取方式, MT6835_READ_ANGLE_METHOD_NORMAL 或 MT6835_READ_ANGLE_METHOD_BURST * MT6835_READ_ANGLE_METHOD_BURST 会更快*/raw_angle = mt6835_get_raw_angle(mt6835, MT6835_READ_ANGLE_METHOD_BURST);radian_angle = raw_angle * (M_PI * 2.0f) / MT6835_ANGLE_RESOLUTION;// radian_angle = mt6835_get_angle(motor1_mt6835, MT6835_READ_ANGLE_METHOD_BURST);if (!mt6835->crc_res) {printf("crc error\n\r");}printf("raw_angle: %d, radian_angle: %f\n\r", raw_angle, radian_angle);HAL_Delay(500);}
}

若不想自己实现,则直接将example/stm32/mt6835_stm32_spi_port.h文件中宏定义MT6835_STM32_SPI_PORT_ENABLE改为1即可:
然后按照如下调用:

int main() {uint32_t raw_angle = 0;float radian_angle = 0.0f;mt6835_t * mt6835 = mt6835_stm32_spi_port_init();while(1) {raw_angle = mt6835_get_raw_angle(mt6835, MT6835_READ_ANGLE_METHOD_BURST);radian_angle = raw_angle * (M_PI * 2.0f) / MT6835_ANGLE_RESOLUTION;// radian_angle = mt6835_get_angle(motor1_mt6835, MT6835_READ_ANGLE_METHOD_BURST);printf("raw_angle: %d, radian_angle: %f\n\r", raw_angle, radian_angle);HAL_Delay(500);}
}

4. 更多函数说明

  • 创建销毁函数:

    • mt6835_create() 创建一个 mt6835 对象
    • mt6835_destroy() 销毁一个 mt6835 对象
  • 链接函数:

    • mt6835_link_spi_send() 链接 SPI 发送函数
    • mt6835_link_spi_recv() 链接 SPI 接收函数
    • mt6835_link_spi_send_recv() 链接 SPI 收发函数
    • mt6835_link_spi_cs_control() 链接 CS 引脚操作函数
  • CRC:

    • mt6835_enable_crc_check() 使能 CRC 校验(查表法)
    • mt6835_disable_crc_check() 失能 CRC 校验
  • GET 和 SET 函数

    • mt6835_get_id() 读取 MT6835 ID 寄存器
    • mt6835_set_id() 暂时写入 MT6835 ID 寄存器(配合 mt6835_write_eeprom() 函数)
    • mt6835_get_raw_angle() 读取原始角度数据(21位)
    • mt6835_get_angle() 读取原始角度数据并换算为弧度值(0 ~ 2*PI)
    • mt6835_get_raw_zero_angle() 读取原始零位值(12位)(零位值在 FOC 中很重要,最好是将正确零位固化在EEPROM中)
    • mt6835_get_zero_angle() 读取零位值,并换算成弧度值
    • mt6835_set_zero_angle() 暂时写入零位值(以弧度)到寄存器,(配合 mt6835_write_eeprom() 函数)
  • 底层函数:

    • mt6835_read_reg() 单字节读取寄存器
    • mt6835_write_reg() 单字节暂时写入寄存器(配合 mt6835_write_eeprom() 函数)
    • mt6835_write_eeprom() 将当前所有的可固化寄存器值写入(固化)到 EEPROM,所有写入操作最后都要额外调用这个函数

5. 写入 EEPROM 示例

存在寄存器的值断电消失,所以只是写入寄存器并不等于固化到 EEPROM

int main() {/* 写入 ID */mt6835_set_id(mt6835, 0xDA); // 写入 ID 0xDA 到寄存器,只在寄存器,断电消失HAL_Delay(1);uint8_t id = mt6835_get_id(mt6835); // 读取 IDprintf("id: 0x%x\r\n", id);/* 写入特定零位值,零位值需要你自己校准并读取角度 */// 2.0943951024f 是 120 度,假设 120 度是零位mt6835_set_zero_angle(mt6835, 2.0943951024f); //写入到零位寄存器,断电消失HAL_Delay(1);float zero_angle = mt6835_get_zero_angle(mt6835); // 读取零位值printf("zero_angle: %f rad\r\n", zero_angle);/*下面进行统一固化,为保证断电后 MT6835 数据不丢失,需要固化到 EEPROM *//** 直接调用 mt6835_write_eeprom 发送固化命令,* 命令成功发送则返回 0x55, 若正确收到 0x55,* 则程序 respond 为 true,错误则 false*/bool respond = mt6835_write_eeprom(mt6835);if (!respond) {printf("write eeprom failed\r\n");} else {printf("write eeprom success\r\n");}HAL_Delay(6000);    // 写入后至少 6 秒钟不能断电/* * 将上面的所有写入和固化操作注释掉,重新烧录程序,* 并重新给编码器上电,进行读取验证 */ id = mt6835_get_id(mt6835); // 读取 IDprintf("id: 0x%x\r\n", id);zero_angle = mt6835_get_zero_angle(mt6835); // 读取零位值printf("zero_angle: %f rad\r\n", zero_angle);// 若任何环节失败,请优先检测你的 SPI 初始化代码是否有问题,// 因为所有代码均经过正确验证
}

最后附上简单的 STM32CubeMX配置图片:
注意另配 CS 引脚,任意引脚设置为 OUTPUT 即可
在这里插入图片描述

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

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

相关文章

24、深度学习-自学之路-卷积神经网络

一、你怎么理解卷积神经网络呢&#xff0c;我的理解是当你看一个东西的时候&#xff0c;你的眼睛距离图片越近&#xff0c;你看到的东西就越清晰&#xff0c;但是如果你看到的图片只是整个物体的一小部分&#xff0c;那么你将不知道你看到的物品是什么&#xff0c;因为关注整体…

集合家族详情

一、Java集合框架全景图 1.1 核心接口层次结构 graph TDA[Iterable] --> B[Collection]B --> C1[List]B --> C2[Set]B --> C3[Queue]C1 --> D1[ArrayList]C1 --> D2[LinkedList]C2 --> E1[HashSet]C2 --> E2[TreeSet]C3 --> F1[PriorityQueue]G[M…

ArcGIS基础知识之ArcMap基础设置——ArcMap选项:常规选项卡设置及作用

作为一名 GIS 从业者,ArcMap 是我们日常工作中不可或缺的工具。对于初学者来说,掌握 ArcMap 的基础设置是迈向 GIS 分析与制图的第一步。今天,就让我们一起深入了解 ArcMap 选项中常规选项卡的各个设置,帮助大家更好地使用这款强大的软件。 在 ArcMap 中,常规选项卡是用户…

在fedora41中安装钉钉dingtalk_7.6.25.4122001_amd64

在Fedora-Workstation-Live-x86_64-41-1.4中安装钉钉dingtalk_7.6.25.4122001_amd64.deb 到官网下载钉钉Linux客户端com.alibabainc.dingtalk_7.6.25.4122001_amd64.deb https://page.dingtalk.com/wow/z/dingtalk/simple/ddhomedownload#/ 一、直接使用dpkg命令安装deb包报错…

设置mysql的主从复制模式

mysql设置主从复制模式似乎很容易&#xff0c;关键在于1&#xff09;主库启用二进制日志&#xff0c;2&#xff09;从库将主库设为主库。另外&#xff0c;主从复制&#xff0c;复制些什么&#xff1f;从我现在获得的还很少的经验来看&#xff0c;复制的内容有表&#xff0c;用户…

面试真题 | B站C++渲染引擎

一、基础与语法 自我介绍 请简要介绍自己的背景、专业技能和工作经验。实习介绍 详细描述你在实习期间参与的项目、职责和成果。二、智能指针相关问题回答 unique_ptr 是如何实现的?它有哪些特点和优势? unique_ptr 是C++11引入的一种智能指针,用于管理动态分配的内存资源…

【算法学习】拓扑排序(Topological Sorting)

目录 定义 例子 拓扑排序的实现 核心思想 实现方法 1&#xff0c;Kahn算法&#xff08;基于贪心策略&#xff09; 步骤&#xff1a; 用二维数组存储图的例子 用哈希表存储图的例子 2&#xff0c;基于DFS的后序遍历法 总结 拓扑排序的应用场景 1&#xff0c;任务调度 …

AGI时代的认知重塑:人类文明的范式转移与思维革命

文章目录 引言:站在文明转型的临界点一、认知危机:当机器开始理解世界1.1 AGI的本质突破:从模式识别到世界建模1.2 人类认知的脆弱性暴露二、认知革命:重构思维的四个维度2.1 元认知升级:从直觉思维到二阶观察2.2 混合智能:人机认知回路的构建2.3 认知安全:防御机器思维…

解释和对比“application/octet-stream“与“application/x-protobuf“

介绍 在现代 Web 和分布式系统的开发中&#xff0c;数据的传输和交换格式扮演着关键角色。为了确保数据在不同系统之间的传输过程中保持一致性&#xff0c;MIME 类型&#xff08;Multipurpose Internet Mail Extensions&#xff09;被广泛应用于描述数据的格式和内容类型。在 …

零基础学CocosCreator·第九季-网络游戏同步策略与ESC架构

课程里的版本好像是1.9&#xff0c;目前使用版本为3.8.3 开始~ 目录 状态同步帧同步帧同步客户端帧同步服务端ECS框架概念ECS的解释ECS的特点EntityComponentSystemWorld ECS实现逻辑帧&渲染帧 ECS框架使用帧同步&ECS 状态同步 一般游戏的同步策略有两种&#xff1a;…

实现限制同一个账号最多只能在3个客户端(有电脑、手机等)登录(附关键源码)

如上图&#xff0c;我的百度网盘已登录设备列表&#xff0c;有一个手机&#xff0c;2个windows客户端。手机设备有型号、最后登录时间、IP等。windows客户端信息有最后登录时间、操作系统类型、IP地址等。这些具体是如何实现的&#xff1f;下面分别给出android APP中采集手机信…

算法基础:贪心|双指针|二分|倍增

贪心 算法思想&#xff1a; 把整个问题分解成多个步骤&#xff0c;在每个步骤都选取当前步骤的最优方案&#xff0c;直到所有步骤结束&#xff1b;每个步骤都不会影响后续步骤。 核心&#xff1a;采取局部最优&#xff0c;最终结果就全局最优。 双指针 反向扫描 同向扫描 二…

在本地校验密码或弱口令 (windows)

# 0x00 背景 需求是验证服务器的弱口令&#xff0c;如果通过网络侧校验可能会造成账户锁定风险。在本地校验不会有锁定风险或频率限制。 # 0x01 实践 ## 1 使用 net use 命令 可以通过命令行使用 net use 命令来验证本地账户的密码。打开命令提示符&#xff08;CMD&#xff0…

Expo运行模拟器失败错误解决(xcrun simctl )

根据你的描述&#xff0c;问题主要涉及两个方面&#xff1a;xcrun simctl 错误和 Expo 依赖版本不兼容。以下是针对这两个问题的解决方案&#xff1a; 解决 xcrun simctl 错误 错误代码 72 通常表明 simctl 工具未正确配置或路径未正确设置。以下是解决步骤&#xff1a; 确保 …

具身智能训练新思路!将生成视频用于训练机器人

将生成视频用于训练具身智能(Embodied AI)确实是近年来备受关注的前沿方向,这一思路通过结合生成式AI(如扩散模型、神经辐射场等)与机器人学习,为解决真实世界数据稀缺、训练成本高等问题提供了新可能。以下从技术逻辑、潜在优势、挑战及案例方向展开分析: 一、技术逻辑…

【设计模式】【行为型模式】观察者模式(Observer)

&#x1f44b;hi&#xff0c;我不是一名外包公司的员工&#xff0c;也不会偷吃茶水间的零食&#xff0c;我的梦想是能写高端CRUD &#x1f525; 2025本人正在沉淀中… 博客更新速度 &#x1f4eb; 欢迎V&#xff1a; flzjcsg2&#xff0c;我们共同讨论Java深渊的奥秘 &#x1f…

git 克隆指定 tag 的项目

git 克隆指定 tag 的项目 一、克隆指定tag的项目二、验证克隆结果 一、克隆指定tag的项目 以 tinyxml2项目 为例说明&#xff1a; git clone --branch V10.0.0 https://github.com/leethomason/tinyxml2.git解释&#xff1a; git clone&#xff1a;这是克隆一个远程仓库的命…

OSPF高级特性(3):安全特效

引言 OSPF的基础我们已经结束学习了&#xff0c;接下来我们继续学习OSPF的高级特性。为了方便大家阅读&#xff0c;我会将高级特性的几篇链接放在末尾&#xff0c;所有链接都是站内的&#xff0c;大家点击即可阅读&#xff1a; OSPF基础&#xff08;1&#xff09;&#xff1a;工…

把 DeepSeek1.5b 部署在显卡小于4G的电脑上

这里写自定义目录标题 介绍准备安装 Ollama查看CUDA需要版本安装CudaToolkit检查Cuda是否装好二、设置Ollama环境变量三、验证是否跑在GPU上ollama如何导入本地下载的模型安装及配置docker安装open-webui启动open-webui开始对话介绍 Deepseek1.5b能够运行在只用cpu和gpu内存小…

WebSocket与Socket.io的区别

文章目录 引言一、WebSocket&#xff1a;原生的实时通信协议&#xff08;一&#xff09;WebSocket 是什么&#xff08;二&#xff09;WebSocket 的工作原理&#xff08;三&#xff09;WebSocket 的使用方法&#xff08;四&#xff09;WebSocket 的优势&#xff08;五&#xff0…