2015-05-11 00:18:22 +02:00
|
|
|
#include <iostream>
|
2015-05-14 19:36:49 +02:00
|
|
|
|
2015-05-11 00:18:22 +02:00
|
|
|
#include <wiringPi.h>
|
2015-05-14 19:36:49 +02:00
|
|
|
#include <unistd.h>
|
2015-05-11 00:18:22 +02:00
|
|
|
|
|
|
|
#include "logger.h"
|
|
|
|
#include "door.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
2015-05-14 19:36:49 +02:00
|
|
|
Door::Door() :
|
|
|
|
_l(Logger::get())
|
2015-05-11 00:18:22 +02:00
|
|
|
{
|
|
|
|
_l(LogLevel::info, "Initializing Raspberry Pi GPIOs");
|
|
|
|
wiringPiSetup();
|
2015-05-21 13:35:30 +02:00
|
|
|
pinMode(_HEARTBEATPIN, OUTPUT);
|
|
|
|
pinMode(_SCHNAPPERPIN, OUTPUT);
|
2015-05-11 00:18:22 +02:00
|
|
|
lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
Door::~Door()
|
|
|
|
{
|
2015-05-14 19:36:49 +02:00
|
|
|
lock();
|
2015-05-11 00:18:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Door &Door::get()
|
|
|
|
{
|
|
|
|
static Door d;
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Door::lock()
|
|
|
|
{
|
2015-05-21 13:35:30 +02:00
|
|
|
digitalWrite(_SCHNAPPERPIN, HIGH);
|
2015-05-11 00:18:22 +02:00
|
|
|
_l(LogLevel::info, "Door closed");
|
2015-05-21 13:35:30 +02:00
|
|
|
|
|
|
|
if (_state == State::Unlocked)
|
2015-05-14 19:36:49 +02:00
|
|
|
{
|
2015-05-21 13:35:30 +02:00
|
|
|
// Stop the Heartbeat Thread
|
|
|
|
_state = State::Locked;
|
2015-05-14 19:36:49 +02:00
|
|
|
_heartbeat.join();
|
|
|
|
}
|
2015-05-18 22:22:08 +02:00
|
|
|
|
2015-05-21 13:35:30 +02:00
|
|
|
// Turn off all lights
|
2015-05-18 22:25:47 +02:00
|
|
|
system("wget -O /dev/null --timeout 3 \"http://homer.binary.kitchen:8080/set?color=000000\" > /dev/null 2>&1");
|
2015-05-11 00:18:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Door::unlock()
|
|
|
|
{
|
2015-05-21 13:35:30 +02:00
|
|
|
// In any case, klacker the schnapper
|
2015-05-14 19:36:49 +02:00
|
|
|
_schnapper = true;
|
|
|
|
|
2015-05-21 13:35:30 +02:00
|
|
|
// If heartbeat is already running, return
|
|
|
|
if (_state == State::Unlocked)
|
2015-05-14 19:36:49 +02:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-05-21 13:35:30 +02:00
|
|
|
// If not, first set state to unlocked
|
|
|
|
_state = State::Unlocked;
|
|
|
|
|
|
|
|
// Start the Heartbeat Thread
|
2015-05-14 19:36:49 +02:00
|
|
|
_heartbeat = std::thread([this] () {
|
2015-05-21 13:35:30 +02:00
|
|
|
|
|
|
|
// One "beat" is one complete cycle of the heartbeat clock
|
2015-05-14 19:36:49 +02:00
|
|
|
auto beat = [] () {
|
2015-05-21 13:35:30 +02:00
|
|
|
digitalWrite(_HEARTBEATPIN, HIGH);
|
2015-05-14 19:36:49 +02:00
|
|
|
usleep(10000);
|
2015-05-21 13:35:30 +02:00
|
|
|
digitalWrite(_HEARTBEATPIN, LOW);
|
2015-05-14 19:36:49 +02:00
|
|
|
usleep(10000);
|
|
|
|
};
|
2015-05-20 22:42:20 +02:00
|
|
|
|
2015-05-21 13:35:30 +02:00
|
|
|
// The default of the Schnapperpin: always high
|
|
|
|
digitalWrite(_SCHNAPPERPIN, HIGH);
|
|
|
|
|
|
|
|
// Heartbeat while the state is unlocked
|
|
|
|
while (_state == State::Unlocked) {
|
|
|
|
|
|
|
|
// In case of schnapper, send 0x55 resp. 0xaa to the schnapperpin
|
2015-05-14 19:36:49 +02:00
|
|
|
if (_schnapper == true)
|
|
|
|
{
|
2015-05-20 22:42:20 +02:00
|
|
|
for (int i = 0; i < 32 ; i++)
|
|
|
|
{
|
2015-05-21 13:35:30 +02:00
|
|
|
// Set '0'
|
|
|
|
digitalWrite(_SCHNAPPERPIN, LOW);
|
|
|
|
// cycle and send
|
2015-05-20 22:42:20 +02:00
|
|
|
beat();
|
2015-05-21 13:35:30 +02:00
|
|
|
// Set '1'
|
|
|
|
digitalWrite(_SCHNAPPERPIN, HIGH);
|
|
|
|
// cycle and send
|
2015-05-20 22:42:20 +02:00
|
|
|
beat();
|
|
|
|
}
|
2015-05-21 13:35:30 +02:00
|
|
|
|
|
|
|
// Reset schnapperpin
|
|
|
|
digitalWrite(_SCHNAPPERPIN, HIGH);
|
|
|
|
// and deactivate schnapper for the next round
|
2015-05-14 19:36:49 +02:00
|
|
|
_schnapper = false;
|
|
|
|
}
|
2015-05-21 13:35:30 +02:00
|
|
|
|
|
|
|
// Heartbeat
|
2015-05-14 19:36:49 +02:00
|
|
|
beat();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2015-05-11 00:18:22 +02:00
|
|
|
_l(LogLevel::info, "Door opened");
|
|
|
|
}
|