forked from ggerganov/kbd-audio
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathkey_detector.cpp
86 lines (74 loc) · 2.54 KB
/
key_detector.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
/*! \file key_detector.cpp
* \brief Enter description here.
* \author Georgi Gerganov
*/
#include "audio_logger.h"
#include <array>
#include <cmath>
#include <cstdio>
#include <chrono>
#include <thread>
int main(int argc, const char ** argv) {
constexpr float kBufferSize_s = 0.150f;
constexpr uint64_t kSampleRate = 96000;
constexpr uint64_t kRingBufferSize = 16*1024;
constexpr int bkgrStep_samples = 7;
constexpr int keyDuration_samples = 0.150f*kSampleRate;
// rig buffer
int rbBegin = 0;
float rbAverage = 0.0f;
std::array<float, kRingBufferSize> rbSamples;
rbSamples.fill(0.0f);
bool doRecord = true;
AudioLogger audioLogger;
AudioLogger::Callback cbAudio = [&](const auto & frames) {
doRecord = true;
float amax = 0.0f;
for (int f = 0; f < frames.size(); ++f) {
for (int s = 0; s < frames[f].size(); s += bkgrStep_samples) {
rbAverage *= rbSamples.size();
rbAverage -= rbSamples[rbBegin];
auto acur = std::abs(frames[f][s]);
rbSamples[rbBegin] = acur;
if (acur > amax) amax = acur;
rbAverage += acur;
rbAverage /= rbSamples.size();
if (++rbBegin >= rbSamples.size()) rbBegin = 0;
}
}
int skip_samples = 0;
int nFrames = frames.size();
int nFrames2 = std::max(1, nFrames/2);
for (int f = nFrames2 - nFrames2/2; f <= nFrames2 + nFrames2/2; ++f) {
for (int s = 0; s < frames[f].size(); ++s) {
if (s + skip_samples >= frames[f].size()) {
skip_samples -= frames[f].size() - s;
s += skip_samples;
continue;
} else {
s += skip_samples;
skip_samples = 0;
}
auto acur = frames[f][s];
if (acur > 10.0f*rbAverage) {
skip_samples = keyDuration_samples;
printf("Key press detected\n");
}
}
}
printf("Average = %10.8f, max = %10.8f\n", rbAverage, amax);
};
if (audioLogger.install(kSampleRate, cbAudio) == false) {
fprintf(stderr, "Failed to install audio logger\n");
return -1;
}
while (true) {
if (doRecord) {
doRecord = false;
audioLogger.record(kBufferSize_s);
} else {
std::this_thread::sleep_for(std::chrono::milliseconds(1));
}
}
return 0;
}