Doppio dimmer con attiny85 e triac

Se hai bisogno di una mano coi timer, fai un fischio. Se poi spieghi per bene cosa deve fare il tuo codice, magari possiamo risolvere con un unico timer.

leo, ho bisogno di una mano...
ho letto la guida che mi hai dato, ma purtroppo non ho ancora le competenze per comprenderla...

ecco il codice che uso:

volatile int i=0, y=0;               // variabili contatore...
volatile boolean zero_cross1=0, zero_cross2=0;  // booleane per lo zero crossing
int out1 = 0, out2 = 1;                // Output >> ai moc...
int pot1 = 3, pot2=4;             // ingressi trimmer
int bright1 = 0, bright2 = 0;   // livello di luminosità (0-128)  0 = on, 128 = 0ff 

int freqStep = 78;    // ritardo del timer in microsecondi
void setup() {
pinMode(out1, OUTPUT);
pinMode(out2, OUTPUT);
attachInterrupt(0, zero_cross_detect, RISING); //zero crossing detection con interrupt esterno 

}

void zero_cross_detect() {    
  zero_cross1 = true;
zero_cross2 = true;
i=0;
y=0;
  digitalWrite(out1, LOW);
digitalWrite(out2, LOW);
}                                 


void br_check() {              // SUB CHE DEVE ESSERE ESEGUITA OGNI "FREQSTEP [uS]" (78 microsecondi)    
  if(zero_cross1 == true) {              
    if(i>=bright1) {                     
      digitalWrite(out1, HIGH);       
      i=0;                         
      zero_cross1 = false;                
    } 
    else {
      i++;                           
    }                                
  }  
if(zero_cross2 == true) {              
    if(y>=bright2) {                     
      digitalWrite(out2, HIGH);       
      y=0;                         
      zero_cross2 = false;                
    } 
    else {
      y++;                           
    }                                
  }                                                                  
}                                   

void loop() {                        
  bright1 = analogRead(pot1) / 8;  
  bright2 = analogRead(pot2) / 8; 
}

come scritto nel codice, voglio che la sub br_check() venga eseguita indipendentemente dal loop, ogni 78uS.

saresti in grado di inviarmi un pezzo di codice che possa risolvere il mio problema usando un timer?
se possibile, potresti commentare lo sketch, così che io riesca a comprendere determinati passaggi, e se non chiedo troppo, anche una breve spiegazione sull'uso dei timer?

Grazie :wink:

Vediamo un po'...
Sui Tiny85 c'è libero il timer 0, che è ad 8 bit.
1 us, se non ho fatto male i conti, sono 0,000001 secondi.
Avere 78 uS dobbiamo avere una frequenza di overflow del timer di 1/0,000078=12820,50 volte al secondo.
1/12820,50=0,000078 us.
Ok sì è questo :wink:

Allora, poniamo il clock del Tiny a 8 MHz, per avere tale periodo bisogna impostare il tutto così:
(8000000*0,000078)/prescaler /8 = 78
Questo è il valore massimo che deve essere conteggiato dal timer per ottenere 1 overflow ogni 78 us, quindi avendo un contatore a 8 bit, con 256 valori, il valore da caricarci dentro è : 256-78=178
Rifacciamo i conti:
8000000/8/178=12820,5 Hz, perfetto :wink:

Prova con questo codice (richiamalo dal setup solo 1 volta):

cli();//fermo gli interrupt
TCCR0A &= ~((1<<COM0A0) | (1<<COM0A1)); //disattivi gli interrupt del timer
//modalità contatore fino all'overflow
TCCR0A &= ~((1<<WGM01) | (1<<WGM00));
TCCR0B &= ~(1<<WGM02);
//prescaler a /8
TCCR0B &= ~((1<<CS02) | (1<<CS00));
TCCR0B |= (1<<CS01);
TCNT0 = 178; //valore iniziale
TIMSK |= (1<<TOIE0); //attivo un interrutp all'overflow
sei(); //riattivo gli interrupt

Ora inserisci questa ISR per gestire l'overflow:

ISR (TIM0_OVF_vect) {
    TCNT0 = 178;
    ....... //qui il resto del codice
}

Dove metti il resto del tuo codice.

L'ho scritto senza poterlo provare, dimmi se va.

Ciao leo,

ho provato il tuo codice, ma durante la verifica mi da quest'errore:

