2021年9月10日 星期五

ESP32 CAMRA 二維碼(QR Code)辨識之門鎖控制 02-週邊裝置的控制

網頁最後修改時間:2021/09/10

ESP32-CAM 二維碼辨識之門鎖控制(下面簡稱 ESP32QRDoorLock)的主要週邊裝置包括:攝像頭(不在本篇討論)、門磁(磁簧)開關、WS2812B和電磁鎖與其驅動模組。

本篇是該主題系列的第二篇,內容是關於這些週邊裝置如何用 ESP32-CAM 做控制的說明,有:


/*-/--*-*/*/*/*/***//-*-*-**-*/*-*-/*/*/*-*-/-////--/**/**--**/--///--//**----**//--**//**----***//*-**//*

** 本篇內容都是搭配 ESP32-CAM 燒錄與接腳擴充二合一底板 (下文簡稱底板)所進行的測試,參數設定不一定適用於其他的搭配!

/*-/--*-*/*/*/*/***//-*-*-**-*/*-*-/*/*/*-*-/-////--/**/**--**/--///--//**----**//--**//**----***//*-**//*


【(00)簡介】

ESP32-CAM 因為另外搭配了記憶卡和攝像頭的緣故,其實能用的接腳所剩無幾,若想要外接週邊裝置時,其實有點難度!但若考慮捨棄不用記憶卡,則可空出這些接腳另作他用。不過也因為這幾個記憶卡用的接腳或多或少在電路上都有再經過設計,因此在外接週邊裝置到 ESP32-CAM 上時,參考電路圖或是其文件的說明就顯得非常重要,避免不能用而多走了冤枉路!

** 關於 ESP32-CAM 開發板說明書、電路圖和其他資料,請上下面網址去下載。

到 ESP32-CAM 官網下載資料文件


【(01)參考接線】

電磁鎖(<DOORLOCK>)和 WS2812B 全彩 LED(<RGBLED>)要選用 ESP32-CAM 上可作為輸出的接腳;門磁(磁簧)開關(<LIMITSWITCH>)要選用 ESP32-CAM 上可作為輸入的接腳,至於到底要用哪三個?請參照 ESP32-CAM 電路圖再自己選擇。

底板上的每個接腳都附有 <+5V><GND> 的電源接腳對,所以請參考下圖來接線即可。


【(02)參考程式碼】

程式碼是以 x01_ESP32CamOTAProgramming.ino 為基礎,額外再加入驅動和測試的程式部分所組成。


/*--*//**---/*///**---*-*////***--*/*///***----*///--*/*///**--*/*//**--**/*//
*(02-01)WS2812B 全彩 LED 驅動

WS2812B 全彩 LED 驅動的部分,採用比較有名的函式庫 FastLED,所以需要在程式裡開頭處引入它的標頭檔

16
17
18
#include <Arduino.h>
#include <WiFi.h>
#include <FastLED.h>

接著,在下面宣告幾個會用到的全局變數。

首先,記得要先把選好的接腳號碼取代掉 #1#2

31
32
33
34
35
36
/*================== FASTLED =================*/
const int NUM_LEDS      = #1;       // 填入 LED 的數量取代 #1
const uint8_t DATA_PIN  = #2;       // 填入 WS2812 通訊接腳號碼取代 #2

// 定義 LED 陣列
CRGB ws2812[NUM_LEDS];

然後再定義幾個常用的顏色,和一個測試會用到的儲放顏色的陣列。

65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
/*================== FASTLED =================*/
// Colors definition for WS2812
#define BLACK  CRGB (  0,   0 ,  0)     // Black
#define WHITE  CRGB (255, 255, 255)     // White
#define RED    CRGB (255,   0,   0)     // Red
#define ORANGE CRGB (255, 127,   0)     // Orange
#define YELLOW CRGB (255, 255,   0)     // Yellow
#define GREEN  CRGB (  0, 255,   0)     // Green
#define BLUE   CRGB (  0,   0, 255)     // Blue
#define INDIGO CRGB (  0, 255, 255)
#define VIOLET CRGB (127,   0, 255)     // Violet

