ESP32修改BootLoader:在boot中添加GPIO和IIC驱动方式

ESP32修改BootLoader:在boot中添加GPIO和IIC驱动方式

1. ESP Bootloader简介

ESP32有着强大的引导加载程序(Bootloader)功能:
主要执行以下任务:
内部模块的最小化初始配置;
根据分区表和 ota_data(如果存在)选择需要引导的应用程序(app)分区;
将此应用程序映像加载到 RAM(IRAM 和 DRAM)中,最后把控制权转交给应用程序。
引导加载程序位于 Flash 的 0x1000 偏移地址处。

2. Bootloader修改方式

这里引用C站一个作者的文章,写的不错:点这里
文章分为上下两篇,下篇。

当涉及到用户有特殊BootLoader功能需求时,需要用户自行修改,修改方式主要有:
通过钩子函数。扩展原来的bootloader流程,打个补丁。
通过覆盖bootloader。重写一个bootloader,覆盖原来bootloader的逻辑来运行。
可以发现,这两种方式一个针对小改动需求,一个针对大改动的需要。
怎么添加自己的功能
如果在应用层开发,会做些什么?

嗯,你会在component目录下新建自己功能名的文件夹
之后呢?不知道了… 那就看看component其他组件都放了啥吧。哦,有CMakeLists.txt,component.mk,还有代码目录。那咱们依葫芦画瓢,也放上这些东西。
然后?既然有Cmake的东西,那就看看里面都说了啥。哦,原来里面就是设置好了要编译哪些源文件,头文件目录在哪里,还有配置好编译选项啊,库路径什么的。

是的,这些东西在bootloader里面都有,方法类似。就我的开发方式而言,mycode放在bootloader_support目录下之后,里面的源文件、头文件、库这些的,都跟应用层开发一样。该在CMakeLists.txt加什么就加。
这跟在boot还是在app其实就没啥关系,是CMake这种跨平台编译方式的基本操作。

3. 我的修改方式

本文主要讲解其中一种修改方式:mycode放在bootloader_support目录下之后,里面的源文件、头文件、库这些的,都跟应用层开发一样。该在CMakeLists.txt加什么就加。

下图是本人在bootloader_support文件夹中添加的个人代码“tc2x”
在这里插入图片描述

并在CmakeLists.txt中添加需要编译的源文件和头文件目录
在这里插入图片描述

此时,完成了在bootloader_support添加自己的代码。

4. 修改bootloader_utility.c文件

如图,我的boot修改内容主要是基于boot跳转APP失败时,跑一段自己的驱动和应用。
在这里插入图片描述

5. 在bootloader中实现GPIO控制

实现代码:

#include <string.h>
#include <stdint.h>
#include <limits.h>
#include "esp_attr.h"
#include "esp_log.h"
#include "esp_err.h"
#include "esp_rom_sys.h"
#include "esp_rom_uart.h"
#include "sdkconfig.h"
#include "esp_rom_crc.h"
#include "esp_rom_gpio.h"
#include "esp_rom_efuse.h"
#include "soc/gpio_periph.h"
#include "soc/soc_caps.h"
#include "hal/gpio_ll.h"
#include "hal/gpio_types.h"#define CFG_I2C_SCL_PORT   GPIO_PORT_0
#define CFG_I2C_SCL_PIN    18
#define CFG_I2C_SDA_PORT   GPIO_PORT_0
#define CFG_I2C_SDA_PIN    17#define PIN_OUTPUT_ENABLE(PIN_NAME)               SET_PERI_REG_MASK(PIN_NAME,FUN_IE)
#define PIN_OUTPUT_DISABLE(PIN_NAME)              CLEAR_PERI_REG_MASK(PIN_NAME,FUN_IE)
void HAL_GPIO_WritePin(int gpio_port, int gpio_pin, int level)
{gpio_ll_set_level(&GPIO, gpio_pin, level);
}
uint8_t HAL_GPIO_ReadPin(int gpio_port, int gpio_pin)
{return gpio_ll_get_level(&GPIO, gpio_pin);
}
void HAL_GPIO_SetOut(int gpio_port, int gpio_pin)
{esp_rom_gpio_pad_select_gpio(gpio_pin);// if (GPIO_PIN_MUX_REG[gpio_pin]) {//     PIN_INPUT_DISABLE(GPIO_PIN_MUX_REG[gpio_pin]);//     PIN_SLP_OUTPUT_ENABLE(GPIO_PIN_MUX_REG[gpio_pin]);// }esp_rom_gpio_pad_pullup_only(gpio_pin);gpio_ll_output_enable(&GPIO, gpio_pin);//gpio_ll_sleep_output_enable(&GPIO, gpio_pin);
}
void HAL_GPIO_SetIn(int gpio_port, int gpio_pin)
{esp_rom_gpio_pad_select_gpio(gpio_pin);// if (GPIO_PIN_MUX_REG[gpio_pin]) {//     PIN_INPUT_ENABLE(GPIO_PIN_MUX_REG[gpio_pin]);//     PIN_SLP_OUTPUT_DISABLE(GPIO_PIN_MUX_REG[gpio_pin]);// }esp_rom_gpio_pad_pullup_only(gpio_pin);gpio_ll_input_enable(&GPIO, gpio_pin);// gpio_ll_sleep_input_enable(&GPIO, gpio_pin);
}

