summaryrefslogtreecommitdiffstats
path: root/dom/media/webrtc/SineWaveGenerator.h
blob: 73bab53f19a149d95ecc2e0806d4fd1ec5095ec4 (plain)
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
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this file,
 * You can obtain one at http://mozilla.org/MPL/2.0/. */

#ifndef SINEWAVEGENERATOR_H_
#define SINEWAVEGENERATOR_H_

#include "MediaSegment.h"
#include "prtime.h"

namespace mozilla {

// generate 1k sine wave per second
template <typename Sample>
class SineWaveGenerator {
  static_assert(std::is_same<Sample, int16_t>::value ||
                std::is_same<Sample, float>::value);

 public:
  static const int bytesPerSample = sizeof(Sample);
  static const int millisecondsPerSecond = PR_MSEC_PER_SEC;
  static constexpr float twopi = 2 * M_PI;

  /* If more than 1 channel, generated samples are interleaved. */
  SineWaveGenerator(uint32_t aSampleRate, uint32_t aFrequency)
      : mPhase(0.), mPhaseIncrement(twopi * aFrequency / aSampleRate) {}

  // NOTE: only safely called from a single thread (MTG callback)
  void generate(Sample* aBuffer, TrackTicks aFrameCount,
                uint32_t aChannelCount = 1) {
    while (aFrameCount--) {
      Sample value = sin(mPhase) * Amplitude();
      for (uint32_t channel = 0; channel < aChannelCount; channel++) {
        *aBuffer++ = value;
      }
      mPhase += mPhaseIncrement;
      if (mPhase > twopi) {
        mPhase -= twopi;
      }
    }
  }

  static float Amplitude() {
    // Set volume to -20db.
    if (std::is_same<Sample, int16_t>::value) {
      return 3276.8;  // 32768.0 * 10^(-20/20) = 3276.8
    }
    return 0.1f;  // 1.0 * 10^(-20/20) = 0.1
  }

 private:
  double mPhase;
  const double mPhaseIncrement;
};

}  // namespace mozilla

#endif /* SINEWAVEGENERATOR_H_ */