hallo,
ich verwende hier die timelib.h sowie timealarms.h als Zeitschaltuhr.
Die aktuelle Uhrzeit erhält der Arduino von einem lokalen Zeitserver.
Die Unix-zeit wird dann in einer Funktion umgerechnet.
Soweit funktioniert das auch ganz gut.
1 x pro Stunde wird die zeit via Zeitserver aktualisiert mit
setSyncInterval(interval);
setSyncInterval verwendet aber intern wohl hour().
Nach einer Stunde habe ich also wieder die Unix-zeit und nicht die Sommerzeit.
Da ich timealarms verwende ist das nicht schön und meine Zeitschaltuhr funktioniert so nicht sauber.
hier der Code
#include <Ethernet.h>
#include <EthernetUdp.h>
#include <TimeLib.h>
#include <TimeAlarms.h>
// Ethernet W5100
byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED };
IPAddress ip(192, 168, 2, 9 );
IPAddress myDns(192, 168, 2, 1);
IPAddress gateway(192, 168, 2, 1);
IPAddress subnet(255, 255, 255, 0);
// NTP Servers
IPAddress timeServer(192, 168, 2, 10); // lokaler Raspberry Zeitserver
const int timeZone = 1; // Central European Time
int stunde = 0;
int minuten = 0;
int sekunden = 0;
const int relaisLicht_1 = 43;
const int relaisLicht_2 = 42;
EthernetUDP Udp;
unsigned int localPort = 8888; // local port to listen for UDP packets
void setup() {
Serial.begin(9600);
// Ethernet Start
Ethernet.begin(mac, ip, myDns, gateway, subnet);
Udp.begin(localPort);
//int interval = 3600; // 1 x pro Stunde Zeit synchronisieren
int interval = 60; // 1 x pro Stunde Zeit synchronisieren
setSyncInterval(interval);
setSyncProvider(getNtpTime);
Serial.print(" vor Settime hour(): ");Serial.print(hour());
Serial.print(" vor Settime minute(): ");Serial.println(minute());
unixtime_umrechnen_aktualisieren();
setTime(stunde, minuten, sekunden, month(), day(), year());
Serial.print(" nach Settime hour(): ");Serial.print(hour());
Serial.print(" nach Settime minute(): ");Serial.println(minute());
Alarm.alarmRepeat(8,30,0, startBeleuchtung); //
Alarm.alarmRepeat(20,30,0, stopBeleuchtung); //
pinMode(relaisLicht_1, OUTPUT);
pinMode(relaisLicht_2, OUTPUT);
startBeleuchtung();
}
time_t prevDisplay = 0; // when the digital clock was displayed
void loop() {
if (timeStatus() != timeNotSet) {
if (now() != prevDisplay) { //update the display only if time has changed
prevDisplay = now();
digitalClockDisplay();
}
}
Serial.print(" hour(): ");Serial.print(hour());
Serial.print(" minute(): ");Serial.println(minute());
Serial.print(" stunde : ");Serial.print(stunde);
Serial.print(" minuten : ");Serial.println(minuten);
delay(3000);
}
void startBeleuchtung(){
digitalWrite(relaisLicht_1, LOW);
digitalWrite(relaisLicht_2, LOW);
}
void stopBeleuchtung(){
digitalWrite(relaisLicht_1, HIGH);
digitalWrite(relaisLicht_1, HIGH);
}
void digitalClockDisplay()
{
// unixtime_umrechnen_aktualisieren();
Serial.print(stunde);
//printDigits(minute());
printDigits(minuten);
//printDigits(second());
printDigits(sekunden);
Serial.println();
Serial.print(day());
Serial.print(".");
Serial.print(month());
Serial.print(".");
Serial.print(year());
Serial.println();
}
void printDigits(int digits){
// utility for digital clock display: prints preceding colon and leading 0
Serial.print(":");
if(digits < 10)
Serial.print('0');
Serial.print(digits);
}
/*-------- NTP code ----------*/
const int NTP_PACKET_SIZE = 48; // NTP time is in the first 48 bytes of message
byte packetBuffer[NTP_PACKET_SIZE]; //buffer to hold incoming & outgoing packets
time_t getNtpTime()
{
while (Udp.parsePacket() > 0) ; // discard any previously received packets
Serial.println("Transmit NTP Request");
sendNTPpacket(timeServer);
uint32_t beginWait = millis();
while (millis() - beginWait < 1500) {
int size = Udp.parsePacket();
if (size >= NTP_PACKET_SIZE) {
Serial.println("Receive NTP Response");
Udp.read(packetBuffer, NTP_PACKET_SIZE); // read packet into the buffer
unsigned long secsSince1900;
// convert four bytes starting at location 40 to a long integer
secsSince1900 = (unsigned long)packetBuffer[40] << 24;
secsSince1900 |= (unsigned long)packetBuffer[41] << 16;
secsSince1900 |= (unsigned long)packetBuffer[42] << 8;
secsSince1900 |= (unsigned long)packetBuffer[43];
return secsSince1900 - 2208988800UL + timeZone * SECS_PER_HOUR;
}
}
Serial.println("No NTP Response :-(");
return 0; // return 0 if unable to get the time
}
void unixtime_umrechnen_aktualisieren()
{
stunde=hour()+(int)summertime_EU(year(),month(),day(),hour(),1);
minuten=minute();
sekunden=second();
if (stunde == 24){
stunde = 00;
}
}
boolean summertime_EU(int year, byte month, byte day, byte hour, byte tzHours)
{
if (month<3 || month>10) return false; // keine Sommerzeit in Jan, Feb, Nov, Dez
if (month>3 && month<10) return true; // Sommerzeit in Apr, Mai, Jun, Jul, Aug, Sep
if (month==3 && (hour + 24 * day)>=(1 + tzHours + 24*(31 - (5 * year /4 + 4) % 7)) || month==10 && (hour + 24 * day)<(1 + tzHours + 24*(31 - (5 * year /4 + 1) % 7)))
return true;
else
return false;
}
// send an NTP request to the time server at the given address
void sendNTPpacket(IPAddress &address)
{
// set all bytes in the buffer to 0
memset(packetBuffer, 0, NTP_PACKET_SIZE);
// Initialize values needed to form NTP request
// (see URL above for details on the packets)
packetBuffer[0] = 0b11100011; // LI, Version, Mode
packetBuffer[1] = 0; // Stratum, or type of clock
packetBuffer[2] = 6; // Polling Interval
packetBuffer[3] = 0xEC; // Peer Clock Precision
// 8 bytes of zero for Root Delay & Root Dispersion
packetBuffer[12] = 49;
packetBuffer[13] = 0x4E;
packetBuffer[14] = 49;
packetBuffer[15] = 52;
// all NTP fields have been given values, now
// you can send a packet requesting a timestamp:
Udp.beginPacket(address, 123); //NTP requests are to port 123
Udp.write(packetBuffer, NTP_PACKET_SIZE);
Udp.endPacket();
}
ich habe mir schon überlegt das Umrechnen zur Sommerzeit in der Timlib und dort in der setSyncInterval Funktion mit einzubauen....
geht das ? oder gibt es da eine bessere Lösung?
Grüße Sepp
edit: Interval habe ich auf 60 sekunden gesetzt nur zum testen! normal = 3600 sekunden