LabVIEWForum.de - FFT-Analyse von EEG-Daten via RS232

LabVIEWForum.de

Normale Version: FFT-Analyse von EEG-Daten via RS232
Du siehst gerade eine vereinfachte Darstellung unserer Inhalte. Normale Ansicht mit richtiger Formatierung.
Seiten: 1 2 3
Hallo,

ich habe mir nach den Anleitungen der OpenEEG-community ein EEG-Gerät gebaut.
Da ich mich in Labview einarbeiten möchte, versuche ich nun die Daten, welche das Gerät über die serielle
Schnittstelle an den Computer sendet, aufzubereiten und grafisch zu visualisieren.
Mein Ziel ist die Darstellung der gemessenen Gehirnpotentiale als:

1. Oszilloskopbild
2. FFT der Gehirnströme (4Hz bis 30Hz)
3. Balken, der sich auf und abbewegt, nach Ausfilterung einer bestimmten Frequenz durch einen Bandpass

Nun habe ich folgende Probleme:
-Störfrequenzen im FFT
-Bei Einfügen von neuen Filtern oder neuen Strukturen (z.B. Casestruktur)
schafft es Labview nicht mehr alle Daten von der RS232 abzuholen und verpasst einige Pakete.

Zur Erklärung:

Das EEG-Gerät sendet periodisch 17 Bytes an den PC.
Davon interessant sind:

Byte1/Byte2: Synchronisationsbytes (A5 5A)
Byte3:Versionsnummer
Byte4:Rahmennummer (Zähler)
Byte5:Kanal1 Low Byte
Byte6:Kanal1 High Byte
Byte7 - Byte16:enthalten von mir nicht genutzte Kanäle
Byte17:nicht relevant

Also nach jeweils 17 Bytes erhalte ich einen digitalen 10Bit-Messwert (8Bit und 2 Bit von Byte5 und Byte6 zusammengesetzt).

Das VI habe ich folgendermassen aufgebaut:
Über VISA lese ich kontinuierlich die Daten von der RS232 ein.
Diese werden in der Haupt-Whileschleife am Ausgang in ein Schieberegister geschoben und
erscheinen somit am Eingang der Whileschleife sequentiell nacheinander.
Dort warte ich dann auf die SyncBytes A5 5A (Startkennung).
Sobald diese aufgetreten sind, was die SyncBit LED signalisiert, wird die
Case-Struktur "True" und nimmt den 10Bit-breiten Messwert auf.
Dieser Messwert wird vorher aus Byte5 und Byte6 zusammengesetzt.
In der Case-Struktur verwende ich wieder ein Schieberegister.
Das Oszilloskop ist direkt am ersten Eingang des Schieberegisters angeschlossen und
scheint gut zu funktionieren.
Über "Build Array" fasse ich 64 Messwerte zusammen und übergebe sie "Build Waveform".
Danach erfolgt eine Filterung, um z.B. das 50Hz Brummen auszufiltern und dann folgt das FFT-Modul.

Selbst wenn ich kein Signal vom EEG-Gerät bekomme (Messelektroden auf Masse gelegt), erscheinen im FFT-Diagramm Frequenzen.
In meinem Fall bei 0Hz, 4Hz, 8Hz.

[attachment=32625]

Wenn ich ein 20Hz Sinussignal vom EEG einspeise,
bekomme ich 16Hz, 20Hz, 24Hz angezeigt, also die Summe und Differenz dazu.

[attachment=32622]

Bei Einspeisung von 30Hz fehlt diese Frequenz komplett und es erscheinen 28Hz und 32Hz

[attachment=32623]

Ich gehe davon aus, dass, wenn ich die 4 Hz beseitigen könnte,
auch die Summen- und Differenzenfrequenzen verschwinden werden, oder?
Woher könnten die 0Hz und 4 Hz Störfrequenzen kommen?
Nutze ich zu wenig Daten für das "Build Array" oder ist mein Ansatz falsch?

