1 /*
2  *  Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
3  *
4  *  Use of this source code is governed by a BSD-style license
5  *  that can be found in the LICENSE file in the root of the source
6  *  tree. An additional intellectual property rights grant can be found
7  *  in the file PATENTS.  All contributing project authors may
8  *  be found in the AUTHORS file in the root of the source tree.
9  */
10 
11 #include "modules/audio_device/fine_audio_buffer.h"
12 
13 #include <limits.h>
14 
15 #include <memory>
16 
17 #include "api/array_view.h"
18 #include "api/task_queue/default_task_queue_factory.h"
19 #include "modules/audio_device/mock_audio_device_buffer.h"
20 #include "test/gmock.h"
21 #include "test/gtest.h"
22 
23 using ::testing::_;
24 using ::testing::AtLeast;
25 using ::testing::InSequence;
26 using ::testing::Return;
27 
28 namespace webrtc {
29 
30 const int kSampleRate = 44100;
31 const int kChannels = 2;
32 const int kSamplesPer10Ms = kSampleRate * 10 / 1000;
33 
34 // The fake audio data is 0,1,..SCHAR_MAX-1,0,1,... This is to make it easy
35 // to detect errors. This function verifies that the buffers contain such data.
36 // E.g. if there are two buffers of size 3, buffer 1 would contain 0,1,2 and
37 // buffer 2 would contain 3,4,5. Note that SCHAR_MAX is 127 so wrap-around
38 // will happen.
39 // |buffer| is the audio buffer to verify.
VerifyBuffer(const int16_t * buffer,int buffer_number,int size)40 bool VerifyBuffer(const int16_t* buffer, int buffer_number, int size) {
41   int start_value = (buffer_number * size) % SCHAR_MAX;
42   for (int i = 0; i < size; ++i) {
43     if (buffer[i] != (i + start_value) % SCHAR_MAX) {
44       return false;
45     }
46   }
47   return true;
48 }
49 
50 // This function replaces the real AudioDeviceBuffer::GetPlayoutData when it's
51 // called (which is done implicitly when calling GetBufferData). It writes the
52 // sequence 0,1,..SCHAR_MAX-1,0,1,... to the buffer. Note that this is likely a
53 // buffer of different size than the one VerifyBuffer verifies.
54 // |iteration| is the number of calls made to UpdateBuffer prior to this call.
55 // |samples_per_10_ms| is the number of samples that should be written to the
56 // buffer (|arg0|).
ACTION_P2(UpdateBuffer,iteration,samples_per_10_ms)57 ACTION_P2(UpdateBuffer, iteration, samples_per_10_ms) {
58   int16_t* buffer = static_cast<int16_t*>(arg0);
59   int start_value = (iteration * samples_per_10_ms) % SCHAR_MAX;
60   for (int i = 0; i < samples_per_10_ms; ++i) {
61     buffer[i] = (i + start_value) % SCHAR_MAX;
62   }
63   // Should return samples per channel.
64   return samples_per_10_ms / kChannels;
65 }
66 
67 // Writes a periodic ramp pattern to the supplied |buffer|. See UpdateBuffer()
68 // for details.
UpdateInputBuffer(int16_t * buffer,int iteration,int size)69 void UpdateInputBuffer(int16_t* buffer, int iteration, int size) {
70   int start_value = (iteration * size) % SCHAR_MAX;
71   for (int i = 0; i < size; ++i) {
72     buffer[i] = (i + start_value) % SCHAR_MAX;
73   }
74 }
75 
76 // Action macro which verifies that the recorded 10ms chunk of audio data
77 // (in |arg0|) contains the correct reference values even if they have been
78 // supplied using a buffer size that is smaller or larger than 10ms.
79 // See VerifyBuffer() for details.
ACTION_P2(VerifyInputBuffer,iteration,samples_per_10_ms)80 ACTION_P2(VerifyInputBuffer, iteration, samples_per_10_ms) {
81   const int16_t* buffer = static_cast<const int16_t*>(arg0);
82   int start_value = (iteration * samples_per_10_ms) % SCHAR_MAX;
83   for (int i = 0; i < samples_per_10_ms; ++i) {
84     EXPECT_EQ(buffer[i], (i + start_value) % SCHAR_MAX);
85   }
86   return 0;
87 }
88 
RunFineBufferTest(int frame_size_in_samples)89 void RunFineBufferTest(int frame_size_in_samples) {
90   const int kFrameSizeSamples = frame_size_in_samples;
91   const int kNumberOfFrames = 5;
92   // Ceiling of integer division: 1 + ((x - 1) / y)
93   const int kNumberOfUpdateBufferCalls =
94       1 + ((kNumberOfFrames * frame_size_in_samples - 1) / kSamplesPer10Ms);
95 
96   auto task_queue_factory = CreateDefaultTaskQueueFactory();
97   MockAudioDeviceBuffer audio_device_buffer(task_queue_factory.get());
98   audio_device_buffer.SetPlayoutSampleRate(kSampleRate);
99   audio_device_buffer.SetPlayoutChannels(kChannels);
100   audio_device_buffer.SetRecordingSampleRate(kSampleRate);
101   audio_device_buffer.SetRecordingChannels(kChannels);
102 
103   EXPECT_CALL(audio_device_buffer, RequestPlayoutData(_))
104       .WillRepeatedly(Return(kSamplesPer10Ms));
105   {
106     InSequence s;
107     for (int i = 0; i < kNumberOfUpdateBufferCalls; ++i) {
108       EXPECT_CALL(audio_device_buffer, GetPlayoutData(_))
109           .WillOnce(UpdateBuffer(i, kChannels * kSamplesPer10Ms))
110           .RetiresOnSaturation();
111     }
112   }
113   {
114     InSequence s;
115     for (int j = 0; j < kNumberOfUpdateBufferCalls - 1; ++j) {
116       EXPECT_CALL(audio_device_buffer, SetRecordedBuffer(_, kSamplesPer10Ms))
117           .WillOnce(VerifyInputBuffer(j, kChannels * kSamplesPer10Ms))
118           .RetiresOnSaturation();
119     }
120   }
121   EXPECT_CALL(audio_device_buffer, SetVQEData(_, _))
122       .Times(kNumberOfUpdateBufferCalls - 1);
123   EXPECT_CALL(audio_device_buffer, DeliverRecordedData())
124       .Times(kNumberOfUpdateBufferCalls - 1)
125       .WillRepeatedly(Return(0));
126 
127   FineAudioBuffer fine_buffer(&audio_device_buffer);
128   std::unique_ptr<int16_t[]> out_buffer(
129       new int16_t[kChannels * kFrameSizeSamples]);
130   std::unique_ptr<int16_t[]> in_buffer(
131       new int16_t[kChannels * kFrameSizeSamples]);
132 
133   for (int i = 0; i < kNumberOfFrames; ++i) {
134     fine_buffer.GetPlayoutData(
135         rtc::ArrayView<int16_t>(out_buffer.get(),
136                                 kChannels * kFrameSizeSamples),
137         0);
138     EXPECT_TRUE(
139         VerifyBuffer(out_buffer.get(), i, kChannels * kFrameSizeSamples));
140     UpdateInputBuffer(in_buffer.get(), i, kChannels * kFrameSizeSamples);
141     fine_buffer.DeliverRecordedData(
142         rtc::ArrayView<const int16_t>(in_buffer.get(),
143                                       kChannels * kFrameSizeSamples),
144         0);
145   }
146 }
147 
TEST(FineBufferTest,BufferLessThan10ms)148 TEST(FineBufferTest, BufferLessThan10ms) {
149   const int kFrameSizeSamples = kSamplesPer10Ms - 50;
150   RunFineBufferTest(kFrameSizeSamples);
151 }
152 
TEST(FineBufferTest,GreaterThan10ms)153 TEST(FineBufferTest, GreaterThan10ms) {
154   const int kFrameSizeSamples = kSamplesPer10Ms + 50;
155   RunFineBufferTest(kFrameSizeSamples);
156 }
157 
158 }  // namespace webrtc
159