-
Notifications
You must be signed in to change notification settings - Fork 0
/
Long-Range-RFID-Sniffer.ino
310 lines (248 loc) · 11.3 KB
/
Long-Range-RFID-Sniffer.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
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
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
/***************************************************************
* Long Range RFID Sniffer *
* *
* Reads RFID cards from a long/medium distance, and displays *
* the card IDs on a web page on a WiFi network (SSID of a *
* printer by default) created by the device. *
* *
* This is intended to be loaded onto a WiFi-capable *
* Arduino-compatible board and used with a Wiegand long-range *
* reader (like the HID MaxiProx 5375.) *
* *
* by Ege F, 2022 *
* credit to Tom Igoe and Adafruit for parts of the WiFi code *
***************************************************************/
#include <SPI.h>
#include <WiFiNINA.h>
#include <Wiegand.h>
// Wiegand data pins
#define PIN_D0 2
#define PIN_D1 3
// The object that handles the wiegand protocol
Wiegand wiegand;
// String that will store all the capture messages
String captures = "";
// <WiFi Stuff>
char ssid[] = "DIRECT-04-HP OfficeJet 250"; // Pretend to be a printer since we can't hide SSID
char pass[] = "KeepItSimple";
int status = WL_IDLE_STATUS;
// </WiFi Stuff>
// Web server
WiFiServer server(80);
// Favicon of the captures page (id-card-clip from Font Awesome)
String favicon = "";
void setup() {
// <Wiegand stuff>
//Initialize serial
Serial.begin(9600);
//Install listeners and initialize Wiegand reader
wiegand.onReceive(receivedData, "Card read: ");
wiegand.onReceiveError(receivedDataError, "Card read error: ");
wiegand.onStateChange(stateChanged, "State changed: ");
wiegand.begin(Wiegand::LENGTH_ANY, true);
//initialize pins as INPUT and attaches interruptions
pinMode(PIN_D0, INPUT);
pinMode(PIN_D1, INPUT);
attachInterrupt(digitalPinToInterrupt(PIN_D0), pinStateChanged, CHANGE);
attachInterrupt(digitalPinToInterrupt(PIN_D1), pinStateChanged, CHANGE);
//Sends the initial pin state to the Wiegand library
pinStateChanged();
// </Wiegand stuff>
// <WiFi stuff>
wifi: // Label for retrying WiFi
Serial.println("Starting AP");
// check for the WiFi module:
if (WiFi.status() == WL_NO_MODULE) {
Serial.println("Communication with WiFi module failed!");
// Wait and try again
delay(3000);
goto wifi;
}
// Check NINA firmware version and warn if it's old
String fv = WiFi.firmwareVersion();
if (fv < WIFI_FIRMWARE_LATEST_VERSION) {
Serial.print("Please upgrade the firmware. Current firmware is ");
Serial.println(fv);
}
// Create the WiFi network
Serial.print("Creating access point named: ");
Serial.println(ssid);
status = WiFi.beginAP(ssid, pass);
if (status != WL_AP_LISTENING) {
Serial.println("Creating access point failed");
// Wait and try again
delay(3000);
goto wifi;
}
// Wait 10 seconds (example has this, not sure how useful)
delay(10000);
// </WiFi stuff>
// <web server stuff>
// start the web server on port 80
server.begin();
// Print connection status
printWiFiStatus();
// </web server stuff>
Serial.println("setup done");
}
// Tends to the Wiegand library
void doWiegand(){
noInterrupts();
wiegand.flush();
interrupts();
}
void loop() {
// Tend to the wiegand library
doWiegand();
// <WiFi stuff>
// Compare the previous status to the current status
if (status != WiFi.status()) {
// It has changed, update the variable
status = WiFi.status();
if (status == WL_AP_CONNECTED) {
// A device has connected to the AP
Serial.println("Device connected to AP");
} else {
// A device has disconnected from the AP, and we are back in listening mode
Serial.println("Device disconnected from AP");
}
}
// </WiFi stuff>
// <web server stuff>
WiFiClient client = server.available(); // Listen for incoming clients
if (client) { // if we get a client,
Serial.println("new client"); // print a message out to serial
String currentLine = ""; // Current line of the request
while (client.connected()) { // Loop while the client's connected
// We still have to tend to the Wiegand library when we're connected to a client
doWiegand();
if (client.available()) { // If there're bytes to read from the client,
char c = client.read(); // read a byte, then
Serial.write(c); // print it out to serial
if (c == '\n') { // If the byte is a newline character, we're at the end of a line
// If the current line is blank, we got two newline characters in a row.
// That's the end of the client HTTP request, so send a response:
if (currentLine.length() == 0) {
// Headers
client.println("HTTP/1.1 200 OK");
client.println("Content-type:text/html");
client.println();
// Body
// Get MAC to identify device in case multiple are deployes
byte mac[6];
WiFi.macAddress(mac);
// Page
client.println("<html><head><title>RFID Sniffer</title><link rel='shortcut icon' href = '");
client.println(favicon);
client.println("'></head><body><p>Device MAC: ");
client.print(mac[5],HEX);
client.print(":");
client.print(mac[4],HEX);
client.print(":");
client.print(mac[3],HEX);
client.print(":");
client.print(mac[2],HEX);
client.print(":");
client.print(mac[1],HEX);
client.print(":");
client.println(mac[0],HEX);
client.println("</p>");
// If we have anything captured, echo
if(captures.length() == 0){
client.println("<p>Nothing yet</p>");
}else{
client.println("<p>Dump:");
client.println("<textarea style='width: 100%; height: 80vh'>");
client.println(captures);
client.println("</textarea></p>");
}
client.println("<button onclick='location.reload();'>Refresh</button></body></html>");
// End of page
// The HTTP response ends with another blank line
client.println();
// Break out of the while loop:
break;
}
else { // If we got a newline, clear currentLine
currentLine = "";
}
}
else if (c != '\r') { // Otherwise, add it to currentLine (ignore CR)
currentLine += c;
}
}
}
// If there isn't a client connected anymore, close the connection
client.stop();
Serial.println("client disconnected");
}
// </web server stuff>
}
// Prints debug information
void printWiFiStatus() {
Serial.print("SSID: ");
Serial.println(WiFi.SSID());
IPAddress ip = WiFi.localIP();
Serial.print("IP Address: ");
Serial.println(ip);
}
// <Wiegand helper functions/ISRs>
// When any of the pins have changed, update the state of the wiegand library
void pinStateChanged() {
wiegand.setPin0State(digitalRead(PIN_D0));
wiegand.setPin1State(digitalRead(PIN_D1));
}
// Notifies when a reader has been connected or disconnected.
void stateChanged(bool plugged, const char* message) {
Serial.print(message);
Serial.println(plugged ? "CONNECTED" : "DISCONNECTED");
}
// Notifies when a card was read.
void receivedData(uint8_t* data, uint8_t bits, const char* message) {
// Debug output of the message
Serial.print(message);
Serial.print(bits);
Serial.print("bits / ");
// Store the message
captures += message;
captures += bits;
captures += "bits / ";
// Serial print and store the card ID in hex
uint8_t bytes = (bits+7)/8;
for (int i=0; i<bytes; i++) {
Serial.print(data[i] >> 4, 16);
Serial.print(data[i] & 0xF, 16);
captures += String(data[i] >> 4, 16);
captures += String(data[i] & 0xF, 16);
}
// NL signifies end of record
captures += "\n";
Serial.println();
}
// Notifies when an invalid transmission is detected (invalid bit length, malformed symbol, etc.)
void receivedDataError(Wiegand::DataError error, uint8_t* rawData, uint8_t rawBits, const char* message) {
// Debug output of the message
Serial.print(message);
Serial.print(Wiegand::DataErrorStr(error));
Serial.print(" - Raw data: ");
Serial.print(rawBits);
Serial.print("bits / ");
// Store the message
captures += message;
captures += Wiegand::DataErrorStr(error);
captures += " - Raw data: ";
captures += rawBits;
captures += "bits / ";
// Serial print and store the received data (likely a card ID of non-standard length) in hex
uint8_t bytes = (rawBits+7)/8;
for (int i=0; i<bytes; i++) {
Serial.print(rawData[i] >> 4, 16);
Serial.print(rawData[i] & 0xF, 16);
captures += String(rawData[i] >> 4, 16);
captures += String(rawData[i] & 0xF, 16);
}
// NL signifies end of record
captures += "\n";
Serial.println();
}
// <Wiegand helper functions/ISRs>