CRGB x02_colors[9] = {
    RED, ORANGE, YELLOW, GREEN,
    BLUE, INDIGO, VIOLET, WHITE, BLACK
};

預先宣告 blink () 函式,避免編譯時出現找不到該函式的錯誤。

84
void blink (const int8_t blink_count, const TickType_t delay, const CRGB color);

撰寫 blink () 函式的主程式。

112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
/**========================================================================
 **                           blink()
 *?  系統運行狀態的指示
 *@param blink_count const int8_t  
 *@param color       CRGB
 *========================================================================**/
void blink (const int8_t blink_count,   // 閃爍次數
            const CRGB color)           // WS2812 顏色
{
    uint8_t state = LOW;
    for (int x = 0; x < (blink_count << 1); ++x) {
        if (state ^= HIGH) {
            ws2812[0] = color;
        } else {
            ws2812[0] = BLACK;
        }
        FastLED.show();
        delay (50);
    }
}

ESP32QRDoorLock 預設只使用一顆 WS2812B 全彩 LED。以現有的硬體配置,是可以直接用 PC 的 USB 埠來做整體的供電,不會有所問題!

不過一但確定要增加 LED 的數量,那麼當電流不足以供給整體電路電流消耗的話,就有可能會導致整個系統重置或是崩潰。所以如果出現這種非正常程式運作的情況導致的系統重置,請先使用預設數量的 LED 進行測試,確定是電源導致的問題後,再考慮外加電源給 WS2812B。

FastLED 提供不少的 LED 燈色變化,不過在 ESP32QRDoorLock 只使用來做閃爍和顏色的變化而已,如果看倌有需要可以再自行做變化。


/*--*//**---/*///**---*-*////***--*/*///***----*///--*/*///**--*/*//**--**/*//
*(02-02)電磁鎖驅動

電磁鎖就像是繼電器一樣,不能使用 GPIO 直接驅動,需要外接驅動模組才不至於損壞 GPIO,而且也才能保證有足夠的電流驅動電磁鎖。

ESP32QRDoorLock 所採用的電磁鎖是屬於彈簧自動回歸式的。觸發時,電磁鎖頭會縮回,門屬於會自動彈出的設計會自然開啟;非觸發時,電磁鎖頭會自動彈出,由於電磁鎖頭是一個滑頭形狀,此時就可以用手動的方式將門關回,會自動鎖住。所以設計上,必須確保電磁鎖頭有足夠的觸發時間讓門彈出。

首先,宣告和定義幾個需要的全局變數,記得要先把選好的接腳號碼取代掉 #3

45
46
47
/*================== GPIO =================*/
/** 電磁鎖 */
const uint8_t DOORLOCK      = #3;   // 填入電磁鎖驅動模組控制接腳號碼取代 #3

接著就可以撰寫門鎖控制的程式。

這部分的程式預期要用於電動鎖和電磁鎖兩種鎖頭,不過現只實現電磁鎖一種,使用時只要填入 false 做引數就可以。

133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
/**========================================================================
 **                           doorControl()
 *?  門鎖開關控制
 *@param isMotor    const bool
 *      門鎖的形式。true: DC Motor; false: 電磁開關
 *========================================================================**/
void doorControl (const bool isMotor) {
    if (isMotor) {
        // 如果使用電動鎖頭,寫程式在這裡
        // ...
    } else {
        digitalWrite (DOORLOCK, HIGH);
        delay (1000);
        digitalWrite (DOORLOCK, LOW);
    }
}


/*--*//**---/*///**---*-*////***--*/*///***----*///--*/*///**--*/*//**--**/*//
*(02-03)門磁開關訊號接收

門磁裡面內嵌有一個磁簧開關,沒有磁性物體靠近時是開路的狀態,反之就是閉路。為了要能得知它的狀態,要選擇 ESP32-CAM 可以做為輸入且能上拉的接腳。

