이번에는 아두이노를 활용한 지문 인식 장치에 대해 이야기 해보려고 한다.
이전에 프로젝트 쪽에 올렸던 지문 인식 장치에 대한 것이기도 하니 참고하자.
( https://blog.naver.com/darknisia/222204861701 )
아마 이제 대부분 한번쯤은 써봤을 정도로 지문 인식이라는 것은 요즘 주변에서 쉽게 볼 수 있는 보안장치 중 하나다.
쉽게 생각해보면 항상 들고 다니는 스마트폰에서도 볼 수 있을 것이다.
그리고 일을 하시는 분이라면 결제 시스템이라던가 등록시스템에서 별도의 지문 인식 장치를 사용하는 사람들도 있다.
그런 것처럼 지문 인식은 쉽게 접할 수 있고 보안 장치를 이야기 할 때 빠지지 않고 등장하는 것 중 하나다.
지문 인식은 아마 스마트 폰의 스펙에 대해 관심이 많은 사람들이라면 그 원리에 대해 대략 알고 있을 것이다.
폰의 기종에 따라 잠금을 해제하는 지문 인식 방법이 조금씩 다르기 때문이다.
지문 인식 방법을 간단히 살펴보면 크게 광학식, 정전 용량식, 초음파식으로 나뉘게 된다.
먼저 광학식을 이야기하자면 우리가 주변에서 쉽게 볼 수 있는 방식이 이 광학식이다.
가시광선에 반사된 지문 영상을 획득하는 방식으로 카메라나 광학 스캐너를 이용한다.
쉽게 말해 강한 빛을 손 끝에 쏘아 지문 형태가 반사되면 그 이미지를 추출하는 방식이다.
LG 벨벳이 광학식 지문인식을 사용하고 있다.
정전 용량식은 지문의 굴곡을 활용하여 인식하는 방식이다.
정전 용량식은 매우 민감한 센서를 사용하며 100um이하로 촘촘한 전극 간격을 생성하고 그 위로 지문을 얹히면 지문의 들어간 부분과 나온 부분의 전기량 차이를 인식하여 이미지를 추출하게 된다.
현대 자동차의 보안장치로 정전 용량식 지문인식을 활용한다고 한다.
초음파식은 초음파 검사와 유사한 방법으로 초음파를 이용해 피부 표피층의 미세한 특징을 스캔하는 방식이다.
이 방식은 초음파의 일부 물체를 투과 할 수 있다는 장점을 활용하여 손가락의 물과 같은 일부 이물질이 묻은 상태라도 인식률이 높다.
그래서 차세대 지문 인식 방식으로 알려져 있기도 하다.
갤럭시 S21이 이 방식으로 지문 인식을 활용하고 있다.
이왕 지문 인식에 대해 공부하는 것이니 간단한 개념 정도는 알아두는 것이 좋을 것 같아 소개했다.
자, 그럼 우리가 사용 할 지문 인식 센서에 대해 알아보자.
우리가 쉽게 구할 수 있는 지문 인식 센서은 Adafruit, Seeed, DFROBOT의 센서들일 것이다.
위의 그림이 DFROBOT의 지문 인식 센서인 SEN0188 이다.
앞에서 언급한 3개의 회사 모두 형태는 동일하며 차이점은 센서에서 출력되는 데이터 선의 종류 정도가 차이가 있다.
그리고 방식 또한 짐작한 사람들도 있겠지만 대부분 광학식을 사용하고 있다.
정전 용량식의 센서도 판매를 하긴 하지만 구하기가 쉽지 않고 그건 모양이 좀 다르게 생겼다.
그래서 이 포스팅에서는 광학식을 사용하는 지문 인식 센서에 대해 포스팅 할 것이다.
아마 지문 인식 센서를 사용하기 위해서는 구입을 해야 될텐데 위에서 언급한 3개의 회사의 제품들을 검색을 해보면 사실 구입하기 꺼려 질 수도 있다.
지금 그림으로 표현된 DFROBOT의 센서의 가격이 보통 60,000원대에 판매가 되고 있다.
쉽게 테스트나 재미를 위해 구입하기에는 선뜻 꺼려지는 가격일 것이다.
그래서 그 대안이라고 할 수 있는 중국산 센서를 사용해 볼 것이다.
참고로 중국산 센서라고 해도 성능은 위의 3사의 제품과 거의 동일하고 사용법도 동일하니 다른 센서를 가지고 있다고 해도 같이 따라해도 된다.
중국산 센서는 JM-101B 센서로 가격은 15,000원대에 해당한다.
위의 그림이 JM-101B 센서인데 사실 생긴 모양은 큰 차이가 없다.
동작방법도 마찬가지다.
사실 지문인식에는 지문을 저장하고 비교하고 하는 다양한 동작이 필요하다.
그 동작을 아두이노에서 다하게 된다면 코딩도 사실 상당히 복잡할 것이고 그 처리 시간 또한 짧지는 않을 것이다.
다행히 앞에서 언급한 지문인식 센서들은 대부분 자체 컨트롤러를 가지고 있다.
꽤나 고성능 DSP 칩이 부착되어 있기 때문에 센서에서 측정되는 모든 이미지의 렌더링, 계산, 찾기 및 검색 기능을 수행 할 수 있다.
지문인식 또한 보통의 시중에 판매되는 센서만큼의 성능을 낼 수 있도록 500dpi의 해상도를 가지고 있다.
500dpi의 성능은 위의 그림으로 대략 비교를 할 수 있을 것이다.
또한 인식 시간에는 1초 이내의 시간 밖에 들지 않는다.
아무튼 시중에 판매되는 성능 정도는 낼 수 있다는 말이다.
동작 전압은 3~5V로 아두이노로 충분히 동작을 할 수 있는 센서다.
그리고 이 센서의 사용방법은 크게 2가지다.
우리가 사용할 아두이노와 같은 컨트롤러를 연결해서 사용하는 방법과 호환되는 프로그램을 이용하여 윈도우에서 바로 동작을 하든가 하는 방법이다.
두 방법의 차이는 아두이노를 활용한다면 저장되는 지문의 모양을 볼 수 없지만 프로그램을 활용하면 시중에 판매되는 지문인식 장치에서 볼 수 있는 지문의 모양을 볼 수 있다.
그래서 센서의 연결법도 2가지 방법이 있다.
센서의 연결부를 보면 총 연결선은 8개로 이루어져 있다.
(센서에 따라 핀맵이 다를 수 있으니 구입처에서 꼭 확인하자)
각각 연결선의 번호는 위의 그림을 기준으로 위에서부터 순서대로 1~8에 해당한다.
하지만 여기서 보면 알다시피 우리가 사용할 선은 사실 몇 개 없다.
전원선인 Vin과 GND는 당연히 연결해야 하고 시리얼(Serial) 통신을 활용할 것이면 Tx Rx 핀을 사용하면 되고 바로 컴퓨터에 연결 할 것이면 USB 핀인 D+ D-핀을 사용하면 된다.
다만 연결선이 조금 아두이노에 바로 연결하기에는 문제가 있다.
위의 그림과 같이 선이 아주 얇기 때문에 끝에 연장을 해주어야 한다.
아두이노에 연결할 때는 일반 점퍼선을 연결하면 되지만 USB로 바로 동작하기 위해서는 4핀 USB 커넥터에 연결해야 하는데 이게 일반적으로는 쉽지 않을 것이다.
그래서 일단은 이 포스팅에선 점퍼선을 연결하여 사용하는 방법에 대해 설명하겠다.
선은 위의 그림과 같이 연결했다.
아두이노를 활용할 것이기 때문에 1~4번 선만 필요하다.
그래서 4개의 선만 연장한 상태다.
그럼 이 선을 센서에 연결하고 본격적으로 사용해보자.
먼저 윈도우 프로그램을 활용해서 동작하는 방법을 간단히 살펴보자.
위의 그림은 참고사항이니 선 연결부는 잘 확인하고 하면 된다.
프로그램과 연동을 위해서는 원래 중간 USB to Serial 변환장치가 필요하다.
(사실 D+ D-를 바로 사용한다면 이 작업이 필요치 않다)
하지만 그런 장치가 없는 사람들이 대부분일테니 그럴 때는 위의 그림처럼 아두이노를 활용하면 된다.
먼저 아두이노 보드에 빈 코드를 업로드하자.
빈 코드라고 하면 아두이노 스케치를 새창으로 띄웠을 때 void setup() { } void loop() { }만 있는 상태를 말한다.
빈 코드를 올렸다면 이제 지문 센서를 연결하자.
전원선은 당연히 전원에 연결하고 Rx Tx는 교차 연결이 아니라 바로 연결하면 된다.
쉽게 말해 센서의 Rx 핀을 아두이노의 0핀에 Tx핀을 아두이노의 1번 핀에 연결하면 된다.
참고로 이와 같은 방식으로 동작 할 수 있는 보드는 아두이노 우노, 메가와 같이 USB to Serial 장치가 별도로 부착되어 있는 보드에만 활용할 수 있다.
레오나르도, 마이크로, 제로와 같이 메인 컨트롤러가 USB to Serial 역할을 함께 하는 경우에는 빈코드가 아닌 아래의 코드를 업로드 하고 사용해야 한다.
레오나르도 코드 void setup() { Serial1.begin(57600); Serial.begin(57600); }
void loop() { while(Serial.available()){ Serial1.write(Serial.read()); } while(Serial1.available()){ Serial.write(Serial1.read()); } }
그럼 이번엔 센서 사용을 위해 프로그램을 켜보자.
파일은 첨부된 것을 활용해도 되고 아래의 링크를 활용해도 된다.
( https://bc-robotics.com/shop/fingerprint-sensor-jm-101/ )
위의 링크로 들어가서 아래의 그림처럼 downloads에 있는 파일을 다운받으면 된다.
파일을 받아서 압축을 풀고 SFGDemo.exe 파일을 켜보면 아래의 그림과 같이 나타난다.
그림과 같이 프로그램이 나타나면 Open Device 버튼을 눌러 현재 연결된 아두이노의 포트를 선택하자.
그리고 JM-101B 센서는 기본 통신속도가 57600 baud이나 참고하자.
이후에는 Preview 체크 박스를 활성화시키고 Enroll 버튼을 통해 지문을 등록할 수 있다.
Con Enroll 버튼은 여러개의 지문을 한 번에 등록할 때 사용하는 버튼이다.
프로그램은 이 정도만 설명하고 넘어가겠다.
이 포스팅은 아두이노를 활용하는 것이니 말이다.
그럼 이제 아두이노를 활용하는 방법으로 넘어가자.
이번에는 아두이노의 시리얼 모니터를 활용해야 하기 때문에 센서의 RX TX핀을 각각 아두이노의 3번 2번 핀에 연결하자.
센서를 연결했으면 이번엔 코딩을 해보자.
코딩은 간단하게 라이브러리를 활용하자.
앞에서 이야기 했듯이 JM-101B 센서는 중국에서 만들어진 저가형 지문 인식 센서다.
그래서 이 센서는 기존의 Adafruit와 호환이 가능하게 만들어져 있다.
그럼 Adafrut의 라이브러리를 다운로드 받아보자.
먼저 아두이노 스케치를 켜고 라이브러리 관리를 들어가보자.
라이브러리 관리를 들어가면 이번에는 Adafruit fingerprint를 검색해보자.
검색하게 되면 하나의 라이브러리가 나타난다.
이 라이브러리를 설치하자.
라이브러리 설치가 끝나면 바로 사용해 볼 수 있다.
이번에는 라이브러리를 사용해서 지문 등록을 해보자.
위의 그림처럼 Adafruit fingerprint 라이브로러로 가서 enroll 예제를 불러와보자.
enroll 예제는 아래와 같다.
enroll 코드 /*************************************************** This is an example sketch for our optical Fingerprint sensor
Designed specifically to work with the Adafruit BMP085 Breakout —-> http://www.adafruit.com/products/751
These displays use TTL Serial to communicate, 2 pins are required to interface Adafruit invests time and resources providing this open source code, please support Adafruit and open-source hardware by purchasing products from Adafruit!
Written by Limor Fried/Ladyada for Adafruit Industries. BSD license, all text above must be included in any redistribution ****************************************************/
#include
#if (defined(__AVR__) || defined(ESP8266)) && !defined(__AVR_ATmega2560__) // For UNO and others without hardware serial, we must use software serial… // pin #2 is IN from sensor (GREEN wire) // pin #3 is OUT from arduino (WHITE wire) // Set up the serial port to use softwareserial.. SoftwareSerial mySerial(2, 3);
#else // data-on Leonardo/M0/etc, others with hardware serial, use hardware serial! // #0 is green wire, #1 is white #define mySerial Serial1
#endif
Adafruit_Fingerprint finger = Adafruit_Fingerprint(&mySerial);
uint8_t id;
void setup() { Serial.begin(9600); while (!Serial); // For Yun/Leo/Micro/Zero/… delay(100); Serial.println(”
Adafruit Fingerprint sensor enrollment”);
// set the data rate for the sensor serial port finger.begin(57600);
if (finger.verifyPassword()) { Serial.println(“Found fingerprint sensor!”); } else { Serial.println(“Did not find fingerprint sensor :(“); while (1) { delay(1); } }
Serial.println(F(“Reading sensor parameters”)); finger.getParameters(); Serial.print(F(“Status: 0x”)); Serial.println(finger.status_reg, HEX); Serial.print(F(“Sys ID: 0x”)); Serial.println(finger.system_id, HEX); Serial.print(F(“Capacity: “)); Serial.println(finger.capacity); Serial.print(F(“Security level: “)); Serial.println(finger.security_level); Serial.print(F(“Device address: “)); Serial.println(finger.device_addr, HEX); Serial.print(F(“Packet len: “)); Serial.println(finger.packet_len); Serial.print(F(“Baud rate: “)); Serial.println(finger.baud_rate); }
uint8_t readnumber(void) { uint8_t num = 0;
while (num == 0) { while (! Serial.available()); num = Serial.parseInt(); } return num; }
void loop() // run over and over again { Serial.println(“Ready to enroll a fingerprint!”); Serial.println(“Please type in the ID # (from 1 to 127) you want to save this finger as…”); id = readnumber(); if (id == 0) {// ID #0 not allowed, try again! return; } Serial.print(“Enrolling ID #”); Serial.println(id);
while (! getFingerprintEnroll() ); }
uint8_t getFingerprintEnroll() {
int p = -1; Serial.print(“Waiting for valid finger to enroll as #”); Serial.println(id); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.println(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(1); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
Serial.println(“Remove finger”); delay(2000); p = 0; while (p != FINGERPRINT_NOFINGER) { p = finger.getImage(); } Serial.print(“ID “); Serial.println(id); p = -1; Serial.println(“Place same finger again”); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.print(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(2); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
// OK converted! Serial.print(“Creating model for #”); Serial.println(id);
p = finger.createModel(); if (p == FINGERPRINT_OK) { Serial.println(“Prints matched!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_ENROLLMISMATCH) { Serial.println(“Fingerprints did not match”); return p; } else { Serial.println(“Unknown error”); return p; }
Serial.print(“ID “); Serial.println(id); p = finger.storeModel(id); if (p == FINGERPRINT_OK) { Serial.println(“Stored!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_BADLOCATION) { Serial.println(“Could not store in that location”); return p; } else if (p == FINGERPRINT_FLASHERR) { Serial.println(“Error writing to flash”); return p; } else { Serial.println(“Unknown error”); return p; }
return true; }
상당히 길어보이지만 사실 막상 들여다 보면 별 것 없는 코드다.
일단은 동작부터 해보자.
예제를 업로드하고 시리얼 모니터를 열어보자.
만약 시리얼 모니터에 그림과 같이 나타난다면 현재 센서를 찾지 못하고 있는 것이나 선 연결을 다시 확인하자.
제대로 연결이 됬다면 아래와 같이 나타난다.
그림과 같이 길게 쭉 내용이 나타나는데 뭐 중요한 것은 위의 2줄과 아래의 2줄 정도가 되겠다.
위의 2줄은 센서가 잘 연결되어 찾았다는 뜻이고 아래 2줄은 지문을 등록할 준비가 되었으니 지문을 저장 할 장소를 지정해 달라는 것이다.
그외의 가운데 내용은 현재 센서의 정보를 알려주고 있다.
지금은 이 부분은 필요없으니 넘어가도록 하자.
자 그럼 지문등록을 해야하는데 보면 총 1~127 번까지 지문을 등록할 수 있다고 되어있다.
하지만 알다시피 이 라이브러리는 adafruit의 지문인식 센서에 관한 내용이다.
실제 우리가 사용하는 지문 인식 센서는 스펙상 162개가 등록된다고 되어 있으니 참고하자.
자, 그럼 이번엔 지문을 등록해보자.
위의 그림과 같이 나타난 창에서 지문을 등록하고 싶은 번호를 지정하자.
그럼 3번에 저장을 해보자.
3을 입력하면 3번에 저장하겠다는 내용이 나타나고 지문이 인식되기를 기다린다.
그리고 지문을 인식하게 되면 손을 치워달라는 문구가 나탄난다.
지문 이미지의 변환이 완료되면 확인을 위해 다시 한 번 지문을 올려달라고 한다.
지문을 한 번 더 올려서 2개의 지문이 일치하면 지문이 해당되는 번호에 등록되게 된다.
이렇게 쉽게 지문을 등록 할 수 있다.
지문의 이미지 처리를 센서의 DSP가 다 처리하다 보니 사실 우리가 할 건 그다지 많지 않다.
통신을 통해 ‘지문 등록 할게’ 나 ‘지금 지문 비교해줘’ 정도의 명령만 내리는 것이다.
그리고 참고 사항으로 아 실수로 내가 같은 지문을 여러 개 등록하는 경우가 생길 것이다.
지문을 인식하는 것은 뒤에 할 것이지만 미리 이야기 하자면 같은 지문을 여러개 등록하게 되면 먼저 저장된 곳의 번호가 불려오게 된다.
엄지 손가락지문을 5번과 15번에 저장했다면 인식 할 때는 5번이 우선적으로 나타난다는 말이다.
거기다 지문이 저장되는 위치가 센서의 컨트롤러에 저장되다 보니 아두이노를 리셋하거나 코드를 다른 코드를 넣더라도 지문 정보는 계속 유지되게 된다.
뒤에서 마찬가지로 하겠지만 지문 정보를 삭제하기 위해서는 별도의 코딩을 통해 센서에 지문을 지워달라는 요청을 해야 한다.
자, 그럼 코드를 한 번 살펴보자.
기본적으로 예제 코드다 보니 간단하게 살펴보자.
일단 예제 코드 내에서 주석과 쓸데 없는 부분들을 지워보자.
코드 1 #include
SoftwareSerial mySerial(2, 3);
Adafruit_Fingerprint finger = Adafruit_Fingerprint(&mySerial);
uint8_t id;
void setup() { Serial.begin(9600); while (!Serial); delay(100); Serial.println(”
Adafruit Fingerprint sensor enrollment”);
finger.begin(57600);
if (finger.verifyPassword()) { Serial.println(“Found fingerprint sensor!”); } else { Serial.println(“Did not find fingerprint sensor :(“); while (1) { delay(1); } }
Serial.println(F(“Reading sensor parameters”)); finger.getParameters(); Serial.print(F(“Status: 0x”)); Serial.println(finger.status_reg, HEX); Serial.print(F(“Sys ID: 0x”)); Serial.println(finger.system_id, HEX); Serial.print(F(“Capacity: “)); Serial.println(finger.capacity); Serial.print(F(“Security level: “)); Serial.println(finger.security_level); Serial.print(F(“Device address: “)); Serial.println(finger.device_addr, HEX); Serial.print(F(“Packet len: “)); Serial.println(finger.packet_len); Serial.print(F(“Baud rate: “)); Serial.println(finger.baud_rate); }
uint8_t readnumber(void) { uint8_t num = 0;
while (num == 0) { while (! Serial.available()); num = Serial.parseInt(); } return num; }
void loop() { Serial.println(“Ready to enroll a fingerprint!”); Serial.println(“Please type in the ID # (from 1 to 127) you want to save this finger as…”); id = readnumber(); if (id == 0) {// ID #0 not allowed, try again! return; } Serial.print(“Enrolling ID #”); Serial.println(id);
while (! getFingerprintEnroll() ); }
uint8_t getFingerprintEnroll() {
int p = -1; Serial.print(“Waiting for valid finger to enroll as #”); Serial.println(id); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.println(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(1); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
Serial.println(“Remove finger”); delay(2000); p = 0; while (p != FINGERPRINT_NOFINGER) { p = finger.getImage(); } Serial.print(“ID “); Serial.println(id); p = -1; Serial.println(“Place same finger again”); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.print(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(2); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
// OK converted! Serial.print(“Creating model for #”); Serial.println(id);
p = finger.createModel(); if (p == FINGERPRINT_OK) { Serial.println(“Prints matched!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_ENROLLMISMATCH) { Serial.println(“Fingerprints did not match”); return p; } else { Serial.println(“Unknown error”); return p; }
Serial.print(“ID “); Serial.println(id); p = finger.storeModel(id); if (p == FINGERPRINT_OK) { Serial.println(“Stored!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_BADLOCATION) { Serial.println(“Could not store in that location”); return p; } else if (p == FINGERPRINT_FLASHERR) { Serial.println(“Error writing to flash”); return p; } else { Serial.println(“Unknown error”); return p; }
return true; }
주석을 지웠는데도 사실 코드가 상당히 길다.
그럼 하나씩 살펴보자.
#include
SoftwareSerial mySerial(2, 3);
Adafruit_Fingerprint finger = Adafruit_Fingerprint(&mySerial);
uint8_t id;
void setup(){ Serial.begin(9600); while (!Serial); delay(100); Serial.println(”
Adafruit Fingerprint sensor enrollment”);
finger.begin(57600);
if (finger.verifyPassword()) { Serial.println(“Found fingerprint sensor!”); } else { Serial.println(“Did not find fingerprint sensor :(“); while (1) { delay(1); } }
Serial.println(F(“Reading sensor parameters”)); finger.getParameters(); Serial.print(F(“Status: 0x”)); Serial.println(finger.status_reg, HEX); Serial.print(F(“Sys ID: 0x”)); Serial.println(finger.system_id, HEX); Serial.print(F(“Capacity: “)); Serial.println(finger.capacity); Serial.print(F(“Security level: “)); Serial.println(finger.security_level); Serial.print(F(“Device address: “)); Serial.println(finger.device_addr, HEX); Serial.print(F(“Packet len: “)); Serial.println(finger.packet_len); Serial.print(F(“Baud rate: “)); Serial.println(finger.baud_rate); }
위의 코드는 setup까지 코드를 가져온 것이다.
여기서부터 살펴보자.
처음엔 라이브러리를 불러왔고 그 다음에는 소프트웨어 시리얼을 설정해줬다.
아두이노 우노에서는 시리얼 핀이 모자라기 때문에 사용된 것이다.
여기서 하나 주목 할 점은 소프트웨어 시리얼이 사용됬는데 그 라이브러리는 따로 불러오지 않은 점이다.
이때 소프트웨어 시리얼의 라이브러리는 Adafruit_Fingerprint 라이브러리 안에서 불러오고 있기 때문에 이 코드에서는 따로 선언해 줄 필요가 없는 것이다.
그리고 다음으로 있는 코드가 아래의 코드다.
Adafruit_Fingerprint finger = Adafruit_Fingerprint(&mySerial)
이 코드는 쉽게 말해 지문 인식 센서와 연동될 통신을 설정하는 것과 동시에 지문 인식 라이브러리에 사용될 인스턴스 변수 즉, 이름을 지어준다라고 생각하면 된다.
만약 위에 표시된 finger 자리에 넣고 싶은 이름이 있다면 그 이름으로 넣고 아래의 함수들도 그 이름으로 변경 하면 된다.
다음으로 넘어가 보자.
이번에는 간단한 id라는 변수가 선언되어 있는데 이 변수는 지문 등록을 할 때 선택하는 번호가 저장될 변수로 기억하면 된다.
그럼 setup을 살펴보자.
setup 안도 내용은 꽤 길게 나타나 있다.
먼저 통신 속도 설정이 있는데 Serial 통신, 아두이노와 컴퓨터 간의 통신은 속도를 9600 baud로 해두었다.
하지만 센서와의 통신에 사용될 통식속도는 57600 baud로 설정되었다.
finger.begin(57600)
JM-101b 센서는 기본 통신 속도가 57600으로 설정되어 있다.
그외에도 9600, 19200,28800, 38400 등의 통신 속도를 지원하니 참고하자.
if (finger.verifyPassword()) { Serial.println(“Found fingerprint sensor!”); } else { Serial.println(“Did not find fingerprint sensor :(“); while (1) { delay(1); } }
다음에는 지문 인식 센서를 확인하는 코드다.
finger.verifyPassword()
여기서 사용되는 위의 함수는 지문 인식 센서의 비밀번호를 확인하는 함수로 if 문에 나타난대로 비밀번호가 맞다면 Found ~ 의 내용을 띄우고 아니라면 센서를 찾지 못했다는 내용을 나타낸다.
이 부분에서 나타난 것처럼 사실 이 지문 인식 센서도 보안을 위해 비밀 번호를 설정 할 수 있다.
그리고 비밀 번호가 맞을 때만 동작하는 것이다.
처음 제품을 구입했을 때의 기본 비밀 번호는 0x000000에 해당한다.
사실 이때 확인 할 비밀번호는 앞의 함수에 나타난다.
Adafruit_Fingerprint finger = Adafruit_Fingerprint(&mySerial, Password)
지금은 비밀번호가 0이라 생략되었다.
비밀번호를 변경하는 방법은 추후에 다시 이야기 하도록 하자.
자 그럼 다음 코드를 살펴보자.
Serial.println(F(“Reading sensor parameters”)); finger.getParameters(); Serial.print(F(“Status: 0x”)); Serial.println(finger.status_reg, HEX); Serial.print(F(“Sys ID: 0x”)); Serial.println(finger.system_id, HEX); Serial.print(F(“Capacity: “)); Serial.println(finger.capacity); Serial.print(F(“Security level: “)); Serial.println(finger.security_level); Serial.print(F(“Device address: “)); Serial.println(finger.device_addr, HEX); Serial.print(F(“Packet len: “)); Serial.println(finger.packet_len); Serial.print(F(“Baud rate: “)); Serial.println(finger.baud_rate);
다음 코드는 더욱 간단하다.
이 코드의 핵심은 현재 센서의 파라미터를 읽어오는 것이다.
finger.getParameters()
읽어오는 파라미터는 통신속도, 센서의 주소 등등 7가지 정보를 나타낸다.
뭐 사실 우리가 사용할 때는 딱히 필요없는 부분이니 넘어가자.
다음은 loop문을 살펴보자.
사실 이 코드에서 loop문은 상당히 짧다.
대부분 별도의 함수로 나타냈기 때문이다.
void loop() { Serial.println(“Ready to enroll a fingerprint!”); Serial.println(“Please type in the ID # (from 1 to 127) you want to save this finger as…”); id = readnumber(); if (id == 0) {// ID #0 not allowed, try again! return; } Serial.print(“Enrolling ID #”); Serial.println(id);
while (! getFingerprintEnroll() ); }
loop문을 보면 처음에는 지문을 등록할 준비가 되었다는 내용과 등록할 번호(ID)를 적으라는 문구를 시리얼 모니터에 나타낸다.
그리고 그 이후에 있는 것이 readnumber() 함수다.
이 함수는 loop 문 바로 위에 정의되어 있다.
uint8_t readnumber(void) { uint8_t num = 0;
while (num == 0) { while (! Serial.available()); num = Serial.parseInt(); } return num; }
readnumber에서는 간단히 시리얼 모니터를 통해 글자를 읽어오는 것을 나타낸다.
이때 데이터를 읽어오는 함수로 사용된 시리얼 함수가 있다.
Serial.parseInt();
이 함수는 시리얼 모니터 상에서 입력되는 내용 중에 int 형태의 문자열을 불러오는 함수다.
만약 시리얼 모니터에 asd12라고 적는다면 이 중에서 12라는 숫자만 읽어오게 되는 것이다.
여기서 한 번 더 꼬아서 만약 a1s2라고 적는다면 아마 1과 2를 각각 읽어오게 될 것이다.
당연히 지문 등록 작업을 2번해야 한다.
아무튼 readnumber는 시리얼 모니터에서 입력되는 숫자를 읽어오는 함수에 해당한다.
그리고 읽어온 값은 id 변수에 저장된다.
다시 loop 문으로 돌아가보면 readnumber 이후에는 읽어 온 수를 시리얼 모니터에 나타내 주는 동작을 하고 이제 지문을 등록하는 함수를 불러오게 된다.
uint8_t getFingerprintEnroll() {
int p = -1; Serial.print(“Waiting for valid finger to enroll as #”); Serial.println(id); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.println(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(1); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
Serial.println(“Remove finger”); delay(2000); p = 0; while (p != FINGERPRINT_NOFINGER) { p = finger.getImage(); } Serial.print(“ID “); Serial.println(id); p = -1; Serial.println(“Place same finger again”); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.print(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(2); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
// OK converted! Serial.print(“Creating model for #”); Serial.println(id);
p = finger.createModel(); if (p == FINGERPRINT_OK) { Serial.println(“Prints matched!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_ENROLLMISMATCH) { Serial.println(“Fingerprints did not match”); return p; } else { Serial.println(“Unknown error”); return p; }
Serial.print(“ID “); Serial.println(id); p = finger.storeModel(id); if (p == FINGERPRINT_OK) { Serial.println(“Stored!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_BADLOCATION) { Serial.println(“Could not store in that location”); return p; } else if (p == FINGERPRINT_FLASHERR) { Serial.println(“Error writing to flash”); return p; } else { Serial.println(“Unknown error”); return p; }
return true; }
getFingerprintEnroll 함수는 사실상 이 예제의 핵심이 되는 함수다.
지문을 등록하고 등록시 발생하는 오류에 대해 나타내는 내용이기 때문이다.
이 함수를 한 번 살펴보자.
getFingerprintEnroll 함수는 크게 3 부분으로 나뉘어져 있다.
먼저 첫 번째 지문 등록에 관한 내용이다.
int p = -1; Serial.print(“Waiting for valid finger to enroll as #”); Serial.println(id); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.println(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(1); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
사실 이 부분은 case 문 때문에 코드가 길어보이지만 내용을 이해한다면 코드가 긴 것이 아니라는 것을 알 것이다.
먼저 p라는 변수가 나오는데 이 변수는 후에 센서에 돌아오는 대답이 저장될 곳이다.
그럼 코드를 살펴보자.
코드의 첫번째 while 문 안을 보자.
가장 먼저 나타나 있는 함수가 있을 것이다.
finger.getImage()
이 함수는 센서에 손가락의 이미지를 촬영하도록 요청하는 함수다.
이 함수를 사용하면 상황에 따라 다른 값이 반환된다.
이때 반환되는 값들이 위의 코드에 나타나있는 case 문의 내용들이다.
반환 값 내용 FINGERPRINT_OK 이미지 촬영 성공 FINGERPRINT_NOFINGER 지문을 찾지 못함 FINGERPRINT_PACKETRECIEVEERR 센서와의 통신 에러 FINGERPRINT_IMAGEFAIL 촬영 중 이미지 에러
반환되는 내용은 위의 표와 같다.
그렇기 때문에 코드 상에서 나타나 있는 case 문의 내용들도 표를 바탕으로 나타나고 있다.
다음 사용된 함수를 살펴보자.
finger.image2Tz(1)
이 함수는 간단히 말해 지문을 등록하기 전 임시로 저장하는 함수다.
이미지를 feature template(특징 탬플릿)으로 변환하여 저장하게 되는데 저장공간은 1과 2 이렇게 2가지 공간이 있다.
추후에 2 이미지가 모두 저장이 되면 서로 비교하여 지문이 등록되게 된다.
이 함수 역시 다양한 값을 반환하게 된다.
반환 값 내용 FINGERPRINT_OK 이미지 변환 성공 FINGERPRINT_IMAGEMESS 이미지가 너무 지저분 함 FINGERPRINT_PACKETRECIEVEERR 센서와의 통신 에러 FINGERPRINT_FINGERPRINT_FEATUREFAIL 지문의 특징 확인 실패 FINGERPRINT_INVALIDIMAGE 지문의 특징 확인 실패
위의 표와 같은 값을 반환하게 된다.
이 또한 코드 상에 case문으로 나타나 있다.
이렇게 첫 번째 지문 등록이 끝나면 이제 두 번째 지문 등록이 남았다.
Serial.println(“Remove finger”); delay(2000); p = 0; while (p != FINGERPRINT_NOFINGER) { p = finger.getImage(); } Serial.print(“ID “); Serial.println(id); p = -1; Serial.println(“Place same finger again”); while (p != FINGERPRINT_OK) { p = finger.getImage(); switch (p) { case FINGERPRINT_OK: Serial.println(“Image taken”); break; case FINGERPRINT_NOFINGER: Serial.print(“.”); break; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); break; case FINGERPRINT_IMAGEFAIL: Serial.println(“Imaging error”); break; default: Serial.println(“Unknown error”); break; } }
// OK success!
p = finger.image2Tz(2); switch (p) { case FINGERPRINT_OK: Serial.println(“Image converted”); break; case FINGERPRINT_IMAGEMESS: Serial.println(“Image too messy”); return p; case FINGERPRINT_PACKETRECIEVEERR: Serial.println(“Communication error”); return p; case FINGERPRINT_FEATUREFAIL: Serial.println(“Could not find fingerprint features”); return p; case FINGERPRINT_INVALIDIMAGE: Serial.println(“Could not find fingerprint features”); return p; default: Serial.println(“Unknown error”); return p; }
두 번째 지문 등록 역시 내용은 동일하다.
처음에는 손가락을 치워달라는 문구를 2초간 표시하고 다시 finger.getImage 함수를 통해 지문을 읽어온다.
그리고 지문을 성공적으로 읽어오면 지문 이미지를 feature template으로 변환하여 저장해준다.
이번에는 finger.image2Tz(2)에 저장하게 된다.
두 번의 지문의 feature template 모두 등록되었다면 이제 실제 지문이 저장될 차례다.
Serial.print(“Creating model for #”); Serial.println(id);
p = finger.createModel(); if (p == FINGERPRINT_OK) { Serial.println(“Prints matched!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_ENROLLMISMATCH) { Serial.println(“Fingerprints did not match”); return p; } else { Serial.println(“Unknown error”); return p; }
Serial.print(“ID “); Serial.println(id); p = finger.storeModel(id); if (p == FINGERPRINT_OK) { Serial.println(“Stored!”); } else if (p == FINGERPRINT_PACKETRECIEVEERR) { Serial.println(“Communication error”); return p; } else if (p == FINGERPRINT_BADLOCATION) { Serial.println(“Could not store in that location”); return p; } else if (p == FINGERPRINT_FLASHERR) { Serial.println(“Error writing to flash”); return p; } else { Serial.println(“Unknown error”); return p; }
이제 실제 지문 정보가 등록 될 차례다.
코드를 보면 먼저 저장될 장소를 다시 한 번 말해준다.
그리고 나오는 함수가 실제 지문 모델을 만드는 함수다.
finger.createModel()
이 함수는 앞서 finger.image2Tz() 함수로 저장된 2가지의 feature template을 활용하여 지문의 모델을 만들도록 요청하는 함수다.
저장되어 있는 2가지의 지문 특징을 활용하여 하나의 지문 정보를 만드는 것이다.
이 함수 또한 앞서 본 함수들과 마찬가지로 동작 결과에 따라 반환하는 값이 달라진다.
반환 값 내용 FINGERPRINT_OK 지문 모델 생성 성공 FINGERPRINT_PACKETRECIEVEERR 센서와의 통신 에러 FINGERPRINT_ENROLLMISMATCH 두 지문이 일치 하지 않음
이렇게 지문 모델이 완성이 되고 마지막으론 지문을 저장해야 한다.
finger.storeModel(id)
마지막으로 사용된 이 함수가 지문 모델을 저장 할 장소를 뜻한다.
그래서 괄호 안에 저장될 위치를 같이 넣어 줘야 한다.
마지막으로 이 함수의 반환 값은 아래와 같다.
반환 값 내용 FINGERPRINT_OK 지문 모델 저장 성공 FINGERPRINT_BADLOCATION 저장 위치 오류 FINGERPRINT_FLASHERR 플래시 메모리의 모델 저장 오류 FINGERPRINT_PACKETRECIEVEERR 센서와의 통신 에러
이렇게 지문 센서 JM-101B를 활용해 지문 등록하는 방법에 대해 알아보았다.
사실 이번 포스팅에서 지문 인식과, 지문 정보 삭제, 비밀번호 설정까지 다 하려고 했지만….
알다시피 내용이 너무 길어져 다음 포스팅으로 넘어가도록 하겠다.
이렇게까지 길어질 내용이 아니었는데 욕심이었나 싶은 생각도 든다…..