Forum: Mikrocontroller und Digitale Elektronik SPI-Bits manuell auslesen?


von max (Gast)


Lesenswert?

Hey profis,

habe gerade mein praktikum angefange und habe schon folgendes problem :(

Ich soll hier ein mikrocontroller programmieren , der von einem 
AD-Wandler die daten bekommt und diese zu einem bluetooth modul schickt.

Das problem:
Ich muss einen MSP430 F 2274 benutzen. Der mikrocontroller hat leider 
SPI und UART auf den gleichen pins. SPI brauche ich um mit dem 
AD-wandler zu kommunizieren und UART um mit dem bluetooth-modul.

Ich habe mir überlegt SPI manuell in C zu programmieren, also die 
leitung auf irrgendein pin legen und diesen zu richtigen zeit abzuhören. 
Der AD-Wandler arbeitet mit 0,8 MHz und der MSP430*** ist 16MHz schnell.

Hat jemand schon erfahrungen damit? das problem kommt doch bestimmt 
öffters vor.

PS: leider habe ich noch gar nichts zu ausprobieren hier ausser MSP430F 
2274 EVO-board, das alleirdings auch noch nicht läuft

MFG max

von Falk B. (falk)


Lesenswert?


von Peter Diener (Gast)


Lesenswert?

Hallo,

nimm doch nen MSP430F2417 dafür, der hat genug Schnittstellen. Das in 
Software wird eine reine Datenschlacht.

Peter

von max (Gast)


Lesenswert?

danke für die idee.

das geht bestimmt auch mit internen schieberegistern von MSP430 F 2274.

werde mich nochmal melden, wenn was beim ausprobieren schief geht ;)

von Peter D. (peda)


Lesenswert?

Peter Diener wrote:
> Das in
> Software wird eine reine Datenschlacht.

Was ist denn an den paar Bytes Code ne Datenschlacht?

SPI ist sehr einfach in Software zu machen und kaum größer als der Code 
für HW-SPI.
Daher findet man auch nen Haufen Beispiele im Netz dafür.


Peter

von Maria (Gast)


Lesenswert?

uint16_t Exchange(GPIO ucSS, uint8_t ucCommand, BOOL b16BitAnswer)
{
  uint16_t usMask, usAnswer;

  PSet(ucSS, FALSE);    // /SS low

  for (usMask = 0x0080; usMask; usMask >>= 1)
  {  // send 8 bit command
    PSet(GYRO_SCK, FALSE);
    PSet(GYRO_MOSI, ucCommand & usMask);
    PSet(GYRO_SCK, TRUE);
  }

  for (usAnswer = 0, usMask = b16BitAnswer ? 0x8000 : 0x0080; usMask; 
usMask >>= 1)
  {  // receive 16 bit Answer
    PSet(GYRO_SCK, FALSE);
    PSet(GYRO_SCK, TRUE);
    usAnswer = (usAnswer << 1) | PGet(GYRO_MISO);
  }

  PSet(ucSS, TRUE);    // /SS high

  return usAnswer;
}

Soft SPI: Diese Funktion sendet ein 8 Bit Kommando end empfängt 8 oder 
16 Bit als Antwort.

Die Hilfsfunktion PSet() setzt ein Port Pin high/low.
PGet() fragt ein Port Pin ab.

von max (Gast)


Lesenswert?

@peter diener: danke für die antwort, leider ist ein µ-controller dieser 
art doppelt so teuer wie F 2274.

@danneger : werde versuchen beispiel code zu finden. habe versucht bist 
jetzt "SPI simulieren" einzugeben, da kamm nichts gescheites raus.
aber "SPI software" ist bestimmt besser, als suchwort  ;)

danke

von Jörg S. (joerg-s)


Lesenswert?

>Ich muss einen MSP430 F 2274 benutzen. Der mikrocontroller hat leider
>SPI und UART auf den gleichen pins.
Der MSP430F2274 hat 2 USCI Channels (A0, B0) die (natürlich) auf 
verschiedene Pins gehen. Somit kannst du SPI und UART gleichzeitig 
nutzen.

von Peter D. (peda)


Lesenswert?

Maria wrote:

> Die Hilfsfunktion PSet() setzt ein Port Pin high/low.
> PGet() fragt ein Port Pin ab.


Beim AVR-GCC geht das einfacher, da kann man direkt auf die IO-Pins per 
Bitvariablen zugreifen:
1
#include <avr\io.h>
2
3
4
struct bits {
5
  uint8_t b0:1;
6
  uint8_t b1:1;
7
  uint8_t b2:1;
8
  uint8_t b3:1;
9
  uint8_t b4:1;
10
  uint8_t b5:1;
11
  uint8_t b6:1;
12
  uint8_t b7:1;
13
} __attribute__((__packed__));
14
15
#define SBIT(port,pin) ((*(volatile struct bits*)&port).b##pin)
16
17
18
#define SPI_CLK         SBIT( PORTB, 0 )        // clock
19
#define SPI_CLK_DDR     SBIT( DDRB,  0 )
20
#define SPI_MOSI        SBIT( PORTB, 1 )        // data out
21
#define SPI_MOSI_DDR    SBIT( DDRB,  1 )
22
#define SPI_MISO_PIN    SBIT( PINB,  2 )        // data in
23
24
25
uint8_t shift_io( uint8_t b )   // send / receive byte
26
{
27
  uint8_t i;
28
29
  SPI_CLK_DDR = 1;              // set as output
30
  SPI_MOSI_DDR = 1;
31
32
  for( i = 8; i; i-- ){         // 8 bits
33
    SPI_MOSI = 0;
34
    if( b & 0x80 )              // high bit first
35
      SPI_MOSI = 1;
36
    b <<= 1;
37
    SPI_CLK = 1;
38
    if( SPI_MISO_PIN )
39
      b++;
40
    SPI_CLK = 0;
41
  }
42
  return b;
43
}

Sind dann riesige 26 Byte Code.


Peter

von max (Gast)


Lesenswert?

@jörg s.:
hier ein auszug aus data-sheet.

Pin 3.1-->USCI_B0 slave in/master out in SPI mode, SDA I2C data in I2C 
mode

Pin 3.2-->USCI_B0 slave out/master in in SPI mode, SCL I2C clock in I2C 
mode

Pin 3.3-->USCI_B0 clock input/output / USCI_A0 slave transmit enable

Pin 3.4-->USCI_A0 transmit data output inUARTmode, slave in/master out 
in SPImode

Pins 3.5-->USCI_A0 receive data input in UART mode, slave out/master in 
in SPI mode



verstehe ich da was falsch? USCI_B0 und USCI_A0 überschneiden sich soch 
am Pin 3.3, allerdings würde ja sonst usci_a0 und usci_b0 keinen sinn 
machen, hm...habe jetzt im user guide auch die bestätigung gefunden, 
dass es gehen müsste. danke

von max (Gast)


Lesenswert?

danke nochmal an alle, habe leider noch keine erfahrung mit 
mikrocontroller, deswegen übersehe ich wohl auch gerne sachen im 
datenblatt. werde mich bestimmt wieder melden, wenn es ans basteln geht 
:)

von Jörg S. (joerg-s)


Lesenswert?

>deswegen übersehe ich wohl auch gerne sachen im datenblatt.
Kann passieren. Das mit der Überschneindung auf P3.3 hatte ich auch 
nicht gesehen :) Aber bei UART + SPI sollte es gehen.

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.