Ich habe es noch nicht geschafft, das VI mit einer Textdatei anstatt Daten von der RS232 zu speisen,
damit man das Verhalten unabhängig von dem EEG-Gerät ausprobieren kann.
Logisch erschien mir, anstelle des VISA Read Elementes ein Datei Read Element zu setzen, aber
ohne Daten von der RS232 läuft die While-Schleife nicht an. Oder sollte ich besser das komplette VI kopieren,
von allen VISA-Elementen befreien und dann mittels Case-Struktur, mal das Datei-Lese und mal das VISA Blockdiagramm aufrufen?
Vielleicht habt Ihr auch da einen Tip.

Da dies mein allererstes VI ist vermute ich auch noch generelle Fehler,
welche die Performance so verschlechtern, dass es zu den verlorenen Paketen kommt.
Das merke ich vor allem daran, dass nach Ausschalten des EEG-Gerätes Labview noch 2-3sec
weiter Daten im Oszilloskop anzeigt, also noch Daten irgendwo im Puffer liegen.

Damit das VI übersichtlich bleibt, habe ich es auf das Wesentliche reduziert und z.B. die
Paketfehlererkennung rausgenommen.

Danke für Eure Hilfe.
[attachment=32624]
Hallo Frank,

das hier meinst nicht ernst, oder?
[attachment=32626]
Also auf meinem Laptop sehe ich nicht mal annähernd, was das VI so machen soll. Wie soll man da helfen?

Außerdem gibt es sicherlich bessere Methoden, um die geschätzt 50 letzten Messwerte in einem Array zu halten!
Idee: ein Array mit 50 Werten initialisieren und dann mit ReplaceArraySubset jeweils einen Wert ersetzen. Schwups braucht man nur noch ein einfaches Schieberegister...
(05.03.2011 18:44 )GerdW schrieb: [ -> ]das hier meinst nicht ernst, oder?
Also auf meinem Laptop sehe ich nicht mal annähernd, was das VI so machen soll. Wie soll man da helfen?

Ursprünglich waren es nur 8 Leitungen.
Diese habe ich dann auf 64 hochgesetzt um zu schauen, ob sich die Qualität des Spektrums verbessert, da ich gelesen hatte, dass
die FFT umso besser wird, je mehr Werte im Array stehen.
Wie gesagt, das ist mein erstes VI und ich hatte gehofft, dass mir hier konstruktiv auf die Sprünge geholfen wird.


(05.03.2011 18:44 )GerdW schrieb: [ -> ]Außerdem gibt es sicherlich bessere Methoden, um die geschätzt 50 letzten Messwerte in einem Array zu halten!
Idee: ein Array mit 50 Werten initialisieren und dann mit ReplaceArraySubset jeweils einen Wert ersetzen. Schwups braucht man nur noch ein einfaches Schieberegister...

Danke für die Idee, das werde ich umsetzen und dann posten.
(05.03.2011 18:44 )GerdW schrieb: [ -> ]Außerdem gibt es sicherlich bessere Methoden, um die geschätzt 50 letzten Messwerte in einem Array zu halten!
Idee: ein Array mit 50 Werten initialisieren und dann mit ReplaceArraySubset jeweils einen Wert ersetzen. Schwups braucht man nur noch ein einfaches Schieberegister...

Nach nunmehr 8 Stunden muss ich passen.
Ich bekomme es nicht in mein VI integriert.
Die vielen Einzel VI's, die ich heute gemacht hatte, liefen nicht so, wie sie sollten und liessen sich nicht in mein VI integrieren.
Brauche ich zwangsläufig den Rotate1DArray Baustein?
Bin ich beim Konzept Ringpuffer richtig?
Hallo Frank,

man muss nicht "zwangsläufig" ein Rotate verwenden, aber es vereinfacht das Vorgehen ungemein:
[attachment=32636]
Ja, das Konzept "Ringbuffer" trifft es ganz gut...

Zitat:Ursprünglich waren es nur 8 Leitungen. Diese habe ich dann auf 64 hochgesetzt um zu schauen, ob sich die Qualität des Spektrums verbessert, da ich gelesen hatte, dass die FFT umso besser wird, je mehr Werte im Array stehen.
Willst du damit sagen, dass du aus nur 8 Werten eine FFT bildest? Echt?