core.a(wiring.c.o): In function __vector_5': C:\Documents and Settings\user\Desktop\arduino-1.0.1\hardware\arduino\cores\arduino/wiring.c:49: multiple definition of __vector_5'
dimmer.cpp.o:C:\DOCUME~1\user\IMPOST~1\Temp\build3703977071899918311.tmp/dimmer.cpp:243: first defined here

il codice è così adesso...

volatile int i=0, y=0;               // variabili contatore...
volatile boolean zero_cross1=0, zero_cross2=0;  // booleane per lo zero crossing
int out1 = 0, out2 = 1;                // Output >> ai moc...
int pot1 = 3, pot2=4;             // ingressi trimmer
int bright1 = 0, bright2 = 0;   // livello di luminosità (0-128)  0 = on, 128 = 0ff 

//int freqStep = 78;    // ritardo del timer in microsecondi
void setup() {
cli();//fermo gli interrupt
TCCR0A &= ~((1<<COM0A0) | (1<<COM0A1)); //disattivi gli interrupt del timer
//modalità contatore fino all'overflow
TCCR0A &= ~((1<<WGM01) | (1<<WGM00));
TCCR0B &= ~(1<<WGM02);
//prescaler a /8
TCCR0B &= ~((1<<CS02) | (1<<CS00));
TCCR0B |= (1<<CS01);
TCNT0 = 178; //valore iniziale
TIMSK |= (1<<TOIE0); //attivo un interrutp all'overflow
sei(); //riattivo gli interrupt
pinMode(out1, OUTPUT);
pinMode(out2, OUTPUT);
attachInterrupt(0, zero_cross_detect, RISING); //zero crossing detection con interrupt esterno 

}

void zero_cross_detect() {    
  zero_cross1 = true;
zero_cross2 = true;
i=0;
y=0;
  digitalWrite(out1, LOW);
digitalWrite(out2, LOW);
}                                 


ISR (TIM0_OVF_vect) {
    TCNT0 = 178;
  if(zero_cross1 == true) {              
    if(i>=bright1) {                     
      digitalWrite(out1, HIGH);       
      i=0;                         
      zero_cross1 = false;                
    } 
    else {
      i++;                           
    }                                
  }  
if(zero_cross2 == true) {              
    if(y>=bright2) {                     
      digitalWrite(out2, HIGH);       
      y=0;                         
      zero_cross2 = false;                
    } 
    else {
      y++;                           
    }                                
  }                                                                  
}                                   

void loop() {                        
  bright1 = analogRead(pot1) / 8;  
  bright2 = analogRead(pot2) / 8; 
}

EDIT: confermo che il chip è stato settato ad 8mhz con quarzo interno... ho tolto il ckdiv
EDIT2:
seguendo questa guida: http://www.atmel.com/Images/doc2505.pdf

a pagina 9 ho visto TIMER0_OVF0_vect, e l'ho modificato nel codice che mi hai dato... adesso non genera errori, ma provandolo, il loop è diventato velocissimo! e la funzione che dovrebbe farmi, non me la fa...

Grazie

Scusa se ti rispondo solo ora ma in questi giorni ho poco tempo libero ed inoltre ieri sera sono stato anche escluso dal forum per problemi di login.

  1. non mi torna il vettore che hai usato: TIM0_OVR_vect funziona benissimo, lo uso sia nella swRTC sia nel leOS. Perché a te dà errori? Ma stai usando il core Tiny oppure un altro core, per il Tiny85? Se usi il core Tiny, potresti controllare nel file /hardware/tiny/cores/tiny/core_build_options.h se intorno alla riga 112, nelle opzioni per il Tinyx5, la voce TIMER_TO_USE_FOR_MILLIS ha il valore 1?
  2. che intendi per "il loop è diventato velocissimo"? Come fai ad accorgertene? Hai compilato per un micro ad 8 MHz, dopo aver impostato il chip a questa frequenza?

@leo non so che dirti...

io ho solo visto quella stringa in quella guida che ho linkato, e in quel modo non genera errori.

mettendo un semplice blink nel loop, disabilitando la routine ISR... ecc ecc, il loop è moolto veloce, cioè un delay di 1000mS è diventato di neanche metà...

il core che uso l'ho preso da qui: http://hlt.media.mit.edu/?p=1695 e nella cartella "hardware" trovo la cartella "attiny" nella quale trovo "variants" e "boards.txt", in "variants" trovo "tiny8" e "tiny14" nelle quali c'è un file "pins_arduino.h".