首先,宣告和定義幾個需要的全局變數,記得要先把選好的接腳號碼取代掉 #4

48
49
50
51
52
53
/** 門磁(磁簧)開關 */
const uint8_t LIMITSWITCH   = #4;   // 填入門磁開關訊號接接腳號碼取代 #4
// 門的狀態
#define OPEN                HIGH
#define CLOSE               LOW
#define dDoorState          digitalRead(LIMITSWITCH)

門磁訊號的接收可以使用輪詢或是中斷的方式來取得,但對於這個例子來說,採用中斷服務會比較適合。

另外,開關在閉合的時候會發生訊號抖動的情形,容易造成重複觸發誤動作,這樣的情形可用硬體電路或軟體程式的方式來去除抖動訊號;對於這個例子來說,採用軟體程式比較方便。

首先,宣告幾個中斷服務函式會用到的全局變數。

55
56
57
/*================== lsInterrupt =================*/
volatile bool bIntTrigger = false;
unsigned long ulLastMicros;

接著,建立 ESP32 中斷服務函式。

門磁開關的去抖動時間設定為 150ms。不同的開關會有不同的抖動時間,要依實際情況適當地做調整,才能正確的取得穩定的訊號。

在取得穩定的訊號之後,會設定一個旗標並且解除中斷服務函式。中斷服務函式只有在需要的時候才會由程式裡做啟動,一但完成後就會自動被解除。

100
101
102
103
104
105
106
107
108
109
110
/**========================================================================
 **                           lsInterrupt()
 *?  中斷處理函式
 *========================================================================**/
void IRAM_ATTR lsInterrupt () {
    if((long)(micros() - ulLastMicros) >= 150000L) {
        bIntTrigger = true;
        detachInterrupt (digitalPinToInterrupt(LIMITSWITCH));
    }
    ulLastMicros = micros();
}


/*--*//**---/*///**---*-*////***--*/*///***----*///--*/*///**--*/*//**--**/*//
*(02-04)接腳與函式初始化

根據之前所討論關於接腳的使用,初始化接腳和函式庫,並關閉 LED。

86
87
88
89
90
91
92
93
94
95
96
97
98
/**========================================================================
 **                           ioInit()
 *?  ESP32 接腳和 FastLED 初始化
 *========================================================================**/
void ioInit () {
    pinMode (DOORLOCK, OUTPUT);
    digitalWrite (DOORLOCK, LOW);
    pinMode (LIMITSWITCH, INPUT_PULLUP);
    FastLED.addLeds<WS2812B, DATA_PIN, GRB>(ws2812, NUM_LEDS);
    //* 開機關閉 LED
    ws2812[0] = BLACK;
    FastLED.show ();
}


/*--*//**---/*///**---*-*////***--*/*///***----*///--*/*///**--*/*//**--**/*//
*(02-05)外部裝置功能測試函式

藉由使用者的輸入,決定要測試的外部裝置。

191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
/**========================================================================
 **                           x02_functionTest
 *?  選擇外部裝置功能測試
 *========================================================================**/
void x02_functionTest () {
    if (Serial.available()) {
        char c = Serial.read ();
        switch (c) {
            case 0x31:
                Serial.print (F("Test LED, "));
                x02_testLED ();
                Serial.println (F("done."));
                break;
            case 0x32:
                Serial.print (F("Test Door Lock/Unlock"));
                x02_testDoorLockUnlock ();
                Serial.println (F("done."));
                   break;
             default:
                return;
        }
        Serial.println (F("Enter 1 for LED test, 2 for door Lock/Unlock test"));
    }
}

選擇 1,依序閃爍顯示 x02_colors[] 陣列裡面的顏色。

150
151
152
153
154
155
156
157
158
159
/**========================================================================
 **                           testLED
 *?  顏色顯示測試
 *========================================================================**/
void x02_testLED () {
    for (int i = 0; i < 9; i++) {
        blink (5, x02_colors[i]);
        delay (1450);
    }
}