Zitat:ich hatte gehofft, dass mir hier konstruktiv auf die Sprünge geholfen wird.
Was verstehst du unter "konstruktiv"? Gut gemeinte Ratschläge & Denkanstöße? Hinweise auf Fehlerquellen?
Oder erwartest du etwa fertige Lösungen, die dir kostenlos programmiert werden?
(06.03.2011 18:29 )GerdW schrieb: [ -> ]Willst du damit sagen, dass du aus nur 8 Werten eine FFT bildest? Echt?

Anfangs ja, da ich mir dachte, dass der FFT-Baustein die Daten intern erst puffert und dann aus z.B. 100*8 Werten seine Analyse macht.
Da das Ergebnis aber schlecht war (Störfrequenzen), habe ich schliesslich auf 64 Leitungen erweitert.


Zitat:Was verstehst du unter "konstruktiv"? Gut gemeinte Ratschläge & Denkanstöße? Hinweise auf Fehlerquellen?
Oder erwartest du etwa fertige Lösungen, die dir kostenlos programmiert werden?

Ups, wie falsch manchmal Text rüberkommen kann.
Das war nicht als Kritik oder Vorwurf gemeint.
Ich wollte damit sagen, dass ich erwartet hatte, dass an meinem Konzept mit den 64 Leitungen etwas verbesserungsfähig ist.
Der Satz war so gemeint:
"Ich hatte innerlich gehofft, dass die Lösung mit den 64 Leitungen falsch ist, und jemand so was sagt wie, "das ist Blödsinn".
denn das würde bedeuten dass das VI dann kleiner und überschaubarer gemacht werden kann."

Ich versuche mich seit 3 Monaten an diesem VI, habe viel wieder verworfen, umgebaut und wieder gelöscht, weil die Performance bei zu vielen Elementen zusammenbrach.
Eigentlich wollte ich es allein schaffen, aber jetzt bin ich an einem Punkt, wo ich Hilfe brauche, da für mein Problem unter anderem theoretisches Verständnis von Signaltheorie notwendig ist und das, was ich mir angelesen und verstanden habe, bei weitem nicht ausreicht.
Ausserdem habe ich Schwierigkeiten, wenn mehrere Strukturen ineinanderverschachtelt sind und dann Daten nach rein und raus gereicht werden müssen.
Für mich als Anfänger ist das Projekt ne Nummer zu gross.
Ich will es aber trotzdem versuchen, da man am besten an konkreten Projekten lernt.

Zitat:Ratschläge & Denkanstöße? Hinweise auf Fehlerquellen?
Genau das suche ich.
Fertige Lösungen natürlich nicht.
Ich hoffe nicht, so rübergekommen zu sein.

Danke bis hier, ich werde mich nächste Woche am Ringpuffer versuchen.
Solo lief er, aber in mein VI habe ich ihn nicht integriert bekommen.
Ich habe jetzt mal alles, was nicht unbedingt notwendig war, inclusive der FFT, bei welcher mein eigentliches Problem auftrat, rausgenommen
und versuche nochmal die Funktion zu beschreiben:

VISA gibt die Daten an die äussere Whileschleife.
VISA Read schickt den eingelesenen String an das rechte Schieberegister.
Nun werden die Daten solange an die linken Schieberegisterzellen durchgereicht,
bis die Startkennung A55A an den beiden unteren Schieberegisterzellen anliegt.
In diesem Moment befinden sich die Daten von dem zu messenden Kanal an den beiden oberen Schieberegisterzellen,
werden zusammengesetzt und mittels Type Cast in einen 16Bit Integer Wert umgewandelt.
An der Stelle, wo der schwarze Pfeil hinzeigt, gehen die Daten dann in die Casestruktur, allerdings immer nur dann,
wenn die Synchronisationsbytes A5,5A die Casestruktur auf True setzen.

Die innere For-Schleife habe ich versuchsweise mit 1 Schleifendurchlauf eingefügt, damit ich die Schieberegister
darin platzieren kann.

