215 lines
4.7 KiB
Arduino
Raw Normal View History

2021-01-29 12:31:58 +01:00
//
// FILE: DHT_simulator.ino
// AUTHOR: Rob Tillaart
// VERSION: 0.2.1
// PURPOSE: Simulation of the DHT protocol
// DATE: 2014-06-14
// URL: https://github.com/RobTillaart/DHT_Simulator
// TODO
// - split loop into some functions?
2023-10-24 15:56:14 +02:00
// SET ACTUAL PINS PER PLATFORM
const int dataPin = 5; // connect to MCU ( !! also connect GND !! )
const int CRCPin = 6; // connect switch to simulate CRC errors.
const int TimeOutPin = 7; // connect switch to simulate timeout errors.
const int PulseLenPin = 8; // connect switch to simulate pulselength errors.
2021-01-29 12:31:58 +01:00
#if defined(__AVR__)
2023-10-24 15:56:14 +02:00
const int humPin = A0; // analog pins for potmeters.
2021-01-29 12:31:58 +01:00
const int tempPin = A2;
#elif defined(ESP32)
const int humPin = 14;
const int tempPin = 15;
#elif defined(ESP8266)
const int humPin = 2;
const int tempPin = 3;
2023-10-24 15:56:14 +02:00
#else // CI
2021-01-29 12:31:58 +01:00
const int humPin = A0;
const int tempPin = A2;
#endif
2023-10-24 15:56:14 +02:00
// DATA TO SEND
byte b[5]; // actual bytes sent
int humidity; // humidity * 10 - prevent float operations
int temperature; // temperature * 10
2021-01-29 12:31:58 +01:00
2023-10-24 15:56:14 +02:00
// CONFIGURE
const bool randomize = true; // use random generator
const bool debug = false; // test data generation
2021-01-29 12:31:58 +01:00
2023-10-24 15:56:14 +02:00
bool CRCerror = false; // inject CRC error
bool TimeOutError = false; //
bool PulseLenError = false; //
uint32_t count = 0; // count values per second generated
uint32_t lastTime = 0; // keep track of timing
2021-01-29 12:31:58 +01:00
/////////////////////////////////////////
2023-10-24 15:56:14 +02:00
2021-01-29 12:31:58 +01:00
void setup()
{
Serial.begin(115200);
Serial.print("Start ");
Serial.println(__FILE__);
pinMode(dataPin, INPUT_PULLUP);
pinMode(CRCPin, INPUT_PULLUP);
pinMode(TimeOutPin, INPUT_PULLUP);
pinMode(PulseLenPin, INPUT_PULLUP);
}
2023-10-24 15:56:14 +02:00
2021-01-29 12:31:58 +01:00
void loop()
{
2023-10-24 15:56:14 +02:00
yield(); // keep ESP happy
2021-01-29 12:31:58 +01:00
count++;
uint32_t now = millis();
if (now - lastTime >= 1000)
{
uint32_t nps = round((1000.0 * count) / (now - lastTime));
Serial.print("DATA PER SECOND: ");
Serial.println(nps);
lastTime = now;
count = 0;
}
if (randomize)
{
humidity = random(20, 1000);
temperature = random(-200, 1800);
}
else
{
analogRead(humPin);
humidity = analogRead(humPin);
analogRead(tempPin);
temperature = analogRead(tempPin) * 2 - 200;
}
humidity = constrain(humidity, 0, 1000);
temperature = constrain(temperature, -200, 1800);
if (debug)
{
Serial.print(humidity);
Serial.print("\t");
Serial.print(temperature);
Serial.println();
}
2023-10-24 15:56:14 +02:00
// READ "ERROR" PINS
2021-01-29 12:31:58 +01:00
CRCerror = digitalRead(CRCPin) == LOW;
TimeOutError = digitalRead(TimeOutPin) == LOW;
PulseLenError = digitalRead(PulseLenPin) == LOW;
2023-10-24 15:56:14 +02:00
// WAKE UP SIGNAL DETECTED
2021-01-29 12:31:58 +01:00
if (digitalRead(dataPin) == LOW)
{
uint32_t start = micros();
2023-10-24 15:56:14 +02:00
// wait max 1500 us until signal goes high
2021-01-29 12:31:58 +01:00
while (digitalRead(dataPin) == LOW)
{
if (micros() - start > 1500)
{
// Serial.println("ERROR: low puise too long");
return;
}
}
2023-10-24 15:56:14 +02:00
if (micros() - start > 500) // serious request...
2021-01-29 12:31:58 +01:00
{
DHTsend(humidity, temperature);
Serial.print(humidity);
Serial.print("\t");
Serial.print(temperature);
Serial.print("\t");
for (int i = 0; i < 5; i++)
{
if (b[i] < 0x10) Serial.print('0');
Serial.print(b[i], HEX);
Serial.print(' ');
}
Serial.println();
}
else
{
Serial.println("ERROR: low puise too short");
}
}
}
void DHTsend(int H, int T)
{
pinMode(dataPin, OUTPUT);
2023-10-24 15:56:14 +02:00
// SEND ACK
2021-01-29 12:31:58 +01:00
digitalWrite(dataPin, LOW);
2023-10-24 15:56:14 +02:00
delayMicroseconds(80); // 80 us
2021-01-29 12:31:58 +01:00
digitalWrite(dataPin, HIGH);
2023-10-24 15:56:14 +02:00
delayMicroseconds(80); // 80 us
2021-01-29 12:31:58 +01:00
if (TimeOutError)
{
2023-10-24 15:56:14 +02:00
delayMicroseconds(100); // inject extra 100 microseconds
2021-01-29 12:31:58 +01:00
}
2023-10-24 15:56:14 +02:00
// PREPARE DATA
2021-01-29 12:31:58 +01:00
b[0] = H / 256;
b[1] = H & 255;
b[2] = 0;
if (T < 0)
{
T = -T;
b[2] = 0x80;
}
b[2] |= T / 256;
b[3] = T & 255;
2023-10-24 15:56:14 +02:00
// CRC
2021-01-29 12:31:58 +01:00
b[4] = b[0] + b[1] + b[2] + b[3];
2023-10-24 15:56:14 +02:00
if (CRCerror) b[4]++; // inject CRC error
2021-01-29 12:31:58 +01:00
2023-10-24 15:56:14 +02:00
// SEND DATA
2021-01-29 12:31:58 +01:00
for (int i = 0; i < 5; i++)
{
DHTsendbyte(b[i]);
}
2023-10-24 15:56:14 +02:00
// END OF TRANSMISSION SIGNAL
2021-01-29 12:31:58 +01:00
digitalWrite(dataPin, LOW);
2023-10-24 15:56:14 +02:00
delayMicroseconds(50); // 50 us
2021-01-29 12:31:58 +01:00
pinMode(dataPin, INPUT_PULLUP);
}
2023-10-24 15:56:14 +02:00
// timing manual tuned
2021-01-29 12:31:58 +01:00
void DHTsendbyte(byte b)
{
byte mask = 128;
for (int i = 0; i < 8; i++)
{
digitalWrite(dataPin, LOW);
2023-10-24 15:56:14 +02:00
delayMicroseconds(45); // 50 us
2021-01-29 12:31:58 +01:00
if (PulseLenError)
{
2023-10-24 15:56:14 +02:00
delayMicroseconds(10); // inject extra pulse length // TWEAK amount
2021-01-29 12:31:58 +01:00
}
digitalWrite(dataPin, HIGH);
2023-10-24 15:56:14 +02:00
if (b & mask) delayMicroseconds(60); // 70 us
else delayMicroseconds(24); // 26 us
2021-01-29 12:31:58 +01:00
mask >>= 1;
}
}
2023-10-24 15:56:14 +02:00
// -- END OF FILE --