Aus RN-Wissen.de
Wechseln zu: Navigation, Suche


K
(neue Version der twislave.c)
Zeile 28: Zeile 28:
 
</pre>
 
</pre>
  
Die twislave.c für den Slave
+
Die twislave.c für den Slave. Stand: 23.03.07
 
<pre>
 
<pre>
 
#ifndef _TWISLAVE_H
 
#ifndef _TWISLAVE_H
Zeile 51: Zeile 51:
 
Autor: Uwe Große-Wortmann (uwegw)
 
Autor: Uwe Große-Wortmann (uwegw)
 
Status: Testphase, keine Garantie für ordnungsgemäße Funktion!  
 
Status: Testphase, keine Garantie für ordnungsgemäße Funktion!  
letze Änderungen: Kommentare ergänzt, Schutz gegen mehrfaches einbinden eingebaut.
+
letze Änderungen:  
 +
23.03.07 Makros für TWCR eingefügt. Abbruch des Sendens, wenn der TXbuffer komplett gesendet wurde.  
 +
 
  */  
 
  */  
 
   
 
   
#include <util/twi.h>
+
 
 +
 
  
 
//%%%%%%%% von Benutzer konfigurierbare Einstellungen %%%%%%%%
 
//%%%%%%%% von Benutzer konfigurierbare Einstellungen %%%%%%%%
Zeile 82: Zeile 85:
 
//%%%%%%%% ab hier sind normalerweise keine weiteren Änderungen erforderlich! %%%%%%%%
 
//%%%%%%%% ab hier sind normalerweise keine weiteren Änderungen erforderlich! %%%%%%%%
 
//_____________________________________________________________________________________________
 
//_____________________________________________________________________________________________
 +
 +
#include <util/twi.h> //enthält z.B. die Bezeichnungen für die Statuscodes in TWSR
 +
 +
 +
//Bei zu alten AVR-GCC-Versionen werden die Interrupts anders genutzt, daher in diesem Fall mit Fehlermeldung abbrechen
 +
#if (__GNUC__ * 100 + __GNUC_MINOR__) < 304
 +
#error "This library requires AVR-GCC 3.4.5 or later, update to newer AVR-GCC compiler !"
 +
#endif
  
  
Zeile 92: Zeile 103:
 
