Hallo,
ich will mit einem Master Controller mehrere Slave Controller über spi
steuern, aber mache wohl irgendwelche Fehler. Hab noch relativ wenig
Ahnung von Microcontrollern ;)
Ich vermute, dass irgendwas an der Initialisierung nicht stimmt..
Master:
#include <util/delay.h>
#include <avr/io.h>
#include <avr/interrupt.h>
void SPI_Transmit(unsigned char cData){
/* Start transmission */
SPDR = cData;
/* Wait for transmission complete */
while(!(SPSR & (1<<SPIF)));
}
void main(){
/*Set MOSI, SCK output, */
DDRB = 0x28;
//SS connections
DDRC=0xFF;
//enable global interrupt
sei();
//enable SPI and Master mode
SPCR=0x50;
while(1){
//transmit data to first controller
PORTC|=0x01;
SPI_Transmit('a');
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
SPI_Transmit('b');
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
SPI_Transmit('c');
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
SPI_Transmit('d');
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
SPI_Transmit('e');
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
SPI_Transmit('f');
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
_delay_ms(40);
PORTC&=!0x01;
}
}
Slave:
#include <avr/io.h>
#include <avr/interrupt.h>
#include <util/delay.h>
//SPI Transfer Complete Interrupt starting on page 124 in datasheet
ISR( SPI_STC_vect ){
unsigned char tmp = SPDR;
switch(tmp){
case 'a':
PORTD&=!0x01;
break;
case 'b':
PORTD|=0x01;
break;
case 'c':
PORTD&=!0x02;
break;
case 'd':
PORTD|=0x02;
break;
case 'e':
PORTD&=!0x02;
break;
case 'f':
PORTD|=0x02;
break;
}
PORTC=~PORTC;
}
void main(){
// Set MISO and PB0 output
DDRB=0x11;
DDRD=0xFF;
DDRC=0xFF;
//enable global interrupt
sei();
//enable SPI,slave etc
SPCR=0xC0;
PORTD=0xFF;
PORTC=0xFF;
PORTB|=0x01;
while(1){
}
}
>_delay_ms(40); was soll das? Sollte man vermeiden. >void SPI_Transmit(unsigned char cData) >{ > /* Start transmission */ > SPDR = cData; > /* Wait for transmission complete */ > while(!(SPSR & (1<<SPIF))); >} Heul.... Ist leider alles ganz falsch. Der SPI Interrupt kommt wenn die Uebertragung vorbei ist. Dann hat man 2 Moeglichkeiten : - Nachladen, naechstes Byte - Beenden, dh Interrupt Bit des SPI befriedigen
Die transmit funktion stand so im Datenblatt. Dachte die könnte ich übernehmen.. Das empfangen ist in Ordnung? Danke schonmal ;)
>PORTD&=!0x02; Damit wirst du nicht das erreichen, was du vorhast. >Heul.... >Ist leider alles ganz falsch. Der SPI Interrupt kommt wenn die >Uebertragung vorbei ist. Dann hat man 2 Moeglichkeiten : >- Nachladen, naechstes Byte >- Beenden, dh Interrupt Bit des SPI befriedigen Es ist eine synchrone Übertragung. Da ist es dem Slave ziemlich egal, wann das nächste Byte kommt, solange er genug Zeit hat, das letzte wegzupacken. Wenn der Slave-Select an PortC0 hängt, sollte es mit der "!"durch"~"Ersetzung besser funktionieren...
Hallo,
hab die vielen fehler versucht zu beheben.. die sleeps sind nur drin,
damit ich auf den leds am slave erkenne ob das klappt.. Es tut aber noch
nich 100%ig das was ich gedacht hätte. Muss man SS nach jedem byte
wieder auf high und dann auf low setzen?
Danke schonmal ;)
Master:
#include <util/delay.h>
#include <avr/io.h>
#include <avr/interrupt.h>
void SPI_Transmit(unsigned char cData){
/* Start transmission */
SPDR = cData;
/* Wait for transmission complete */
while(!(SPSR & (1<<SPIF)));
}
void main(){
/*Set MOSI, SCK output, */
DDRB = 0x2C;
//SS connections
DDRC=0xFF;
//enable global interrupt
sei();
//enable SPI and Master mode
SPCR=0x50;
while(1){
//transmit data to first controller
PORTC&=~0x01;
SPI_Transmit('a');
PORTC|=0x01;
_delay_ms(100);
PORTC&=~0x01;
SPI_Transmit('b');
PORTC|=0x01;
_delay_ms(100);
PORTC&=~0x01;
SPI_Transmit('c');
PORTC|=0x01;
_delay_ms(100);
PORTC&=~0x01;
SPI_Transmit('d');
PORTC|=0x01;
_delay_ms(100);
PORTC&=~0x01;
SPI_Transmit('e');
PORTC|=0x01;
_delay_ms(100);
PORTC&=~0x01;
SPI_Transmit('f');
PORTC|=0x01;
_delay_ms(100);
}
}
slave:
#include <avr/io.h>
#include <avr/interrupt.h>
#include <util/delay.h>
//SPI Transfer Complete Interrupt starting on page 124 in datasheet
ISR( SPI_STC_vect ){
unsigned char tmp = SPDR;
switch(tmp){
case 'a':
PORTD&=~0x01;
break;
case 'b':
PORTD|=0x09;
break;
case 'c':
PORTD&=~0x1B;
break;
case 'd':
PORTD|=0x1A;
break;
case 'e':
PORTD&=~0x05;
break;
case 'f':
PORTD|=0x04;
break;
}
PORTC=~PORTC;
}
void main(){
// Set MISO and PB0 output
DDRB=0x11;
DDRD=0xFF;
DDRC=0xFF;
//enable global interrupt
sei();
//enable SPI,slave etc
SPCR=0xC0;
PORTD=0xFF;
PORTC=0xFF;
PORTB|=0x01;
while(1){
}
}
Bitte melde dich an um einen Beitrag zu schreiben. Anmeldung ist kostenlos und dauert nur eine Minute.
Bestehender Account
Schon ein Account bei Google/GoogleMail? Keine Anmeldung erforderlich!
Mit Google-Account einloggen
Mit Google-Account einloggen
Noch kein Account? Hier anmelden.