Go Down

Topic: [Résolu] Mettre mon ds1307 à l'heure (Read 8044 times) previous topic - next topic

Uzuma

Mar 31, 2014, 07:36 pm Last Edit: Apr 09, 2014, 05:54 pm by Uzuma Reason: 1
Bonsoir,

comme dit dans le titre j'aimerais pouvoir mettre mon ds1307 à l'heure. Oui bien sûr un tutoriel existe déjà sur le forum mais l'ennui c'est que j'ai déjà testé le code mais il ne donne pas les résultats escomptés. J'ai cherché ailleurs sur le net sans rien trouver bon. Tout ce qui m'importe c'est que mon horloge récupère l'heure courante de l'ordinateur. Le but est de pouvoir me servir de l'heure pour d'autres applications.

Dans l'espoir de trouver quelques choses de concluant, je vous dit merci d'avance.

Artouste


Bonsoir,

comme dit dans le titre j'aimerais pouvoir mettre mon ds1307 à l'heure. Oui bien sûr un tutoriel existe déjà sur le forum mais l'ennui c'est que j'ai déjà testé le code mais il ne donne pas les résultats escomptés. J'ai cherché ailleurs sur le net sans rien trouver bon. Tout ce qui m'importe c'est que mon horloge récupère l'heure courante de l'ordinateur. Le but est de pouvoir me servir de l'heure pour d'autres applications.

Dans l'espoir de trouver quelques choses de concluant, je vous dit merci d'avance.

J'ai utilisé le tuto de JF , ça fonctionnait "nickel" , c'est quoi exactement ton soucis avec ?

jfs

Oups... un week end un peu mouvementé et du coup j'ai oublié de refaire les tests avec l'IDE 1.xx et Processing 2.xx

Le tout sans lcd  :smiley-sweat:

Est ce que tu as essayé avec un lcd ?
Pas d'aide par MP !!!

Concernant le fonctionnement du forum tout se trouve dans les messages épinglés en tête de page.

Artouste


Oups... un week end un peu mouvementé et du coup j'ai oublié de refaire les tests avec l'IDE 1.xx et Processing 2.xx

Le tout sans lcd  :smiley-sweat:

Est ce que tu as essayé avec un lcd ?

Salut JF
coté arduino depuis la 1.00 les fonctions lecture/ecriture  I²C ont changées

Code: [Select]
/*
Permet de mettre une horloge ds1307 en synchronisation avec l'heure NTP
Par le biais d'une application Arduino-Processing
*/

/*

Comporte des parties de code écrites par :
- Maurice Ribble
  4-17-2008
  http://www.glacialwanderer.com/hobbyrobotics
  Pour le ds1307

- X. HINAULT
   01/2010
   http://www.mon-club-elec.fr
   Pour le LCD

*/

#include "Wire.h"
#define DS1307_I2C_ADDRESS 0x68

// Convert normal decimal numbers to binary coded decimal
byte decToBcd(byte val)
{
  return ( (val/10*16) + (val%10) );
}

// Convert binary coded decimal to normal decimal numbers
byte bcdToDec(byte val)
{
  return ( (val/16*10) + (val%16) );
}

// Stops the DS1307, but it has the side effect of setting seconds to 0
// Probably only want to use this for testing
/*void stopDs1307()
{
Wire.beginTransmission(DS1307_I2C_ADDRESS);
Wire.write(0);
Wire.write(0x80);
Wire.endTransmission();
}*/

// 1) Sets the date and time on the ds1307
// 2) Starts the clock
// 3) Sets hour mode to 24 hour clock
// Assumes you're passing in valid numbers
void setDateDs1307(byte second,        // 0-59
byte minute,        // 0-59
byte hour,          // 1-23
byte dayOfWeek,     // 1-7
byte dayOfMonth,    // 1-28/29/30/31
byte month,         // 1-12
byte year)          // 0-99
{
  Wire.beginTransmission(DS1307_I2C_ADDRESS);
  Wire.write(0);
  Wire.write(decToBcd(second));    // 0 to bit 7 starts the clock
  Wire.write(decToBcd(minute));
  Wire.write(decToBcd(hour));      // If you want 12 hour am/pm you need to set
  // bit 6 (also need to change readDateDs1307)
  Wire.write(decToBcd(dayOfWeek));
  Wire.write(decToBcd(dayOfMonth));
  Wire.write(decToBcd(month));
  Wire.write(decToBcd(year));
  Wire.endTransmission();
}

