-
Notifications
You must be signed in to change notification settings - Fork 14
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
2nd (actually nth) order clock offset correction? #16
Comments
Please make sure this GitHub issue in pyxdf in on your radar when thinking about this. As Matthew mentioned in that thread, he prefers a rolling window linear fit. I kind of like the rolling window linear fit idea because it's closer to what liblsl does if someone enables online clock sync with dejittering. Either method is a major improvement, and there's no reason the xdf importer can't do both:
That's interesting. I wouldn't think a 4 msec shift can affect ERSP. What's the application? I understand how this would affect gamma and phase alignment but that's not normally in the "ERSP" bucket. I agree the current implementation should be improved; I'm just looking to learn about other high-temporal-precision applications. |
Cool, I wasn't aware of that other issue. I may have some time/incentive to implement this option in Matlab. I will aim for the rolling window if that is the preferred interpolation. It could be that 4ms isn't a show stopper with ERSPs. I am not an expert in EEG analysis. I do know that even a few ms will attenuate your ERPs. It depends, of course, on how many trials you have etc. Also, setup time and experiment time. At SCCN, a tolerance of 1ms was always the mantra so I just assumed that it was important for ERSP as well---but maybe it was just an arbitrary figure, like 10000 steps per day. Then again, if you are timelocking ERSPs to things like people jogging on a treadmill or turning their head, which will probably require time warping to do the averaging, it seems to me that you want to be as accurate as possible. |
Hey, I agree that offering a sliding window (or adaptive RLS) fit would be a somewhat neater solution for a non-linear fit than trying to do a polynomial fit, mainly since it'd be the same that LSL does online (which we'd indeed want to have as a choice for offline simulation in any case), and more alternatives beyond that would probably mainly serve to create confusion (there'd be a number of ways to do a polynomial fit, e.g., Legendre polynomials or a series of band-limited sinusoids etc, and if the fits are supposed to be robust to outliers that'd add even more complexity and uncertainty as to whether the fit actually worked or got derailed somehow). A note re the 4ms above -- that should be without the first-order correction (but correct me if I'm wrong, because otherwise there shouldn't be a linear / slope component in there), so the actual residual error between linear and a hypothetical non-linear fit should be more like 1ms at the point of max deviation. As for the RLS time constant, the one that LSL uses had been tuned to be a more or less optimal tradeoff between the achievable precision (sub-ms), which requires a longer integration window, and the ability to track more or less rapid rate changes (which requires a shorter window, and which we reasoned are mostly driven by device/ambient temperature changes that play out over multi-minute time scales). I recall having written something on that but not sure if it was on github or in some code comments. |
I'm working with a user who happens to have recordings where the outlet was from a PC with a pretty bad clock (drifts 1.5 s / hr and highly variable). They are using pyxdf and the _robust_fit linear regression is doing its best but I think they'd be better off with an adaptive fit.
@chkothe , I couldn't find a sliding window for clock offset in the liblsl source code. All that I found was that online clock correction uses the "best" offset from the last probe burst (see The dejitterer updates smoothly, and for continuous streams this probably compensates for outlier offset measurements, but only if enabled and it doesn't apply to irregular streams. Should the online clock offset estimation also have some smoothing? |
Interesting idea, yeah. One could have an option in the config for doing online smoothing for particularly bad clocks. For recording of course one would want to have raw offsets. Theoretically one could use the same RLS code for that, but getting the time constant right can be a bit crafty, especially considering that those offsets would normally be at much lower sampling rate (0.2 hz). One could up the offset sampling rate maybe to 1hz and do the math to see what would be a good default time constant, or alternatively look into sliding window estimation. |
Has anyone ever evaluated the accuracy of load_xdf's drift correction algorithm? It is a non-standard algorithm (i.e. not part of Matlab) for fitting a first order curve to correct for clock drift between sending and receiving devices
https://github.com/xdf-modules/xdf-Matlab/blob/master/load_xdf.m#L505-L516
xdf-Matlab/load_xdf.m
Lines 773 to 815 in 0cdf054
The reason I ask is that it has come to my attention that 1st order curves may not be appropriate for typical drift. For example, Here is the difference between LSL Markers and a screen flip signal recorded via an EEG device.
The total marker/EEG drift changes from about 27 to 23 ms over the course of several hours. This is definitely enough to muck up ERP/ERSP analysis. Such second order curves are pretty typical but an nth order curve is probably what is best.
If I were to rewrite this, I would use polyfit and do away with the ADMM method using the Huber loss function as minimization criteria but maybe there is a compelling reason for using such a robust tool. I confess that the main reason I would choose polyfit is that I don't fully understand how to adapt the code to return an nth order set of mapping coefficients. scipy has analogs to polyfit but I don't know quite what to do in C++, but there must a be a good lib out there for that stuff.
The text was updated successfully, but these errors were encountered: