DS3231 a nefungující přerušení s ATTiny84

Diskuze a poradna o programátorech a programování různých obvodů

Moderátor: Moderátoři

Odpovědět
Zpráva
Autor
Uživatelský avatar
Darthy
Příspěvky: 187
Registrován: 15 led 2008, 01:00
Bydliště: Olomouc

DS3231 a nefungující přerušení s ATTiny84

#1 Příspěvek od Darthy »

Bastlím zrovna s modulem DS3231 a ATTiny84. Pro komunikaci používám knihovnu i2cmaster. V podstatě modul běží jak má, až na alarm, ten chci aby spustil přerušení každou minutu (případně každou sekundu). Přerušení se ale nevyvolá už ani na úrovni modulu. Pokud mám nastavený SQW výstup na 1Hz (RS0 a RS1 bity na 0), tak vše funguje a na výstupu dostávám 1Hz signál a přerušení ATTiny beží. Pochopil jsem nastavení registrů nějak špatně? Dle datasheetu po zapsaní bytu se adresa inkrementuje, takže v podstatě zapisuji jen hodnoty po určení adresy prvního registru. Pro kontrolu jsem měl na osciloskopu výstup z DS3231 na SQW pinu, ale nic.

Kód: Vybrat vše

#include <avr/io.h>
extern "C" {
  #include <i2cmaster.h>
}

#define DS3231_ReadMode_U8          (0xD1)
#define DS3231_WriteMode_U8         (0xD0)
#define DS3231_REG_ALARM_1          (0x07)
#define DS3231_REG_ALARM_2          (0x0B)
#define DS3231_REG_CONTROL          (0x0E)

#define LED PA5

int main(){
  
  DDRA |= (1 << LED);

  init_ds3231();
  
  ds3231_SetDateTime(__TIME__);
  ds3231_SendMinuteInterrupt();

  init_interrupt();
  sei();

  while(1){
  }    
}

void init_interrupt(){
    MCUCR |= (1 << ISC01) | (1<<ISC00);   // Nastavi interrupt na náběžnou hranu INT0 pinu
    GIMSK |= (1 << INT0);                           // Zapne externí přerušení na INT0 pinu  
}

ISR(EXT_INT0_vect){
   PORTA ^= (1 << LED);
}

void init_ds3231()
{
    i2c_init();                             
    i2c_start(DS3231_WriteMode_U8);          
    i2c_write(DS3231_REG_CONTROL);         
    i2c_write(0x06);                      // Nastaví bity INTCN a A2IE
    i2c_stop();                             
}
  
void ds3231_SendMinuteInterrupt(){
  i2c_start(DS3231_WriteMode_U8);
  i2c_write(DS3231_REG_ALARM_2);  // Registr 0Bh
  i2c_write(0x80);                // Nastaví A2M2 bit registru 0Bh
  i2c_write(0x80);                // Nastaví A2M3 bit registru 0Ch
  i2c_write(0x80);                // Nastaví A2M4 bit registru 0Dh
  i2c_stop();
}

void ds3231_SendSecondsInterrupt(){
  i2c_start(DS3231_WriteMode_U8);
  i2c_write(DS3231_REG_ALARM_1);  // Registr 07h
  i2c_write(0x80);                // Nastaví A1M1 bit registru 07h 
  i2c_write(0x80);                // Nastaví A1M2 bit registru 08h
  i2c_write(0x80);                // Nastaví A1M3 bit registru 09h
  i2c_write(0x80);                // Nastaví A1M4 bit registru 0Ah
  i2c_stop();
}
void ds3231_SetDateTime()
void ds3231_GetDateTime(){...}
byte dec2bcd(char p_num1, char p_num2){...}
uint8_t bcd2dec(byte p_num, byte p_decimal){...}
Zkoušel jsem i posílat hodnotu jednotlivě do registrů, ale beze změny.

Kód: Vybrat vše