// Gets the date and time from the ds1307
void getDateDs1307(
byte *second,
byte *minute,
byte *hour,
byte *dayOfWeek,
byte *dayOfMonth,
byte *month,
byte *year)
{
  // Reset the register pointer
  Wire.beginTransmission(DS1307_I2C_ADDRESS);
  Wire.write(0);
  Wire.endTransmission();

  Wire.requestFrom(DS1307_I2C_ADDRESS, 7);

  // A few of these need masks because certain bits are control bits
  *second     = bcdToDec(Wire.read() & 0x7f);
  *minute     = bcdToDec(Wire.read());
  *hour       = bcdToDec(Wire.read() & 0x3f);  // Need to change this if 12 hour am/pm
  *dayOfWeek  = bcdToDec(Wire.read());
  *dayOfMonth = bcdToDec(Wire.read());
  *month      = bcdToDec(Wire.read());
  *year       = bcdToDec(Wire.read());
}


#include <LiquidCrystal.h>

const int RS=2; //declaration constante de broche
const int E=3; //declaration constante de broche
const int D4=4; //declaration constante de broche
const int D5=5; //declaration constante de broche
const int D6=6; //declaration constante de broche
const int D7=7; //declaration constante de broche
LiquidCrystal lcd(RS, E, D4, D5, D6, D7);

int octetReception=0; // variable de stockage des valeurs reçues sur le port Série (ASCII)
char caractereRecu=0; // variable pour stockage caractère recu
int compt=0; // variable comptage caractères reçus

String chaineReception=""; // déclare un objet String vide pour reception chaine
int oldSecond;

void setup() {
  byte second, minute, hour, dayOfWeek, dayOfMonth, month, year;
  Wire.begin(); // initialise 1wire

  Serial.begin(115200); // Initialise la communication série

  // lit les valeurs de l'horloge pour les replacer si elle est déjà à l'heure
  getDateDs1307(&second, &minute, &hour, &dayOfWeek, &dayOfMonth, &month, &year);
  if ((year)== NULL){ // si l'année = NULL
  // preinitialise l'horloge mets les valeurs lue précédement (à 0 si premier allumage)
  setDateDs1307(int(second), int(minute), int(hour), int(dayOfWeek), int(dayOfMonth), int(month), int(year));
  }

  //***** LCD *****//
  lcd.begin(20,4); // Initialise le LCD avec 20 colonnes x 4 lignes
  delay(10); // pause rapide pour laisser temps initialisation
  lcd.print("initialisation") ; // affiche la chaîne texte - message de test
  lcd.setCursor(0, 1) ;
  lcd.print("de l'horloge") ;
  delay(2000); // pause de 2 secondes
  lcd.clear(); // // efface écran et met le curseur en haut à gauche
  delay(100); // pour laisser temps effacer écran

  //**** liaison pour l'heure NTP ****//
  Serial.println('A'); // envoi un "top" à Processing
  while (millis()<5000) {  // attente pendant 5 secondes d'une liaison série (ne fonctionne pas en dessous)
    while (Serial.available()>0) { // tant qu'un octet est dans la liaison série

      octetReception=Serial.read(); // Lit le 1er octet reçu et le met dans la variable     

      if (octetReception==13) { // si l'octet reçu est le retour chariot (CR ou 13)

        // converti la chaine en entier pour les valeurs de temps
        second = int((chaineReception.charAt(0)-48)*10)+int(chaineReception.charAt(1)-48) ;
        minute = int((chaineReception.charAt(2)-48)*10)+int(chaineReception.charAt(3)-48);
        hour = int((chaineReception.charAt(4)-48)*10)+int(chaineReception.charAt(5)-48);
        dayOfWeek = 1;
        dayOfMonth = int((chaineReception.charAt(6)-48)*10)+int(chaineReception.charAt(7)-48);
        month = int((chaineReception.charAt(8)-48)*10)+int(chaineReception.charAt(9)-48);
        year = int((chaineReception.charAt(12)-48)*10)+int(chaineReception.charAt(13)-48); //int((chaineReception.charAt(10)-48)*1000)+int((chaineReception.charAt(11)-48)*100)+

        //mets le DS1307 en syncronisation avec l'horloge de l'ordinateur
        setDateDs1307(int(second), int(minute), int(hour), int(dayOfWeek), int(dayOfMonth), int(month), int(year));

        chaineReception=""; //RAZ de la chaine de réception
        break; // sort de la boucle while
      }
      else { // si le caractère reçu n'est pas un saut de ligne
        caractereRecu=char(octetReception); // convertit l'octet reçu en caractère
        chaineReception=chaineReception+caractereRecu; // ajoute le caratère à la chaine
      }
    }
  }
}

