102-ESP32学习开发(SDK)-GPIO

<p><iframe name="ifd" src="https://mnifdv.cn/resource/cnblogs/LearnESP32" frameborder="0" scrolling="auto" width="100%" height="1500"></iframe></p>

 

控制GPIO25输出高低电平

1.原理图

 

 

2.参考官方例程

 

 

 

3.程序

 

 

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
#include "driver/gpio.h"


#define gpio_pin 25

void app_main(void)
{
    //gpio配置结构体
    gpio_config_t io_conf;
    //禁止中断
    io_conf.intr_type = GPIO_PIN_INTR_DISABLE;
    //输出模式
    io_conf.mode = GPIO_MODE_OUTPUT;
    //配置要设置的引脚
    io_conf.pin_bit_mask = (unsigned long long)1<<gpio_pin;
    //禁止下拉
    io_conf.pull_down_en = 0;
    //禁止上拉
    io_conf.pull_up_en = 0;
    //配置gpio(不设置上下拉默认输出低电平)
    gpio_config(&io_conf);

    while(1) {
        gpio_set_level(gpio_pin, 0);//设置引脚输出低电平
        vTaskDelay(3000 / portTICK_RATE_MS);//延时约3S
        gpio_set_level(gpio_pin, 1);//设置引脚输出高电平
        vTaskDelay(3000 / portTICK_RATE_MS);//延时约3S
    }
}

 

 

 

 

控制GPIO25 和 GPIO26 输出高低电平

 

 

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
#include "driver/gpio.h"


#define gpio_pin 25
#define gpio_pin1 26

void app_main(void)
{
    //gpio配置结构体
    gpio_config_t io_conf;
    //禁止中断
    io_conf.intr_type = GPIO_PIN_INTR_DISABLE;
    //输出模式
    io_conf.mode = GPIO_MODE_OUTPUT;
    //配置要设置的引脚
    io_conf.pin_bit_mask = (((unsigned long long)1<<gpio_pin) | ((unsigned long long)1<<gpio_pin1));
    //禁止下拉
    io_conf.pull_down_en = 0;
    //禁止上拉
    io_conf.pull_up_en = 0;
    //配置gpio(不设置上下拉默认输出低电平)
    gpio_config(&io_conf);

    while(1) {
        gpio_set_level(gpio_pin, 0);//设置引脚输出低电平
        gpio_set_level(gpio_pin1, 0);//设置引脚输出低电平

        vTaskDelay(3000 / portTICK_RATE_MS);//延时约3S

        gpio_set_level(gpio_pin, 1);//设置引脚输出高电平
        gpio_set_level(gpio_pin1, 1);//设置引脚输出高电平

        vTaskDelay(3000 / portTICK_RATE_MS);//延时约3S
    }
}

 

 

 

 

补充:

配置gpio还有一个参数  driver

GPIO_DRIVE_CAP_0 弱 weak
GPIO_DRIVE_CAP_1 强
GPIO_DRIVE_CAP_2 默认值
GPIO_DRIVE_CAP_DEFAULT 默认值
GPIO_DRIVE_CAP_3 最强

 

io_conf.driver = GPIO_DRIVE_CAP_3;

 

 

提示

GPIO的模式

GPIO_MODE_INPUT 输入
GPIO_MODE_OUTPUT 输出
GPIO_MODE_OUTPUT_OD 开漏输出
GPIO_MODE_INPUT_OUTPUT_OD 开漏输入输出
GPIO_MODE_INPUT_OUTPUT 输入输出(如果想让模块即做输入检测又做输出控制,需要设置这个模式)

 

 

 

配置GPIO0作为输入输出模式,检测引脚输出状态

 

 

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
#include "driver/gpio.h"


#define gpio_pin 0