Das Problem bei dieser Anordnung ist aber nun, dass das Element <Initialize Array> bei jedem erneuten Truesetzen der Casestruktur immer wieder
das Array auf 0 setzt.
Das Array am rechten Schieberegister wird nie nach links durchgereicht, bzw anscheinend immer wieder überschrieben.
Verschiedene andere Variationen funktionierten auch nicht.
Was mache ich dabei falsch?
(Das Array ist erst mal nur zum Testen so klein.)

Das Snippet kann ich nicht in mein Labview hineinziehen.
Ist es mit LV 2010 erstellt?
Wie kann ich mir anzeigen lassen, mit welcher LV-Version ein Snippet erstellt wurde?


[attachment=32684]
Hallo Frank,

Zitat:Das Array am rechten Schieberegister wird nie nach links durchgereicht, bzw anscheinend immer wieder überschrieben.
... Was mache ich dabei falsch?
Nun ja, du initialisierst das Shiftregister der FOR-Loop immer wieder neu. Was erwartest du also für Werte im Register, wenn die Schleife erneut aufgerufen wird?
Du solltest das Schieberegister nur genau einmal initialisieren...

Zitat:Die innere For-Schleife habe ich versuchsweise mit 1 Schleifendurchlauf eingefügt, damit ich die Schieberegister darin platzieren kann.
Dein Fehler: Du hast doch schon eine While-Schleife, um dort Shiftregister zu erzeugen...

Siehe Anhang (Lv09_img2)

P.S.: Dein Umgang mit der seriellen Schnittstelle ist "suboptimal". Lese doch gleich wortweise oder komplette Messages a 6 Byte ein. Du kannst dein "Sync"-Wort auch als Terminierungszeichen verwenden und bräuchtest dann nicht erst noch der korrekten Zeichenzuordnung suchen... Es gibt so viele Threads, die das Thema Serielle Schnittstelle behandeln, einfach mal suchen und lesen...
Hallo Gerd,

Toll, es funktioniert besser und ist viel aufgeräumter.
Ich habe die Anzahl der Arrayelemente auf 1024 gesetzt und jetzt sind die Störfrequenzen fast weg.
Sie liegen nun bei < 1 Hz und da stören sie die Nutzsignale nicht mehr.
Die Idee mit dem Terminierungszeichen habe ich eingebaut und lese alle 17 Bytes aufeinmal ein.
Jetzt sind noch ein paar Erweiterungen geplant.
Wenn dann alles läuft würde ich mich freuen, wenn Du nochmal drüberschauen kannst.

Zitat:Nun ja, du initialisierst das Shiftregister der FOR-Loop immer wieder neu. Was erwartest du also für Werte im Register, wenn die Schleife erneut aufgerufen wird?
Du solltest das Schieberegister nur genau einmal initialisieren...
Ich dachte, LV macht eine Initialisierung am Schieberegister nur genau einmal, egal in welchen Strukturen befindlich.
Das hatte ich in einem englischen Beitrag gelesen und wahrscheinlich missverstanden oder falsch übersetzt. So macht es jetzt auf alle Fälle mehr Sinn.

Zitat:Dein Fehler: Du hast doch schon eine While-Schleife, um dort Shiftregister zu erzeugen...
Ich hatte fast genau Deine Version auch mal probiert und wieder verworfen, weil sie nicht funktionierte, da hab ich wohl etwas Anderes falsch gemacht.

[attachment=32737]

[attachment=32738][attachment=32736]
Hallo Frank,

ein Fehler von mir: SerialInit versteht nur U8-Werte als Terminierungszeichen, du kannst also nicht "A55A" dort vorgeben. (Es wird wohl stattdessen "FF" verwendet, da der Wert angepasst wird.) Vielleicht kommen deine PacketErrors dort her...

Zitat:Ich habe die Anzahl der Arrayelemente auf 1024 gesetzt und jetzt sind die Störfrequenzen fast weg. Sie liegen nun bei < 1 Hz und da stören sie die Nutzsignale nicht mehr.
Ja, das hat irgendwas mit den mathematischen Grundlagen der FFT zu tun... Smile
Seiten: 1 2 3
Referenz-URLs