void ds3231_SendMinuteInterrupt(){
  i2c_start(DS3231_WriteMode_U8);
  i2c_write(0x0B); 
  i2c_write(0x80);               
  i2c_stop();
  
  i2c_start(DS3231_WriteMode_U8);
  i2c_write(0x0C);  
  i2c_write(0x80);             
  i2c_stop();
  
  i2c_start(DS3231_WriteMode_U8);
  i2c_write(0x0D);  
  i2c_write(0x80);                
  i2c_stop();
}

Uživatelský avatar
bdn
Příspěvky: 436
Registrován: 16 led 2020, 01:00

#2 Příspěvek od bdn »

Jestli máš Arduino, zkus stáhnout knihovnu k DS3231 a vyzkoušet příklad na nastavení alarmu na tvém harware DS3231. Pokud to pojede, tak od toho se lze odpíchnout a dohledat, co a jak nastavit...
https://www.arduino.cc/reference/en/libraries/ds3231/

Uživatelský avatar
lesana87
Příspěvky: 3296
Registrován: 20 zář 2014, 02:00

#3 Příspěvek od lesana87 »

Tuším problém v přerušení nastaveném na náběžnou hranu, ale při alarmu je generovaná sestupná hrana. A protože neshodíš bity A1F/A2F ve Status registru, tak se náběžná hrana nikdy neobjeví.

Uživatelský avatar
Darthy
Příspěvky: 187
Registrován: 15 led 2008, 01:00
Bydliště: Olomouc

#4 Příspěvek od Darthy »

bdn píše:Jestli máš Arduino, zkus stáhnout knihovnu k DS3231 a vyzkoušet příklad na nastavení alarmu na tvém harware DS3231. Pokud to pojede, tak od toho se lze odpíchnout a dohledat, co a jak nastavit...
https://www.arduino.cc/reference/en/libraries/ds3231/
Nemám arduino, musím si vystačit jen s LEDkami případně "těžkým" seriovým výstupem, který né vždy funguje. Díval jsem se na nějaké knihovny pro arduino, ale nejsem z toho moc chytrý.
lesana87 píše:Tuším problém v přerušení nastaveném na náběžnou hranu, ale při alarmu je generovaná sestupná hrana. A protože neshodíš bity A1F/A2F ve Status registru, tak se náběžná hrana nikdy neobjeví.
Můžeš mi trošku přiblížit problematiku flag bitů prosím? Nejsem si jistý jestli ji dostatečně chápu. Měl jsem za to, že flag bity se automaticky nastavují, podle funkce a není třeba do nich jakkoliv zasahovat. Stejně tak pro časovače, nebo přerušení na ATTiny/ATmega flag bity tam jsou, ale vůbec s něma nepracuji a časovače i přerušení mi běžně fungují.

Pokusil jsem se po nastavení Control registru na alarm a samotného alarmu nastavit i 0 pro A1F a A2F. Ale bez úspěchu.

Kód: Vybrat vše

i2c_start(DS3231_WriteMode_U8);
  i2c_write(DS3231_REG_STATUS);
  i2c_write(0x88); 
  i2c_stop(); 

Uživatelský avatar
lesana87
Příspěvky: 3296
Registrován: 20 zář 2014, 02:00

#5 Příspěvek od lesana87 »

Musíš nastavit přerušení AVR na sestupnou hranu a až po vzniku přerušení shodit ty AxF bity v RTC, aby jsi umožnil další přerušení.

Uživatelský avatar
Darthy
Příspěvky: 187
Registrován: 15 led 2008, 01:00
Bydliště: Olomouc

#6 Příspěvek od Darthy »

To problém vyřešilo. Děkuji. Nějak jsem ale nenašel zmínku v datasheetu, že je potřeba flag bit vždy po přerušení resetovat.

Odpovědět

Zpět na „Programování PIC, ATMEL, EEPROM a dalších obvodů“