6. 在bootloader中实现IIC控制

代码实现:

void I2C_DELAY(uint32_t us)
{esp_rom_delay_us(us);
}
static const char* TAG = "[GPIOIIC]";#define IIC1 0
#define IIC2 1
#define IICX 0XFF 
#define GPIO_PIN_SET   1 
#define GPIO_PIN_RESET 0  
static int I2C_SCL_PORT = 0;
static int I2C_SCL_PIN  = 18;
static int I2C_SDA_PORT = 0;
static int I2C_SDA_PIN  = 17;
#define us_num 2
#define I2C_READ_BIT                 (0x01)        //!< If this bit is set in the address field, transfer direction is from slave to master.
#define I2C_WRITE_BIT                (0xFE)static void I2C_SEL(uint8_t I2Cn, uint8_t scl, uint8_t sda)
{switch(I2Cn){case IIC1:I2C_SCL_PORT = CFG_I2C_SCL_PORT;I2C_SCL_PIN  = CFG_I2C_SCL_PIN;I2C_SDA_PORT = CFG_I2C_SDA_PORT;I2C_SDA_PIN  = CFG_I2C_SDA_PIN;break;case IIC2:I2C_SCL_PORT = CFG_I2C_SCL_PORT;I2C_SCL_PIN  = CFG_I2C_SCL_PIN;I2C_SDA_PORT = CFG_I2C_SDA_PORT;I2C_SDA_PIN  = CFG_I2C_SDA_PIN;break;default: I2C_SCL_PORT = CFG_I2C_SCL_PORT;I2C_SCL_PIN  = scl;I2C_SDA_PORT = CFG_I2C_SDA_PORT;I2C_SDA_PIN  = sda;break;}
}
static void I2C_SCL_OUTPUT(void)
{HAL_GPIO_SetOut(I2C_SCL_PORT, I2C_SCL_PIN);
}
static void I2C_SCL_INPUT(void)
{HAL_GPIO_SetIn(I2C_SCL_PORT, I2C_SCL_PIN);
}
static void I2C_SCL_HIGH(void)
{HAL_GPIO_WritePin(I2C_SCL_PORT, I2C_SCL_PIN, GPIO_PIN_SET);
}
static void I2C_SCL_LOW(void)
{HAL_GPIO_WritePin(I2C_SCL_PORT, I2C_SCL_PIN, GPIO_PIN_RESET);
}
static void I2C_SDA_OUTPUT(void)
{HAL_GPIO_SetOut(I2C_SDA_PORT, I2C_SDA_PIN);
}
static void I2C_SDA_INPUT(void)
{HAL_GPIO_SetIn(I2C_SDA_PORT, I2C_SDA_PIN);
}
static void I2C_SDA_HIGH(void)
{HAL_GPIO_WritePin(I2C_SDA_PORT, I2C_SDA_PIN, GPIO_PIN_SET);
}
static void I2C_SDA_LOW(void)
{HAL_GPIO_WritePin(I2C_SDA_PORT, I2C_SDA_PIN, GPIO_PIN_RESET);
}
static uint8_t I2C_SDA_READ(void)
{return HAL_GPIO_ReadPin(I2C_SDA_PORT, I2C_SDA_PIN);
}
static uint8_t I2C_SCL_READ(void)
{return HAL_GPIO_ReadPin(I2C_SCL_PORT, I2C_SCL_PIN);
}
void I2C_SCL_SDA_TEST(void)
{I2C_SCL_OUTPUT();I2C_SDA_OUTPUT();I2C_SDA_HIGH();I2C_SDA_LOW();
}
static void I2C_Init(int scl, int sda)
{I2C_SEL(IICX, scl, sda);I2C_SCL_INPUT();I2C_SDA_INPUT();I2C_SCL_OUTPUT();I2C_SCL_HIGH();I2C_SDA_OUTPUT();I2C_SDA_HIGH();
}
static bool i2c_master_clear_bus(void);
static bool i2c_master_issue_startcondition(void);
static bool i2c_master_issue_stopcondition(void);
static bool i2c_master_clock_byte(uint_fast8_t databyte);
static bool i2c_master_clock_byte_in(uint8_t *databyte, bool ack);
static bool i2c_master_wait_while_scl_low(void);
bool gpioi2c_master_init(uint8_t sclpin, uint8_t sdapin)
{I2C_Init(sclpin, sdapin);return i2c_master_clear_bus();
}
static bool i2c_master_clear_bus(void)
{bool bus_clear;I2C_SDA_HIGH();I2C_SCL_HIGH();I2C_DELAY(1);if (I2C_SDA_READ() == 1 && I2C_SCL_READ() == 1){bus_clear = true;}else if (I2C_SCL_READ() == 1){bus_clear = false;ESP_LOGE(TAG, "i2c_master_clear_bus 1 fail");for (uint_fast8_t i = 18; i--;){I2C_SCL_LOW();I2C_DELAY(1);I2C_SCL_HIGH();I2C_DELAY(1);if (I2C_SDA_READ() == 1){bus_clear = true;break;}}}else{bus_clear = false;ESP_LOGE(TAG, "i2c_master_clear_bus 2 fail");}return bus_clear;
}
static bool i2c_master_issue_startcondition(void)
{I2C_SCL_OUTPUT();I2C_SDA_OUTPUT();I2C_SDA_HIGH();I2C_DELAY(1);if (!i2c_master_wait_while_scl_low()){ESP_LOGE(TAG, "i2c_master_wait_while_scl_low fail");return false;}I2C_SDA_LOW();I2C_DELAY(1);// Other module function expect SCL to be lowI2C_SCL_LOW();I2C_DELAY(1);return true;
}
static bool i2c_master_issue_stopcondition(void)
{I2C_SDA_LOW();I2C_DELAY(1);if (!i2c_master_wait_while_scl_low()){ESP_LOGE(TAG, "i2c_master_issue_stopcondition fail");return false;}I2C_SDA_HIGH();I2C_DELAY(1);return true;
}
static bool i2c_master_clock_byte(uint_fast8_t databyte)
{bool transfer_succeeded = true;I2C_SDA_OUTPUT();// MSB firstfor (uint_fast8_t i = 0x80; i != 0; i >>= 1){I2C_SCL_LOW();I2C_DELAY(1);if (databyte & i){I2C_SDA_HIGH();}else{I2C_SDA_LOW();}if (!i2c_master_wait_while_scl_low()){ESP_LOGE(TAG, "i2c_master_clock_byte fail");transfer_succeeded = false; // Timeoutbreak;}}I2C_SCL_LOW();I2C_DELAY(1);I2C_SDA_INPUT();I2C_DELAY(1);transfer_succeeded &= i2c_master_wait_while_scl_low();transfer_succeeded &= !(I2C_SDA_READ());I2C_SCL_LOW();I2C_DELAY(1);I2C_SDA_OUTPUT();return transfer_succeeded;
}
static bool i2c_master_clock_byte_in(uint8_t *databyte, bool ack)
{uint_fast8_t byte_read          = 0;bool         transfer_succeeded = true;I2C_SDA_INPUT();for (uint_fast8_t i = 0x80; i != 0; i >>= 1){if (!i2c_master_wait_while_scl_low()){ESP_LOGE(TAG, "i2c_master_clock_byte_in 1 fail");transfer_succeeded = false;break;}if (I2C_SDA_READ()){byte_read |= i;}else{// No need to do anything}I2C_SCL_LOW();I2C_DELAY(1);}I2C_SDA_OUTPUT();*databyte = (uint8_t)byte_read;if (ack){I2C_SDA_LOW();}else{I2C_SDA_HIGH();}I2C_DELAY(1);if (!i2c_master_wait_while_scl_low()){ESP_LOGE(TAG, "i2c_master_clock_byte_in 2 fail");transfer_succeeded = false; // Timeout}I2C_SCL_LOW();I2C_DELAY(1);return transfer_succeeded;
}
static bool i2c_master_wait_while_scl_low()
{uint8_t timeout_counter = 30;I2C_SCL_HIGH();I2C_DELAY(1);while (I2C_SCL_READ() == 0){if (timeout_counter-- == 0){ESP_LOGE(TAG, "i2c_master_wait_while_scl_low fail");return false;}else{I2C_DELAY(3);}}return true;
}
bool gpioi2c_master_read(uint8_t address, uint8_t reg_addr, uint8_t * data, uint8_t data_length, bool issue_stop_condition)
{bool transfer_succeeded = true;transfer_succeeded &= i2c_master_issue_startcondition();transfer_succeeded &= i2c_master_clock_byte(address&I2C_WRITE_BIT);/** send the address of register which you want to read*/transfer_succeeded &= i2c_master_clock_byte(reg_addr);/** must send 2 times start condition */transfer_succeeded &= i2c_master_issue_startcondition();transfer_succeeded &= i2c_master_clock_byte(address|I2C_READ_BIT);/* Transfer direction is from Slave to Master */while (data_length-- && transfer_succeeded){// To indicate to slave that we've finished transferring last data byte// we need to NACK the last transfer.if (data_length == 0){transfer_succeeded &= i2c_master_clock_byte_in(data, (bool)false);}else{transfer_succeeded &= i2c_master_clock_byte_in(data, (bool)true);}data++;}if (issue_stop_condition || !transfer_succeeded){transfer_succeeded &= i2c_master_issue_stopcondition();}return transfer_succeeded;
}bool gpioi2c_master_write(uint8_t address, uint8_t reg_addr, uint8_t * data, uint8_t data_length, bool issue_stop_condition)
{bool transfer_succeeded = true;transfer_succeeded &= i2c_master_issue_startcondition();
ESP_LOGD(TAG, "gpioi2c_master_write transfer_succeeded1=%d",transfer_succeeded);transfer_succeeded &= i2c_master_clock_byte(address&I2C_WRITE_BIT);
ESP_LOGD(TAG, "gpioi2c_master_write transfer_succeeded2=%d",transfer_succeeded);/** send the address of register which you want to write */transfer_succeeded &= i2c_master_clock_byte(reg_addr);
ESP_LOGD(TAG, "gpioi2c_master_write transfer_succeeded3=%d",transfer_succeeded);/* Transfer direction is from Master to Slave */while (data_length-- && transfer_succeeded){transfer_succeeded &= i2c_master_clock_byte(*data);data++;}if (issue_stop_condition || !transfer_succeeded){transfer_succeeded &= i2c_master_issue_stopcondition();
ESP_LOGD(TAG, "gpioi2c_master_write transfer_succeeded4=%d",transfer_succeeded);}return transfer_succeeded;
}

7. 总结

通过以上方式,可以实现在esp32修改BootLoader代码,并实现GPIO和IIC驱动,从而实现自己的修改需求。

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

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

相关文章

ESP32 基础篇: 启动时 rst cause 和 boot mode

本文参考 ESP32 技术参考手册 和 ESP32-WROOM-32 datasheet 简介 在 ESP32 启动时, ROM CODE 会读取 GPIO 状态和 rst cause 状态, 进而决定 ESP32 工作模式。 通过了解和掌握 rst cause 和 boot mode, 有助于定位某些系统问题。 例如: ESP32 启动时会有如下打印: rst:0x1 (…

微信小程序 自动对对联

微信小程序 自动对对联 微信扫描二维码体验吧

使用scoped让样式只对当前页面其作用时/deep/ 可以使其对子组件起作用

官方文档https://vue-loader.vuejs.org/guide/scoped-css.html#mixing-local-and-global-styles 使其样式只对当前页面起作用 影响到里面的子组件

uniapp-微信小程序,对子组件实现onShow效果

背景&#xff1a;使用uni-app开发一个微信小程序&#xff0c;使用的vue子组件&#xff0c;发现在子组件里面不能使用onShow生命周期&#xff0c;但是可以在page里面可以调用&#xff0c;需要在使用uni.navigateBack()返回后&#xff0c;该组件能请求接口数据刷新 解决&#xf…

基于强化学习的大规模多任务机器人训练

发布人:Google 机器人团队高级研究员 Karol Hausman 和研究员 Yevgen Chebotar 通用机器人要想发挥最大的作用,就需要能够完成一系列的任务,如清洁、维护和运送。但是,使用离线强化学习 (RL)(智能体使用以前收集的数据开展训练,在试验和错误中学习的一种方法)来训练,即…

数据库MySQL详解

全网最详细MySQL教程&#xff0c;应付大学考试、考研复试、求职笔试应该说是完全足够的&#xff0c; 有兴趣的朋友可以看我的MySQL专栏&#xff0c;都是MySQL原理和底层一点的东西&#xff0c;可能比一般的面试文都深入。 本文篇幅较长&#xff0c;笔误之处在所难免&#xff0c…

关于对Vue中slot插槽理解

关于slot插槽理解 1.何时需要使用插槽 在开发中&#xff0c;我们需要将共性内容抽取到组件中&#xff0c;将不同的暴露为插槽。 插槽的益处便是&#xff0c;一旦预留了插槽&#xff0c;使用者便可以根据自己的需求来决定插槽中插入的的内容2. slot的基本使用 <div idapp&…

记录一次 AGP 调研过程中的思考,我从一个事故搞出了一个故事!

背景 看过我博客的老铁应该知道&#xff0c;我在 18 年五月写过一个小 gradle 插件https://github.com/yanbober/app-tiny-R-gradle-plugin&#xff0c;其作用就是将 app 生成的 R 常量进行内联操作。对&#xff0c;就是前不久很火的滴滴 booster 和字节跳动 ByteX 提供的 R 资…

hive中对子查询如in,exists等支持情况和使用

案例情况&#xff1a;同事使用公司数据探查跑一段代码&#xff0c;部分代码如下&#xff0c;报错&#xff0c;显示不支持in内的子查询。但是直接用虚拟机去跑的话代码没有任何报错&#xff0c;也出结果&#xff0c;很奇怪。 SELECT t1.SIGN_CODE AS bus_src,t1.ORGANIZATI…

overflow属性对before、after伪元素的影响

div中有before伪元素&#xff0c;如图&#xff1a; 当该div内容增多时&#xff0c;添加了纵向滚动条的样式&#xff0c;如下&#xff1a; max-height:300px; overflow: auto; 随后伪元素就消失了&#xff0c;如图&#xff0c;小箭头不见了。 overflow的说明&#xff1a; http…

如何提升对编程的兴趣,在编程中找到快乐?

上周有同学和我交流&#xff0c;问我怎么能在编程中找到快乐&#xff0c;提升编程的兴趣。 今天正好又是周末&#xff0c;对于这个问题&#xff0c;小编就要祭出大招了。 首先&#xff0c;打开浏览器&#xff0c;访问一个神奇的地址&#xff1a; https://github.com/ 。 对的…

一个会对对联的AI项目

编辑文章 声明&#xff1a;本文首发微信公众号【菜鸟要飞】&#xff0c;如有转载&#xff0c;请标明出处&#xff01; 快过年了&#xff0c;贴对联是必不可少的传统风俗。不知道各位读者有没有自己写过对联呢&#xff1f;写对联可不是一件简单的事情&#xff0c;如果不是满腹…

字符串匹配算法知多少?

文章目录 BF算法RK算法编辑器中的全局替换方法&#xff1a;BM算法坏字符好后缀规则代码实现 KMP算法 一说到字符串匹配算法&#xff0c;不知道会有多少小伙伴不由自主的想起那个kmp算法呢&#xff1f; 想到是很正常的&#xff0c;谁让它那么优秀呢。 BF算法 不要被事物的表面…

量化股票查询代码是什么?

量化股票查询代码是什么&#xff1f;接下来用一些代码来分析一下&#xff0c;如下&#xff1a; 做空95&#xff1a;HHV((HIGHLOWOPEN2*CLOSE)/5H-L,5),COLORBLUE;做空68: HHV((HIGH-LOWOPEN2*CLOSE)/5*2-L,5),COLORRED&#xff1b; 平衡点&#xff1a;LLV((HIGHLOWOPEN2*CLOSE…

voipdiscount免费拨打全球电话(无需手机注册)

我测试过了的&#xff0c;能给我手机打通&#xff0c;我也给无题打了的感觉还不错。现推荐给大家&#xff01; voipdiscount免费拨打全球电话&#xff08;无需手机注册&#xff09;通话效果极好到www.voipdiscount.com下载一个软件voipdiscount,申请一个用户&#xff08;不需手…

企业使用虚拟码号的优势!

其实用不用隐私码号&#xff0c;或者怎么用隐私码号&#xff0c;是和企业的基本业务场景有关的。我们在这将近5年的服务过程中&#xff0c;遇上的行业千差万别&#xff0c;需求也是完全不同。如果非要总结一些优势的话&#xff0c;那么简单的做个应用场景分类。 隐私码号&#…

VOS网络电话如何注册IMS

IMS注册需要IMS方提供账号的注册信息 比如 用户名称&#xff1a;862584372919 认证密码&#xff1a;123456 服务器地址&#xff1a;ims.jx.chinamobile.com 认证用户&#xff1a;862584372919 SIP代理&#xff1a;172.16.5.144 第一步: 在vos的业务管理—>注册管理中按下图…

趣图:程序员假发攻略

&#xff08;给程序员的那些事加星标&#xff0c;每天看趣图&#xff09; 15 号字 ↓↓↓ (漫画原作者&#xff1a;tango2010 投稿&#xff1a;遇见) 往期趣图&#xff08;点击下方图片可跳转阅读&#xff09; 关注「程序员的那些事」加星标&#xff0c;不错过趣图 &#xff0…

老大“秃”伤悲的年轻人,正靠假发维持最后的体面

本文转载自虎嗅网 如今90后的潮流&#xff0c;已经逐渐让人看不懂了。 现在见面第一句话都不是“今天你基金绿了吗&#xff1f;”“今天你cp发糖了吗&#xff1f;”&#xff0c;而是&#xff1a; “你今天戴的是假发吗&#xff1f;” 随着越来越多的90后涌入脱发大军&#…

计算机毕业设计php假发销售商城网站(源码+系统+mysql数据库+Lw文档)

项目介绍 假发销售网站使用php,mysql实现了如用户注册、用户登录、假发商品的预览查询、对假发商品的购买通过购物车实现、可进入留言本留言等等&#xff0c;从而实现了网站与客户之间的交流和沟通。 功能&#xff1a;客户功能和管理员功能两个部分。 设计题目&#xff1a;假…