void loop(){

  byte second, minute, hour, dayOfWeek, dayOfMonth, month, year;

  getDateDs1307(&second, &minute, &hour, &dayOfWeek, &dayOfMonth, &month, &year);

  if(second!=oldSecond){  // change l'affichage uniquement au changement de seconde
    Serial.print(hour, DEC);
    Serial.print(":");
    Serial.print(minute, DEC);
    Serial.print(":");
    Serial.print(second, DEC);
    Serial.print("  ");
    Serial.print(month, DEC);
    Serial.print("/");
    Serial.print(dayOfMonth, DEC);
    Serial.print("/");
    Serial.print(year, DEC);
    Serial.print("  Day_of_week:");
    Serial.println(dayOfWeek, DEC);

    lcd.home();
    if (int(hour)<=9)lcd.print(0);
    lcd.print(int(hour));
    lcd.print (":");
    if (int(minute)<=9)lcd.print(0);
    lcd.print(int(minute));
    lcd.print (":");
    if (int(second)<=9)lcd.print(0);
    lcd.print(int(second));
    lcd.setCursor (0,1);
    if (int(dayOfMonth)<=9)lcd.print(0);
    lcd.print(int(dayOfMonth));
    lcd.print ("/");
    if (int(month)<=9)lcd.print(0);
    lcd.print(int(month));
    lcd.print ("/20");
    lcd.print(int(year));

  }
  oldSecond=second;  // réinitialise oldSecond

}

jfs

Oui, sur la deuxième page du tuto j'avais fait cette correction.
Pas d'aide par MP !!!

Concernant le fonctionnement du forum tout se trouve dans les messages épinglés en tête de page.

Uzuma

Bonsoir,


J'ai utilisé le tuto de JF , ça fonctionnait "nickel" , c'est quoi exactement ton soucis avec ?



