mirror of https://github.com/arendst/Tasmota.git
Experimental support for fingerprint reader
Experimental support for fingerprint reader (#9568)
This commit is contained in:
parent
daec8b9eb5
commit
5f0f594347
|
@ -0,0 +1,301 @@
|
|||
/*
|
||||
xsns_79_as608.ino - AS608 and R503 fingerprint sensor support for Tasmota
|
||||
|
||||
Copyright (C) 2020 boaschti and Theo Arends
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifdef USE_AS608
|
||||
/*********************************************************************************************\
|
||||
* AS608 optical and R503 capacitive Fingerprint sensor
|
||||
*
|
||||
* Currently supports:
|
||||
* - English only as there are no tanslations supported
|
||||
* - AS608 only as R503 fails to read stored fingerprints
|
||||
\*********************************************************************************************/
|
||||
|
||||
#define XSNS_79 79
|
||||
|
||||
#define D_JSON_FPRINT "FPrint"
|
||||
|
||||
#define D_PRFX_FP "Fp"
|
||||
#define D_CMND_FP_ENROLL "Enroll"
|
||||
#define D_CMND_FP_DELETE "Delete"
|
||||
#define D_CMND_FP_COUNT "Count"
|
||||
|
||||
const char kAs608Commands[] PROGMEM = D_PRFX_FP "|" D_CMND_FP_ENROLL "|" D_CMND_FP_DELETE "|" D_CMND_FP_COUNT;
|
||||
|
||||
void (*const As608Commands[])(void) PROGMEM = { &CmndFpEnroll, &CmndFpDelete, &CmndFpCount };
|
||||
|
||||
const char kAs608Messages[] PROGMEM =
|
||||
"Done|Comms error||Imaging error|Unknown error|Image too messy|Could not find fingerprint features|No match|Did not find a match|Fingerprint did not match|"
|
||||
"Bad location|DB range error|Upload feature error|Packet response error|Upload error|Delete error|DB Clear error|Password error|Image invalid|"
|
||||
"Error writing to flash|Invalid reg|Address code|Password verify";
|
||||
|
||||
const uint8_t As608Reference[] PROGMEM = { 0, 1, 2, 3, 4, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 4, 17, 4, 18, 4, 4, 19, 4, 20, 4, 4, 4, 4, 4, 21, 22 };
|
||||
|
||||
#include <TasmotaSerial.h>
|
||||
#include <Adafruit_Fingerprint.h>
|
||||
|
||||
Adafruit_Fingerprint *As608Finger;
|
||||
TasmotaSerial *As608Serial;
|
||||
|
||||
struct AS608 {
|
||||
bool selected = false;
|
||||
uint8_t enroll_step = 0;
|
||||
uint8_t model_number = 0;
|
||||
} As608;
|
||||
|
||||
char* As608Message(char* response, uint32_t index) {
|
||||
if (index > sizeof(As608Reference)) { index = 4; }
|
||||
uint32_t i = pgm_read_byte(&As608Reference[index]);
|
||||
return GetTextIndexed(response, TOPSZ, i, kAs608Messages);
|
||||
}
|
||||
|
||||
void As608PublishMessage(const char* message) {
|
||||
char romram[TOPSZ];
|
||||
snprintf_P(romram, sizeof(romram), message);
|
||||
if (strlen(romram) > 0) {
|
||||
char json_name[20];
|
||||
if (As608.enroll_step) {
|
||||
strcpy_P(json_name, PSTR(D_PRFX_FP D_CMND_FP_ENROLL));
|
||||
} else {
|
||||
strcpy_P(json_name, PSTR(D_JSON_FPRINT));
|
||||
}
|
||||
Response_P(S_JSON_COMMAND_SVALUE, json_name, romram);
|
||||
MqttPublishPrefixTopicRulesProcess_P(RESULT_OR_STAT, json_name);
|
||||
}
|
||||
}
|
||||
|
||||
void As608Init(void) {
|
||||
if (PinUsed(GPIO_AS608_RX) && PinUsed(GPIO_AS608_TX)) {
|
||||
As608Serial = new TasmotaSerial(Pin(GPIO_AS608_RX), Pin(GPIO_AS608_TX), 0);
|
||||
As608Finger = new Adafruit_Fingerprint(As608Serial, 0);
|
||||
|
||||
As608Finger->begin(57600);
|
||||
if (As608Serial->hardwareSerial()) { ClaimSerial(); }
|
||||
|
||||
if (As608Finger->verifyPassword()) {
|
||||
As608Finger->getTemplateCount();
|
||||
AddLog_P(LOG_LEVEL_INFO, PSTR("AS6: Detected with %d fingerprint(s) stored"), As608Finger->templateCount);
|
||||
As608.selected = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int As608GetFingerImage(void) {
|
||||
int p = As608Finger->getImage();
|
||||
if (p != FINGERPRINT_OK) {
|
||||
char response[TOPSZ];
|
||||
As608PublishMessage(As608Message(response, p));
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
int As608ConvertFingerImage(uint8_t slot) {
|
||||
int p = As608Finger->image2Tz(slot);
|
||||
if (p != FINGERPRINT_OK) {
|
||||
char response[TOPSZ];
|
||||
As608PublishMessage(As608Message(response, p));
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
void As608Loop(void) {
|
||||
uint32_t p = 0;
|
||||
|
||||
if (!As608.enroll_step) {
|
||||
// Search for Finger
|
||||
|
||||
// As608Finger->LEDcontrol(FINGERPRINT_LED_OFF, 0, FINGERPRINT_LED_RED);
|
||||
// As608Finger->LEDcontrol(FINGERPRINT_LED_OFF, 0, FINGERPRINT_LED_BLUE);
|
||||
// As608Finger->LEDcontrol(FINGERPRINT_LED_OFF, 0, FINGERPRINT_LED_PURPLE);
|
||||
// As608Finger->LEDcontrol(0);
|
||||
|
||||
p = As608Finger->getImage(); // Take image
|
||||
if (p != FINGERPRINT_OK) { return; }
|
||||
|
||||
p = As608Finger->image2Tz(); // Convert image
|
||||
if (p != FINGERPRINT_OK) { return; }
|
||||
|
||||
p = As608Finger->fingerFastSearch(); // Match found
|
||||
if (p != FINGERPRINT_OK) {
|
||||
// AddLog_P(LOG_LEVEL_DEBUG, PSTR("AS6: No matching finger"));
|
||||
return;
|
||||
}
|
||||
|
||||
// Found a match
|
||||
Response_P(PSTR("{\"" D_JSON_FPRINT "\":{\"Id\":%d,\"Confidence\":%d}}"), As608Finger->fingerID, As608Finger->confidence);
|
||||
MqttPublishPrefixTopicRulesProcess_P(RESULT_OR_STAT, PSTR(D_JSON_FPRINT));
|
||||
return;
|
||||
} else {
|
||||
// enroll is active
|
||||
switch (As608.enroll_step) {
|
||||
case 1:
|
||||
As608PublishMessage(PSTR("Place finger"));
|
||||
// As608Finger->LEDcontrol(FINGERPRINT_LED_ON, 0, FINGERPRINT_LED_BLUE);
|
||||
As608.enroll_step++;
|
||||
break;
|
||||
case 2:
|
||||
// get first image
|
||||
if (As608GetFingerImage() == FINGERPRINT_OK) {
|
||||
As608.enroll_step++;
|
||||
}
|
||||
break;
|
||||
case 3:
|
||||
// convert image
|
||||
if (As608ConvertFingerImage(1) == FINGERPRINT_OK) {
|
||||
As608.enroll_step++;
|
||||
} else {
|
||||
As608.enroll_step--;
|
||||
}
|
||||
break;
|
||||
case 4:
|
||||
As608PublishMessage(PSTR("Remove finger"));
|
||||
As608.enroll_step++;
|
||||
break;
|
||||
case 5:
|
||||
// Remove finger
|
||||
p = As608Finger->getImage();
|
||||
if (p == FINGERPRINT_NOFINGER) {
|
||||
As608.enroll_step++;
|
||||
}
|
||||
break;
|
||||
case 6:
|
||||
As608PublishMessage(PSTR("Place same finger again"));
|
||||
// As608Finger->LEDcontrol(FINGERPRINT_LED_OFF, 0, FINGERPRINT_LED_PURPLE);
|
||||
As608.enroll_step++;
|
||||
break;
|
||||
case 7:
|
||||
// get second image of finger
|
||||
if (As608GetFingerImage() == FINGERPRINT_OK) {
|
||||
As608.enroll_step++;
|
||||
}
|
||||
break;
|
||||
case 8:
|
||||
// convert second image
|
||||
if (As608ConvertFingerImage(2) != FINGERPRINT_OK) {
|
||||
As608PublishMessage(PSTR("Not Ok so try again"));
|
||||
As608.enroll_step -= 2;
|
||||
break;
|
||||
}
|
||||
// Create model
|
||||
p = As608Finger->createModel();
|
||||
if (p != FINGERPRINT_OK) {
|
||||
char response[TOPSZ];
|
||||
As608PublishMessage(As608Message(response, p));
|
||||
As608.enroll_step = 99;
|
||||
break;
|
||||
}
|
||||
// Store model
|
||||
p = As608Finger->storeModel(As608.model_number);
|
||||
char response[TOPSZ];
|
||||
As608PublishMessage(As608Message(response, p));
|
||||
if (p == FINGERPRINT_OK) {
|
||||
As608.enroll_step = 0;
|
||||
As608.model_number = 0;
|
||||
} else {
|
||||
As608.enroll_step = 99;
|
||||
}
|
||||
break;
|
||||
case 99:
|
||||
As608PublishMessage(PSTR("Restart"));
|
||||
As608.enroll_step = 1;
|
||||
break;
|
||||
default:
|
||||
As608PublishMessage(PSTR("Error"));
|
||||
As608.enroll_step = 0;
|
||||
As608.model_number = 0;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*********************************************************************************************\
|
||||
* Commands
|
||||
\*********************************************************************************************/
|
||||
|
||||
void CmndFpEnroll(void) {
|
||||
if (As608.enroll_step) {
|
||||
if (0 == XdrvMailbox.payload) {
|
||||
// FpEnroll 0 - Stop enrollement
|
||||
As608.enroll_step = 0;
|
||||
ResponseCmndChar_P(PSTR("Reset"));
|
||||
} else {
|
||||
// FpEnroll - Enrollement state
|
||||
ResponseCmndChar_P(PSTR("Active"));
|
||||
}
|
||||
} else {
|
||||
if ((XdrvMailbox.payload > 0) && (XdrvMailbox.payload <= 128)) {
|
||||
// FpEnroll 1..128 - Start enrollement into slot x
|
||||
As608.enroll_step = 1;
|
||||
As608.model_number = XdrvMailbox.payload;
|
||||
ResponseClear(); // Will use loop start message
|
||||
} else {
|
||||
// FpEnroll - Enrollement state
|
||||
ResponseCmndChar_P(PSTR("Inactive"));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CmndFpDelete(void) {
|
||||
if (0 == XdrvMailbox.payload) {
|
||||
// FpDelete 0 - Clear database
|
||||
As608Finger->emptyDatabase();
|
||||
As608Finger->getTemplateCount();
|
||||
if (As608Finger->templateCount) {
|
||||
ResponseCmndChar_P(PSTR("Error"));
|
||||
} else {
|
||||
ResponseCmndDone();
|
||||
}
|
||||
}
|
||||
else if ((XdrvMailbox.payload > 0) && (XdrvMailbox.payload <= 128)) {
|
||||
// FpDelete 1..128 - Delete single entry from database
|
||||
int p = As608Finger->deleteModel(XdrvMailbox.payload);
|
||||
char response[TOPSZ];
|
||||
ResponseCmndChar(As608Message(response, p));
|
||||
}
|
||||
}
|
||||
|
||||
void CmndFpCount(void) {
|
||||
// FpCount - Show number of slots used
|
||||
As608Finger->getTemplateCount();
|
||||
ResponseCmndNumber(As608Finger->templateCount);
|
||||
}
|
||||
|
||||
/*********************************************************************************************\
|
||||
* Interface
|
||||
\*********************************************************************************************/
|
||||
|
||||
bool Xsns79(uint8_t function) {
|
||||
bool result = false;
|
||||
|
||||
if (FUNC_INIT == function) {
|
||||
As608Init();
|
||||
}
|
||||
else if (As608.selected) {
|
||||
switch (function) {
|
||||
case FUNC_EVERY_250_MSECOND:
|
||||
As608Loop();
|
||||
break;
|
||||
case FUNC_COMMAND:
|
||||
result = DecodeCommand(kAs608Commands, As608Commands);
|
||||
break;
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
#endif // USE_AS608
|
Loading…
Reference in New Issue