選擇 2,發送門開啟的訊號並等待門被開啟;門一但確認開啟後,閃綠燈且啟用中斷服務函式;等待門被關閉且中斷旗標被設置,閃紅燈且完成開關門測試。

161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
/**========================================================================
 **                           x02_testDoorLockUnlock
 *?  開關門測試
 *========================================================================**/
void x02_testDoorLockUnlock () {
    Serial.print (F("Check door status, "));
    if (dDoorState == OPEN) {
        Serial.println(F("OPEN\nCLOSE DOOR TRY AGAIN!\n\n"));
        return;
    }
    Serial.println (F("CLOSE"));
    Serial.println (F("01-Open the door"));
    doorControl (false);
    // 等待門開啟(要移開門磁開關)
    while (dDoorState == CLOSE);
    // 亮燈表示門已開啟
    blink (5, GREEN);
    Serial.println (F("02-Sets an interrupt for external pin"));
    // 啟用中斷服務函式
    attachInterrupt (
        digitalPinToInterrupt (LIMITSWITCH), 
        lsInterrupt, 
        FALLING
    );
    Serial.println (F("03-Waiting for door close"));
    while (dDoorState == OPEN || bIntTrigger == false);
    blink (5, RED);
    Serial.print (F("04-Closed the door, "));
}


/*--*//**---/*///**---*-*////***--*/*///***----*///--*/*///**--*/*//**--**/*//
*(02-06)完整程式碼與實測影片

以 x01_ESP32CamOTAProgramming.ino 為基礎加入上面所說的部份,將 ioInit () 新增到 setup () 中,x02_functionTest () 新增到 loop () 中,就完成整個程式,請新增一個 Arduino IDE 檔案並儲存下面程式碼為 x02_ESP32CamPeripheral

