Post Image
By Matt GaidicaOctober 5, 2020In Uncategorized

Forecasting Slow-wave Activity from EEG using FFT

Providing closed-loop, phase-locked neural stimulation often requires some degree of prediction or forecasting of the observed neural (i.e. EEG) data to account for software, hardware, or wireless transmission delays. Slow-wave activity (SWA; ~0.5-4 Hz) is a common target because of it characterizes non-rapid eye movement sleep (NREM; or ‘deep sleep’) and can be augmented or enhanced using transcranial, direct-current, or audio stimulation. There are a few notable approaches for detecting SWA:

  1. Deep sleep maintains learning efficiency of the human brain disturbed SWA when it crossd a -30μV threshold in a 2-second window using a Butterworth filter (0.5–2 Hz, stop-band<0.1 and >10 Hz, stopband attenuation 20 dB, passband attenuation 0.1 dB) using LabVIEW. A human maintained watch of the electromyogram (EMG) to ensure stimulation was not cued if the subject was awake/aroused.

  2. Auditory closed-loop stimulation of the sleep slow oscillation enhances memory identified the negative half-wave peak of slow oscillations (0.5–2Hz) in real-time using Spike2 software triggering stimulation (by default) at -80μV.

  3. The shaky ground truth of real-time phase estimation builds on Real-Time Brain Oscillation Detection and Phase-Locked Stimulation Using Autoregressive Spectral Estimation and Time-Series Forward Prediction to, in brief, forward-predict signal phase based on an optimized signal passband (see Phastimate on Github). These papers and the one below (4) are concerned with faster, theta (4–9Hz) EEG data.

  4. Continuous Phase Estimation for Phase-Locked Neural Stimulation Using an Autoregressive Model for Signal Prediction discusses the work in (3) and instead of using zero-phase filtering, applies a Butterworth filter in the forward-only direction. Phase prediction was performed using a Hilbert transform after applying an autoregressive model to fill in the predicted future filtered data. This method is wrapped into an OpenEphys module.

  5. A Fast EEG Forecasting Algorithm for Phase-Locked Transcranial Electrical Stimulation of the Human Brain is a compelling response to (3,4) that finds phase estimation can be as accurate using a simpler approach. This method is expanded on below.

I find the approach in (5) the easiest software approach to phase estimation in an embedded system using C code. It uses a simple and straight-forward IIR filter along with a fast Fourier transform (FFT) that both have precedence in C (see KISS FFT, FFTW, or CMSIS DSP). Below is a prototype of the system implemented in MATLAB, which follows the simplified procedure of (5, Figure 1).

Row 1: Simulated 2Hz oscillation (blue) filtered using an IIR elliptical filter (0-5–4Hz, red). Row 2: Single-sided power spectrum of filtered signal deteremined by FFT. Row 3: Phase of filtered signal deteremined by FFT. Row 4: Phase-shifted, forec…

Row 1: Simulated 2Hz oscillation (blue) filtered using an IIR elliptical filter (0-5–4Hz, red). Row 2: Single-sided power spectrum of filtered signal deteremined by FFT. Row 3: Phase of filtered signal deteremined by FFT.
Row 4: Phase-shifted, forecasted signal (black).

Note, the forecasted signal has undergone an additional phase shift due to the filter design, which needs to be corrected (not shown). I discuss how to empirically derive the filter delay in the Mathworks Answers forum, Practical use of Phase Delay for IIR Filter on Pure Sinusoid. In brief, the phase delay can be adequetely modeled by a polynomial function. With this information, the forecasted signal can be used along with known system delays to cue phase-precise neural stimulation.

Simulating with Real Data

The Predict brain deep sleep slow oscillation has a database of labelled slow-wave data which I am using for testing. Building on the previous section, a forecasted (and phase-corrected) signal is actually not the ultimate goal: the goal is to know when a slow-wave will approximate a particular phase to be used in delivering phase-precise stimulation.

The composite phase delay described above (the offset of the original signal corrected by the phase shift of the filter) applies to t=0, not the where the original signal ends. Therefore, the phase at the end of the original signal—which is simulating a data buffer—needs to be calculated. From that, a phase difference between a target phase and the current phase can be converted into samples or time.

Below, the top row shows the raw data (4s of SWA), the filtered data (0.25–4Hz elliptical filter), along with the forecasted and corrected sinusoids. In this case, the FFT identified a dominant rhythm of 0.63Hz, and when overlaid with a phase-corrected cosine (bottom row), this matches reality. I chose a target stimulation phase of -π (i.e. the negative deflection) and plotted the next three stimulation opportunities based on the end of the raw data.

Screen Shot 2020-10-06 at 6.05.39 PM.png

As noted at the end of the previous section, the only thing left to determine is if system/transmission delays would allow for stimulation to take place at one of the marked time points.


MATLAB Gist for first figure

fs = 125; % sampling frequency
t = 0 : 1/fs : 5; % time (0-5s)
fmod = 2; % Hz
x = sin((2*pi*fmod*t) + pi/2); % create test signal
% setup plot
rows = 4;
cols = 1;
close all
figure('position',[0 0 1000 700]);
set(gcf,'color','w');
subplot(rows,cols,1);
plot(t,x,'b'); % plot test signal
hold on;
xlabel('Time (s)');
ylabel('Amplitude');
ylim([-1 1]);
% setup filter
f1 = 0.5;
f2 = 4;
[A,B,C,D] = ellip(10,0.5,40,[f1/fs*2 f2/fs*2]);
sos = ss2sos(A,B,C,D);
y = sosfilt(sos,x); % filter test signal
plot(t,y,'r'); % plot filtered signal
legend({'original','filtered'},'location','northwest');
% setup FFT
L = numel(t);
nPad = 5;
n = (2^nextpow2(L)) * nPad; % force zero padding for interpolation
Y = fft(y,n); % remember, Y is complex
f = fs*(0:(n/2))/n;
P = abs(Y/n).^2; % power of FFT
A = angle(Y); % phase of FFT
grid
Psub = P(1:n/2+1); % make power one-sided
subplot(rows,cols,2);
plot(f,Psub,'r')
xlabel('Frequency (f)')
ylabel('|P(f)|^2')
xlim([0 10]);
grid
[v,k] = max(Psub); % find dominant frequency of filtered signal
freq = f(k);
title(sprintf('Dominant @ %1.2fHz',freq));
hold on;
plot(freq,Psub(k),'*');
Asub = A(1:n/2+1); % make phase one-sided
subplot(rows,cols,3);
plot(f,Asub,'r');
xlabel('Frequency (f)')
ylabel('Phase (rad)');
xlim([fmod-2 fmod+2]);
grid
phase = Asub(k); % use k (key) to identify dominant phase
title(sprintf('Phase @ %1.2fHz = %1.2frad',freq,phase));
hold on;
plot(freq,phase,'*');
subplot(rows,cols,4);
t_sig = max(t) + (0 : 1/fs : 1); % time, forecast 1 second
sig = sin((2*pi*freq*t_sig) + pi/2 + phase); % phase-shifted forecast
plot(t,x);
xlabel('Time (s)');
ylabel('Amplitude');
hold on;
plot(t,y,'r');
plot(t_sig,sig,'k');
legend({'original','filtered','forecasted'},'location','northwest');
grid
ylim([-1 1]);

Helical Electrodes for Electrophysiology in Wild Animals
Data Required to Detect and Forecast EEG Slow Waves