{
 
{
 
TWAR= adr; //Adresse setzen
 
TWAR= adr; //Adresse setzen
 
 
TWCR|= (1<<TWEA) | (1<<TWEN)|(1<<TWIE);  
 
TWCR|= (1<<TWEA) | (1<<TWEN)|(1<<TWIE);  
 
TWCR &= ~(1<<TWSTA)|(1<<TWSTO);
 
TWCR &= ~(1<<TWSTA)|(1<<TWSTO);
Zeile 100: Zeile 110:
  
  
 +
//Je nach Statuscode in TWSR müssen verschiedene Bitmuster in TWCR geschreiben werden(siehe Tabellen im Datenblatt!).
 +
//Makros für die verwendeten Bitmuster:
 +
 +
//ACK nach empfangenen Daten senden/ ACK nach gesendeten Daten erwarten
 +
#define TWCR_ACK TWCR = (1<<TWEN)|(1<<TWIE)|(1<<TWINT)|(1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|(0<<TWWC); 
 +
//NACK nach empfangenen Daten senden/ NACK nach gesendeten Daten erwarten   
 +
#define TWCR_NACK TWCR = (1<<TWEN)|(1<<TWIE)|(1<<TWINT)|(0<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|(0<<TWWC);
 +
//switched to the non adressed slave mode...
 +
#define TWCR_RESET TWCR = (1<<TWEN)|(1<<TWIE)|(1<<TWINT)|(1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|(0<<TWWC); 
  
 
/*ISR, die bei einem Ereignis auf dem Bus ausgelöst wird. Im Register TWSR befindet sich dann  
 
/*ISR, die bei einem Ereignis auf dem Bus ausgelöst wird. Im Register TWSR befindet sich dann  
Zeile 111: Zeile 130:
 
switch (TW_STATUS) //TWI-Statusregister prüfen und nötige Aktion bestimmen  
 
switch (TW_STATUS) //TWI-Statusregister prüfen und nötige Aktion bestimmen  
 
{
 
{
 +
 
case TW_SR_SLA_ACK: // 0x60 Slave Receiver, wurde adressiert
 
case TW_SR_SLA_ACK: // 0x60 Slave Receiver, wurde adressiert
      TWCR = (1<<TWEN)|                                // TWI Interface enabled
+
TWCR_ACK;
            (1<<TWIE)|(1<<TWINT)|                      // Enable TWI Interupt and clear the flag to send byte
+
            (1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|          // Expect ACK on this transmission
+
            (0<<TWWC);                                 // 
+
 
buffer_adr=0xFF; //Bufferposition ist undefiniert
 
buffer_adr=0xFF; //Bufferposition ist undefiniert
 
break;
 
break;
Zeile 124: Zeile 141:
 
  {
 
  {
 
buffer_adr= data; //Bufferposition wie adressiert setzen
 
buffer_adr= data; //Bufferposition wie adressiert setzen
       TWCR = (1<<TWEN)|                                // TWI Interface enabled
+
       TWCR_ACK;
            (1<<TWIE)|(1<<TWINT)|                      // Enable TWI Interupt and clear the flag to send byte
+
            (1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|          // Send ACK after next reception
+
            (0<<TWWC);
+
 
  }
 
  }
 
else //weiterer Zugriff, Daten empfangen
 
else //weiterer Zugriff, Daten empfangen
Zeile 136: Zeile 150:
 
{
 
{
 
//nächstes Byte lesen, ACK danach, um nächstes Byte anzufordern
 
//nächstes Byte lesen, ACK danach, um nächstes Byte anzufordern
       TWCR = (1<<TWEN)|                                // TWI Interface enabled
+
       TWCR_ACK;
            (1<<TWIE)|(1<<TWINT)|                      // Enable TWI Interupt and clear the flag to send byte
+
            (1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|          // Send ACK after next reception
+
            (0<<TWWC);
+
 
}
 
}
 
else  //es kann nur noch ein Byte kommen, dann ist der Buffer voll
 
else  //es kann nur noch ein Byte kommen, dann ist der Buffer voll
 
{
 
{
 
//letztes Byte lesen, dann NACK, um vollen Buffer zu signaliseren
 
//letztes Byte lesen, dann NACK, um vollen Buffer zu signaliseren
      TWCR = (1<<TWEN)|                                // TWI Interface enabled
+
TWCR_NACK;
            (1<<TWIE)|(1<<TWINT)|                      // Enable TWI Interupt and clear the flag to send byte
+
            (0<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|       
+
            (0<<TWWC);
+
 
}
 
}
 
  }
 
  }
Zeile 154: Zeile 162:
  
 
case TW_ST_SLA_ACK:
 
case TW_ST_SLA_ACK:
case TW_ST_DATA_ACK: //0xB0 Slave Transmitter, weitere Daten wurden angefordert
+
case TW_ST_DATA_ACK: //0xB8 Slave Transmitter, weitere Daten wurden angefordert
 
TWDR = txbuffer[buffer_adr]; //Datenbyte senden  
 
TWDR = txbuffer[buffer_adr]; //Datenbyte senden  
buffer_adr++; //bhufferadresse für nächstes Byte weiterzählen
+
buffer_adr++; //bufferadresse für nächstes Byte weiterzählen
TWCR = (1<<TWEN)|                               
+
        (1<<TWIE)|(1<<TWINT)|                   
+
if(buffer_adr<(buffer_size-1)) //im Buffer ist mehr als ein Byte, das gesendet werden kann
        (1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|         
+
{
        (0<<TWWC);
+
TWCR_ACK; //nächstes Byte senden, danach ACK erwarten
 +
}
 +
else
 +
{
 +
TWCR_NACK; //letztes Byte senden, danach NACK erwarten
 +
}
 +
 
 +
 
break;
 
break;
 +
  
 
case TW_ST_DATA_NACK: //0xC0 Keine Daten mehr gefordert  
 
case TW_ST_DATA_NACK: //0xC0 Keine Daten mehr gefordert  
Zeile 168: Zeile 184:
 
case TW_SR_STOP: // 0xA0 STOP empfangen
 
case TW_SR_STOP: // 0xA0 STOP empfangen
 
default:  
 
default:  
//Übertragung beenden
+
//Übertragung beenden, warten bis zur nächsten Adressierung
     TWCR = (1<<TWEN)|                               
+
     TWCR_RESET;  
            (1<<TWIE)|(1<<TWINT)|                   
+
            (1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|         
+
            (0<<TWWC);
+
//buffer_adr=0xFF; //Bufferposition ist undefiniert
+
 
break;
 
break;
  
 
 
} //end.switch
+
} //end.switch (TW_STATUS)
 
} //end.ISR(TWI_vect)
 
} //end.ISR(TWI_vect)
  
  
 +
#endif //#ifdef _TWISLAVE_H
 +
////Ende von twislave.c////
  
  
 
#endif //#ifdef _TWISLAVE_H
 
////Ende von twislave.c////
 
 
</pre>
 
</pre>
  

Version vom 23. März 2007, 16:13 Uhr

Programm für einen AVR mit TWI (Hardware-I2C)-Schnittstelle als Slave.

Noch nicht 100% getestet! Es können noch Fehler enthalten sein. Bei mir (uwegw) funktioniert das System problemlos. Es müssten aber noch diverse Sonderfälle wie falsch adressierte Bufferzugriffe abgefangen werden... Also insgesamt eher Alpha-Status!


Ein Codeschnipsel für den Master. Es wird die I2C-Master-Bibliothek von Peter Fleury verwendet. Es wird geprüft, ob der Slave bereit ist, dann werden die ersten drei Bytes aus dem txbuffer des Slaves gelesen und in byte0..2 abgespeichert.

#include "i2cmaster.h"  //I2C-Master-Routinen von Peter Fleury verwenden (siehe www.jump.to/fleury)
#define SLAVE_ADRESSE 0x50
uint8_t byte0;
uint8_t byte1;
uint8_t byte2;

      if(!(i2c_start(SLAVE_ADRESSE+I2C_WRITE))) //Slave bereit zum lesen?
      {
         i2c_write(0x00); //Buffer Startadresse zum Auslesen
         i2c_rep_start(SLAVE_ADRESSE+I2C_READ); //Lesen beginnen

            byte0= i2c_readAck();
            byte1= i2c_readAck();
            byte2= i2c_readAck();
         i2c_stop();
      } 

Die twislave.c für den Slave. Stand: 23.03.07

#ifndef _TWISLAVE_H
#define _TWISLAVE_H

/*
Dieses Programm in einer separaten Datei (z.B. twislave.c) abspeichern und in das eigene Programm
einbinden.

Betrieb eines AVRs mit Hardware-TWI-Schnittstelle als Slave. Zu Beginn muss init_twi_slave mit der gewünschten
Slave-Adresse als Parameter aufgerufen werden. Der Datenaustausch mit dem Master erfolgt über die Buffer 
rxbuffer und txbuffer, auf die von Master und Slave zugegriffen werden kann. 
rxbuffer und txbuffer sind globale Variablen (Array aus uint8_t). 
Die Ansteuerung des rxbuffers, in den der Master schreiben kann, erfolgt ähnlich wie bei einem normalen I2C-EEPROM.
Man sendet zunächst die Bufferposition, an die man schreiben will, und dann die Daten. Die Bufferposition wird 
automatisch hochgezählt, sodass man mehrere Datenbytes hintereinander schreiben kann, ohne jedesmal
die Bufferadresse zu schreiben.
Um den txbuffer vom Master aus zu lesen, überträgt man zunächst in einem Schreibzugriff die gewünschte Bufferposition und
liest dann nach einem repeated start die Daten aus. Die Bufferposition wird automatisch hochgezählt, sodass man mehrere
Datenbytes hintereinander lesen kann, ohne jedesmal die Bufferposition zu schreiben.

Autor: Uwe Große-Wortmann (uwegw)
Status: Testphase, keine Garantie für ordnungsgemäße Funktion! 
letze Änderungen: 
23.03.07 Makros für TWCR eingefügt. Abbruch des Sendens, wenn der TXbuffer komplett gesendet wurde. 
 
 */ 
 



//%%%%%%%% von Benutzer konfigurierbare Einstellungen %%%%%%%%

#define buffer_size 8 //Größe der Buffer in Byte (1..255)


//%%%%%%%% Globale Variablen, die vom Hauptprogramm genutzt werden %%%%%%%%

/*Der Buffer, in dem die empfangenen Daten gespeichert werden. Der Slave funktioniert ähnlich  wie ein normales
 Speicher-IC [I2C-EEPROM], man sendet die Adresse, an die man schreiben will, dann die Daten, die interne Speicher-Adresse
 wird dabei automatisch hochgezählt*/
volatile uint8_t rxbuffer[buffer_size];

/*Der Sendebuffer, der vom Master ausgelesen werden kann.*/
volatile uint8_t txbuffer[buffer_size];


//%%%%%%%% Funktionen, die vom Hauptprogramm aufgerufen werden können %%%%%%%%
 
/*Initaliserung des TWI-Inteface. Muss zu Beginn aufgerufen werden, sowie bei einem Wechsel der Slave Adresse
Parameter: adr: gewünschte Slave-Adresse*/
void init_twi_slave (uint8_t adr);



//%%%%%%%% ab hier sind normalerweise keine weiteren Änderungen erforderlich! %%%%%%%%
//_____________________________________________________________________________________________

#include <util/twi.h> //enthält z.B. die Bezeichnungen für die Statuscodes in TWSR


//Bei zu alten AVR-GCC-Versionen werden die Interrupts anders genutzt, daher in diesem Fall mit Fehlermeldung abbrechen
#if (__GNUC__ * 100 + __GNUC_MINOR__) < 304
#error "This library requires AVR-GCC 3.4.5 or later, update to newer AVR-GCC compiler !"
#endif


uint8_t buffer_adr; //"Adressregister" für den Buffer

/*Initaliserung des TWI-Inteface. Muss zu Beginn aufgerufen werden, sowie bei einem Wechsel der Slave Adresse
Parameter: adr: gewünschte Slave-Adresse
*/
void init_twi_slave (uint8_t adr)
{
TWAR= adr; //Adresse setzen
TWCR|= (1<<TWEA) | (1<<TWEN)|(1<<TWIE); 
TWCR &= ~(1<<TWSTA)|(1<<TWSTO);
buffer_adr=0xFF;  
sei();
}


//Je nach Statuscode in TWSR müssen verschiedene Bitmuster in TWCR geschreiben werden(siehe Tabellen im Datenblatt!). 
//Makros für die verwendeten Bitmuster:

//ACK nach empfangenen Daten senden/ ACK nach gesendeten Daten erwarten
#define TWCR_ACK TWCR = (1<<TWEN)|(1<<TWIE)|(1<<TWINT)|(1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|(0<<TWWC);  
//NACK nach empfangenen Daten senden/ NACK nach gesendeten Daten erwarten     
#define TWCR_NACK TWCR = (1<<TWEN)|(1<<TWIE)|(1<<TWINT)|(0<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|(0<<TWWC);
//switched to the non adressed slave mode...
#define TWCR_RESET TWCR = (1<<TWEN)|(1<<TWIE)|(1<<TWINT)|(1<<TWEA)|(0<<TWSTA)|(0<<TWSTO)|(0<<TWWC);  

/*ISR, die bei einem Ereignis auf dem Bus ausgelöst wird. Im Register TWSR befindet sich dann 
ein Statuscode, anhand dessen die Situation festgestellt werden kann.
*/
ISR (TWI_vect)  
{
uint8_t data=0;

//alte version: switch(TWSR)
switch (TW_STATUS) //TWI-Statusregister prüfen und nötige Aktion bestimmen 
{

case TW_SR_SLA_ACK: // 0x60 Slave Receiver, wurde adressiert	
	TWCR_ACK;
		buffer_adr=0xFF; //Bufferposition ist undefiniert
break;
	
case TW_SR_DATA_ACK: // 0x80 Slave Receiver,Daten empfangen
	data=TWDR; //Empfangene Daten auslesen
	if (buffer_adr == 0xFF) //erster Zugriff, Bufferposition setzen
	  {
		buffer_adr= data; //Bufferposition wie adressiert setzen
      TWCR_ACK;		
	  }
	else //weiterer Zugriff, Daten empfangen
      {
		rxbuffer[buffer_adr]=data; //Daten in Buffer schreiben
		buffer_adr++; //Buffer-Adresse weiterzählen für nächsten Schtreibzugriff
		if(buffer_adr<(buffer_size-1)) //im Buffer ist noch Platz für mehr als ein Byte
		{
		//nächstes Byte lesen, ACK danach, um nächstes Byte anzufordern		
      TWCR_ACK;
		}
		else   //es kann nur noch ein Byte kommen, dann ist der Buffer voll
		{
		//letztes Byte lesen, dann NACK, um vollen Buffer zu signaliseren
		TWCR_NACK;
		}
	  }
break;


case TW_ST_SLA_ACK:
case TW_ST_DATA_ACK: //0xB8 Slave Transmitter, weitere Daten wurden angefordert
	TWDR = txbuffer[buffer_adr]; //Datenbyte senden 
	buffer_adr++; //bufferadresse für nächstes Byte weiterzählen
	
	if(buffer_adr<(buffer_size-1)) //im Buffer ist mehr als ein Byte, das gesendet werden kann
	{
	TWCR_ACK; //nächstes Byte senden, danach ACK erwarten
	}
	else
	{
	TWCR_NACK; //letztes Byte senden, danach NACK erwarten
	}

	
break;


case TW_ST_DATA_NACK: //0xC0 Keine Daten mehr gefordert 
case TW_SR_DATA_NACK: //0x88 
case TW_ST_LAST_DATA: //0xC8  Last data byte in TWDR has been transmitted (TWEA = “0”); ACK has been received
case TW_SR_STOP: // 0xA0 STOP empfangen
default: 
	//Übertragung beenden, warten bis zur nächsten Adressierung
    TWCR_RESET; 
break;

	
} //end.switch (TW_STATUS)
} //end.ISR(TWI_vect)


#endif //#ifdef _TWISLAVE_H
////Ende von twislave.c////