/**------------------------------------------------------------------------
* x02_ESP32CamPeripheral.ino
* ESP32-CAM 週邊裝置的測試。
*
* {功能介紹}
* + [o] 測試 FastLED 函式庫驅動 WS2812B RGB LED 以不同顏色顯示並閃爍;
* + [o] 測試電磁鎖開鎖 -> LS OPEN (閃綠燈) -> LS CLOSE 偵測關門 (閃紅燈);
* + [o] OTA 上傳;
*
* Adapted by Ruten.Proteus
* Test Date: 2021/09/08
*
* {Result} Wrok !!!
*------------------------------------------------------------------------**/
#include <Arduino.h>
#include <WiFi.h>
#include <FastLED.h>
/** 是否使用 OTA 功能? 不使用就註解下一行 */
#define ENABLE_OTA
#ifdef ENABLE_OTA
#include <ArduinoOTA.h>
#endif
/**------------------------------------------------------------------------
* GLOBAL VARIABLES
*------------------------------------------------------------------------**/
/*================== WIFI =================*/
//** 無線分享器名稱與密碼 */
const char* SSID = "填入無線分享器名稱";
const char* PASSWORD = "填入無線分享器連線密碼";
// 裝置名稱
const char* DEVICENAME = "ESP32QRDoorLock";
/*================== FASTLED =================*/
const int NUM_LEDS = #1; // 填入 LED 的數量取代 #1
const uint8_t DATA_PIN = #2; // 填入 WS2812 通訊接腳號碼取代 #2
// 定義 LED 陣列
CRGB ws2812[NUM_LEDS];
/*================== GPIO =================*/
/** 電磁鎖 */
const uint8_t DOORLOCK = #3; // 填入電磁鎖驅動模組控制接腳號碼取代 #3
/** 門磁(磁簧)開關 */
const uint8_t LIMITSWITCH = #4; // 填入門磁開關訊號接接腳號碼取代 #4
// 門的狀態
#define OPEN HIGH
#define CLOSE LOW
#define dDoorState digitalRead(LIMITSWITCH)
/*================== lsInterrupt =================*/
volatile bool bIntTrigger = false;
unsigned long ulLastMicros;
/*======================== END OF GLOBAL VARIABLES ========================*/
/**------------------------------------------------------------------------
* ENUM AND STRUCT
*------------------------------------------------------------------------**/
/*================== FASTLED =================*/
// Colors definition for WS2812
#define BLACK CRGB ( 0, 0 , 0) // Black
#define WHITE CRGB (255, 255, 255) // White
#define RED CRGB (255, 0, 0) // Red
#define ORANGE CRGB (255, 127, 0) // Orange
#define YELLOW CRGB (255, 255, 0) // Yellow
#define GREEN CRGB ( 0, 255, 0) // Green
#define BLUE CRGB ( 0, 0, 255) // Blue
#define INDIGO CRGB ( 0, 255, 255)
#define VIOLET CRGB (127, 0, 255) // Violet
CRGB x02_colors[9] = {
RED, ORANGE, YELLOW, GREEN,
BLUE, INDIGO, VIOLET, WHITE, BLACK
};
/*========================= END OF ENUM AND STRUCT ========================*/
void blink (const int8_t blink_count, const TickType_t delay, const CRGB color);
/**========================================================================
** ioInit()
*? ESP32 接腳和 FastLED 初始化
*========================================================================**/
void ioInit () {
pinMode (DOORLOCK, OUTPUT);
digitalWrite (DOORLOCK, LOW);
pinMode (LIMITSWITCH, INPUT_PULLUP);
FastLED.addLeds<WS2812B, DATA_PIN, GRB>(ws2812, NUM_LEDS);
//* 開機關閉 LED
ws2812[0] = BLACK;
FastLED.show ();
}
/**========================================================================
** lsInterrupt()
*? 中斷處理函式
*========================================================================**/
void IRAM_ATTR lsInterrupt () {
if((long)(micros() - ulLastMicros) >= 150000L) {
bIntTrigger = true;
detachInterrupt (digitalPinToInterrupt(LIMITSWITCH));
}
ulLastMicros = micros();
}
/**========================================================================
** blink()
*? 系統運行狀態的指示
*@param blink_count const int8_t
*@param color CRGB
*========================================================================**/
void blink (const int8_t blink_count, // 閃爍次數
const CRGB color) // WS2812 顏色
{
uint8_t state = LOW;
for (int x = 0; x < (blink_count << 1); ++x) {
if (state ^= HIGH) {
ws2812[0] = color;
} else {
ws2812[0] = BLACK;
}
FastLED.show();
delay (50);
}
}
/**========================================================================
** doorControl()
*? 門鎖開關控制
*@param isMotor const bool
* 門鎖的形式。true: DC Motor; false: 電磁開關
*========================================================================**/
void doorControl (const bool isMotor) {
if (isMotor) {
// 如果使用電動鎖頭,寫程式在這裡
// ...
} else {
digitalWrite (DOORLOCK, HIGH);
delay (1000);
digitalWrite (DOORLOCK, LOW);
}
}
/**========================================================================
** testLED
*? 顏色顯示測試
*========================================================================**/
void x02_testLED () {
for (int i = 0; i < 9; i++) {
blink (5, x02_colors[i]);
delay (1450);
}
}
/**========================================================================
** x02_testDoorLockUnlock
*? 開關門測試
*========================================================================**/
void x02_testDoorLockUnlock () {
Serial.print (F("Check door status, "));
if (dDoorState == OPEN) {
Serial.println(F("OPEN\nCLOSE DOOR TRY AGAIN!\n\n"));
return;
}
Serial.println (F("CLOSE"));
Serial.println (F("01-Open the door"));
doorControl (false);
// 等待門開啟(要移開門磁開關)
while (dDoorState == CLOSE);
// 亮燈表示門已開啟
blink (5, GREEN);
Serial.println (F("02-Sets an interrupt for external pin"));
// 啟用中斷服務函式
attachInterrupt (
digitalPinToInterrupt (LIMITSWITCH),
lsInterrupt,
FALLING
);
Serial.println (F("03-Waiting for door close"));
while (dDoorState == OPEN || bIntTrigger == false);
blink (5, RED);
Serial.print (F("04-Closed the door, "));
}
/**========================================================================
** x02_functionTest
*? 選擇外部裝置功能測試
*========================================================================**/
void x02_functionTest () {
if (Serial.available()) {
char c = Serial.read ();
switch (c) {
case 0x31:
Serial.print (F("Test LED, "));
x02_testLED ();
Serial.println (F("done."));
break;
case 0x32:
Serial.print (F("Test Door Lock/Unlock"));
x02_testDoorLockUnlock ();
Serial.println (F("done."));
break;
default:
return;
}
Serial.println (F("Enter 1 for LED test, 2 for door Lock/Unlock test"));
}
}
/**========================================================================
** setupOTA
*? ESP32 OTA 韌體上傳功能
*========================================================================**/
void setupOTA () {
#ifdef ENABLE_OTA
ArduinoOTA
.onStart([]() {
String type;
if (ArduinoOTA.getCommand() == U_FLASH)
type = "sketch";
else // U_SPIFFS
type = "filesystem";
// NOTE: if updating SPIFFS this would be the place
// to unmount SPIFFS using SPIFFS.end()
Serial.println("Start updating " + type);
})
.onEnd([]() {
Serial.println(F("\nEnd"));
})
.onProgress([](unsigned int progress, unsigned int total) {
Serial.printf("Progress: %u%%\r", (progress / (total / 100)));
})
.onError([](ota_error_t error) {
Serial.printf("Error[%u]: ", error);
if (error == OTA_AUTH_ERROR) Serial.println(F("Auth Failed"));
else if (error == OTA_BEGIN_ERROR) Serial.println(F("Begin Failed"));
else if (error == OTA_CONNECT_ERROR) Serial.println(F("Connect Failed"));
else if (error == OTA_RECEIVE_ERROR) Serial.println(F("Receive Failed"));
else if (error == OTA_END_ERROR) Serial.println(F("End Failed"));
});
ArduinoOTA.setHostname (DEVICENAME);
ArduinoOTA.begin();
#endif
}
void setup () {
delay (2000);
//* UART 初始化
Serial.begin (115200);
Serial.println (F("\nx02_ESP32CamPeripheral Booting..."));
//* WiFi 設定
WiFi.mode (WIFI_STA);
WiFi.setHostname (DEVICENAME);
WiFi.begin (SSID, PASSWORD);
while (WiFi.status() != WL_CONNECTED) {
delay (500);
Serial.print (".");
}
Serial.println ("");
//* 外設初始化
ioInit ();
//* OTA 設定
setupOTA ();
Serial.println (F("Enter 1 for LED test, 2 for door Lock/Unlock test"));
//* 印出 ESP32 Heap 和 PSRAM 的資訊
log_d("Total heap: %d", ESP.getHeapSize ());
log_d("Free heap: %d", ESP.getFreeHeap ());
log_d("Total PSRAM: %d", ESP.getPsramSize ());
log_d("Free PSRAM: %d", ESP.getFreePsram ());
}
void loop () {
ArduinoOTA.handle ();
x02_functionTest ();
}

編譯並上傳之後,實際測試的輸出與影片。



【(03)結論】

ESP32-CAM 開發板可以利用捨棄記憶卡的功能取得多支可用的接腳,但由於電路設計的關係會被侷限住,因此在使用前請先參考電路圖再選用。若在使用的過程中出現問題,建議先上網路尋找是否有相同問題的解決方法,再試著自己解決。

為了避免程式陷入無窮循環以及方便多任務的安排,下一篇要來看看怎麼把 Watchdog 加入到程式中,以及加入 Watchdog 之後對原本程式的影響和其衍生問題解決的方法。


.

.


<<部落格相關文章>>


.

.

沒有留言:

張貼留言

留言屬名為"Unknown"或"不明"的用戶,大多這樣的留言都會直接被刪除掉,不會得到任何回覆!

發問問題,請描述清楚你(妳)的問題,別人回答前不會想去 "猜" 問題是什麼?

不知道怎麼發問,請看 [公告] 部落格提問須知 - 如何問問題 !