IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Arduino Discussion :

Comment se gère l'environnement (dossiers) de l'Arduino sur Windows ?


Sujet :

Arduino

  1. #1
    Membre du Club
    Homme Profil pro
    retraité
    Inscrit en
    Décembre 2019
    Messages
    122
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Yvelines (Île de France)

    Informations professionnelles :
    Activité : retraité

    Informations forums :
    Inscription : Décembre 2019
    Messages : 122
    Points : 60
    Points
    60
    Par défaut Comment se gère l'environnement (dossiers) de l'Arduino sur Windows ?
    Bonjour à tous

    Encore une question basique à laquelle j'ai du mal à répondre: les dossiers d'installation.

    Sur mon PC, Arduino est installé, mais pour dire la vérité, je ne sais pas vraiment , même si je suppose que c'est en C:.

    La question pourrait sembler bête, mais, pour ne pas surcharger mon disque C, j'utilise au maximum un disque dur externe, sur lequel je sauvegarde mes sketches, et d'où je les lance.

    J'ai constaté, un peu à mes dépends, que la sauvegarde des .ino exige que cet ino soit dans un répertorie portant le même nom.

    Là où ça se complique, quand j'ai envie de rajouter du code dans un sketch, et pour lequel je vais devoir manipuler des données qui devront être accessibles dans toute l'étendue du programme, je veux ajouter des déclarations de variables.

    Je veux me créer un fichier de déclarations (que je pourrai appeler .lib mais que j'appelais .c en Turbo C)qui devra être inclus dans mon programme. Par exemple

    #include "ce_fichier.lib".

    Si je le mets dans le répertoire de mon .ino, la compilation me répond qu'il ne le trouve pas.

    J'en conclus que le compilateur s'attend à le voir ailleurs, et j'ai beau chercher, je ne vois pas où.

    Est-ce sur C, où il existe bien un répertoire arduino, mais que je n'ai pas créé et qui contient entre autres, fichiers, le dossier arduino-nightly.


    Où dois-je placer ce fichier de déclaration, mais il y aura aussi, je pense, des fonctions à venir que je vais créer au fur et à mesure de mon avancée de mon programme?

    Peut-être suis-je maladroit dans la rédaction de mon explication, aussi, je vous demande pardon.


    Un tuto me dit qu'il y a un dossier arduino dans le dossier Program files (86) en Windows 64 bits mais ce dossier Program files n'existe pas chez moi et encore moins arduino, ni en Programmes ni en Programmes (86).

  2. #2
    Expert confirmé

    Homme Profil pro
    mad scientist :)
    Inscrit en
    Septembre 2019
    Messages
    2 816
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : Etats-Unis

    Informations professionnelles :
    Activité : mad scientist :)

    Informations forums :
    Inscription : Septembre 2019
    Messages : 2 816
    Points : 5 673
    Points
    5 673
    Par défaut
    je suis sur Mac donc je ne peux pas trop vérifier mais de mémoire vous allez avoir les "outils" dans C:\Users\votre_user\AppData\Local\Arduino15

    ensuite dans vos Documents vous devez trouver le répertoire Arduino et c'est là qu'il y aura les Bibliothèques, autres outils, etc

    J'ai constaté, un peu à mes dépends, que la sauvegarde des .ino exige que cet ino soit dans un répertorie portant le même nom.
    Oui, c'est une obligation...

    il faut comprendre un peu ce que fait l'IDE lors de la demande de compilation pour simplifier la vie:

    - il commence par créer un espace temporaire ailleurs sur le disque
    - il duplique le répertoire courant et tout son contenu dans cet espace temporaire dans un répertoire 'sketch'
    - il prend tous les .ino et les ajoute dans l'ordre alphabétique à la fin du sketch principal (celui qui porte le même nom que le répertoire)
    - il analyse analyse le contenu du fichier résultant et génère des prototypes des fonctions en tout début de ce fichier (de façon à ce que l'ordre de déclaration des fonctions ne soit pas important - normalement en C ou C++ vous ne pouvez pas appeler une fonction ou classe non déclaré avant dans le code)
    - il analyse les #include pour essayer d'être intelligent et trouver les bonnes Bibliothèques à compiler avec votre code. Il "copie" ces Bibliothèques dans l'espace temporaire sous un répertoire 'libraries'
    - ensuite il compile les Bibliothèques, le code résultant des .ino, les fichiers .c .cpp etc qu'il trouve dans le répertoire courant
    - il compile un main.cpp qui est le point d'entrée principal du code C++
    - il fait l'édition de lien de tout cela pour générer le binaire HEX qui sera chargé sur la carte


    Là où ça se complique, quand j'ai envie de rajouter du code dans un sketch, et pour lequel je vais devoir manipuler des données qui devront être accessibles dans toute l'étendue du programme, je veux ajouter des déclarations de variables.
    Je veux me créer un fichier de déclarations (que je pourrai appeler .lib mais que j'appelais .c en Turbo C)qui devra être inclus dans mon programme. Par exemple
    #include "ce_fichier.lib".
    Si je le mets dans le répertoire de mon .ino, la compilation me répond qu'il ne le trouve pas.
    le .lib n'est pas une extension connue de l'IDE et donc ça va être ignoré lors de la copie dans le répertoire temporaire. Appelez le .h et tout se passera bien si vous respectez le C/C++, c'est à dire que vous ne pouvez définir une variable qu'une seule fois.

    ==> par exemple vous définissez la variable int foo = 8; dans le sketch principal et si vous voulez l'utiliser dans d'autres modules, alors vous pouvez mettre dans un fichier variablesPartagees.h et dans les autres fichiers vous feriez
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    #include "variablesPartagees.h"
    Les guillemets sont importants car ça dit au compilateur de chercher le fichier variablesPartagees.h dans le répertoire courant

    Pour créer un second fichier dans le répertorie de travail et le voir dans l'IDE, le plus simple est d'utiliser la petite flèche à droite de la fenêtre
    Nom : onglet.png
Affichages : 377
Taille : 96,4 Ko
    et taper le nom dans le petit champ de texte qui s'affiche sous la zone de code.

    -------


    Cela dit, dans un premier temps vous devriez essayer de jouer avec tout le code dans le même sketch... c'est quand même plus simple

  3. #3
    Membre du Club
    Homme Profil pro
    retraité
    Inscrit en
    Décembre 2019
    Messages
    122
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Yvelines (Île de France)

    Informations professionnelles :
    Activité : retraité

    Informations forums :
    Inscription : Décembre 2019
    Messages : 122
    Points : 60
    Points
    60
    Par défaut
    Bonjour à tous

    Citation Envoyé par Jay M Voir le message
    ... il faut comprendre un peu ce que fait l'IDE lors de la demande de compilation pour simplifier la vie ...
    Ça a bien l'air de ressembler à ça, mais je suis encore à me poser une question, car il y a un hiatus pour l'instant inexplicable.

    J'ai recensé tous mes RTClib.h dispersés sur mon disque C:, suite à des installations ou des copy. Ceci pour copier mon dossier contenant le .ino, sur une partition B: de mon disque dur, pour pouvoir essayer de compiler sans le disque dur externe.

    J'avais trouvé le seul .h qui contenait ma déclaration de variable nouvelle, ajoutée dans mon RTClib.h qui contenait ma déclaration:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    // ajout dans RTClib.h du contenu de mes déclarations globales et initiales 19032021
    // supposé déclarations globales en lib
     
    unsigned int Minute_de_semaine ; // RTC converti en minute actuelle
    Par élimination, j'avais renommé les trois autres .h qui les rendait inutilisables.

    Après transfert du .ino sur B:, la compilation bloque par RTClib.h not found.

    En rétablissant les renommés un par un, ça compile bien avec le

    C:\Users\Guy\Documents\Arduino\libraries\RTClib-1-12-5\RTClib.h

    qui ne contient pas mon ajout de déclaration. Il contient

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    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
    311
    312
    313
    314
    315
    316
    317
    318
    319
    320
    321
    322
    323
    324
    325
    326
    327
    328
    329
    330
    331
    332
    333
    334
    335
    336
    337
    338
    339
    340
    341
    342
    343
    344
    345
    346
    347
    348
    349
    350
    351
    352
    353
    354
    355
    356
    357
    358
    359
    360
    361
    362
    363
    364
    365
    366
    367
    368
    369
    370
    371
    372
    373
    374
    375
    376
    377
    378
    379
    380
    381
    382
    383
    384
    385
    386
    387
    388
    389
    390
    391
    392
    393
    394
    395
    396
    397
    398
    399
    400
    401
    402
    403
    404
    405
    406
    407
    408
    409
    410
    411
    412
    413
    414
    415
    416
    417
    418
    419
    420
    421
    422
    423
    424
    425
    426
    427
    428
    429
    430
    431
    432
    433
    434
    435
    436
    437
    438
    439
    440
    441
    442
    443
    444
    445
    446
    447
    448
    449
    450
    451
    452
    453
    454
    455
    456
    457
    458
    459
    460
    461
    462
    463
    464
    465
    466
    467
    468
    469
    470
    471
    472
    473
    474
    475
    476
    477
    478
    479
    480
    481
    482
    483
    484
    485
    486
    487
    488
    489
    490
    491
    492
    493
    /**************************************************************************/
    /*!
      @file     RTClib.h
     
      Original library by JeeLabs http://news.jeelabs.org/code/, released to the
      public domain
     
      License: MIT (see LICENSE)
     
      This is a fork of JeeLab's fantastic real time clock library for Arduino.
     
      For details on using this library with an RTC module like the DS1307, PCF8523,
      or DS3231, see the guide at:
      https://learn.adafruit.com/ds1307-real-time-clock-breakout-board-kit/overview
     
      Adafruit invests time and resources providing this open source code,
      please support Adafruit and open-source hardware by purchasing
      products from Adafruit!
    */
    /**************************************************************************/
     
    #ifndef _RTCLIB_H_
    #define _RTCLIB_H_
     
    #include <Arduino.h>
    class TimeSpan;
     
    /** Registers */
    #define PCF8523_ADDRESS 0x68       ///< I2C address for PCF8523
    #define PCF8523_CLKOUTCONTROL 0x0F ///< Timer and CLKOUT control register
    #define PCF8523_CONTROL_1 0x00     ///< Control and status register 1
    #define PCF8523_CONTROL_2 0x01     ///< Control and status register 2
    #define PCF8523_CONTROL_3 0x02     ///< Control and status register 3
    #define PCF8523_TIMER_B_FRCTL 0x12 ///< Timer B source clock frequency control
    #define PCF8523_TIMER_B_VALUE 0x13 ///< Timer B value (number clock periods)
    #define PCF8523_OFFSET 0x0E        ///< Offset register
    #define PCF8523_STATUSREG 0x03     ///< Status register
     
    #define PCF8563_ADDRESS 0x51       ///< I2C address for PCF8563
    #define PCF8563_CLKOUTCONTROL 0x0D ///< CLKOUT control register
    #define PCF8563_CONTROL_1 0x00     ///< Control and status register 1
    #define PCF8563_CONTROL_2 0x01     ///< Control and status register 2
    #define PCF8563_VL_SECONDS 0x02    ///< register address for VL_SECONDS
    #define PCF8563_CLKOUT_MASK 0x83   ///< bitmask for SqwPinMode on CLKOUT pin
     
    #define DS1307_ADDRESS 0x68 ///< I2C address for DS1307
    #define DS1307_CONTROL 0x07 ///< Control register
    #define DS1307_NVRAM 0x08   ///< Start of RAM registers - 56 bytes, 0x08 to 0x3f
     
    #define DS3231_ADDRESS 0x68   ///< I2C address for DS3231
    #define DS3231_TIME 0x00      ///< Time register
    #define DS3231_ALARM1 0x07    ///< Alarm 1 register
    #define DS3231_ALARM2 0x0B    ///< Alarm 2 register
    #define DS3231_CONTROL 0x0E   ///< Control register
    #define DS3231_STATUSREG 0x0F ///< Status register
    #define DS3231_TEMPERATUREREG                                                  \
      0x11 ///< Temperature register (high byte - low byte is at 0x12), 10-bit
           ///< temperature value
     
    /** Constants */
    #define SECONDS_PER_DAY 86400L ///< 60 * 60 * 24
    #define SECONDS_FROM_1970_TO_2000                                              \
      946684800 ///< Unixtime for 2000-01-01 00:00:00, useful for initialization
     
    /**************************************************************************/
    /*!
        @brief  Simple general-purpose date/time class (no TZ / DST / leap
                seconds).
     
        This class stores date and time information in a broken-down form, as a
        tuple (year, month, day, hour, minute, second). The day of the week is
        not stored, but computed on request. The class has no notion of time
        zones, daylight saving time, or
        [leap seconds](http://en.wikipedia.org/wiki/Leap_second): time is stored
        in whatever time zone the user chooses to use.
     
        The class supports dates in the range from 1 Jan 2000 to 31 Dec 2099
        inclusive.
    */
    /**************************************************************************/
    class DateTime {
    public:
      DateTime(uint32_t t = SECONDS_FROM_1970_TO_2000);
      DateTime(uint16_t year, uint8_t month, uint8_t day, uint8_t hour = 0,
               uint8_t min = 0, uint8_t sec = 0);
      DateTime(const DateTime &copy);
      DateTime(const char *date, const char *time);
      DateTime(const __FlashStringHelper *date, const __FlashStringHelper *time);
      DateTime(const char *iso8601date);
      bool isValid() const;
      char *toString(char *buffer);
     
      /*!
          @brief  Return the year.
          @return Year (range: 2000--2099).
      */
      uint16_t year() const { return 2000U + yOff; }
      /*!
          @brief  Return the month.
          @return Month number (1--12).
      */
      uint8_t month() const { return m; }
      /*!
          @brief  Return the day of the month.
          @return Day of the month (1--31).
      */
      uint8_t day() const { return d; }
      /*!
          @brief  Return the hour
          @return Hour (0--23).
      */
      uint8_t hour() const { return hh; }
     
      uint8_t twelveHour() const;
      /*!
          @brief  Return whether the time is PM.
          @return 0 if the time is AM, 1 if it's PM.
      */
      uint8_t isPM() const { return hh >= 12; }
      /*!
          @brief  Return the minute.
          @return Minute (0--59).
      */
      uint8_t minute() const { return mm; }
      /*!
          @brief  Return the second.
          @return Second (0--59).
      */
      uint8_t second() const { return ss; }
     
      uint8_t dayOfTheWeek() const;
     
      /* 32-bit times as seconds since 2000-01-01. */
      uint32_t secondstime() const;
     
      /* 32-bit times as seconds since 1970-01-01. */
      uint32_t unixtime(void) const;
     
      /*!
          Format of the ISO 8601 timestamp generated by `timestamp()`. Each
          option corresponds to a `toString()` format as follows:
      */
      enum timestampOpt {
        TIMESTAMP_FULL, //!< `YYYY-MM-DDThh:mm:ss`
        TIMESTAMP_TIME, //!< `hh:mm:ss`
        TIMESTAMP_DATE  //!< `YYYY-MM-DD`
      };
      String timestamp(timestampOpt opt = TIMESTAMP_FULL);
     
      DateTime operator+(const TimeSpan &span);
      DateTime operator-(const TimeSpan &span);
      TimeSpan operator-(const DateTime &right);
      bool operator<(const DateTime &right) const;
     
      /*!
          @brief  Test if one DateTime is greater (later) than another.
          @warning if one or both DateTime objects are invalid, returned value is
            meaningless
          @see use `isValid()` method to check if DateTime object is valid
          @param right DateTime object to compare
          @return True if the left DateTime is later than the right one,
            false otherwise
      */
      bool operator>(const DateTime &right) const { return right < *this; }
     
      /*!
          @brief  Test if one DateTime is less (earlier) than or equal to another
          @warning if one or both DateTime objects are invalid, returned value is
            meaningless
          @see use `isValid()` method to check if DateTime object is valid
          @param right DateTime object to compare
          @return True if the left DateTime is earlier than or equal to the
            right one, false otherwise
      */
      bool operator<=(const DateTime &right) const { return !(*this > right); }
     
      /*!
          @brief  Test if one DateTime is greater (later) than or equal to another
          @warning if one or both DateTime objects are invalid, returned value is
            meaningless
          @see use `isValid()` method to check if DateTime object is valid
          @param right DateTime object to compare
          @return True if the left DateTime is later than or equal to the right
            one, false otherwise
      */
      bool operator>=(const DateTime &right) const { return !(*this < right); }
      bool operator==(const DateTime &right) const;
     
      /*!
          @brief  Test if two DateTime objects are not equal.
          @warning if one or both DateTime objects are invalid, returned value is
            meaningless
          @see use `isValid()` method to check if DateTime object is valid
          @param right DateTime object to compare
          @return True if the two objects are not equal, false if they are
      */
      bool operator!=(const DateTime &right) const { return !(*this == right); }
     
    protected:
      uint8_t yOff; ///< Year offset from 2000
      uint8_t m;    ///< Month 1-12
      uint8_t d;    ///< Day 1-31
      uint8_t hh;   ///< Hours 0-23
      uint8_t mm;   ///< Minutes 0-59
      uint8_t ss;   ///< Seconds 0-59
    };
     
    /**************************************************************************/
    /*!
        @brief  Timespan which can represent changes in time with seconds accuracy.
    */
    /**************************************************************************/
    class TimeSpan {
    public:
      TimeSpan(int32_t seconds = 0);
      TimeSpan(int16_t days, int8_t hours, int8_t minutes, int8_t seconds);
      TimeSpan(const TimeSpan &copy);
     
      /*!
          @brief  Number of days in the TimeSpan
                  e.g. 4
          @return int16_t days
      */
      int16_t days() const { return _seconds / 86400L; }
      /*!
          @brief  Number of hours in the TimeSpan
                  This is not the total hours, it includes the days
                  e.g. 4 days, 3 hours - NOT 99 hours
          @return int8_t hours
      */
      int8_t hours() const { return _seconds / 3600 % 24; }
      /*!
          @brief  Number of minutes in the TimeSpan
                  This is not the total minutes, it includes days/hours
                  e.g. 4 days, 3 hours, 27 minutes
          @return int8_t minutes
      */
      int8_t minutes() const { return _seconds / 60 % 60; }
      /*!
          @brief  Number of seconds in the TimeSpan
                  This is not the total seconds, it includes the days/hours/minutes
                  e.g. 4 days, 3 hours, 27 minutes, 7 seconds
          @return int8_t seconds
      */
      int8_t seconds() const { return _seconds % 60; }
      /*!
          @brief  Total number of seconds in the TimeSpan, e.g. 358027
          @return int32_t seconds
      */
      int32_t totalseconds() const { return _seconds; }
     
      TimeSpan operator+(const TimeSpan &right);
      TimeSpan operator-(const TimeSpan &right);
     
    protected:
      int32_t _seconds; ///< Actual TimeSpan value is stored as seconds
    };
     
    /** DS1307 SQW pin mode settings */
    enum Ds1307SqwPinMode {
      DS1307_OFF = 0x00,            // Low
      DS1307_ON = 0x80,             // High
      DS1307_SquareWave1HZ = 0x10,  // 1Hz square wave
      DS1307_SquareWave4kHz = 0x11, // 4kHz square wave
      DS1307_SquareWave8kHz = 0x12, // 8kHz square wave
      DS1307_SquareWave32kHz = 0x13 // 32kHz square wave
    };
     
    /**************************************************************************/
    /*!
        @brief  RTC based on the DS1307 chip connected via I2C and the Wire library
    */
    /**************************************************************************/
    class RTC_DS1307 {
    public:
      boolean begin(void);
      static void adjust(const DateTime &dt);
      uint8_t isrunning(void);
      static DateTime now();
      static Ds1307SqwPinMode readSqwPinMode();
      static void writeSqwPinMode(Ds1307SqwPinMode mode);
      uint8_t readnvram(uint8_t address);
      void readnvram(uint8_t *buf, uint8_t size, uint8_t address);
      void writenvram(uint8_t address, uint8_t data);
      void writenvram(uint8_t address, uint8_t *buf, uint8_t size);
    };
     
    /** DS3231 SQW pin mode settings */
    enum Ds3231SqwPinMode {
      DS3231_OFF = 0x1C,            /**< Off */
      DS3231_SquareWave1Hz = 0x00,  /**<  1Hz square wave */
      DS3231_SquareWave1kHz = 0x08, /**<  1kHz square wave */
      DS3231_SquareWave4kHz = 0x10, /**<  4kHz square wave */
      DS3231_SquareWave8kHz = 0x18  /**<  8kHz square wave */
    };
     
    /** DS3231 Alarm modes for alarm 1 */
    enum Ds3231Alarm1Mode {
      DS3231_A1_PerSecond = 0x0F, /**< Alarm once per second */
      DS3231_A1_Second = 0x0E,    /**< Alarm when seconds match */
      DS3231_A1_Minute = 0x0C,    /**< Alarm when minutes and seconds match */
      DS3231_A1_Hour = 0x08,      /**< Alarm when hours, minutes
                                       and seconds match */
      DS3231_A1_Date = 0x00,      /**< Alarm when date (day of month), hours,
                                       minutes and seconds match */
      DS3231_A1_Day = 0x10        /**< Alarm when day (day of week), hours,
                                       minutes and seconds match */
    };
    /** DS3231 Alarm modes for alarm 2 */
    enum Ds3231Alarm2Mode {
      DS3231_A2_PerMinute = 0x7, /**< Alarm once per minute
                                      (whenever seconds are 0) */
      DS3231_A2_Minute = 0x6,    /**< Alarm when minutes match */
      DS3231_A2_Hour = 0x4,      /**< Alarm when hours and minutes match */
      DS3231_A2_Date = 0x0,      /**< Alarm when date (day of month), hours
                                      and minutes match */
      DS3231_A2_Day = 0x8        /**< Alarm when day (day of week), hours
                                      and minutes match */
    };
     
    /**************************************************************************/
    /*!
        @brief  RTC based on the DS3231 chip connected via I2C and the Wire library
    */
    /**************************************************************************/
    class RTC_DS3231 {
    public:
      boolean begin(void);
      static void adjust(const DateTime &dt);
      bool lostPower(void);
      static DateTime now();
      static Ds3231SqwPinMode readSqwPinMode();
      static void writeSqwPinMode(Ds3231SqwPinMode mode);
      bool setAlarm1(const DateTime &dt, Ds3231Alarm1Mode alarm_mode);
      bool setAlarm2(const DateTime &dt, Ds3231Alarm2Mode alarm_mode);
      void disableAlarm(uint8_t alarm_num);
      void clearAlarm(uint8_t alarm_num);
      bool alarmFired(uint8_t alarm_num);
      void enable32K(void);
      void disable32K(void);
      bool isEnabled32K(void);
      static float getTemperature(); // in Celsius degree
    };
     
    /** PCF8523 INT/SQW pin mode settings */
    enum Pcf8523SqwPinMode {
      PCF8523_OFF = 7,             /**< Off */
      PCF8523_SquareWave1HZ = 6,   /**< 1Hz square wave */
      PCF8523_SquareWave32HZ = 5,  /**< 32Hz square wave */
      PCF8523_SquareWave1kHz = 4,  /**< 1kHz square wave */
      PCF8523_SquareWave4kHz = 3,  /**< 4kHz square wave */
      PCF8523_SquareWave8kHz = 2,  /**< 8kHz square wave */
      PCF8523_SquareWave16kHz = 1, /**< 16kHz square wave */
      PCF8523_SquareWave32kHz = 0  /**< 32kHz square wave */
    };
     
    /** PCF8523 Timer Source Clock Frequencies for Timers A and B */
    enum PCF8523TimerClockFreq {
      PCF8523_Frequency4kHz = 0,   /**< 1/4096th second = 244 microseconds,
                                        max 62.256 milliseconds */
      PCF8523_Frequency64Hz = 1,   /**< 1/64th second = 15.625 milliseconds,
                                        max 3.984375 seconds */
      PCF8523_FrequencySecond = 2, /**< 1 second, max 255 seconds = 4.25 minutes */
      PCF8523_FrequencyMinute = 3, /**< 1 minute, max 255 minutes = 4.25 hours */
      PCF8523_FrequencyHour = 4,   /**< 1 hour, max 255 hours = 10.625 days */
    };
     
    /** PCF8523 Timer Interrupt Low Pulse Width options for Timer B only */
    enum PCF8523TimerIntPulse {
      PCF8523_LowPulse3x64Hz = 0,  /**<  46.875 ms   3/64ths second */
      PCF8523_LowPulse4x64Hz = 1,  /**<  62.500 ms   4/64ths second */
      PCF8523_LowPulse5x64Hz = 2,  /**<  78.125 ms   5/64ths second */
      PCF8523_LowPulse6x64Hz = 3,  /**<  93.750 ms   6/64ths second */
      PCF8523_LowPulse8x64Hz = 4,  /**< 125.000 ms   8/64ths second */
      PCF8523_LowPulse10x64Hz = 5, /**< 156.250 ms  10/64ths second */
      PCF8523_LowPulse12x64Hz = 6, /**< 187.500 ms  12/64ths second */
      PCF8523_LowPulse14x64Hz = 7  /**< 218.750 ms  14/64ths second */
    };
     
    /** PCF8523 Offset modes for making temperature/aging/accuracy adjustments */
    enum Pcf8523OffsetMode {
      PCF8523_TwoHours = 0x00, /**< Offset made every two hours */
      PCF8523_OneMinute = 0x80 /**< Offset made every minute */
    };
     
    /**************************************************************************/
    /*!
        @brief  RTC based on the PCF8523 chip connected via I2C and the Wire library
    */
    /**************************************************************************/
    class RTC_PCF8523 {
    public:
      boolean begin(void);
      void adjust(const DateTime &dt);
      boolean lostPower(void);
      boolean initialized(void);
      static DateTime now();
      void start(void);
      void stop(void);
      uint8_t isrunning();
      Pcf8523SqwPinMode readSqwPinMode();
      void writeSqwPinMode(Pcf8523SqwPinMode mode);
      void enableSecondTimer(void);
      void disableSecondTimer(void);
      void enableCountdownTimer(PCF8523TimerClockFreq clkFreq, uint8_t numPeriods,
                                uint8_t lowPulseWidth);
      void enableCountdownTimer(PCF8523TimerClockFreq clkFreq, uint8_t numPeriods);
      void disableCountdownTimer(void);
      void deconfigureAllTimers(void);
      void calibrate(Pcf8523OffsetMode mode, int8_t offset);
    };
     
    /** PCF8563 CLKOUT pin mode settings */
    enum Pcf8563SqwPinMode {
      PCF8563_SquareWaveOFF = 0x00,  /**< Off */
      PCF8563_SquareWave1Hz = 0x83,  /**< 1Hz square wave */
      PCF8563_SquareWave32Hz = 0x82, /**< 32Hz square wave */
      PCF8563_SquareWave1kHz = 0x81, /**< 1kHz square wave */
      PCF8563_SquareWave32kHz = 0x80 /**< 32kHz square wave */
    };
     
    /**************************************************************************/
    /*!
        @brief  RTC based on the PCF8563 chip connected via I2C and the Wire library
    */
    /**************************************************************************/
     
    class RTC_PCF8563 {
    public:
      boolean begin(void);
      boolean lostPower(void);
      void adjust(const DateTime &dt);
      static DateTime now();
      void start(void);
      void stop(void);
      uint8_t isrunning();
      Pcf8563SqwPinMode readSqwPinMode();
      void writeSqwPinMode(Pcf8563SqwPinMode mode);
    };
     
    /**************************************************************************/
    /*!
        @brief  RTC using the internal millis() clock, has to be initialized before
       use. NOTE: this is immune to millis() rollover events.
    */
    /**************************************************************************/
    class RTC_Millis {
    public:
      /*!
          @brief  Start the RTC
          @param dt DateTime object with the date/time to set
      */
      static void begin(const DateTime &dt) { adjust(dt); }
      static void adjust(const DateTime &dt);
      static DateTime now();
     
    protected:
      static uint32_t lastUnix;   ///< Unix time from the previous call to now() -
                                  ///< prevents rollover issues
      static uint32_t lastMillis; ///< the millis() value corresponding to the last
                                  ///< **full second** of Unix time
    };
     
    /**************************************************************************/
    /*!
        @brief  RTC using the internal micros() clock, has to be initialized before
                use. Unlike RTC_Millis, this can be tuned in order to compensate for
                the natural drift of the system clock. Note that now() has to be
                called more frequently than the micros() rollover period, which is
                approximately 71.6 minutes.
    */
    /**************************************************************************/
    class RTC_Micros {
    public:
      /*!
          @brief  Start the RTC
          @param dt DateTime object with the date/time to set
      */
      static void begin(const DateTime &dt) { adjust(dt); }
      static void adjust(const DateTime &dt);
      static void adjustDrift(int ppm);
      static DateTime now();
     
    protected:
      static uint32_t microsPerSecond; ///< Number of microseconds reported by
                                       ///< micros() per "true" (calibrated) second
      static uint32_t lastUnix;   ///< Unix time from the previous call to now() -
                                  ///< prevents rollover issues
      static uint32_t lastMicros; ///< micros() value corresponding to the last full
                                  ///< second of Unix time
    };
     
    #endif // _RTCLIB_H_
    Il ne contient que 19kO alors que les autres en ont 241kO.

    Dans mon .ino, voilà ce que je recopie ici pour essayer de comprendre.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
     ...
     
     
    #include "RTClib.h"  // supposé être complété de déclarations
    #include "LiquidCrystal.h"  //librairie lcd
     
    LiquidCrystal lcd (11, 10, 5, 4, 3, 2);
     
     
    RTC_DS3231 rtc;
    // ... dont lecture de la RTC
     
     // calcul de Minute_de_semaine
     
      unsigned int Minute_de_semaine = (( now.dayOfTheWeek() * 1440) + ((now.hour()) * 60 ) + ( now.minute ())) ; // utilisation de ma variable ajoutée
    Et, sans déclencher d'erreur, mon affichage me donne bien la bonne valeur. Cette grandeur représente "l'heure" de la semaine, exprimée en minutes, qui va servir d'adresse pour pointer les 10080 octets contenant les 10080 minutes où chaque sortie devra être haute (chauffe) ou basse (éteinte).

    Me serais-je trompé, tant que je n'avais pas ajouté ma déclaration dans le .h?

    Mon utilisation sans déclaration préalable faite dans mon .ino est-elle suffisante? (et donc la déclaration initiale inutile)?

  4. #4
    Membre du Club
    Homme Profil pro
    retraité
    Inscrit en
    Décembre 2019
    Messages
    122
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Yvelines (Île de France)

    Informations professionnelles :
    Activité : retraité

    Informations forums :
    Inscription : Décembre 2019
    Messages : 122
    Points : 60
    Points
    60
    Par défaut
    Je me réponds partiellement sur un des points après test systématique.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    //unsigned int Minute_de_semaine = (( now.dayOfTheWeek() * 1440) + ((now.hour()) * 60 ) + ( now.minute ())) ;
     // rappel de l'expression pour rétablissement si modifs infructueuses
    1- en supprimant int ça compile et ça marche.

    2- en supprimant unsigned et int ça déclenche le défaut de non déclaration.

    3- en ne mettant que int ça marche et ça compile.

    La simple mention de int, de unsigned ou des deux suffit-elle pour jouer le rôle de déclaration?

    Il me semble que lors de mes essais préliminaires, je n'avais pas utilisé ni mentionné les l’attribut de ma variable, et donc, la déclaration était obligatoire.

    Mais alors, puisque le .h où cette déclaration se trouvait semblant être erronée ...

  5. #5
    Expert confirmé

    Homme Profil pro
    mad scientist :)
    Inscrit en
    Septembre 2019
    Messages
    2 816
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : Etats-Unis

    Informations professionnelles :
    Activité : mad scientist :)

    Informations forums :
    Inscription : Septembre 2019
    Messages : 2 816
    Points : 5 673
    Points
    5 673
    Par défaut
    // ajout dans RTClib.h du contenu de mes déclarations globales et initiales 19032021
    Pourquoi modifiez vous la bibliothèque?

    Ce n'est pas très pertinent, l'idée c'est vraiment de les laisser comme elles sont (sinon à la prochaine mise à jour vous allez perdre vos changements, ou si vous avez besoin de cette bibliothèque dans d'autres projets vous allez trainer des variables en trop).

    il suffit de définir et calculer Minute_de_semaine dans votre sketch et conserver la bibliothèque intacte... (la convention Arduino est le camelCase, donc ce serait plutôt minuteDeSemaine)

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    #include <RTClib.h>
    RTC_DS3231 rtc;
    
    void setup() {
      Serial.begin(115200);
      rtc.begin();
    }
    
    void loop() {
      DateTime maintenant = rtc.now();
      unsigned int minuteDeSemaine = 1440u * maintenant.dayOfTheWeek() + 60u * maintenant.hour() + maintenant.minute() ;
      ...
    }
    J'ai forcé (avec le u) les constantes numériques en unsigned pour que le calcul s'effectue bien en entier non signé


    qui va servir d'adresse pour pointer les 10080 octets contenant les 10080 minutes où chaque sortie devra être haute (chauffe) ou basse (éteinte).
    rappelez moi quel Arduino vous avez choisi ? un UNO n'a que 2 K octets de RAM et un MEGA n'en a que 8... vous ne pourrez pas déclarer un tableau de quasiment 10k octets...(si vous utilisez qu'un seul octet pour associer les relais)

    Vous ne pouvez pas prendre cette approche sur un petit micro-contrôleur.

    il faut représenter en mémoire que ce qui est utile. par exemple une structure qui définit un intervalle de temps par la minute de début et la minute de fin (au niveau de la semaine) et ensuite vous déclarez (le plus simple c'est statiquement) un nombre max d'intervalles associés à une broche pour définir un programme.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    const uint8_t pinRelais[] = {2, 3, 4, 5, 6, 7, 8, 9};
    const uint8_t nombreDeRelais = sizeof pinRelais / sizeof pinRelais[0];
    const uint8_t maxIntervalle = 21; // au maximum 21 intervalles dans un programme pour une broche (permet 3 par jour)
     
    struct t_intervalle {
      uint16_t minuteSemaineDebut;
      uint16_t minuteSemaineFin;
    } programme[nombreDeRelais][maxIntervalle];
    si vous avez 8 broches pilotables et 21 intervalles par broche (chacun de 4 octets) ça fait 8*21*4=672 octets occupés

    là ça tient raisonnablement sur un UNO et on est bien sur un MEGA

    voici un exemple de code qui montre comment parcourir les tableaux
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    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
    #include <RTClib.h>
    RTC_DS3231 rtc;
     
    const uint8_t pinRelais[] = {2, 3, 4, 5, 6, 7, 8, 9};
    const uint8_t nombreDeRelais = sizeof pinRelais / sizeof pinRelais[0];
    const uint8_t maxIntervalle = 21; // au maximum 21 intervalles dans un programme pour une broche (3 par jour par exemple)
     
    struct t_intervalle {
      uint16_t minuteSemaineDebut;
      uint16_t minuteSemaineFin;
    } programme[nombreDeRelais][maxIntervalle];
     
    inline uint16_t minuteSemaine(uint8_t j, uint8_t h, uint8_t m) {
      return 1440u * j + 60u * h + m;
    }
     
    void setup() {
      Serial.begin(115200);
      rtc.begin();
     
     
      for (uint8_t relais = 0; relais < nombreDeRelais; relais++) {
        // on met la pin du relai en sortie et on la désactive
        pinMode(pinRelais[relais], OUTPUT);
        digitalWrite(pinRelais[relais], LOW); // au repos
     
        // on invalide tous les programmes
        for (uint8_t intervalle = 0; intervalle < maxIntervalle; intervalle++) {
          programme[relais][intervalle].minuteSemaineDebut = 0xFFFF; // inactif
          programme[relais][intervalle].minuteSemaineFin = 0xFFFF;   // inactif
        }
      }
     
      // on crée deux programmes pour le premier relais (0)
      programme[0][0].minuteSemaineDebut  = minuteSemaine(0, 5, 0);  // Lundi (jour 0) à 5h00
      programme[0][0].minuteSemaineFin    = minuteSemaine(0, 7, 30); // Lundi (jour 0) à 7h30
     
      programme[0][1].minuteSemaineDebut  = minuteSemaine(1, 11, 0); // Mardi (jour 1) à 11h00
      programme[0][1].minuteSemaineFin    = minuteSemaine(1, 14, 30); // Mardi (jour 1) à 114h30
     
      // on crée un programme pour le deuxième relais (1)
      programme[1][0].minuteSemaineDebut  = minuteSemaine(3, 9, 0);  // Mercredi (jour 3) à 9h00
      programme[1][0].minuteSemaineFin    = minuteSemaine(3, 19, 15); // Mercredi (jour 3) à 19h15
     
    }
     
    void loop() {
      DateTime maintenant = rtc.now();
      uint16_t moment = minuteSemaine(maintenant.dayOfTheWeek(), maintenant.hour(), maintenant.minute()) ;
     
      for (uint8_t relais = 0; relais < nombreDeRelais; relais++) {
        Serial.print(F("Relais #")); Serial.println(relais);
        for (uint8_t intervalle = 0; intervalle < maxIntervalle; intervalle++) {
          if (programme[relais][intervalle].minuteSemaineDebut != 0xFFFF) { // si le programme est actif
            Serial.print(F("\tIntervalle #")); Serial.print(intervalle);
            if (moment < programme[relais][intervalle].minuteSemaineDebut) {
              Serial.println(F(" -> déclenchement futur"));
            } else if (moment > programme[relais][intervalle].minuteSemaineFin) {
              Serial.println(F(" -> déclenchement terminé"));
            } else {
              Serial.println(F(" -> en cours"));
            }
          } // end if programme actif
        } // end for intervalle
      } // end for relais
     
      while (Serial.read() != '\n'); // on attend de recevoir un passage à la ligne pour recommencer
    }
    bien sûr les programmes ne seront pas codés en dur dans le setup, l'interface utilisateur permettra de les éditer et on pourra sauver en EEPROM (par exemple) les programmes de façon à survivre un reboot

    Ce n'est qu'un exemple - On pourrait rajouter une indirection de plus pour qu'un programme soit indépendant d'une broche donnée de façon à pouvoir associer un programme à plusieurs broches sans avoir à répéter

  6. #6
    Membre du Club
    Homme Profil pro
    retraité
    Inscrit en
    Décembre 2019
    Messages
    122
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Yvelines (Île de France)

    Informations professionnelles :
    Activité : retraité

    Informations forums :
    Inscription : Décembre 2019
    Messages : 122
    Points : 60
    Points
    60
    Par défaut
    Merci pour les commentaires et surtout les idées qui ne peuvent qu'être bonnes à prendre.

    Citation Envoyé par Jay M Voir le message
    Pourquoi modifiez vous la bibliothèque?

    Ce n'est pas très pertinent, l'idée c'est vraiment de les laisser comme elles sont (sinon à la prochaine mise à jour vous allez perdre vos changements, ou si vous avez besoin de cette bibliothèque dans d'autres projets vous allez trainer des variables en trop) ...
    C'est bien mon avis, mais la "tentative" que j'avais faite venait du fait que j'avais échoué (je ne sais pas encore vraiment pourquoi) et j'avais, pour une raison tout aussi inexpliquée, à passer le cap en modifiant le .h

    Depuis, la situation s'est éclaircie, puisque cette modification n'existe plus et et je veillerai à ce qu'il en soit toujours ainsi.

    Citation Envoyé par Jay M Voir le message
    ...il suffit de définir et calculer Minute_de_semaine dans votre sketch et conserver la bibliothèque intacte... (la convention Arduino est le camelCase, donc ce serait plutôt minuteDeSemaine)

    J'ai forcé (avec le u) les constantes numériques en unsigned pour que le calcul s'effectue bien en entier non signé ...
    C'est à présent le cas. Pour la convention camelCase, que je ne connais pas, je tâcherai de la suivre. Je suis habitué à utiliser les underscores dans le noms de fichiers, dont les noms à rallonge permettent de bien distinguer les sauvegardes successives.

    Comme c'est ma première variable, je promets de ne plus recommencer.

    Citation Envoyé par Jay M Voir le message
    ... rappelez moi quel Arduino vous avez choisi? ...
    Là, je suis surtout en phase d'apprentissage, et je commence avec UNO, qui, avec un peu de chance, pourrait remplir mon cahier des charges. Je commence à le sentir.


    Citation Envoyé par Jay M Voir le message
    ... un UNO n'a que 2 K octets de RAM et un MEGA n'en a que 8... vous ne pourrez pas déclarer un tableau de quasiment 10k octets...(si vous utilisez qu'un seul octet pour associer les relais) ...
    Mon idée, c'est de stocker le programme, celui qui contient les 10080 octets pour la semaine, dans une EEPROM car elle doit être non volatile. Strictement, je n'ai maintenant plus besoin que de quatre lignes, depuis que la maison est munie d'une pompe à chaleur. Seules certaines pièces (SB WC, cuisine, pour rester autre SB sont programmables en plus de Noël soit quatre lignes. Côté mémoire, pour rester plus cohérent, je n'utiliserai que les quatre bits nécessaires.

    Le nombre de phases d'allumage et d'extinctions est "illimité". A la limite, une voie pourrait être allumée toutes les minutes paires et éteinte toutes les minutes impaires. C'est la présence des bits à 1 ou 0 dans la mémoire du programme, qui se chargent de sortir la valeur correspondante.

    Mon programme arduino comportera le moyen, à partir d'une EEPROM resettée de programmer indépendamment pour chaque ligne, les périodes d'allumage et d'extinction. Une telle commande sera prévue pour fixer

    {jour;heure;minute=allumage;jour;heure;minute=extinction} -> validation/enregistrement.

    La variable minuteDeSemaine qui représente l'heure à la minute près à tout moment, permet d'adresser la mémoire et de savoir quel doit être l'état des lignes de sortie.

    Un calcul similaire les date/heure d'allumage et date/heure d'extinction permettra d'aller mettre des 1 dans toutes les minutes intervalles.

    Citation Envoyé par Jay M Voir le message
    ... si vous avez 8 broches pilotables et 21 intervalles par broche (chacun de 4 octets) ça fait 8*21*4=672 octets occupés

    là ça tient raisonnablement sur un UNO et on est bien sur un MEGA ...
    Mon approche est très différente et c'est ce que j'avais sur la version PC.

    Si je dois économiser sur la mémoire, je préfèrerai diminuer la résolution. Par exemple, la passer à 2minutes, ce qui divise par 2 la capacité.

    Citation Envoyé par Jay M Voir le message
    ... bien sûr les programmes ne seront pas codés en dur dans le setup, l'interface utilisateur permettra de les éditer et on pourra sauver en EEPROM (par exemple) les programmes de façon à survivre un reboot ...
    Justement, tel que je vois la chose, le reboot se recale tout seul et sort ce qui est programmé.

    La seule perte pourrait être celle des commandes manuelles, que je voudrais bien mettre en RAM sauvegardée. Le départ d'une commande manuelle part pour une durée X et se décrémente jusqu'à zéro à chaque minute écoulée. Tant que cette valeur n'est pas nulle, la voie chauffe et une perte d'alimentation repart des dernières valeurs, même au reboot.

    Il me semblait avoir lu quelque part que la RTC disposait d'une toute petite mémoire RAM sauvegardée, mais mon analyse de ma RTC ne me confirme pas cette possibilité.

    Est-ce exact, et/ou y a-t-il une RTC qui le permet? S'il me faut changer le modèle de la RTC, ce ne devrait pas être un problème.

    Ma prochaine étape importante va être l'entrée des données de contrôle. IL y a le clavier (en fait 5 poussoirs à commande exclusive) pour gérer les affichages programmations (montée, descente, gauche droite et validation) et les quatre modes de fonctionnement de l'automate (normal, enregistrement du programme, lecture du programme, programmation des commandes manuelles.

    Citation Envoyé par Jay M Voir le message
    ... voici un exemple de code qui montre comment parcourir les tableaux ...
    Je pense que je m'en inspirerai. Merci encore pour les aides qui s'avèrent très constructives.

Discussions similaires

  1. Réponses: 2
    Dernier message: 03/12/2018, 18h13
  2. Installer un environnement de dev pour GTK sur Windows
    Par Carambouille dans le forum GTK+ avec C & C++
    Réponses: 15
    Dernier message: 04/11/2017, 11h44
  3. Réponses: 1
    Dernier message: 01/06/2009, 17h43
  4. [ADO.Net][C#]Comment se gère les bases de données ?
    Par Sion_Sempai dans le forum Accès aux données
    Réponses: 7
    Dernier message: 28/03/2006, 16h30
  5. [ADO.NET][2.0] Comment se gère les pools de connection ?
    Par brousaille dans le forum Accès aux données
    Réponses: 8
    Dernier message: 04/03/2006, 19h11

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo