1 /* 2 * Copyright (C) 2020 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #ifndef ANALYZER_GLITCH_ANALYZER_H 18 #define ANALYZER_GLITCH_ANALYZER_H 19 20 #include <algorithm> 21 #include <cctype> 22 #include <iomanip> 23 #include <iostream> 24 25 #include "LatencyAnalyzer.h" 26 #include "PseudoRandom.h" 27 28 /** 29 * Output a steady sine wave and analyze the return signal. 30 * 31 * Use a cosine transform to measure the predicted magnitude and relative phase of the 32 * looped back sine wave. Then generate a predicted signal and compare with the actual signal. 33 */ 34 class GlitchAnalyzer : public LoopbackProcessor { 35 public: 36 getState()37 int32_t getState() const { 38 return mState; 39 } 40 getPeakAmplitude()41 double getPeakAmplitude() const { 42 return mPeakFollower.getLevel(); 43 } 44 getTolerance()45 double getTolerance() { 46 return mTolerance; 47 } 48 setTolerance(double tolerance)49 void setTolerance(double tolerance) { 50 mTolerance = tolerance; 51 mScaledTolerance = mMagnitude * mTolerance; 52 } 53 setMagnitude(double magnitude)54 void setMagnitude(double magnitude) { 55 mMagnitude = magnitude; 56 mScaledTolerance = mMagnitude * mTolerance; 57 } 58 getGlitchCount()59 int32_t getGlitchCount() const { 60 return mGlitchCount; 61 } 62 getStateFrameCount(int state)63 int32_t getStateFrameCount(int state) const { 64 return mStateFrameCounters[state]; 65 } 66 getSignalToNoiseDB()67 double getSignalToNoiseDB() { 68 static const double threshold = 1.0e-14; 69 if (mMeanSquareSignal < threshold || mMeanSquareNoise < threshold) { 70 return 0.0; 71 } else { 72 double signalToNoise = mMeanSquareSignal / mMeanSquareNoise; // power ratio 73 double signalToNoiseDB = 10.0 * log(signalToNoise); 74 if (signalToNoiseDB < MIN_SNR_DB) { 75 ALOGD("ERROR - signal to noise ratio is too low! < %d dB. Adjust volume.", 76 MIN_SNR_DB); 77 setResult(ERROR_VOLUME_TOO_LOW); 78 } 79 return signalToNoiseDB; 80 } 81 } 82 analyze()83 std::string analyze() override { 84 std::stringstream report; 85 report << "GlitchAnalyzer ------------------\n"; 86 report << LOOPBACK_RESULT_TAG "peak.amplitude = " << std::setw(8) 87 << getPeakAmplitude() << "\n"; 88 report << LOOPBACK_RESULT_TAG "sine.magnitude = " << std::setw(8) 89 << mMagnitude << "\n"; 90 report << LOOPBACK_RESULT_TAG "rms.noise = " << std::setw(8) 91 << mMeanSquareNoise << "\n"; 92 report << LOOPBACK_RESULT_TAG "signal.to.noise.db = " << std::setw(8) 93 << getSignalToNoiseDB() << "\n"; 94 report << LOOPBACK_RESULT_TAG "frames.accumulated = " << std::setw(8) 95 << mFramesAccumulated << "\n"; 96 report << LOOPBACK_RESULT_TAG "sine.period = " << std::setw(8) 97 << mSinePeriod << "\n"; 98 report << LOOPBACK_RESULT_TAG "test.state = " << std::setw(8) 99 << mState << "\n"; 100 report << LOOPBACK_RESULT_TAG "frame.count = " << std::setw(8) 101 << mFrameCounter << "\n"; 102 // Did we ever get a lock? 103 bool gotLock = (mState == STATE_LOCKED) || (mGlitchCount > 0); 104 if (!gotLock) { 105 report << "ERROR - failed to lock on reference sine tone.\n"; 106 setResult(ERROR_NO_LOCK); 107 } else { 108 // Only print if meaningful. 109 report << LOOPBACK_RESULT_TAG "glitch.count = " << std::setw(8) 110 << mGlitchCount << "\n"; 111 report << LOOPBACK_RESULT_TAG "max.glitch = " << std::setw(8) 112 << mMaxGlitchDelta << "\n"; 113 if (mGlitchCount > 0) { 114 report << "ERROR - number of glitches > 0\n"; 115 setResult(ERROR_GLITCHES); 116 } 117 } 118 return report.str(); 119 } 120 printStatus()121 void printStatus() override { 122 ALOGD("st = %d, #gl = %3d,", mState, mGlitchCount); 123 } 124 /** 125 * Calculate the magnitude of the component of the input signal 126 * that matches the analysis frequency. 127 * Also calculate the phase that we can use to create a 128 * signal that matches that component. 129 * The phase will be between -PI and +PI. 130 */ 131 double calculateMagnitude(double *phasePtr = nullptr) { 132 if (mFramesAccumulated == 0) { 133 return 0.0; 134 } 135 double sinMean = mSinAccumulator / mFramesAccumulated; 136 double cosMean = mCosAccumulator / mFramesAccumulated; 137 double magnitude = 2.0 * sqrt((sinMean * sinMean) + (cosMean * cosMean)); 138 if (phasePtr != nullptr) { 139 double phase = M_PI_2 - atan2(sinMean, cosMean); 140 *phasePtr = phase; 141 } 142 return magnitude; 143 } 144 145 /** 146 * @param frameData contains microphone data with sine signal feedback 147 * @param channelCount 148 */ processInputFrame(float * frameData,int)149 result_code processInputFrame(float *frameData, int /* channelCount */) override { 150 result_code result = RESULT_OK; 151 152 float sample = frameData[0]; 153 float peak = mPeakFollower.process(sample); 154 155 // Force a periodic glitch to test the detector! 156 if (mForceGlitchDuration > 0) { 157 if (mForceGlitchCounter == 0) { 158 ALOGE("%s: force a glitch!!", __func__); 159 mForceGlitchCounter = getSampleRate(); 160 } else if (mForceGlitchCounter <= mForceGlitchDuration) { 161 // Force an abrupt offset. 162 sample += (sample > 0.0) ? -0.5f : 0.5f; 163 } 164 --mForceGlitchCounter; 165 } 166 167 mStateFrameCounters[mState]++; // count how many frames we are in each state 168 169 switch (mState) { 170 case STATE_IDLE: 171 mDownCounter--; 172 if (mDownCounter <= 0) { 173 mState = STATE_IMMUNE; 174 mDownCounter = IMMUNE_FRAME_COUNT; 175 mInputPhase = 0.0; // prevent spike at start 176 mOutputPhase = 0.0; 177 } 178 break; 179 180 case STATE_IMMUNE: 181 mDownCounter--; 182 if (mDownCounter <= 0) { 183 mState = STATE_WAITING_FOR_SIGNAL; 184 } 185 break; 186 187 case STATE_WAITING_FOR_SIGNAL: 188 if (peak > mThreshold) { 189 mState = STATE_WAITING_FOR_LOCK; 190 //ALOGD("%5d: switch to STATE_WAITING_FOR_LOCK", mFrameCounter); 191 resetAccumulator(); 192 } 193 break; 194 195 case STATE_WAITING_FOR_LOCK: 196 mSinAccumulator += sample * sinf(mInputPhase); 197 mCosAccumulator += sample * cosf(mInputPhase); 198 mFramesAccumulated++; 199 // Must be a multiple of the period or the calculation will not be accurate. 200 if (mFramesAccumulated == mSinePeriod * PERIODS_NEEDED_FOR_LOCK) { 201 double phaseOffset = 0.0; 202 setMagnitude(calculateMagnitude(&phaseOffset)); 203 // ALOGD("%s() mag = %f, offset = %f, prev = %f", 204 // __func__, mMagnitude, mPhaseOffset, mPreviousPhaseOffset); 205 if (mMagnitude > mThreshold) { 206 if (abs(phaseOffset) < kMaxPhaseError) { 207 mState = STATE_LOCKED; 208 // ALOGD("%5d: switch to STATE_LOCKED", mFrameCounter); 209 } 210 // Adjust mInputPhase to match measured phase 211 mInputPhase += phaseOffset; 212 } 213 resetAccumulator(); 214 } 215 incrementInputPhase(); 216 break; 217 218 case STATE_LOCKED: { 219 // Predict next sine value 220 double predicted = sinf(mInputPhase) * mMagnitude; 221 double diff = predicted - sample; 222 double absDiff = fabs(diff); 223 mMaxGlitchDelta = std::max(mMaxGlitchDelta, absDiff); 224 if (absDiff > mScaledTolerance) { 225 result = ERROR_GLITCHES; 226 onGlitchStart(); 227 // LOGI("diff glitch detected, absDiff = %g", absDiff); 228 } else { 229 mSumSquareSignal += predicted * predicted; 230 mSumSquareNoise += diff * diff; 231 // Track incoming signal and slowly adjust magnitude to account 232 // for drift in the DRC or AGC. 233 mSinAccumulator += sample * sinf(mInputPhase); 234 mCosAccumulator += sample * cosf(mInputPhase); 235 mFramesAccumulated++; 236 // Must be a multiple of the period or the calculation will not be accurate. 237 if (mFramesAccumulated == mSinePeriod) { 238 const double coefficient = 0.1; 239 double phaseOffset = 0.0; 240 double magnitude = calculateMagnitude(&phaseOffset); 241 // One pole averaging filter. 242 setMagnitude((mMagnitude * (1.0 - coefficient)) + (magnitude * coefficient)); 243 244 mMeanSquareNoise = mSumSquareNoise * mInverseSinePeriod; 245 mMeanSquareSignal = mSumSquareSignal * mInverseSinePeriod; 246 resetAccumulator(); 247 248 if (abs(phaseOffset) > kMaxPhaseError) { 249 result = ERROR_GLITCHES; 250 onGlitchStart(); 251 ALOGD("phase glitch detected, phaseOffset = %g", phaseOffset); 252 } else if (mMagnitude < mThreshold) { 253 result = ERROR_GLITCHES; 254 onGlitchStart(); 255 ALOGD("magnitude glitch detected, mMagnitude = %g", mMagnitude); 256 } 257 } 258 } 259 incrementInputPhase(); 260 } break; 261 262 case STATE_GLITCHING: { 263 // Predict next sine value 264 mGlitchLength++; 265 double predicted = sinf(mInputPhase) * mMagnitude; 266 double diff = predicted - sample; 267 double absDiff = fabs(diff); 268 mMaxGlitchDelta = std::max(mMaxGlitchDelta, absDiff); 269 if (absDiff < mScaledTolerance) { // close enough? 270 // If we get a full sine period of non-glitch samples in a row then consider the glitch over. 271 // We don't want to just consider a zero crossing the end of a glitch. 272 if (mNonGlitchCount++ > mSinePeriod) { 273 onGlitchEnd(); 274 } 275 } else { 276 mNonGlitchCount = 0; 277 if (mGlitchLength > (4 * mSinePeriod)) { 278 relock(); 279 } 280 } 281 incrementInputPhase(); 282 } break; 283 284 case NUM_STATES: // not a real state 285 break; 286 } 287 288 mFrameCounter++; 289 290 return result; 291 } 292 293 // advance and wrap phase incrementInputPhase()294 void incrementInputPhase() { 295 mInputPhase += mPhaseIncrement; 296 if (mInputPhase > M_PI) { 297 mInputPhase -= (2.0 * M_PI); 298 } 299 } 300 301 // advance and wrap phase incrementOutputPhase()302 void incrementOutputPhase() { 303 mOutputPhase += mPhaseIncrement; 304 if (mOutputPhase > M_PI) { 305 mOutputPhase -= (2.0 * M_PI); 306 } 307 } 308 309 /** 310 * @param frameData upon return, contains the reference sine wave 311 * @param channelCount 312 */ processOutputFrame(float * frameData,int channelCount)313 result_code processOutputFrame(float *frameData, int channelCount) override { 314 float output = 0.0f; 315 // Output sine wave so we can measure it. 316 if (mState != STATE_IDLE) { 317 float sinOut = sinf(mOutputPhase); 318 incrementOutputPhase(); 319 output = (sinOut * mOutputAmplitude) 320 + (mWhiteNoise.nextRandomDouble() * kNoiseAmplitude); 321 // ALOGD("sin(%f) = %f, %f\n", mOutputPhase, sinOut, mPhaseIncrement); 322 } 323 frameData[0] = output; 324 for (int i = 1; i < channelCount; i++) { 325 frameData[i] = 0.0f; 326 } 327 return RESULT_OK; 328 } 329 onGlitchStart()330 void onGlitchStart() { 331 mGlitchCount++; 332 // ALOGD("%5d: STARTED a glitch # %d", mFrameCounter, mGlitchCount); 333 mState = STATE_GLITCHING; 334 mGlitchLength = 1; 335 mNonGlitchCount = 0; 336 } 337 onGlitchEnd()338 void onGlitchEnd() { 339 // ALOGD("%5d: ENDED a glitch # %d, length = %d", mFrameCounter, mGlitchCount, mGlitchLength); 340 mState = STATE_LOCKED; 341 resetAccumulator(); 342 } 343 344 // reset the sine wave detector resetAccumulator()345 void resetAccumulator() { 346 mFramesAccumulated = 0; 347 mSinAccumulator = 0.0; 348 mCosAccumulator = 0.0; 349 mSumSquareSignal = 0.0; 350 mSumSquareNoise = 0.0; 351 } 352 relock()353 void relock() { 354 // ALOGD("relock: %d because of a very long %d glitch", mFrameCounter, mGlitchLength); 355 mState = STATE_WAITING_FOR_LOCK; 356 resetAccumulator(); 357 } 358 reset()359 void reset() override { 360 LoopbackProcessor::reset(); 361 mState = STATE_IDLE; 362 mDownCounter = IDLE_FRAME_COUNT; 363 resetAccumulator(); 364 } 365 prepareToTest()366 void prepareToTest() override { 367 LoopbackProcessor::prepareToTest(); 368 mSinePeriod = getSampleRate() / kTargetGlitchFrequency; 369 mOutputPhase = 0.0f; 370 mInverseSinePeriod = 1.0 / mSinePeriod; 371 mPhaseIncrement = 2.0 * M_PI * mInverseSinePeriod; 372 mGlitchCount = 0; 373 mMaxGlitchDelta = 0.0; 374 for (int i = 0; i < NUM_STATES; i++) { 375 mStateFrameCounters[i] = 0; 376 } 377 } 378 379 private: 380 381 // These must match the values in GlitchActivity.java 382 enum sine_state_t { 383 STATE_IDLE, // beginning 384 STATE_IMMUNE, // ignoring input, waiting fo HW to settle 385 STATE_WAITING_FOR_SIGNAL, // looking for a loud signal 386 STATE_WAITING_FOR_LOCK, // trying to lock onto the phase of the sine 387 STATE_LOCKED, // locked on the sine wave, looking for glitches 388 STATE_GLITCHING, // locked on the sine wave but glitching 389 NUM_STATES 390 }; 391 392 enum constants { 393 // Arbitrary durations, assuming 48000 Hz 394 IDLE_FRAME_COUNT = 48 * 100, 395 IMMUNE_FRAME_COUNT = 48 * 100, 396 PERIODS_NEEDED_FOR_LOCK = 8, 397 MIN_SNR_DB = 65 398 }; 399 400 static constexpr float kNoiseAmplitude = 0.00; // Used to experiment with warbling caused by DRC. 401 static constexpr int kTargetGlitchFrequency = 607; 402 static constexpr double kMaxPhaseError = M_PI * 0.05; 403 404 float mTolerance = 0.10; // scaled from 0.0 to 1.0 405 double mThreshold = 0.005; 406 int mSinePeriod = 1; // this will be set before use 407 double mInverseSinePeriod = 1.0; 408 409 int32_t mStateFrameCounters[NUM_STATES]; 410 411 double mPhaseIncrement = 0.0; 412 double mInputPhase = 0.0; 413 double mOutputPhase = 0.0; 414 double mMagnitude = 0.0; 415 int32_t mFramesAccumulated = 0; 416 double mSinAccumulator = 0.0; 417 double mCosAccumulator = 0.0; 418 double mMaxGlitchDelta = 0.0; 419 int32_t mGlitchCount = 0; 420 int32_t mNonGlitchCount = 0; 421 int32_t mGlitchLength = 0; 422 // This is used for processing every frame so we cache it here. 423 double mScaledTolerance = 0.0; 424 int mDownCounter = IDLE_FRAME_COUNT; 425 int32_t mFrameCounter = 0; 426 double mOutputAmplitude = 0.75; 427 428 int32_t mForceGlitchDuration = 0; // if > 0 then force a glitch for debugging 429 int32_t mForceGlitchCounter = 4 * 48000; // count down and trigger at zero 430 431 // measure background noise continuously as a deviation from the expected signal 432 double mSumSquareSignal = 0.0; 433 double mSumSquareNoise = 0.0; 434 double mMeanSquareSignal = 0.0; 435 double mMeanSquareNoise = 0.0; 436 437 PeakDetector mPeakFollower; 438 439 PseudoRandom mWhiteNoise; 440 441 sine_state_t mState = STATE_IDLE; 442 }; 443 444 445 #endif //ANALYZER_GLITCH_ANALYZER_H 446