- Labornetzteil AliExpress         
Ergebnis 1 bis 10 von 59

Thema: Interrupt-Abfrage >>> Routine vereinfachen

Hybrid-Darstellung

Vorheriger Beitrag Vorheriger Beitrag   Nächster Beitrag Nächster Beitrag
  1. #1
    Erfahrener Benutzer Robotik Einstein
    Registriert seit
    11.12.2007
    Ort
    weit weg von nahe Bonn
    Alter
    40
    Beiträge
    3.416
    Code:
    volatile uint8_t RegisterA = PINA;
    
    ISR (PCINT0_vect)					// Interrupt, ABFRAGE, aus Bank0, wird ausgelöst
    {
            uint8_t resetVal = Reset_Status();
    	if(RegisterA & (1<<PINA2)) != 0)		// auf Zustandveränderung PA2/RESET abfragen
    	{
    		switch (resetVal)		// "Reset" entprellen // return 0=low, 1=high
    		{
    			case 0: RegisterA=PINA;	Reset_off();	break;  // Reset wird ausgeführt
    			case 1: RegisterA=PINA;	Reset_on();	break;  // Reset wird nicht ausgeführt
    		}
    	}
    	if(RegisterA & (1<<PINA3)) != 0)		// auf Zustandveränderung PA3/RESET abfragen
    	{
    		switch (resetVal)		// "Reset" entprellen // return 0=low, 1=high
    		{
    			case 0: RegisterA=PINA;	Reset_off();	break;  // Reset wird ausgeführt
    			case 1: RegisterA=PINA;	Reset_on();	break;  // Reset wird nicht ausgeführt
    		}
    	}
     
          // solch eine Abfrage noch für PA4, PA5 hintereinander gehangen
    }
    Es gibt 10 Sorten von Menschen: Die einen können binär zählen, die anderen
    nicht.

  2. #2
    Erfahrener Benutzer Fleißiges Mitglied
    Registriert seit
    07.06.2019
    Beiträge
    148
    Ahhh - Missverständnis!

    Reset_Status() testet nur die "Reset"-Taste an PA2,
    PA3 heisst Start_Status(), usw.
    Alles eigenständige Funktionen, in dehnen entprellt und der letzte Zustand ausgegeben wird.

    Dann scheint mein Weg doch nicht so schlecht.

    Nur bei der Auswertung bin ich mir noch nicht sicher.
    Code:
    if(RegisterA & (1<<PINA2)) != 0)
    Die Zeile sollte die "alte, gespeicherte" Pinbelegung PA2 (RegisterA, vor dem Interrupt) mit dem "neuen, aktuellen" Zustand PA2 vergleichen um heraus zu finden, ob PA2 den Interrupt ausgelöst hat.
    1- Ist eine Veränderung zu erkennen, wird PA2 entprellt, danach Zustand 0/1 abfragen und weiter bearbeiten.
    2 -Wenn nicht, wird PA3 usw. abgefragt.

    - - - Aktualisiert - - -

    Müsste der alt/neu-Vergleich nicht eher folgend aussehen?
    Code:
    volatile uint8_t RegA_alt = PINA;
    
    ISR (PCINT0_vect)
    {
      uint8_t RegA_neu = PINA;
      uint8_t RegA_diff = RegA_alt ^ RegA_neu;  // hier sollte in jedem Reg.eintrag eine "1" stehen, wenn sich zum "alten" (Vor-Interrupt-Auslösung) eine Veränderung ergeben hat
      
      If((RegA_diff & (1<<PINA2)) != 1)
      {
        //wenn Ja, mach was
      {
    
      If((RegA_diff & (1<<PINA3)) != 1)
      {
        //wenn Ja, mach was
      }
    
      //usw. mit PA4, PA5
    __________________________________________________ _
    | Sprache: C | Teensy 3.2 | Arduino 2.x | Status: EwigerAnfaenger |

  3. #3
    Erfahrener Benutzer Robotik Einstein
    Registriert seit
    11.12.2007
    Ort
    weit weg von nahe Bonn
    Alter
    40
    Beiträge
    3.416
    entweder machst du deine Variablen noch zusätzlich static oder legst sie global an, sonst wirst du nie einen unterschied sehen weil sie immer neu initialisiert werden

    ob das XOR da nicht lustige seiteneffekte auslöst würde ich gern prüfen, komme heute aber definitiv nicht mehr dazu mich da etwas tiefer reinzudenken oder zu antworten
    Es gibt 10 Sorten von Menschen: Die einen können binär zählen, die anderen
    nicht.

  4. #4
    Erfahrener Benutzer Fleißiges Mitglied
    Registriert seit
    07.06.2019
    Beiträge
    148
    Seiteneffekte?

    Rein logisch müsste der XOR alle Unterschiede mit einer 1 markieren.
    Baue mir gerade ein Testprogramm auf.
    __________________________________________________ _
    | Sprache: C | Teensy 3.2 | Arduino 2.x | Status: EwigerAnfaenger |

  5. #5
    Erfahrener Benutzer Robotik Einstein
    Registriert seit
    18.03.2018
    Beiträge
    2.674
    Blog-Einträge
    1
    so kann man es auch beschreiben.
    Rein logisch müsste der XOR alle Unterschiede mit einer 1 markieren.

    0+0 = 0, 0+1 = 1, 1+0 = 1, 1+1 = 0

  6. #6
    Erfahrener Benutzer Roboter Genie
    Registriert seit
    07.04.2015
    Beiträge
    911
    Geht das jetzt nur um Tastenabfragen an den Inputs?
    Dann würde ich NICHT den PortChange-Interrupt verwenden, sondern einen Timer im ms-Bereich. Der alleine bewerkstelligt schon die Entprellung.
    Der Code gestaltet sich vielleicht mithilfe einer Loop etwas schöner, wobei ich auch gleich das Allgemeingültige vom Anwendungsspezifischen trennen würde:
    Code:
    uint8_t prevPINA;
    
    void InitKeys()
    {
        prevPINA = PINA;
    }
    
    ISR (TIM0_OVF_vect)
    {
      uint8_t actPINA = PINA; //buffer actual state
      uint8_t chPINA = prevPINA ^ actPINA;   //get changes with XOR
      uint8_t setPINA = chPINA & actPINA;    // get new set pins
      uint8_t clPINA = chPINA & ~actPINA;    // get new cleared pins
    
      prevPINA = actPINA; //save actual PINA in prevPINA for next ISR call 
        
      for (uint8_t i = 0; i<8; i++)
      {
        if ((setPINA >>i) & 0x01)
          KeyDown(i);
     
        if ((clPINA >>i) & 0x01)
          KeyUp(i);
      }
    }
    InitKeys wird einmalig nach dem Controllerreset aufgerufen, die ISR wäre hier z.B. der TimerOverflow.

    Das Applikationsspezifische:
    Code:
    #define KEY_ENTER 0
    #define KEY_ESC 1
    #define KEY_UP 6
    #define KEY_DOWN 7
    
    void KeyDown(uint8_t key)
    {
      switch (key)
      {
        case KEY_ENTER:
            //TODO
            break;
    
        case KEY_ESC:
            //TODO
            break;
    
    
      }
    
    
    }
    
    
    void KeyUp(uint8_t key)
    {
        //Das gleiche in grün
    }
    Ob man nun alle Pins mit Tasten belegt hat oder nur einige, ist dem allgemeinen Teil egal. Das wird erst durch die cases in den Switches entschieden.

    (nicht kompiliert, daher ohne Gewähr! Aber mal so als Idee)

  7. #7
    Erfahrener Benutzer Robotik Einstein
    Registriert seit
    11.12.2007
    Ort
    weit weg von nahe Bonn
    Alter
    40
    Beiträge
    3.416
    Als ich die Antwort geschreiben hatte hate ich wohl ein Brett vor dem Kopf, ich dachte du XORst in die gleiche Variable zurück und konnte mir irgendwie keinen Reim drauf machen wie das funktionieren sollte ^^
    Es gibt 10 Sorten von Menschen: Die einen können binär zählen, die anderen
    nicht.

  8. #8
    Erfahrener Benutzer Fleißiges Mitglied
    Registriert seit
    07.06.2019
    Beiträge
    148
    Dankefür eure Ausführungen!

    Zitat Zitat von Holomino Beitrag anzeigen
    Geht das jetzt nur um Tastenabfragen an den Inputs?
    Dann würde ich NICHT den PortChange-Interrupt verwenden, sondern einen Timer im ms-Bereich. Der alleine bewerkstelligt schon die Entprellung.
    Wo siehst du den Vorteil gegenüber einem PortChange-Interrupt ?
    Der Timer-I. würde alle ca.10ms aktiv werden, obwohl Sek, Min oder Std nichts passiert.

    Zitat Zitat von Holomino Beitrag anzeigen
    Code:
      uint8_t setPINA = chPINA & actPINA;    // get new set pins
      uint8_t clPINA = chPINA & ~actPINA;    // get new cleared pins
    
      prevPINA = actPINA; //save actual PINA in prevPINA for next ISR call
    ...warum dieser Aufwand?
    In deinem chPINA stehen doch alle Pin-Änderungen. Danach müssten die entsprechenden Bits PA2-PA5 auf 1 ausgewertet werden. Habe ich hier irgend etwas übersehen?
    Nach jeder chPINA-Auswertung kommt dann sofort eine erneute Aktualisierung+Speicherung in einer volatile-Variable.
    Code:
    prevPINA = PINA;
    __________________________________________________ _
    | Sprache: C | Teensy 3.2 | Arduino 2.x | Status: EwigerAnfaenger |

Ähnliche Themen

  1. [ERLEDIGT] Interrupt Routine
    Von Saturas077 im Forum Assembler-Programmierung
    Antworten: 8
    Letzter Beitrag: 23.04.2014, 12:46
  2. Codebeispiel für Lesen von RC5 Code mit Interrupt-Routine
    Von -tomas- im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 19
    Letzter Beitrag: 25.05.2011, 12:54
  3. Interrupt Routine
    Von luvat im Forum Schaltungen und Boards der Projektseite Mikrocontroller-Elektronik.de
    Antworten: 4
    Letzter Beitrag: 16.03.2008, 20:54
  4. Interrupt in ISR-Routine freigeben
    Von dj5am im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 5
    Letzter Beitrag: 10.08.2007, 08:44
  5. uart interrupt routine
    Von Computerkora im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 4
    Letzter Beitrag: 25.11.2006, 13:45

Berechtigungen

  • Neue Themen erstellen: Nein
  • Themen beantworten: Nein
  • Anhänge hochladen: Nein
  • Beiträge bearbeiten: Nein
  •  

Solar Speicher und Akkus Tests