Je ne comprends pas moi-même, je ne devrais pas avoir de soucis  normalement :(


Oups... un week end un peu mouvementé et du coup j'ai oublié de refaire les tests avec l'IDE 1.xx et Processing 2.xx

Le tout sans lcd  :smiley-sweat:

Est ce que tu as essayé avec un lcd ?


Je  n'ai pas essayé avec le lcd car je n'en ai pas à ma disposition (je n'ai que des afficheurs 7 segments). Mais moi je me disais qu'il ne devrait pas y avoir de problème même sans lcd.
Svp aidez-moi à résoudre ce problème.

john_lenfr

Code: [Select]
// libraries
#include <SPI.h>
#include <Ethernet.h>
#include <Wire.h>
#include "RTClib.h"
#include <EthernetUdp.h>

// Configuration des adresses Ethernet (MAC) et Internet (IP) du serveur TCP/IP de la carte Arduino.
byte mac[] = { 0x90, 0xA2, 0xDA, 0x0D, 0xB5, 0x42 };//Valeur en hexadécimale de l'adresse MAC de la carte Arduino. Valeur unique sur le réseau.
IPAddress ip(192,168,1,12);

// Configuration pour connection à un serveur NTP pour mise à jour de l'heure
unsigned int localPort = 8888;             // local port to listen for UDP packets
IPAddress timeServer(95, 130, 9, 77);      // http://Ntp.crashdump.fr se synchronise sur les serveur de strate 1
//IPAddress timeServer(193, 79, 237, 14);
//IPAddress timeServer(193, 52, 137, 193);   //http://www.pool.ntp.org/fr/
byte pb[48];                  // buffer to hold incoming and outgoing packets // NTP time stamp is in the first 48 bytes of the message

//Déclaration Internet
EthernetServer server(80);//déclaration du serveur internet de la carte Arduino.
// A UDP instance to let us send and receive packets over UDP
EthernetUDP Udp;

// for refresh time day
const unsigned long DISPLAY_INTERVAL_day = 86400000; // wait for 1 day
static unsigned long lastTime_day=0;  // in ms
unsigned long time_day=0;      // in ms

// DS 1307 Time declarations
typedef enum clock__members
{   clock__MemberDay,
   clock__MemberMonth,
   clock__MemberYear,
   clock__MemberHour,
   clock__MemberMinute,
   clock__MemberSecond
} clock__member_t;

static void clock__printDatetime();
static RTC_DS1307  clock__rtc;
static DateTime clock__currentTime;


Code: [Select]

///// SETUP /////
void setup() {
      Serial.begin(9600);
      Ethernet.begin(mac, ip);
      server.begin();
      Wire.begin();

      // Initialize RTC.
      clock__rtc.begin();
      delay(10);

      // update RTC time with NTP protocol if connected to network
      ntpUpdate();        
}


Code: [Select]

///// MAIN PROG /////
void loop(){

 clock__currentTime = clock__rtc.now();
 clock__printDatetime();


 // Update time every day by NTP
 time_day=millis();
 if (time_day-lastTime_day>DISPLAY_INTERVAL_day)  // if at least DISPLAY_INTERVAL ms have passed
 {
   ntpUpdate(); // update RTC time with NTP protocol  
   lastTime_day=time_day;  // reset timer
 }

}


Code: [Select]
static void clock__printDatetime(){
   // Build date string
   char date[] = "XX/XX/XXXX";
   snprintf
   (
       date,
       strlen(date) + 1,
       "%02d/%02d/%04d",
       clock__currentTime.day(),
       clock__currentTime.month(),
       clock__currentTime.year()
   );
   
   // Build time string
   char time[] = "00:00:00";
   snprintf
   (
       time,strlen(time) + 1,
       "%02d:%02d:%02d",
       clock__currentTime.hour(),
       clock__currentTime.minute(),
       clock__currentTime.second()
   );
   
   // Print date
   lcd.setCursor(3, 0);
   lcd.print(date);
   
   // Print time
   lcd.setCursor(4, 1);
   lcd.print(time);
}

// send an NTP request to the time server at the given address
unsigned long sendNTPpacket(IPAddress& address)
{
 // set all bytes in the buffer to 0
 memset(pb, 0, 48);
 // Initialize values needed to form NTP request
 // (see URL above for details on the packets)
 pb[0] = 0b11100011;   // LI, Version, Mode
 pb[1] = 0;     // Stratum, or type of clock
 pb[2] = 6;     // Polling Interval
 pb[3] = 0xEC;  // Peer Clock Precision
 // 8 bytes of zero for Root Delay & Root Dispersion
 pb[12]  = 49;
 pb[13]  = 0x4E;
 pb[14]  = 49;
 pb[15]  = 52;

 // all NTP fields have been given values, now
 // you can send a packet requesting a timestamp:
 // 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(pb,48);
 Udp.endPacket();
}

void ntpUpdate(void)
{
   lcd.setCursor(0,0);
   lcd.print("NTP Time sync...");
   Udp.begin(localPort);
   delay(250);
   // send an NTP packet to a time server
   sendNTPpacket(timeServer);

   // wait to see if a reply is available
   delay(1000);

   if ( Udp.parsePacket() ) {
         // read the packet into the buffer
     
         Udp.read(pb, 48);
     
         // NTP contains four timestamps with an integer part and a fraction part
         // we only use the integer part here
         unsigned long t1, t2, t3, t4;
         t1 = t2 = t3 = t4 = 0;
         for (int i=0; i< 4; i++)
         {
           t1 = t1 << 8 | pb[16+i];      
           t2 = t2 << 8 | pb[24+i];      
           t3 = t3 << 8 | pb[32+i];      
           t4 = t4 << 8 | pb[40+i];
         }
           
         // part of the fractional part
         // could be 4 bytes but this is more precise than the 1307 RTC
         // which has a precision of ONE second
         // in fact one byte is sufficient for 1307
           float f1,f2,f3,f4;
           f1 = ((long)pb[20] * 256 + pb[21]) / 65536.0;      
           f2 = ((long)pb[28] * 256 + pb[29]) / 65536.0;      
           f3 = ((long)pb[36] * 256 + pb[37]) / 65536.0;      
           f4 = ((long)pb[44] * 256 + pb[45]) / 65536.0;
           //Serial.println(f1);
           //Serial.println(f2);
           //Serial.println(f3);
           //Serial.println(f4);
           const unsigned long seventyYears = 2208988800UL;
           t1 -= seventyYears;
           t2 -= seventyYears;
           t3 -= seventyYears;
           t4 -= seventyYears;
     
           //Serial.println("T1 .. T4 && fractional parts");
           //PrintDateTime(DateTime(t1)); Serial.println(f1,4);
           //PrintDateTime(DateTime(t2)); Serial.println(f2,4);
           //PrintDateTime(DateTime(t3)); Serial.println(f3,4);
           
           //PrintDateTime(DateTime(t4)); Serial.println(f4,4);
           //Serial.println();
           // Adjust timezone and DST... in my case substract 4 hours for Chile Time
           // or work in UTC?
           t4 += adjustDstEurope(t4); // Adjustig Time with DST Europe/France: UTC+1h in winter, UTC+2h in summer
           t4 += 1;                // adjust the delay(1000) lost at begin of loop
           if (f4 > 0.4) t4++;     // adjust fractional part, see above
           clock__rtc.adjust(DateTime(t4)); // update RTC DS 1307 Clock
           lcd.setCursor(0,0);
           lcd.print("NTP sync OK!    ");
           delay(2000);
           lcd.clear();
     }
     else
     {
     lcd.setCursor(0,0);
     lcd.print("NTP sync not OK!");
     delay(2000);
     lcd.clear();
     }
}


int adjustDstEurope(DateTime t)
{
/*You can use the following equations to calculate when DST starts and ends.
The divisions are integer divisions, in which remainders are discarded.
"mod" means the remainder when doing integer division, e.g., 20 mod 7 = 6.
That is, 20 divided by 7 is 2 and 6/7th (where six is the remainder).
With: y = year.
       For the United States:
           Begin DST: Sunday April (2+6*y-y/4) mod 7+1
           End DST: Sunday October (31-(y*5/4+1) mod 7)
          Valid for years 1900 to 2006, though DST wasn't adopted until the 1950s-1960s. 2007 and after:
           Begin DST: Sunday March 14 - (1 + y*5/4) mod 7
           End DST: Sunday November 7 - (1 + y*5/4) mod 7;
       European Economic Community:
           Begin DST: Sunday March (31 - (5*y/4 + 4) mod 7) at 1h U.T.
           End DST: Sunday October (31 - (5*y/4 + 1) mod 7) at 1h U.T.
           Since 1996, valid through 2099
(Equations by Wei-Hwa Huang (US), and Robert H. van Gent (EC))

Adjustig Time with DST Europe/France/Paris: UTC+1h in winter, UTC+2h in summer

*/

 // last sunday of march
 int beginDSTDate=  (31 - (5* t.year() /4 + 4) % 7);
 //Serial.println(beginDSTDate);
 int beginDSTMonth=3;
 //last sunday of october
 int endDSTDate= (31 - (5 * t.year() /4 + 1) % 7);
 //Serial.println(endDSTDate);
 int endDSTMonth=10;
 // DST is valid as:
 if (((t.month() > beginDSTMonth) && (t.month() < endDSTMonth))
     || ((t.month() == beginDSTMonth) && (t.day() >= beginDSTDate))
     || ((t.month() == endDSTMonth) && (t.day() <= endDSTDate)))
 return 7200;      // DST europe = utc +2 hour (summer time)
 else return 3600; // nonDST europe = utc +1 hour (winter time)
}


Avec gestion heure d'été, heure d'hiver

Uzuma


........................................
Avec gestion heure d'été, heure d'hiver


J'aimerais que l'on me rassure : Avec ce code, je n'aurai pas besoin de shield ethernet n'est-ce pas ? L'arduino seul peut faire l'affaire n''est-ce pas ?!


john_lenfr

non avec ce code  il faut un shield.

ton arduino sera branché en permanence a ton PC en usb? je n'en voit pas l'interet dans ce cas

Uzuma


non avec ce code  il faut un shield.


Ce n'est ce que je cherchais. Il n'y a que ça comme alternative ?


ton arduino sera branché en permanence a ton PC en usb? je n'en voit pas l'interet dans ce cas


Non. Mon système en entier sera autonome. Le branchement au PC ne servira juste qu'à téléverser le programme.

john_lenfr

Et tu souhaites mettre à l'heure régulièrement ou juste une fois définitivement?
Sinon si tu veux pas de câble ça doit pouvoir passer avec un shield wifi non?


gunsman76

En fait c'est assez simple de mettre à l'heure, je ne comprends pas pourquoi tout le monde se complique la vie...

Tu déclares ta librairie DS1307

Dans le void setup :

Code: [Select]
Serial.begin(57600);
    Wire.begin();
    RTC.begin();
   



  if (! RTC.isrunning()) {
    Serial.println("RTC is NOT running!");
    // following line sets the RTC to the date & time this sketch was compiled
    RTC.adjust(DateTime(__DATE__, __TIME__));

  }



Dans le void loop :

Code: [Select]

  //RTC.adjust(DateTime(__DATE__, __TIME__));

 
  DateTime now = RTC.now();
  lcd.setCursor(0, 1);
  lcd.print (now.hour(),DEC);
  lcd.print(":");
  lcd.print(now.minute(), DEC);



Il faut juste décommenter le ligne   //RTC.adjust(DateTime(__DATE__, __TIME__));  lorsque tu es connecté à ton ordi, une fois à l'heure, il faut téléverser le programme une seconde fois en recommantant la ligne   //RTC.adjust(DateTime(__DATE__, __TIME__));


voilà

john_lenfr

Oui c'est clair que si c'est pour mettre à l'heure une seule fois c'est plus simple.
Ce code est d'ailleurs par défaut dans la lib RTC.

Uzuma


Et tu souhaites mettre à l'heure régulièrement ou juste une fois définitivement?


Une seule fois.



En fait c'est assez simple de mettre à l'heure, je ne comprends pas pourquoi tout le monde se complique la vie...


Effectivement tu as raison ;)

Merci beaucoup pour ton intervention ! Avec le code que tu as laissé, j'ai fait quelques recherches sur google (en tapant les bons mots cette fois) et je suis tombé sur le code d'Adafruit qui m'a beaucoup aidé d'ailleurs. Faut dire que mon problème est presque résolu, car l'heure y est presque : je veux dire qu'il y a un décalage d'une minute environ. Enfin ...... regardez par vous-même :


jfs

#14
Apr 01, 2014, 07:52 pm Last Edit: Apr 01, 2014, 07:54 pm by Jean-François Reason: 1
Bon... j'ai tout ressorti, enlevé la poussière  :smiley-mr-green:

Mis en configuration sans lcd (tout viré ce qui s'y rapporte), branché mon ds1307 sur les pin A4 et A5 (Duemilanova atmega168).

Je téléverse le sketch pour la version 1.xx (j'utilise la 1.04), lance le sketch Processing (1.5.1),cette fois je suis en position 4 de la liste des ports disponibles, je change la position du port dans le sketch, débranche mon arduino, lance le sketch Processing à nouveau, rebranche l'Arduino.... j'attends, dans la console série Processing une série de chiffres s'affiche signe qu'il s'est passé quelques chose, je sors du sketch Processing en appuyant sur le carré, Je lance la console série sur l'IDE Arduino.... et l'heure de mon ds1307 est au 10ème de seconde près le même que l'horloge de mon ordi.

mon ds1307 n'a pas de pile, donc c'est pas une heure qui reste en mémoire.

Voili voilou
Pas d'aide par MP !!!

Concernant le fonctionnement du forum tout se trouve dans les messages épinglés en tête de page.

Go Up