void app_main(void)
{
    //gpio配置结构体
    gpio_config_t io_conf;
    //禁止中断
    io_conf.intr_type = GPIO_PIN_INTR_DISABLE;
    //输入输出模式
    io_conf.mode = GPIO_MODE_INPUT_OUTPUT;
    //配置要设置的引脚
    io_conf.pin_bit_mask = (unsigned long long)1<<gpio_pin;
    //禁止下拉
    io_conf.pull_down_en = 0;
    //禁止上拉
    io_conf.pull_up_en = 0;
    //配置gpio(不设置上下拉默认输出低电平)
    gpio_config(&io_conf);

    while(1) {
        gpio_set_level(gpio_pin, 0);//设置引脚输出低电平
        
        printf("获取引脚状态=%d\r\n",gpio_get_level(gpio_pin));


        vTaskDelay(3000 / portTICK_RATE_MS);//延时约3S

        gpio_set_level(gpio_pin, 1);//设置引脚输出高电平
        printf("获取引脚状态=%d\r\n",gpio_get_level(gpio_pin));
    
        vTaskDelay(3000 / portTICK_RATE_MS);//延时约3S
    }
}

 

 

 

 

 

 

配置GPIO0下降沿中断

1,中断类型

GPIO_INTR_DISABLE 禁用GPIO中断
GPIO_INTR_POSEDGE GPIO中断类型:上升沿
GPIO_INTR_NEGEDGE 下降沿
GPIO_INTR_ANYEDGE 上升沿和下降沿
GPIO_INTR_LOW_LEVEL 输入低电平触发
GPIO_INTR_HIGH_LEVEL 输入高电平触发

2,程序

 

 

 

 

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
#include "driver/gpio.h"


#define gpio_pin 0

static xQueueHandle gpio_evt_queue = NULL;

/*gpio中断回调函数*/
static void IRAM_ATTR gpio_isr_handler(void* arg)
{
    uint32_t gpio_num = (uint32_t) arg;
    //把消息存储到消息队列
    xQueueSendFromISR(gpio_evt_queue, &gpio_num, NULL);
}

/*任务函数*/
static void gpio_task_example(void* arg)
{
    uint32_t io_num;
    for(;;) {
        /*消息队列里面有消息*/
        if(xQueueReceive(gpio_evt_queue, &io_num, portMAX_DELAY)) {
            printf("GPIO[%d] intr, val: %d\n", io_num, gpio_get_level(io_num));
        }
    }
}


void app_main(void)
{
    //gpio配置结构体
    gpio_config_t io_conf;
    //下降沿中断
    io_conf.intr_type = GPIO_INTR_NEGEDGE;
    //输入模式
    io_conf.mode = GPIO_MODE_INPUT;
    //配置要设置的引脚
    io_conf.pin_bit_mask = (unsigned long long)1<<gpio_pin;
    //禁止下拉
    io_conf.pull_down_en = 0;
    //上拉
    io_conf.pull_up_en = 1;
    //配置gpio
    gpio_config(&io_conf);


    /*中断里面不能写printf,写了会死机; 所以就用任务+消息队列打印*/
    //创建消息队列
    gpio_evt_queue = xQueueCreate(10, sizeof(uint32_t));
    //创建任务
    xTaskCreate(gpio_task_example, "gpio_task_example", 2048, NULL, 10, NULL);


    //设置中断优先级(1-8级),如果参数写0,则内部自动从1-3级中分配一个优先级; 7级优先级为最高;
    //如果设置为8,则此中断是共享中断,即多个外设都可触发这个中断(处理起来应该会很麻烦,到时候真的使用到再说)
    gpio_install_isr_service(ESP_INTR_FLAG_LEVEL1);
    //移除中断
    gpio_isr_handler_remove(gpio_pin);
    //添加中断
    gpio_isr_handler_add(gpio_pin, gpio_isr_handler, (void*) gpio_pin);


    while(1) {
        //必须加延时,任务不能没有延时,否则导致任务无法切换.
        vTaskDelay(1000 / portTICK_RATE_MS);
    }
}

 

 

 

3,其实主要的就几句话

 

 

 

 

4,动作 按键 (GPIO0)

 

 

 

 

 

5,提示

按理说下降沿中断,读取到的引脚电平只有0才对,但是呢之所有1,是因为

 

posted on 2021-08-02 04:10  广源时代  阅读(1502)  评论(0编辑  收藏  举报

导航

支付宝 QQ群