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,是因為

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM