Forum: Mikrocontroller und Digitale Elektronik Eingangsschieberegister 74HCT165 in C


von Tom (Gast)


Lesenswert?

Hallo,

bin ja noch relativer Neuling aber arbeite mich grade hoch.

Nur meine 74HCT165 bringen mir Werte, die ich nicht wirklich 
nachvollziehen kann (wenn ich mehrfach nacheinander auslese bleiben die 
16bit (2 stück kaskadiert) aber gleich, also kein Zufallsgenerator ;) )


Folgenden Code lass ich ablaufen, irgendwo muss da wohl ein Fehler sein 
aber ich komm auch nach mehreren Stunden nicht drauf wo...
1
// zur leichteren Behandlung erstmal Makros definieren
2
// DPORT ist der Port, an dem die Daten seriell gelesen werden (sollen)
3
// CPORT clock
4
// RPORT parallel Load
5
6
  #define __SRI_DPORT__ PINC
7
  #define __SRI_DDDR__  DDRC
8
  #define __SRI_CPORT__ PORTD
9
  #define __SRI_CDDR__  DDRD
10
  #define __SRI_RPORT__ PORTD
11
  #define __SRI_RDDR__  DDRD
12
       
13
  #define __SRI_DPIN__ 0
14
  #define __SRI_CPIN__ 3
15
  #define __SRI_RPIN__ 4
16
17
// Nur DPORT ist Eingang, die anderen beiden Ausgaenge
18
  __SRI_DDDR__ &= ~(1 << __SRI_DPIN__);
19
  __SRI_CDDR__ |= (1 << __SRI_CPIN__);
20
  __SRI_RDDR__ |= (1 << __SRI_RPIN__);
21
22
// Die routine zum einlesen (die 16bit werden in eine uint32_t variable
23
// gelesen (sr_Input)
24
25
void readSR_In(void) {
26
  int i;
27
28
  /* CLK auf High, dann PL LOW HIGH*/
29
  __SRI_CPORT__ |= (1 << __SRI_CPIN__);
30
  __SRI_RPORT__ &= ~(1 << __SRI_RPIN__);
31
  __SRI_RPORT__ |= (1 << __SRI_RPIN__);
32
33
  for (i=0; i < MAX_SRP_INPUT; i++) {
34
35
                // Bit pruefen
36
    if ((__SRI_DPORT__ & (1 << __SRI_DPIN__))) {
37
      sr_Input |= (1UL << i);
38
    } else {
39
      sr_Input &= ~(1UL << i);
40
    }
41
42
                //clock
43
    __SRI_CPORT__ &= ~(1 << __SRI_CPIN__);
44
    __SRI_CPORT__ |= (1 << __SRI_CPIN__);
45
  }
46
}


Das sollte doch so funktionieren :-(
Wenn niemand beim drüberschauen nen fatalen fehler findet muss ich als 
nächstes auf der hardware schaun. IC gebraten?!


Vielen vielen Dank!

von Matthias L. (Gast)


Lesenswert?

-> Code ist unvollständig,
-> Schaltplan hinzufügen,
-> Wieso kein Hardware SPI?

von Peter D. (peda)


Lesenswert?

Tom wrote:
> bin ja noch relativer Neuling aber arbeite mich grade hoch.

Dann solltest Du gleich versuchen, nicht unnötig Ressourcen zu 
verschwenden, z.B.:

- variables Schieben ist teuer
- unnötig große Datentypen sind teuer (Schleifenzähler 8 Bit)
- globale Variablen sind teuer (lokale sind oft in Registern).

Und auch bessere Lesbarkeit kann nicht schaden, z.B. durch Bitvariablen.

Hier mal ein Beispiel, wie ich es machen würde:
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_MISO_PIN    SBIT( PINB,  2 )
21
#define HC165_LD        SBIT( PORTB, 3 )
22
#define HC165_LD_DDR    SBIT( DDRB,  3 )
23
24
25
uint16_t read_2xhc165( void )
26
{
27
  uint8_t i;
28
  uint16_t b = 0;
29
30
  SPI_CLK_DDR = 1;
31
  HC165_LD_DDR = 1;
32
  HC165_LD = 0;
33
  HC165_LD = 1;
34
35
  for( i = 16; i; i-- ){
36
    b <<= 1;
37
    if( SPI_MISO_PIN )
38
      b++;
39
    SPI_CLK = 0;
40
    SPI_CLK = 1;
41
  }
42
  return b;
43
}

Bringt eine Codeeinsparung von 80% (162/32Byte)

Prinzipiell sollte Dein Code es aber tun, allerdings liest er gespiegelt 
aus.


Peter

P.S.:
Und dran denken, alle Eingänge des HC165 müssen beschaltet werden (0V 
oder 5V).
"Offen" ist kein gültiger Logikpegel.

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.