blob: ca0a240aec05956b5a76c2a5f68da53260f93d9b [file] [log] [blame]
turaj@webrtc.orgd0631e32013-06-06 19:00:091/*
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 <stdio.h>
12
kwiberg4df87572016-02-15 04:40:5713#include <memory>
14
turaj@webrtc.orgd0631e32013-06-06 19:00:0915#include "gflags/gflags.h"
turaj@webrtc.orgd0631e32013-06-06 19:00:0916#include "webrtc/common_types.h"
kjellander0c1546c2015-11-26 12:44:5417#include "webrtc/modules/audio_coding/include/audio_coding_module.h"
18#include "webrtc/modules/audio_coding/test/Channel.h"
19#include "webrtc/modules/audio_coding/test/PCMFile.h"
Henrik Kjellander36a14b52015-11-04 07:31:5220#include "webrtc/modules/include/module_common_types.h"
Henrik Kjellander78f65d02015-10-28 17:17:4021#include "webrtc/system_wrappers/include/clock.h"
kwiberg36a24792016-10-01 05:29:4322#include "webrtc/test/gtest.h"
turaj@webrtc.orgd0631e32013-06-06 19:00:0923#include "webrtc/test/testsupport/fileutils.h"
24
25// Codec.
26DEFINE_string(codec, "opus", "Codec Name");
27DEFINE_int32(codec_sample_rate_hz, 48000, "Sampling rate in Hertz.");
28DEFINE_int32(codec_channels, 1, "Number of channels of the codec.");
29
30// PCM input/output.
31DEFINE_string(input, "", "Input PCM file at 16 kHz.");
32DEFINE_bool(input_stereo, false, "Input is stereo.");
33DEFINE_int32(input_fs_hz, 32000, "Input sample rate Hz.");
34DEFINE_string(output, "insert_rtp_with_timing_out.pcm", "OutputFile");
35DEFINE_int32(output_fs_hz, 32000, "Output sample rate Hz");
36
37// Timing files
38DEFINE_string(seq_num, "seq_num", "Sequence number file.");
39DEFINE_string(send_ts, "send_timestamp", "Send timestamp file.");
40DEFINE_string(receive_ts, "last_rec_timestamp", "Receive timestamp file");
41
42// Delay logging
43DEFINE_string(delay, "", "Log for delay.");
44
45// Other setups
turaj@webrtc.orgd0631e32013-06-06 19:00:0946DEFINE_bool(verbose, false, "Verbosity.");
47DEFINE_double(loss_rate, 0, "Rate of packet loss < 1");
48
49const int32_t kAudioPlayedOut = 0x00000001;
50const int32_t kPacketPushedIn = 0x00000001 << 1;
51const int kPlayoutPeriodMs = 10;
52
53namespace webrtc {
54
55class InsertPacketWithTiming {
56 public:
57 InsertPacketWithTiming()
58 : sender_clock_(new SimulatedClock(0)),
59 receiver_clock_(new SimulatedClock(0)),
60 send_acm_(AudioCodingModule::Create(0, sender_clock_)),
61 receive_acm_(AudioCodingModule::Create(0, receiver_clock_)),
62 channel_(new Channel),
63 seq_num_fid_(fopen(FLAGS_seq_num.c_str(), "rt")),
64 send_ts_fid_(fopen(FLAGS_send_ts.c_str(), "rt")),
65 receive_ts_fid_(fopen(FLAGS_receive_ts.c_str(), "rt")),
66 pcm_out_fid_(fopen(FLAGS_output.c_str(), "wb")),
67 samples_in_1ms_(48),
68 num_10ms_in_codec_frame_(2), // Typical 20 ms frames.
69 time_to_insert_packet_ms_(3), // An arbitrary offset on pushing packet.
70 next_receive_ts_(0),
71 time_to_playout_audio_ms_(kPlayoutPeriodMs),
72 loss_threshold_(0),
73 playout_timing_fid_(fopen("playout_timing.txt", "wt")) {}
74
75 void SetUp() {
76 ASSERT_TRUE(sender_clock_ != NULL);
77 ASSERT_TRUE(receiver_clock_ != NULL);
78
andrew@webrtc.org5e3379e2013-09-12 01:27:4379 ASSERT_TRUE(send_acm_.get() != NULL);
80 ASSERT_TRUE(receive_acm_.get() != NULL);
turaj@webrtc.orgd0631e32013-06-06 19:00:0981 ASSERT_TRUE(channel_ != NULL);
82
83 ASSERT_TRUE(seq_num_fid_ != NULL);
84 ASSERT_TRUE(send_ts_fid_ != NULL);
85 ASSERT_TRUE(receive_ts_fid_ != NULL);
86
87 ASSERT_TRUE(playout_timing_fid_ != NULL);
88
89 next_receive_ts_ = ReceiveTimestamp();
90
91 CodecInst codec;
92 ASSERT_EQ(0, AudioCodingModule::Codec(FLAGS_codec.c_str(), &codec,
93 FLAGS_codec_sample_rate_hz,
94 FLAGS_codec_channels));
95 ASSERT_EQ(0, receive_acm_->InitializeReceiver());
96 ASSERT_EQ(0, send_acm_->RegisterSendCodec(codec));
97 ASSERT_EQ(0, receive_acm_->RegisterReceiveCodec(codec));
98
99 // Set codec-dependent parameters.
100 samples_in_1ms_ = codec.plfreq / 1000;
101 num_10ms_in_codec_frame_ = codec.pacsize / (codec.plfreq / 100);
102
andrew@webrtc.org5e3379e2013-09-12 01:27:43103 channel_->RegisterReceiverACM(receive_acm_.get());
turaj@webrtc.orgd0631e32013-06-06 19:00:09104 send_acm_->RegisterTransportCallback(channel_);
105
106 if (FLAGS_input.size() == 0) {
107 std::string file_name = test::ResourcePath("audio_coding/testfile32kHz",
108 "pcm");
109 pcm_in_fid_.Open(file_name, 32000, "r", true); // auto-rewind
110 std::cout << "Input file " << file_name << " 32 kHz mono." << std::endl;
111 } else {
112 pcm_in_fid_.Open(FLAGS_input, static_cast<uint16_t>(FLAGS_input_fs_hz),
113 "r", true); // auto-rewind
114 std::cout << "Input file " << FLAGS_input << "at " << FLAGS_input_fs_hz
115 << " Hz in " << ((FLAGS_input_stereo) ? "stereo." : "mono.")
116 << std::endl;
117 pcm_in_fid_.ReadStereo(FLAGS_input_stereo);
118 }
119
120 ASSERT_TRUE(pcm_out_fid_ != NULL);
121 std::cout << "Output file " << FLAGS_output << " at " << FLAGS_output_fs_hz
122 << " Hz." << std::endl;
123
124 // Other setups
turaj@webrtc.orgd0631e32013-06-06 19:00:09125 if (FLAGS_loss_rate > 0)
126 loss_threshold_ = RAND_MAX * FLAGS_loss_rate;
127 else
128 loss_threshold_ = 0;
129 }
130
131 void TickOneMillisecond(uint32_t* action) {
132 // One millisecond passed.
133 time_to_insert_packet_ms_--;
134 time_to_playout_audio_ms_--;
135 sender_clock_->AdvanceTimeMilliseconds(1);
136 receiver_clock_->AdvanceTimeMilliseconds(1);
137
138 // Reset action.
139 *action = 0;
140
141 // Is it time to pull audio?
142 if (time_to_playout_audio_ms_ == 0) {
143 time_to_playout_audio_ms_ = kPlayoutPeriodMs;
henrik.lundinb8896d22016-05-17 19:21:55144 bool muted;
turaj@webrtc.orgd0631e32013-06-06 19:00:09145 receive_acm_->PlayoutData10Ms(static_cast<int>(FLAGS_output_fs_hz),
henrik.lundinb8896d22016-05-17 19:21:55146 &frame_, &muted);
147 ASSERT_FALSE(muted);
turaj@webrtc.orgd0631e32013-06-06 19:00:09148 fwrite(frame_.data_, sizeof(frame_.data_[0]),
149 frame_.samples_per_channel_ * frame_.num_channels_, pcm_out_fid_);
150 *action |= kAudioPlayedOut;
151 }
152
153 // Is it time to push in next packet?
154 if (time_to_insert_packet_ms_ <= .5) {
155 *action |= kPacketPushedIn;
156
157 // Update time-to-insert packet.
158 uint32_t t = next_receive_ts_;
159 next_receive_ts_ = ReceiveTimestamp();
160 time_to_insert_packet_ms_ += static_cast<float>(next_receive_ts_ - t) /
161 samples_in_1ms_;
162
163 // Push in just enough audio.
164 for (int n = 0; n < num_10ms_in_codec_frame_; n++) {
165 pcm_in_fid_.Read10MsData(frame_);
henrik.lundin@webrtc.org57277692015-03-02 12:29:30166 EXPECT_GE(send_acm_->Add10MsData(frame_), 0);
turaj@webrtc.orgd0631e32013-06-06 19:00:09167 }
168
169 // Set the parameters for the packet to be pushed in receiver ACM right
170 // now.
171 uint32_t ts = SendTimestamp();
172 int seq_num = SequenceNumber();
173 bool lost = false;
174 channel_->set_send_timestamp(ts);
175 channel_->set_sequence_number(seq_num);
176 if (loss_threshold_ > 0 && rand() < loss_threshold_) {
177 channel_->set_num_packets_to_drop(1);
178 lost = true;
179 }
180
turaj@webrtc.orgd0631e32013-06-06 19:00:09181 if (FLAGS_verbose) {
182 if (!lost) {
183 std::cout << "\nInserting packet number " << seq_num
184 << " timestamp " << ts << std::endl;
185 } else {
186 std::cout << "\nLost packet number " << seq_num
187 << " timestamp " << ts << std::endl;
188 }
189 }
190 }
191 }
192
193 void TearDown() {
turaj@webrtc.orgd0631e32013-06-06 19:00:09194 delete channel_;
195
196 fclose(seq_num_fid_);
197 fclose(send_ts_fid_);
198 fclose(receive_ts_fid_);
199 fclose(pcm_out_fid_);
200 pcm_in_fid_.Close();
201 }
202
203 ~InsertPacketWithTiming() {
204 delete sender_clock_;
205 delete receiver_clock_;
206 }
207
208 // Are there more info to simulate.
209 bool HasPackets() {
210 if (feof(seq_num_fid_) || feof(send_ts_fid_) || feof(receive_ts_fid_))
211 return false;
212 return true;
213 }
214
215 // Jitter buffer delay.
216 void Delay(int* optimal_delay, int* current_delay) {
minyue@webrtc.org5f8d2882015-02-18 15:24:13217 NetworkStatistics statistics;
218 receive_acm_->GetNetworkStatistics(&statistics);
turaj@webrtc.orgd0631e32013-06-06 19:00:09219 *optimal_delay = statistics.preferredBufferSize;
220 *current_delay = statistics.currentBufferSize;
221 }
222
223 private:
224 uint32_t SendTimestamp() {
225 uint32_t t;
226 EXPECT_EQ(1, fscanf(send_ts_fid_, "%u\n", &t));
227 return t;
228 }
229
230 uint32_t ReceiveTimestamp() {
231 uint32_t t;
232 EXPECT_EQ(1, fscanf(receive_ts_fid_, "%u\n", &t));
233 return t;
234 }
235
236 int SequenceNumber() {
237 int n;
238 EXPECT_EQ(1, fscanf(seq_num_fid_, "%d\n", &n));
239 return n;
240 }
241
242 // This class just creates these pointers, not deleting them. They are deleted
243 // by the associated ACM.
244 SimulatedClock* sender_clock_;
245 SimulatedClock* receiver_clock_;
246
kwiberg4df87572016-02-15 04:40:57247 std::unique_ptr<AudioCodingModule> send_acm_;
248 std::unique_ptr<AudioCodingModule> receive_acm_;
turaj@webrtc.orgd0631e32013-06-06 19:00:09249 Channel* channel_;
250
251 FILE* seq_num_fid_; // Input (text), one sequence number per line.
252 FILE* send_ts_fid_; // Input (text), one send timestamp per line.
253 FILE* receive_ts_fid_; // Input (text), one receive timestamp per line.
254 FILE* pcm_out_fid_; // Output PCM16.
255
256 PCMFile pcm_in_fid_; // Input PCM16.
257
258 int samples_in_1ms_;
259
260 // TODO(turajs): this can be computed from the send timestamp, but there is
261 // some complication to account for lost and reordered packets.
262 int num_10ms_in_codec_frame_;
263
264 float time_to_insert_packet_ms_;
265 uint32_t next_receive_ts_;
266 uint32_t time_to_playout_audio_ms_;
267
268 AudioFrame frame_;
269
270 double loss_threshold_;
271
272 // Output (text), sequence number, playout timestamp, time (ms) of playout,
273 // per line.
274 FILE* playout_timing_fid_;
275};
276
277} // webrtc
278
279int main(int argc, char* argv[]) {
280 google::ParseCommandLineFlags(&argc, &argv, true);
281 webrtc::InsertPacketWithTiming test;
282 test.SetUp();
283
284 FILE* delay_log = NULL;
285 if (FLAGS_delay.size() > 0) {
286 delay_log = fopen(FLAGS_delay.c_str(), "wt");
287 if (delay_log == NULL) {
288 std::cout << "Cannot open the file to log delay values." << std::endl;
289 exit(1);
290 }
291 }
292
293 uint32_t action_taken;
294 int optimal_delay_ms;
295 int current_delay_ms;
296 while (test.HasPackets()) {
297 test.TickOneMillisecond(&action_taken);
298
299 if (action_taken != 0) {
300 test.Delay(&optimal_delay_ms, &current_delay_ms);
301 if (delay_log != NULL) {
302 fprintf(delay_log, "%3d %3d\n", optimal_delay_ms, current_delay_ms);
303 }
304 }
305 }
306 std::cout << std::endl;
307 test.TearDown();
308 if (delay_log != NULL)
309 fclose(delay_log);
310}