' ********************************************************************** ' * File: Corso_BASCOMAVR_094.BAS * ' * Versione: 1.1 * ' * Data: 17.07.11 * ' * Ambiente sviluppo: Bascom-AVR Demo Ver. 1.11.9.1 + * ' * + AVR bootloader grifo(r) Ver. 1.2 * ' * Schede: GMM AM08 + GMM TST3 * ' * Sviluppato da: GRIFO(r) Italian Technology * ' * via Dell'Artigiano 8/6 * ' * 40016 S. Giorgio di Piano (BO) * ' * Tel. +39 051 892052 Fax. +39 051 893661 * ' * http://www.grifo.com http://www.grifo.it * ' * Autore: Gianluca Angelini * ' ********************************************************************** ' Programma di esempio 094 del corso BASCOM AVR. ' Rappresentazione orario su display a 7 segmenti tramite I2C BUS. ' Il programma usa il componente SAA 1064 con cui pilota 4 display a 7 segmenti ' più punto decimale, che rappresentano l'orario attuale prelevato da un RTC ' tipo PCF 8583. ' La gestione di entrambi i componenti avviene tramite l'interfaccia I2C BUS ' del Mini Modulo ed i display sono comandati in modalità dinamica. ' Alla partenza il programma prima attiva tutti i segmenti dei display in modo ' da verificarne il funzionamento, poi inizia la rappresentazione di ore e ' minuti sui display ed il lampeggio del punto decimale ad ogni secondo. ' Il programma si presenta ed usa una console seriale dotata di monitor, con ' un protocollo fisico costante a 19200 Baud, 8 Bit x chr, 1 Stop bit, Nessuna ' parità, per visualizzare ed impostare l'orario attuale sul RTC. ' Questa console può coincidere con un sistema in grado di gestire una ' comunicazione seriale in RS 232. Al fine di semplificarne l'uso si può ' usare un PC dotato di una linea COMx, che esegue un programma di emulazione ' terminale come HYPERTERMINAL o l'omonima modalità offerta dal BASCOM AVR ' (vedere Configurazioni IDE). ' Il programma funziona solo se la GMM AM08 è montata sullo zoccolo Z2 della ' GMM TST3!! ' ' Aggiunte: Nessuna. ' ' 17/07/11: Corso_BASCOMAVR_94.BAS - Ver 1.1 - By G.A. ' Prima versione. ' ' '*************************** Configurazioni IDE ******************************** ' NOTE: per usare correttamente questo programma demo, eseguire i seguenti ' punti: ' 1) Controllare la presenza del file M8DEF.DAT nella directory di ' installazione del BASCOM-AVR e se non disponibile copiarlo e dopo ' rilanciare l'IDE. ' 2) Nella finestra "Options | Compiler | Chip" impostare: ' Chip: m8def.dat ' XRAM: None ' HW Stack: 64 ' Soft Stack: 32 ' Framesize: 64 ' XRAM waitstate: disattivo ' External Access Enable: disattivo ' 3) Nella finestra "Options | Compiler | I2C, SPI, 1WIRE" settare: ' I2C ' SCL port = PORTC.5 ' SDA port = PORTC.4 ' 1Wire ' 1wire = indifferente ' SPI ' Clock = indifferente ' MOSI = indifferente ' MISO = indifferente ' SS = indifferente ' Use Hardware SPI = disattivo ' 4) Nella finestra "Options | Communication" settare: ' COM port = la linea del PC collegata alla GMM AM08, tramite la GMM TST3 ' Baudrate = 19200 ' Parity = None ' Databits = 8 ' Stopbit = 1 ' Handshake = None ' Emulation = TTY ' Font = Terminal, Normale, 12 punti, colore bianco ' Backcolor = Navy ' 5) Al termine della compilazione, della programmazione del codice nella ' GMM AM08, aprire la finestra di emulazione terminale del BASCOM AVR con ' l'opzione: Tools | Terminal emulator (Ctrl+T) ed a questo punto resettare ' o riaccendere il Mini Modulo. '********************** Direttive del compilatore ****************************** $regfile "M8DEF.DAT" ' File di definizione microcontrollore usato $romstart = &H0 ' Indirizzo inizio codice in FLASH $crystal = 7372800 ' Frequenza quarzo del microcontrollore $hwstack = 64 ' Spazio per stack hardware $swstack = 32 ' Spazio per stack software $framesize = 64 ' Spazio per frame $map ' Genera informazioni di debug $baud = 19200 ' Velocità comunicazione seriale: 19200 Baud ' Altri parametri sono fissi a: 8 bit x chr ' 1 Stop bit ' No parity '******************************* Definizioni *********************************** ' Le risorse usate dal programma sono collegate come illustrato nella seguenti ' tabelle: ' !!! N.B. Sulla GMM TST3 i jumper devono essere posizionati come segue: ' J1 in 2-3 ; J2 in 1-2 ; J3 in 1-2 ; J5 in 2-3 ; J7 in 2-3 ; J8 in 2-3 ' J9 in 2-3 !!! ' ' Segnali Risorsa pin Z2 pin Segnale Segnale uP ' esterni GMM TST3 GMM TST3 GMM AM08 GMM AM08 usato ' SCL PCF8583 SCL SAA1064 CN3.15 12 6 PC5 ADC5 SCL PC5 ' SDA PCF8583 SDA SAA1064 CN3.16 13 7 PC4 ADC4 SDA PC4 ' Vdd PCF8583 Vcc SAA1064 CN3.18 34 28 +5 Vdc - ' GND PCF8583 Vss SAA1064 CN3.17 20 14 GND - ' ' Segnale pin COMx pin CN5 pin Z2 pin Segnale Segnale uP ' PC DB9 GMM TST3 GMM TST3 GMM AM08 GMM AM08 usato ' TX 3 3 9 3 RxD RS232 PD0 ' RX 2 2 10 4 TxD RS232 PD1 ' GND 5 5 20 14 GND - ' Da questa tabella si ricava che il cavo di collegamento tra la COM del PC ed ' il CN5 della GMM TST3 è un normale cavo dritto o prolunga. Volendolo ' richiedere alla grifo(r) si deve specificare il codice CCR 9+9E. Pinrx Alias Ddrd.0 ' Bit con direzione segnale collegato a RxD GMM AM08 Pintx Alias Ddrd.1 ' Bit con direzione segnale collegato a TxD GMM AM08 '************************ Dichiarazioni costanti ******************************* Const Cret = &H0D ' Codice ASCII Carriage Return Const Slawsaa1064 = &H70 ' Slave address SAA1064 in scrittura Const Slarsaa1064 = &H71 ' Slave address SA1064 in lettura Const Ctb = 0 ' Indirizzo Control byte SA1064 Const Dig1 = 1 ' Indirizzo Digit 1 SA1064 Const Dig2 = 2 ' Indirizzo Digit 2 SA1064 Const Dig3 = 3 ' Indirizzo Digit 3 SA1064 Const Dig4 = 4 ' Indirizzo Digit 4 SA1064 Const Slawpcf8583 = &HA0 ' Slave address PCF8583 in scrittura Const Slarpcf8583 = &HA1 ' Slave address PCF8583 in lettura '************************ Dichiarazioni variabili ****************************** Dim Hlpb As Byte ' Variabile byte di aiuto ad uso generico Dim Hlpw As Word ' Variabile word di aiuto ad uso generico Dim Choice As Byte ' Scelta utente Dim Patdis(10) As Byte ' Vettore con pattern cifre numeriche da rappresentare su display Dim Disdat(6) As Byte ' Vettore con dati e pattern per accensione segmenti su display Dim Num As Word ' Valore da rappresentare su display Dim Wee As Byte ' Giorno settimana RTC Dim Scn As Byte ' Secondi RTC Dim Mnu As Byte ' Minuti RTC Dim Hou As Byte ' Ore RTC Dim Day As Byte ' Giorno mese RTC Dim Mon As Byte ' Mese RTC Dim Yea As Word ' Anno RTC '************************ Dichiarazioni procedure ****************************** Declare Sub Ini_i2c() ' Inizializza linee usate per l'interfaccia I2C BUS software Declare Sub Ini_disp7seg() ' Inizializza controllore display a 7 segmenti SAA 1064 Declare Sub Shw_disp7seg() ' Visualizza stato dei segmenti su display comandati da SAA 1064 Declare Sub Get_rtc() ' Acquisisce data ed ora da RTC Declare Sub Set_rtc() ' Setta data ed ora su RTC Declare Sub Shw_time() ' Rappresenta ora formattata sulla console Declare Sub Shw_time_disp7seg() ' Rappresenta ora formattata su display a 7 segmenti '************************** Programma principale ******************************* Main: Pinrx = 0 ' Inizializza segnali per comunicazione seriale Pintx = 0 ' come ingressi digitali Call Ini_i2c() ' Inizializza linee usate per l'interfaccia I2C BUS software Print Print " Rappresentazione orario su display 7 segmenti, con GMM AM08 + GMM TST3" Print "Montare Mini Modulo su Z2 della GMM TST3, collegare linea I2C BUS ai componenti" Print "SAA 1064 e PCF 8583, come da schema." Print "Il programma preleva ciclicamente l'ora dall'orologio e rappresenta ore," Print "minuti sui display in forma numerica, con lampeggio del punto decimale ad" Print "secondo. Premere I su console per impostare orologio." Print Call Ini_disp7seg() ' Inizializza controllore display a 7 segmenti SAA 1064 Disdat(2) = &HFF ' Attiva tutti i segmenti dei quattro display Disdat(3) = &HFF Disdat(4) = &HFF Disdat(5) = &HFF Call Shw_disp7seg() ' Visualizza stato impostato su display Wait 1 ' Attesa di 1 secondo per consentire verifica attivazione segmenti dei display Disdat(2) = &H00 ' Disattiva tutti i segmenti dei quattro display Disdat(3) = &H00 Disdat(4) = &H00 Disdat(5) = &H00 Call Shw_disp7seg() ' Visualizza stato impostato su display Wait 1 ' Attesa di 1 secondo per consentire verifica disattivazione segmenti dei display Do ' Inizio ciclo infinito Call Get_rtc() ' Acquisisce data ed ora attuale da RTC Call Shw_time_disp7seg() ' Rappresenta ora formattata su display a 7 segmenti Call Shw_time() ' Rappresenta ora attuale su console Printbin Cret ' Mantiene rappresentazione su stessa riga Waitms 100 ' Ritardo tra acquisizioni e controlli Choice = Inkey() ' Verifica se tasto premuto su console If Choice > 0 Then ' Se tasto premuto su console If Choice >= "a" Then ' Se tasto minuscolo Choice = Choice And &HDF ' Lo converte in in maiuscolo End If If Choice = "I" Then ' Controlla scelta convertita Print ' Separa da precedenti indicazioni, andando su nuova riga Print Print "Inserire dati per impostare orologio:" ' Richiede e preleva data da impostare Input "Giorno settimana (0->lunedi',..,6->Domenica): " , Wee Input "Giorno (1..31): " , Day Input "Mese (1..12): " , Mon Input "Anno (0000..9999): " , Yea Input "Ore (0..23): " , Hou ' Richiede e preleva ora da impostare Input "Minuti (0..59): " , Mnu Input "Secondi (0..59): " , Scn Call Set_rtc() ' Inizializza RTC con dati inseriti Print ' Separa da precedenti indicazioni, andando su nuova riga End If End If Loop ' Fine ciclo infinito End '************************ Fine programma principale **************************** '****************** Inizio procedure usate dal programma *********************** ' Inizializza segnali e variabili usati per l'interfaccia I2C BUS gestita via ' software ' Ingresso: Nulla ' Uscita: Nulla Sub Ini_i2c() ' Questa procedura è stata realizzata per compatibilità con i programmi demo ' delle EEPROM con altri protocolli gestite a basso livello. Per il protocollo ' I2C BUS si usano invece le istruzioni ad alto livello del BASCOM. ' Per le connessioni linee dell'interfaccia I2C BUS software vedere help in ' linea e precedente tabella. Config Scl = Portc.5 ' Segnale del micro usato come segnale SCL dell'I2C BUS Config Sda = Portc.4 ' Segnale del micro usato come segnale SDA dell'I2C BUS I2cinit ' Inizializza segnali I2C BUS End Sub ' Procedura che inizializza controllore display a 7 segmenti SAA 1064, in I2C BUS ' Ingresso: Nulla ' Uscita: Patdis() = Vettore con pattern cifre numeriche da rappresentare su display Sub Ini_disp7seg() Do I2creceive Slarsaa1064 , Hlpb , 0 , 1 ' Legge registro di stato SAA 1064 Loop Until Hlpb = 0 ' Attende accensione SAA 1064 ' Prepara sequenza di inizializzazione dell'SAA1064 Disdat(1) = Ctb ' Punta al registro di controllo Disdat(2) = &B00100111 ' bit0 =1 dynamic mode ' bit1 =1 digit 1+3 not blanked ' bit2 =1 digit 2+4 not blanked ' bit3 =0 no test segment ' bit4 =0 no 3mA segment current ' bit5 =1 6mA segment current ' bit6 =0 no 12mA segment current ' bit7 =0 indifferente Disdat(3) = 0 ' Setta DY1 off Disdat(4) = 0 ' Setta DY2 off Disdat(5) = 0 ' Setta DY3 off Disdat(6) = 0 ' Setta DY4 off I2csend Slawsaa1064 , Disdat(1) , 6 ' Invia sequenza preparata all'SAA 1064 ' Inizializza vettore con pattern per convertire una cifra da 0 a 9 nella codifica a 7 segmenti Patdis(1) = &H3F ' 0 D0 Patdis(2) = &H06 ' 1 +----+ Patdis(3) = &H5B ' 2 | | Patdis(4) = &H4F ' 3 D5 | | D1 Patdis(5) = &H66 ' 4 | D6 | Patdis(6) = &H6D ' 5 +----+ Patdis(7) = &H7D ' 6 | | Patdis(8) = &H07 ' 7 D4 | | D2 Patdis(9) = &H7F ' 8 | D3 | Patdis(10) = &H6F ' 9 +----+ . D7 End Sub ' Visualizza stato dei segmenti su display comandati da SAA 1064 ' Ingresso: Disdat() = Pattern per visualizzazione valore su display ' Uscita: Nulla Sub Shw_disp7seg() Disdat(1) = Dig1 ' Punta al primo display del controllore SAA 1064 I2csend Slawsaa1064 , Disdat(1) , 5 ' Invia sequenza con pattern preparati all'SAA 1064 End Sub ' Procedura che preleva data ed ora attuali dal RTC PCF 8583, in I2C BUS ' Ingresso: Nulla ' Uscita: Scn,Mnu,Hou = orario prelevato ' Day,Mon,Yea,Wee = data prelevata (con anno a 4 cifre) Sub Get_rtc() I2cstart ' Manda start I2cwbyte Slawpcf8583 ' Manda slave address PCF8583+W I2cwbyte &H02 ' Manda indirizzo registro 2 I2cstart ' Manda repeated start I2cwbyte Slarpcf8583 ' Manda slave address PCF8583+R I2crbyte Scn , Ack ' Legge secondi da reg. 2 I2crbyte Mnu , Ack ' Legge minuti da reg. 3 I2crbyte Hou , Ack ' Legge ore da reg. 4 I2crbyte Hlpb , Ack ' Legge anno e giorno da reg. 5 I2crbyte Wee , Nack ' Legge settimana e mese da reg. 6 I2cstop ' Manda stop Scn = Makedec(scn) ' Ottiene secondi Mnu = Makedec(mnu) ' Ottiene minuti Hou = Makedec(hou) ' Ottiene ore Day = Hlpb And &H3F ' Ottiene giorno Day = Makedec(day) Shift Hlpb , Right , 6 ' Ottiene offset anno (0÷3) Yea = Hlpb Mon = Wee And &H1F ' Ottiene mese Mon = Makedec(mon) Shift Wee , Right , 5 ' Ottiene settimana ' Preleva anno a 4 cifre da ultime locazioni RAM del PCF 8583 I2cstart ' Manda start I2cwbyte Slawpcf8583 ' Manda slave address PCF8583+W I2cwbyte &HFE ' Manda indirizzo penultima locazione I2cstart ' Manda repeated start I2cwbyte Slarpcf8583 ' Manda slave address PCF8583+R I2crbyte Hlpb , Ack ' Legge byte high anno a 4 cifre da RAM Hlpw = Hlpb ' Salva byte high anno a 4 cifre I2crbyte Hlpb , Nack ' Legge byte low anno a 4 cifre da RAM I2cstop ' Manda stop Shift Hlpw , Left , 8 ' Ottiene anno a 4 cifre dai due byte letti Hlpw = Hlpw Or Hlpb Yea = Yea + Hlpw ' Aggiunge offset anno letto da RTC End Sub ' Procedura che setta data ed ora attuali sul RTC PCF 8583, in I2C BUS ' Ingresso: Scn,Mnu,Hou = orario prelevato ' Day,Mon,Yea,Wee = data prelevata (con anno a 4 cifre) ' Uscita: Nulla Sub Set_rtc() Scn = Makebcd(scn) ' Trasforma secondi Mnu = Makebcd(mnu) ' Trasforma minuti Hou = Makebcd(hou) ' Trasforma ore Day = Makebcd(day) ' Trasforma giorno Mon = Makebcd(mon) ' Trasforma mese Shift Wee , Left , 5 ' Trasforma settimana Wee = Wee Or Mon ' Ottiene settimana e mese I2cstart ' Manda start I2cwbyte Slawpcf8583 ' Manda slave address PCF8583+W I2cwbyte &H00 ' Manda indirizzo registro 0 I2cwbyte &H84 ' Manda comando stop RTC su reg. 0 I2cstop ' Manda stop I2cstart ' Manda start I2cwbyte Slawpcf8583 ' Manda slave address PCF8583+W I2cwbyte &H01 ' Manda indirizzo registro 1 I2cwbyte &H00 ' Manda centinaia secondi I2cwbyte Scn ' Manda secondi su reg. 2 I2cwbyte Mnu ' Manda minuti su reg. 3 I2cwbyte Hou ' Manda ore su reg. 4 I2cwbyte Day ' Manda anno=0 e giorno su reg. 5 I2cwbyte Wee ' Manda settimana e mese su reg. 6 I2cstop ' Manda stop I2cstart ' Manda start I2cwbyte Slawpcf8583 ' Manda slave address PCF8583+W I2cwbyte &H00 ' Manda indirizzo registro 0 I2cwbyte &H00 ' Manda comando start RTC su reg. 0 I2cstop ' Manda stop ' Setta anno a 4 cifre su ultime locazioni RAM del PCF 8583 I2cstart ' Manda start I2cwbyte Slawpcf8583 ' Manda slave address PCF8583+W I2cwbyte &HFE ' Manda indirizzo penultima locazione Hlpb = High(yea) ' Preleva byte high anno a 4 cifre I2cwbyte Hlpb ' Manda byte high anno a 4 cifre su RAM Hlpb = Low(yea) ' Preleva byte low anno a 4 cifre I2cwbyte Hlpb ' Manda byte low anno a 4 cifre su RAM I2cstop ' Manda stop End Sub ' Rappresenta orario formattato sulla console nel formato oo:mm:ss ' Ingresso: Scn,Mnu,Hou = orario attuale ' Uscita: Nulla Sub Shw_time() If Hou < 10 Then ' Rappresenta ore a due cifre Print "0"; End If Print Hou ; ":" ; ' Con separatore If Mnu < 10 Then ' Rappresenta minuti a due cifre Print "0"; End If Print Mnu ; ":" ; ' Con separatore If Scn < 10 Then ' Rappresenta secondi a due cifre Print "0"; End If Print Scn ; End Sub ' Rappresenta orario formattato sui display a 7 segmenti, nel formato oo.mm, ' con il punto di separazione che lampeggia ad ogni secondo ' Ingresso: Scn,Mnu,Hou = orario attuale ' Uscita: Nulla Sub Shw_time_disp7seg() ' Converte ora attuale Hlpb = Hou \ 10 ' Ottiene cifra delle decine 0..9 Incr Hlpb ' Trasforma decine da 1..10 Disdat(2) = Patdis(hlpb) ' Salva pattern cifra delle decine per primo display (DY1=DY5) Hlpb = Hou Mod 10 ' Ottiene cifra delle unità 0..9 Incr Hlpb ' Trasforma unità da 1..10 Disdat(3) = Patdis(hlpb) ' Salva pattern cifra delle unità per secondo display (DY2=DY6) con punto decimale disattivo ' Converte minuti attuali Hlpb = Mnu \ 10 ' Ottiene cifra delle decine 0..9 Incr Hlpb ' Trasforma decine da 1..10 Disdat(4) = Patdis(hlpb) ' Salva pattern cifra delle decine per terzo display (DY3=DY7) Hlpb = Mnu Mod 10 ' Ottiene cifra delle unità 0..9 Incr Hlpb ' Trasforma unità da 1..10 Disdat(5) = Patdis(hlpb) ' Salva pattern cifra delle unità per quarto display (DY4=DY8) ' Lampeggia punto decimale di separazione, ad ogni secondo, attivandolo nei ' secondi dispari e disattivandolo in quelli pari Hlpb = Scn And &H01 If Hlpb <> 0 Then ' Se secondi dispari Disdat(3) = Disdat(3) Or &H80 ' Attiva punto decimale (lampeggio) End If Call Shw_disp7seg() ' Visualizza stato impostato su display End Sub '******************* Fine procedure usate dal programma ************************