potresti cortesemente passarmi il core che usi tu ?

il chip è stato impostato con internal rc osc ad 8mhz - startup 6ck/14ck + 64ms (quello di default), e settando i fuses tramite avrdude gui.

ho tolto il ckdiv8, facendo così diventare il low fuse da 0x62 a 0xE2

se ti serve sapere altro, sono a disposizione

Usa il core Tiny:
http://code.google.com/p/arduino-tiny/

  1. cancella il precedente core
  2. scarica ed installa il nuovo

il chip è stato impostato con internal rc osc ad 8mhz - startup 6ck/14ck + 64ms (quello di default), e settando i fuses tramite avrdude gui.

ho tolto il ckdiv8, facendo così diventare il low fuse da 0x62 a 0xE2

se ti serve sapere altro, sono a disposizione

Quindi tu hai effettivamente riscritto il nuovo fuse sul micro, giusto? Però il delay lo vedi accelerato? Allora vuol dire che stai compilando ad 1 MHz ed eseguendo ad 8 MHz. Riprova con il core ti ho passato, poi ne riparliamo.

ho provato col tuo core, ma ho lo stesso identico problema...

se metto il code normale non funziona nulla, se faccio il blink commentando la routine isr...ecc mi fa un lampeggio e poi stop... se tolgo anche la parte che va da cli() a sei(), il blink funziona perfettamente...

adesso non so proprio che fare...

aspetto tue risposte... lo so che ti sto facendo perdere un sacco di tempo, e mi scuso per questo...

solo, che voglio riuscire a farlo funzionare, ormai è diventata una questione di principio...

@leo CI SONO RIUSCITOOO!!!!

leo, ti do un bacione a distanza! :*

mi è bastato cambiare TIM0_OVF_vect in TIMER0_OVF_vect...

adesso, mi è però nato un problema hardware... a prescindere dai funzionamenti...

mi sono accorto che la resistenza da 47k 1/4W che ho messo dopo il ponte di diodi, che alimenta il 4n35, riscalda maledettamente!

basterà metterne una da 1/2W, oppure devo cambiare il valore?

non vorrei ricorrere alla Xc (reattanza capacitiva)...

mi sono accorto che la resistenza da 47k 1/4W che ho messo dopo il ponte di diodi, che alimenta il 4n35, riscalda maledettamente!
basterà metterne una da 1/2W, oppure devo cambiare il valore?
non vorrei ricorrere alla Xc (reattanza capacitiva)...

fai un semplice calcolo W=V*I

Il leOS io l'ho compilato, caricato ed usato su molti dei micro che supporta: Attiny85, Attiny84, Atmega168/328, Atmega644/1284, Atmega32U4. E non ho avuto nessun problema usando TIM0_OVF_vect. L'ultimo progetto in ordine di tempo prevede un Attiny85, che sta regolarmente funzionando su una breadboard di test. Perché a te non va? :cold_sweat:
Mi togli una curiosità? Ma che toolchain Avr stai usando? Quella dell'IDE o quella che hai sul sistema?
A proposito, che versione dell'IDE e che SO usi?

@brunello... una resistenza da 5w (perchè col calcolo mi esce circa 3w) è abbastanza grossa, e riscalderà comunque, a quel punto mi conviene il condensatore... grosso + o - nello stesso modo, e non riscalda come la resistenza...

@leo, non so perchè non mi va xD ma con quella modifica è apposto...

cos'è il toolchain avr?

sul pc con cui compilo, ho win xp pro sp2 (originale) (su un pc con i5 e 4gb di ram), ide versione 1.0.1, utilizzo come compiler arduino isp, anche se in realtà ho creato un piccolo clone dell'arduino serial e l'ho assemblato per uso di programmatore... uso un semplice usb > rs232 per la comunicazione.

La toolchain Avr è quel gruppo di programmi e librerie che servono per poter compilare i firmware per i microcontrollori Atmel. Se non sai cos'è ed hai Windows, stai sicuramente utilizzando la toolchain distribuita con l'IDE di Arduino.

Mi potresti fare un test? Potresti scaricare il leOS ed usare l'esempio BlinkWithoutMillis su un pin del microcontrollore e dirmi se:

  1. compilando ricevi errori
  2. una volta uploadato il programma, questo fa effettivamente lampeggiare il led

