BangleApps/apps/hrmmar/fftelim.js

191 lines
6.2 KiB
JavaScript

exports.run = (settings, updateHrm, isInternal) => {
const SAMPLE_RATE = 12.5;
const NUM_POINTS = 256; // fft size
const ACC_PEAKS = 2; // remove this number of ACC peaks
// ringbuffers
const hrmvalues = new Int16Array(8*SAMPLE_RATE);
const accvalues = new Int16Array(8*SAMPLE_RATE);
// fft buffers
const hrmfftbuf = new Int16Array(NUM_POINTS);
const accfftbuf = new Int16Array(NUM_POINTS);
let BPM_est_1 = 0;
let BPM_est_2 = 0;
let hrmdata;
let idx=0, wraps=0;
// init settings
Bangle.setOptions({hrmPollInterval: 40, powerSave: false}); // hrm=25Hz
Bangle.setPollInterval(80); // 12.5Hz
calcfft = (values, idx, normalize, fftbuf) => {
fftbuf.fill(0);
let i_out=0;
let avg = 0;
if (normalize) {
const sum = values.reduce((a, b) => a + b, 0);
avg = sum/values.length;
}
// sort ringbuffer to fft buffer
for(let i_in=idx; i_in<values.length; i_in++, i_out++) {
fftbuf[i_out] = values[i_in]-avg;
}
for(let i_in=0; i_in<idx; i_in++, i_out++) {
fftbuf[i_out] = values[i_in]-avg;
}
E.FFT(fftbuf);
return fftbuf;
};
getMax = (values) => {
let maxVal = -Number.MAX_VALUE;
let maxIdx = 0;
values.forEach((value,i) => {
if (value > maxVal) {
maxVal = value;
maxIdx = i;
}
});
return {idx: maxIdx, val: maxVal};
};
getSign = (value) => {
return value < 0 ? -1 : 1;
};
// idx in fft buffer to frequency
getFftFreq = (idx, rate, size) => {
return idx*rate/(size-1);
};
// frequency to idx in fft buffer
getFftIdx = (freq, rate, size) => {
return Math.round(freq*(size-1)/rate);
};
calc2ndDeriative = (values) => {
const result = new Int16Array(values.length-2);
for(let i=1; i<values.length-1; i++) {
const diff = values[i+1]-2*values[i]+values[i-1];
result[i-1] = diff;
}
return result;
};
const minFreqIdx = getFftIdx(1.0, SAMPLE_RATE, NUM_POINTS); // 60 BPM
const maxFreqIdx = getFftIdx(3.0, SAMPLE_RATE, NUM_POINTS); // 180 BPM
let rangeIdx = [0, maxFreqIdx-minFreqIdx]; // range of search for the next estimates
const freqStep=getFftFreq(1, SAMPLE_RATE, NUM_POINTS)*60;
const maxBpmDiffIdxDown = Math.ceil(5/freqStep); // maximum down BPM
const maxBpmDiffIdxUp = Math.ceil(10/freqStep); // maximum up BPM
calculate = (idx) => {
// fft
const ppg_fft = calcfft(hrmvalues, idx, true, hrmfftbuf).subarray(minFreqIdx, maxFreqIdx+1);
const acc_fft = calcfft(accvalues, idx, false, accfftbuf).subarray(minFreqIdx, maxFreqIdx+1);
// remove spectrum that have peaks in acc fft from ppg fft
const accGlobalMax = getMax(acc_fft);
const acc2nddiff = calc2ndDeriative(acc_fft); // calculate second derivative
for(let iClean=0; iClean < ACC_PEAKS; iClean++) {
// get max peak in ACC
const accMax = getMax(acc_fft);
if (accMax.val >= 10 && accMax.val/accGlobalMax.val > 0.75) {
// set all values in PPG FFT to zero until second derivative of ACC has zero crossing
for (let k = accMax.idx-1; k>=0; k--) {
ppg_fft[k] = 0;
acc_fft[k] = -Math.abs(acc_fft[k]); // max(acc_fft) should no longer find this
if (k-2 > 0 && getSign(acc2nddiff[k-1-2]) != getSign(acc2nddiff[k-2]) && Math.abs(acc_fft[k]) < accMax.val*0.75) {
break;
}
}
// set all values in PPG FFT to zero until second derivative of ACC has zero crossing
for (let k = accMax.idx; k < acc_fft.length-1; k++) {
ppg_fft[k] = 0;
acc_fft[k] = -Math.abs(acc_fft[k]); // max(acc_fft) should no longer find this
if (k-2 >= 0 && getSign(acc2nddiff[k+1-2]) != getSign(acc2nddiff[k-2]) && Math.abs(acc_fft[k]) < accMax.val*0.75) {
break;
}
}
}
}
// bpm result is maximum peak in PPG fft
const hrRangeMax = getMax(ppg_fft.subarray(rangeIdx[0], rangeIdx[1]));
const hrTotalMax = getMax(ppg_fft);
const maxDiff = hrTotalMax.val/hrRangeMax.val;
let idxMaxPPG = hrRangeMax.idx+rangeIdx[0]; // offset range limit
if ((maxDiff > 3 && idxMaxPPG != hrTotalMax.idx) || hrRangeMax.val === 0) { // prevent tracking from loosing the real heart rate by checking the full spectrum
if (hrTotalMax.idx > idxMaxPPG) {
idxMaxPPG = idxMaxPPG+Math.ceil(6/freqStep); // step 6 BPM up into the direction of max peak
} else {
idxMaxPPG = idxMaxPPG-Math.ceil(2/freqStep); // step 2 BPM down into the direction of max peak
}
}
idxMaxPPG = idxMaxPPG + minFreqIdx;
const BPM_est_0 = getFftFreq(idxMaxPPG, SAMPLE_RATE, NUM_POINTS)*60;
// smooth with moving average
let BPM_est_res;
if (BPM_est_2 > 0) {
BPM_est_res = 0.9*BPM_est_0 + 0.05*BPM_est_1 + 0.05*BPM_est_2;
} else {
BPM_est_res = BPM_est_0;
}
return BPM_est_res.toFixed(1);
};
Bangle.on('HRM-raw', (hrm) => {
hrmdata = hrm;
});
Bangle.on('accel', (acc) => {
if (hrmdata !== undefined && isInternal(hrmdata)) {
hrmvalues[idx] = hrmdata.filt;
accvalues[idx] = acc.x*1000 + acc.y*1000 + acc.z*1000;
idx++;
if (idx >= 8*SAMPLE_RATE) {
idx = 0;
wraps++;
}
if (idx % (SAMPLE_RATE*2) == 0) { // every two seconds
if (wraps === 0) { // use rate of firmware until hrmvalues buffer is filled
updateHrm(undefined);
BPM_est_2 = BPM_est_1;
BPM_est_1 = hrmdata.bpm;
} else {
let bpm_result;
if (hrmdata.confidence >= 90) { // display firmware value if good
bpm_result = hrmdata.bpm;
updateHrm(undefined);
} else {
bpm_result = calculate(idx);
bpm_corrected = bpm_result;
updateHrm(bpm_result);
}
BPM_est_2 = BPM_est_1;
BPM_est_1 = bpm_result;
// set search range of next BPM
const est_res_idx = getFftIdx(bpm_result/60, SAMPLE_RATE, NUM_POINTS)-minFreqIdx;
rangeIdx = [est_res_idx-maxBpmDiffIdxDown, est_res_idx+maxBpmDiffIdxUp];
if (rangeIdx[0] < 0) {
rangeIdx[0] = 0;
}
if (rangeIdx[1] > maxFreqIdx-minFreqIdx) {
rangeIdx[1] = maxFreqIdx-minFreqIdx;
}
}
}
}
});
};