Forum: Compiler & IDEs Overflow-Interrupt, Globalvariable


Announcement: there is an English version of this forum on EmbDev.net. Posts you create there will be displayed on Mikrocontroller.net and EmbDev.net.
von Peter R. (dk7ih)


Angehängte Dateien:

Lesenswert?

Moin!

In einem größeren Projekt stellt sich mir gerade ein Problem, das ich 
mal auf den essentiellen Code reduziert habe.

Controller: ATMEGA8
Compiler: GNU C-Compiler für AVR unter Linux

Die einfache Aufgabe: Nach Start des Programms soll eine bestimmte Zeit 
später eine Funktion ausgelöst werden. Simuliert wird dies durch eine 
LED (Led 2), die dann angehen soll, wenn z. B. die Overflowroutine 100x 
durchlaufen wurde. In der ISR ist eine weitere LED (LED 1) angesteuert, 
die jeweils umschaltet, wenn der Interrupt ausgelöst wird. Alles 
funktioniert soweit wie gewünscht.

Nun aber der Knackpunkt: Eine Variable s (global deklariert) zählt die 
Anzahl der Aufrufe der ISR seit Programmstart. Sie wird auf 0 
initialisiert und müsste nun in der ISR mehrfach pro Sekunde erhöht 
werden. Der Wert dieser Variable wird aber in main() immer mit 0 
gelesen, LED 2 geht niemals an.

Komisch, oder? Vielleicht hat jemand eine Idee...

Viele Grüße

Peter

von Georg M. (g_m)


Lesenswert?

Peter R. schrieb:
> Eine Variable s (global deklariert)

volatile?

von Peter R. (dk7ih)


Lesenswert?

Georg M. schrieb:
> Peter R. schrieb:
>> Eine Variable s (global deklariert)
>
> volatile?

Das war's! Danke! :-)

von Adam P. (adamap)


Lesenswert?

Ich würde den Zugriff noch mit Atomic umsetzen, oder was sagt ihr dazu?
1
long get_ticks(void)
2
{
3
  long ticks;
4
  
5
  ATOMIC_BLOCK(ATOMIC_RESTORESTATE)
6
  {
7
    ticks = s;
8
  }
9
  
10
  return ticks;
11
}

Muss ja keine extra Funktion sein, mir gings nur um die 
Veranschaulichung.

: Bearbeitet durch User
von Dirk (devnull)


Lesenswert?

Adam P. schrieb:
> Ich würde den Zugriff noch mit Atomic umsetzen, oder was sagt ihr dazu?

Da die Variable vom Typ long ist, ist der Zugriff über Atomic angezeigt.

Allerdings könnte der TO, sofern er wirklich nur bis 100 zählen will,
einen uint8_t benutzen und sich Atomic sparen. Volatile ist auf jeden
Fall notwendig.

Grüße, Dirk

von Apollo M. (Firma: @home) (majortom)


Lesenswert?

Dirk schrieb:
> Volatile ist auf jeden Fall notwendig.

Nicht unbedingt, die Variable könnte direkt auf eine Register-Adresse 
gelinkt werden!

von Johann L. (gjlayde) Benutzerseite


Lesenswert?

Adam P. schrieb:
> Ich würde den Zugriff noch mit Atomic umsetzen, oder was sagt ihr dazu?
>
>
1
> long get_ticks(void)
2
> {
3
>   long ticks;
4
> 
5
>   ATOMIC_BLOCK(ATOMIC_RESTORESTATE)
6
>   {
7
>     ticks = s;
8
>   }
9
> 
10
>   return ticks;
11
> }
12
>

Kann abgekürzt werden zu:
1
long get_ticks (void)
2
{
3
  ATOMIC_BLOCK (ATOMIC_RESTORESTATE)
4
  {
5
    return s;
6
  }
7
}

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.