Te ne sarei grato per capire se è una cosa della toolchain che sto utilizzando io oppure no.

@leo, ho fatto le prove con attiny45 e attiny85 sia ad 1mhz sia ad 8mhz, ed in tutti e 4 i casi, funziona!

non so perchè a me non funzioni in quel caso, ma a me interessa aver risolto :wink:

Puoi pubblicare il codice che stai usando in questo momento?

eccolo:

volatile int i=0, y=0;               // variabili contatore...
volatile boolean zero_cross1=0, zero_cross2=0;  // booleane per lo zero crossing
int out1 = 0, out2 = 1;                // Output >> ai moc...
int pot1 = 3, pot2=4;             // ingressi trimmer
int bright1 = 0, bright2 = 0;   // livello di luminosità (0-128)  0 = on, 128 = 0ff 

//int freqStep = 78;    // ritardo del timer in microsecondi
void setup() {
cli();//fermo gli interrupt
TCCR0A &= ~((1<<COM0A0) | (1<<COM0A1)); //disattivi gli interrupt del timer
//modalità contatore fino all'overflow
TCCR0A &= ~((1<<WGM01) | (1<<WGM00));
TCCR0B &= ~(1<<WGM02);
//prescaler a /8
TCCR0B &= ~((1<<CS02) | (1<<CS00));
TCCR0B |= (1<<CS01);
TCNT0 = 178; //valore iniziale
TIMSK |= (1<<TOIE0); //attivo un interrutp all'overflow
sei(); //riattivo gli interrupt
pinMode(out1, OUTPUT);
pinMode(out2, OUTPUT);
attachInterrupt(0, zero_cross_detect, RISING); //zero crossing detection con interrupt esterno 

}

void zero_cross_detect() {    
  zero_cross1 = true;
zero_cross2 = true;
i=0;
y=0;
  digitalWrite(out1, LOW);
digitalWrite(out2, LOW);
}                                 


ISR (TIMER0_OVF_vect) {
    TCNT0 = 178;
  if(zero_cross1 == true) {              
    if(i>=bright1) {                     
      digitalWrite(out1, HIGH);       
      i=0;                         
      zero_cross1 = false;                
    } 
    else {
      i++;                           
    }                                
  }  
if(zero_cross2 == true) {              
    if(y>=bright2) {                     
      digitalWrite(out2, HIGH);       
      y=0;                         
      zero_cross2 = false;                
    } 
    else {
      y++;                           
    }                                
  }                                                                  
}                                   

void loop() {                        
  bright1 = analogRead(pot1) / 8;  
  bright2 = analogRead(pot2) / 8; 
}

Ho provato su un computer che non è il mio ma ha Windows XP con IDE 1.0.1 e core Tiny. A me ha compilato sia con TIM0 che con TIMER0 senza errori.

beh, anche a me compila senza errori, ma non mi funziona all'atto pratico...

Ah. Perché inizialmente avevi detto che ti dava errore anche in compilazione

Guarda, ho compilato e caricato ora uno sketch di test.

byte stato=1;
void setup() {
    cli();//fermo gli interrupt
TCCR0A &= ~((1<<COM0A0) | (1<<COM0A1)); //disattivi gli interrupt del timer
//modalità contatore fino all'overflow
TCCR0A &= ~((1<<WGM01) | (1<<WGM00));
TCCR0B &= ~(1<<WGM02);
//prescaler a /8
TCCR0B &= ~((1<<CS02) | (1<<CS00));
TCCR0B |= (1<<CS01);
TCNT0 = 178; //valore iniziale
TIMSK |= (1<<TOIE0); //attivo un interrutp all'overflow
sei(); //riattivo gli interrupt}
pinMode(0, OUTPUT);
}
ISR (TIMER0_OVF_vect) { //funziona anche con TIM0_OVF_vect ovviamente
    TCNT0 = 178;
    digitalWrite(0, stato);
    stato ^=1;
   
}
void loop(){}

Ho messo un led sul pin D0 per verificare che funzionasse correttamente. A me va sia con TIM0 che con TIMER0. Ed ora sono su Linux e con la toolchain ufficiale Atmel.
C'è qualcosa sul tuo PC che non va e che non capisco cosa sia... :cold_sweat:
Ci vorrebbero altri riscontri.