Forum: Mikrocontroller und Digitale Elektronik SPI atmega8 interrupt


von gast (Gast)


Lesenswert?

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){

  }
}

von genau so (Gast)


Lesenswert?

>_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

von gast (Gast)


Lesenswert?

Die transmit funktion stand so im Datenblatt. Dachte die könnte ich 
übernehmen..
Das empfangen ist in Ordnung?

Danke schonmal ;)

von STK500-Besitzer (Gast)


Lesenswert?

>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...

von gast (Gast)


Lesenswert?

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
Noch kein Account? Hier anmelden.