Refactored ViEFileRecorder.
Types and arguments will be done in a later CL. Review URL: http://webrtc-codereview.appspot.com/317008 git-svn-id: http://webrtc.googlecode.com/svn/trunk@1206 4adac7df-926f-26a2-2b94-8c16560cd09d
This commit is contained in:
		| @@ -8,274 +8,233 @@ | |||||||
|  *  be found in the AUTHORS file in the root of the source tree. |  *  be found in the AUTHORS file in the root of the source tree. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| /* | #include "video_engine/vie_file_recorder.h" | ||||||
|  * vie_file_recorder.cc |  | ||||||
|  * | #include "modules/utility/interface/file_player.h" | ||||||
|  */ | #include "modules/utility/interface/file_recorder.h" | ||||||
| #include "vie_file_recorder.h" | #include "system_wrappers/interface/critical_section_wrapper.h" | ||||||
| #include "critical_section_wrapper.h" | #include "system_wrappers/interface/tick_util.h" | ||||||
| #include "trace.h" | #include "system_wrappers/interface/trace.h" | ||||||
| #include "tick_util.h" | #include "video_engine/vie_defines.h" | ||||||
| #include "file_player.h" |  | ||||||
| #include "file_recorder.h" |  | ||||||
| #include "vie_defines.h" |  | ||||||
|  |  | ||||||
| namespace webrtc { | namespace webrtc { | ||||||
|  |  | ||||||
| ViEFileRecorder::ViEFileRecorder(int instanceID) | ViEFileRecorder::ViEFileRecorder(int instanceID) | ||||||
|     :   _ptrCritSec(CriticalSectionWrapper::CreateCriticalSection()), |     : recorder_cs_(CriticalSectionWrapper::CreateCriticalSection()), | ||||||
|         _fileRecorder(NULL), _isFirstFrameRecorded(false), |       file_recorder_(NULL), | ||||||
|         _isOutStreamStarted(false), _instanceID(instanceID), _frameDelay(0), |       is_first_frame_recorded_(false), | ||||||
|         _audioChannel(-1), _audioSource(NO_AUDIO), |       is_out_stream_started_(false), | ||||||
|         _veFileInterface(NULL) |       instance_id_(instanceID), | ||||||
| { |       frame_delay_(0), | ||||||
|  |       audio_channel_(-1), | ||||||
|  |       audio_source_(NO_AUDIO), | ||||||
|  |       voe_file_interface_(NULL) { | ||||||
| } | } | ||||||
|  |  | ||||||
| ViEFileRecorder::~ViEFileRecorder() | ViEFileRecorder::~ViEFileRecorder() { | ||||||
| { |   StopRecording(); | ||||||
|     StopRecording(); |   delete recorder_cs_; | ||||||
|     delete _ptrCritSec; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| int ViEFileRecorder::StartRecording(const char* fileNameUTF8, | int ViEFileRecorder::StartRecording(const char* file_nameUTF8, | ||||||
|                                     const VideoCodec& codecInst, |                                     const VideoCodec& codec_inst, | ||||||
|                                     AudioSource audioSource, |                                     AudioSource audio_source, | ||||||
|                                     int audioChannel, |                                     int audio_channel, | ||||||
|                                     const webrtc::CodecInst audioCodecInst, |                                     const CodecInst audio_codec_inst, | ||||||
|                                     VoiceEngine* vePtr, |                                     VoiceEngine* voe_ptr, | ||||||
|                                     const webrtc::FileFormats fileFormat) |                                     const FileFormats file_format) { | ||||||
| { |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|  |  | ||||||
|     if (_fileRecorder) |   if (file_recorder_) { | ||||||
|     { |     WEBRTC_TRACE(kTraceError, kTraceVideo, instance_id_, | ||||||
|         WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, _instanceID, |                  "ViEFileRecorder::StartRecording() - already recording."); | ||||||
|                    "ViEFileRecorder::StartRecording() failed, already recording."); |  | ||||||
|         return -1; |  | ||||||
|     } |  | ||||||
|     _fileRecorder = FileRecorder::CreateFileRecorder(_instanceID, fileFormat); |  | ||||||
|     if (!_fileRecorder) |  | ||||||
|     { |  | ||||||
|         WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, _instanceID, |  | ||||||
|                    "ViEFileRecorder::StartRecording() failed to create file recoder."); |  | ||||||
|         return -1; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     int error = _fileRecorder->StartRecordingVideoFile(fileNameUTF8, |  | ||||||
|                                                        audioCodecInst, |  | ||||||
|                                                        codecInst, |  | ||||||
|                                                        AMRFileStorage, |  | ||||||
|                                                        audioSource == NO_AUDIO); |  | ||||||
|     if (error) |  | ||||||
|     { |  | ||||||
|         WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, _instanceID, |  | ||||||
|                    "ViEFileRecorder::StartRecording() failed to StartRecordingVideoFile."); |  | ||||||
|         FileRecorder::DestroyFileRecorder(_fileRecorder); |  | ||||||
|         _fileRecorder = NULL; |  | ||||||
|         return -1; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     _audioSource = audioSource; |  | ||||||
|     if (vePtr && audioSource != NO_AUDIO) // VeInterface have been provided and we want to record audio |  | ||||||
|     { |  | ||||||
|         _veFileInterface = VoEFile::GetInterface(vePtr); |  | ||||||
|         if (!_veFileInterface) |  | ||||||
|         { |  | ||||||
|             WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, _instanceID, |  | ||||||
|                        "ViEFileRecorder::StartRecording() failed to get VEFile interface"); |  | ||||||
|             return -1; |  | ||||||
|         } |  | ||||||
|         // always drive VoE in L16 |  | ||||||
|         CodecInst engineAudioCodecInst = { 96, // .pltype |  | ||||||
|                                             "L16", // .plname |  | ||||||
|                                             audioCodecInst.plfreq, // .plfreq |  | ||||||
|                                             audioCodecInst.plfreq / 100, // .pacsize (10ms) |  | ||||||
|                                             1, // .channels |  | ||||||
|                                             audioCodecInst.plfreq * 16 // .rate |  | ||||||
|                                          }; |  | ||||||
|  |  | ||||||
|         switch (audioSource) |  | ||||||
|         { |  | ||||||
|             case MICROPHONE: |  | ||||||
|                 error |  | ||||||
|                     = _veFileInterface->StartRecordingMicrophone( |  | ||||||
|                                                                  this, |  | ||||||
|                                                                  &engineAudioCodecInst); |  | ||||||
|                 break; |  | ||||||
|             case PLAYOUT: |  | ||||||
|                 error |  | ||||||
|                     = _veFileInterface->StartRecordingPlayout( |  | ||||||
|                                                               audioChannel, |  | ||||||
|                                                               this, |  | ||||||
|                                                               &engineAudioCodecInst); |  | ||||||
|                 break; |  | ||||||
|             case NO_AUDIO: |  | ||||||
|                 break; |  | ||||||
|             default: |  | ||||||
|                 assert(!"Unknown audioSource"); |  | ||||||
|         } |  | ||||||
|         if (error != 0) |  | ||||||
|         { |  | ||||||
|             WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, _instanceID, |  | ||||||
|                        "ViEFileRecorder::StartRecording() failed to start recording audio"); |  | ||||||
|             FileRecorder::DestroyFileRecorder(_fileRecorder); |  | ||||||
|             _fileRecorder = NULL; |  | ||||||
|             return -1; |  | ||||||
|         } |  | ||||||
|         _isOutStreamStarted = true; |  | ||||||
|         _audioChannel = audioChannel; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     _isFirstFrameRecorded = false; |  | ||||||
|     return 0; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| int ViEFileRecorder::StopRecording() |  | ||||||
| { |  | ||||||
|  |  | ||||||
|     int error = 0; |  | ||||||
|     // Stop recording audio |  | ||||||
|     // Note - we can not hold the _ptrCritSect while accessing VE functions. It might cause deadlock in Write |  | ||||||
|     if (_veFileInterface) |  | ||||||
|     { |  | ||||||
|         switch (_audioSource) |  | ||||||
|         { |  | ||||||
|             case MICROPHONE: |  | ||||||
|                 error = _veFileInterface->StopRecordingMicrophone(); |  | ||||||
|                 break; |  | ||||||
|             case PLAYOUT: |  | ||||||
|                 error = _veFileInterface->StopRecordingPlayout(_audioChannel); |  | ||||||
|                 break; |  | ||||||
|             case NO_AUDIO: |  | ||||||
|                 break; |  | ||||||
|             default: |  | ||||||
|                 assert(!"Unknown audioSource"); |  | ||||||
|         } |  | ||||||
|         if (error != 0) |  | ||||||
|         { |  | ||||||
|             WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, _instanceID, |  | ||||||
|                        "ViEFileRecorder::StopRecording() failed to stop recording audio"); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|     if (_veFileInterface) |  | ||||||
|     { |  | ||||||
|         _veFileInterface->Release(); |  | ||||||
|         _veFileInterface = NULL; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (_fileRecorder) |  | ||||||
|     { |  | ||||||
|         if (_fileRecorder->IsRecording()) |  | ||||||
|         { |  | ||||||
|             int error = _fileRecorder->StopRecording(); |  | ||||||
|             if (error) |  | ||||||
|             { |  | ||||||
|                 return -1; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         FileRecorder::DestroyFileRecorder(_fileRecorder); |  | ||||||
|         _fileRecorder = NULL; |  | ||||||
|     } |  | ||||||
|     _isFirstFrameRecorded = false; |  | ||||||
|     _isOutStreamStarted = false; |  | ||||||
|     return 0; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void ViEFileRecorder::SetFrameDelay(int frameDelay) |  | ||||||
| { |  | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|     _frameDelay = frameDelay; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool ViEFileRecorder::RecordingStarted() |  | ||||||
| { |  | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|     return _fileRecorder && _fileRecorder->IsRecording(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool ViEFileRecorder::FirstFrameRecorded() |  | ||||||
| { |  | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|     return _isFirstFrameRecorded; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool ViEFileRecorder::IsRecordingFileFormat(const webrtc::FileFormats fileFormat) |  | ||||||
| { |  | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|     return (_fileRecorder->RecordingFileFormat() == fileFormat) ? true : false; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /******************************************************************************* |  | ||||||
|  * void RecordVideoFrame() |  | ||||||
|  * |  | ||||||
|  * Records incoming decoded video frame to AVI-file. |  | ||||||
|  * |  | ||||||
|  */ |  | ||||||
| void ViEFileRecorder::RecordVideoFrame(const VideoFrame& videoFrame) |  | ||||||
| { |  | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|  |  | ||||||
|     if (_fileRecorder && _fileRecorder->IsRecording()) |  | ||||||
|     { |  | ||||||
|         if (!IsRecordingFileFormat(webrtc::kFileFormatAviFile)) |  | ||||||
|         { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         //Compensate for frame delay in order to get audiosync when recording local video. |  | ||||||
|         const WebRtc_UWord32 timeStamp = videoFrame.TimeStamp(); |  | ||||||
|         const WebRtc_Word64 renderTimeStamp = videoFrame.RenderTimeMs(); |  | ||||||
|         VideoFrame& unconstVideoFrame = |  | ||||||
|             const_cast<VideoFrame&> (videoFrame); |  | ||||||
|         unconstVideoFrame.SetTimeStamp(timeStamp - 90 * _frameDelay); |  | ||||||
|         unconstVideoFrame.SetRenderTime(renderTimeStamp - _frameDelay); |  | ||||||
|  |  | ||||||
|         _fileRecorder->RecordVideoToFile(unconstVideoFrame); |  | ||||||
|  |  | ||||||
|         unconstVideoFrame.SetRenderTime(renderTimeStamp); |  | ||||||
|         unconstVideoFrame.SetTimeStamp(timeStamp); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // --------------------- |  | ||||||
| // From OutStream |  | ||||||
| // --------------------- |  | ||||||
| // 10 ms block of PCM 16 |  | ||||||
| bool ViEFileRecorder::Write(const void* buf, int len) |  | ||||||
| { |  | ||||||
|     if (!_isOutStreamStarted) |  | ||||||
|         return true; |  | ||||||
|  |  | ||||||
|     // always L16 from VoCE |  | ||||||
|     if (len % (2 * 80)) // 2 bytes 80 samples |  | ||||||
|     { |  | ||||||
|         WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, _audioChannel, |  | ||||||
|                    "Audio length not supported: %d.", len); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|     AudioFrame audioFrame; |  | ||||||
|     WebRtc_UWord16 lengthInSamples = len / 2; |  | ||||||
|  |  | ||||||
|     audioFrame.UpdateFrame(_audioChannel, 0, (const WebRtc_Word16*) buf, |  | ||||||
|                            lengthInSamples, lengthInSamples * 100, |  | ||||||
|                            AudioFrame::kUndefined, |  | ||||||
|                            AudioFrame::kVadUnknown); |  | ||||||
|  |  | ||||||
|     CriticalSectionScoped lock(*_ptrCritSec); |  | ||||||
|  |  | ||||||
|     if (_fileRecorder && _fileRecorder->IsRecording()) |  | ||||||
|     { |  | ||||||
|         TickTime tickTime = TickTime::Now(); |  | ||||||
|         _fileRecorder->RecordAudioToFile(audioFrame, &tickTime); |  | ||||||
|     } |  | ||||||
|     return true; // Always return true! |  | ||||||
| } |  | ||||||
|  |  | ||||||
| int ViEFileRecorder::Rewind() |  | ||||||
| { |  | ||||||
|     // Not supported! |  | ||||||
|     return -1; |     return -1; | ||||||
| } |   } | ||||||
| } // namespace webrtc |   file_recorder_ = FileRecorder::CreateFileRecorder(instance_id_, file_format); | ||||||
|  |   if (!file_recorder_) { | ||||||
|  |     WEBRTC_TRACE(kTraceError, kTraceVideo, instance_id_, | ||||||
|  |                  "ViEFileRecorder::StartRecording() failed to create recoder."); | ||||||
|  |     return -1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   int error = file_recorder_->StartRecordingVideoFile(file_nameUTF8, | ||||||
|  |                                                       audio_codec_inst, | ||||||
|  |                                                       codec_inst, | ||||||
|  |                                                       AMRFileStorage, | ||||||
|  |                                                       audio_source == NO_AUDIO); | ||||||
|  |   if (error) { | ||||||
|  |     WEBRTC_TRACE(kTraceError, kTraceVideo, instance_id_, | ||||||
|  |                  "ViEFileRecorder::StartRecording() failed to " | ||||||
|  |                  "StartRecordingVideoFile."); | ||||||
|  |     FileRecorder::DestroyFileRecorder(file_recorder_); | ||||||
|  |     file_recorder_ = NULL; | ||||||
|  |     return -1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   audio_source_ = audio_source; | ||||||
|  |   if (voe_ptr && audio_source != NO_AUDIO) { | ||||||
|  |     // VoE interface has been provided and we want to record audio. | ||||||
|  |     voe_file_interface_ = VoEFile::GetInterface(voe_ptr); | ||||||
|  |     if (!voe_file_interface_) { | ||||||
|  |       WEBRTC_TRACE(kTraceError, kTraceVideo, instance_id_, | ||||||
|  |                    "ViEFileRecorder::StartRecording() failed to get VEFile " | ||||||
|  |                    "interface"); | ||||||
|  |       return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Always L16. | ||||||
|  |     CodecInst engine_audio_codec_inst = {96, "L16", audio_codec_inst.plfreq, | ||||||
|  |                                          audio_codec_inst.plfreq / 100, 1, | ||||||
|  |                                          audio_codec_inst.plfreq * 16 }; | ||||||
|  |  | ||||||
|  |     switch (audio_source) { | ||||||
|  |       case MICROPHONE: | ||||||
|  |         error = voe_file_interface_->StartRecordingMicrophone( | ||||||
|  |             this, &engine_audio_codec_inst); | ||||||
|  |         break; | ||||||
|  |       case PLAYOUT: | ||||||
|  |         error = voe_file_interface_->StartRecordingPlayout( | ||||||
|  |             audio_channel, this, &engine_audio_codec_inst); | ||||||
|  |         break; | ||||||
|  |       case NO_AUDIO: | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         assert(!"Unknown audio_source"); | ||||||
|  |     } | ||||||
|  |     if (error != 0) { | ||||||
|  |       WEBRTC_TRACE(kTraceError, kTraceVideo, instance_id_, | ||||||
|  |                    "ViEFileRecorder::StartRecording() failed to start recording" | ||||||
|  |                    " audio"); | ||||||
|  |       FileRecorder::DestroyFileRecorder(file_recorder_); | ||||||
|  |       file_recorder_ = NULL; | ||||||
|  |       return -1; | ||||||
|  |     } | ||||||
|  |     is_out_stream_started_ = true; | ||||||
|  |     audio_channel_ = audio_channel; | ||||||
|  |   } | ||||||
|  |   is_first_frame_recorded_ = false; | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int ViEFileRecorder::StopRecording() { | ||||||
|  |   int error = 0; | ||||||
|  |   // We can not hold the ptr_cs_ while accessing VoE functions. It might cause | ||||||
|  |   // deadlock in Write. | ||||||
|  |   if (voe_file_interface_) { | ||||||
|  |     switch (audio_source_) { | ||||||
|  |       case MICROPHONE: | ||||||
|  |         error = voe_file_interface_->StopRecordingMicrophone(); | ||||||
|  |         break; | ||||||
|  |       case PLAYOUT: | ||||||
|  |         error = voe_file_interface_->StopRecordingPlayout(audio_channel_); | ||||||
|  |         break; | ||||||
|  |       case NO_AUDIO: | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         assert(!"Unknown audio_source"); | ||||||
|  |     } | ||||||
|  |     if (error != 0) { | ||||||
|  |       WEBRTC_TRACE(kTraceError, kTraceVideo, instance_id_, | ||||||
|  |                    "ViEFileRecorder::StopRecording() failed to stop recording " | ||||||
|  |                    "audio"); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|  |   if (voe_file_interface_) { | ||||||
|  |     voe_file_interface_->Release(); | ||||||
|  |     voe_file_interface_ = NULL; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (file_recorder_) { | ||||||
|  |     if (file_recorder_->IsRecording()) { | ||||||
|  |       int error = file_recorder_->StopRecording(); | ||||||
|  |       if (error) { | ||||||
|  |         return -1; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     FileRecorder::DestroyFileRecorder(file_recorder_); | ||||||
|  |     file_recorder_ = NULL; | ||||||
|  |   } | ||||||
|  |   is_first_frame_recorded_ = false; | ||||||
|  |   is_out_stream_started_ = false; | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void ViEFileRecorder::SetFrameDelay(int frame_delay) { | ||||||
|  |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|  |   frame_delay_ = frame_delay; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool ViEFileRecorder::RecordingStarted() { | ||||||
|  |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|  |   return file_recorder_ && file_recorder_->IsRecording(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool ViEFileRecorder::FirstFrameRecorded() { | ||||||
|  |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|  |   return is_first_frame_recorded_; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool ViEFileRecorder::IsRecordingFileFormat(const FileFormats file_format) { | ||||||
|  |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|  |   return (file_recorder_->RecordingFileFormat() == file_format) ? true : false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void ViEFileRecorder::RecordVideoFrame(const VideoFrame& video_frame) { | ||||||
|  |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|  |  | ||||||
|  |   if (file_recorder_ && file_recorder_->IsRecording()) { | ||||||
|  |     if (!IsRecordingFileFormat(kFileFormatAviFile)) | ||||||
|  |       return; | ||||||
|  |  | ||||||
|  |     // Compensate for frame delay in order to get audio/video sync when | ||||||
|  |     // recording local video. | ||||||
|  |     const WebRtc_UWord32 time_stamp = video_frame.TimeStamp(); | ||||||
|  |     const WebRtc_Word64 render_time_stamp = video_frame.RenderTimeMs(); | ||||||
|  |     VideoFrame& unconst_video_frame = const_cast<VideoFrame&>(video_frame); | ||||||
|  |     unconst_video_frame.SetTimeStamp(time_stamp - 90 * frame_delay_); | ||||||
|  |     unconst_video_frame.SetRenderTime(render_time_stamp - frame_delay_); | ||||||
|  |  | ||||||
|  |     file_recorder_->RecordVideoToFile(unconst_video_frame); | ||||||
|  |  | ||||||
|  |     unconst_video_frame.SetRenderTime(render_time_stamp); | ||||||
|  |     unconst_video_frame.SetTimeStamp(time_stamp); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool ViEFileRecorder::Write(const void* buf, int len) { | ||||||
|  |   if (!is_out_stream_started_) | ||||||
|  |     return true; | ||||||
|  |  | ||||||
|  |   // Always 10 ms L16 from VoE. | ||||||
|  |   if (len % (2 * 80)) { | ||||||
|  |     // Not 2 bytes 80 samples. | ||||||
|  |     WEBRTC_TRACE(kTraceError, kTraceVideo, audio_channel_, | ||||||
|  |                  "Audio length not supported: %d.", len); | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   AudioFrame audio_frame; | ||||||
|  |   WebRtc_UWord16 length_in_samples = len / 2; | ||||||
|  |   audio_frame.UpdateFrame(audio_channel_, 0, | ||||||
|  |                           static_cast<const WebRtc_Word16*>(buf), | ||||||
|  |                           length_in_samples, length_in_samples * 100, | ||||||
|  |                           AudioFrame::kUndefined, | ||||||
|  |                           AudioFrame::kVadUnknown); | ||||||
|  |  | ||||||
|  |   CriticalSectionScoped lock(*recorder_cs_); | ||||||
|  |   if (file_recorder_ && file_recorder_->IsRecording()) { | ||||||
|  |     TickTime tick_time = TickTime::Now(); | ||||||
|  |     file_recorder_->RecordAudioToFile(audio_frame, &tick_time); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Always return true to continue recording. | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int ViEFileRecorder::Rewind() { | ||||||
|  |   // Not supported! | ||||||
|  |   return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | }  // namespace webrtc | ||||||
|   | |||||||
| @@ -8,57 +8,58 @@ | |||||||
|  *  be found in the AUTHORS file in the root of the source tree. |  *  be found in the AUTHORS file in the root of the source tree. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| /* | #ifndef WEBRTC_VIDEO_ENGINE_VIE_FILE_RECORDER_H_ | ||||||
|  * vie_file_recorder.h | #define WEBRTC_VIDEO_ENGINE_VIE_FILE_RECORDER_H_ | ||||||
|  */ |  | ||||||
|  |  | ||||||
| #ifndef WEBRTC_VIDEO_ENGINE_MAIN_SOURCE_VIE_FILE_RECORDER_H_ |  | ||||||
| #define WEBRTC_VIDEO_ENGINE_MAIN_SOURCE_VIE_FILE_RECORDER_H_ |  | ||||||
|  |  | ||||||
|  | #include "modules/utility/interface/file_recorder.h" | ||||||
| #include "typedefs.h" | #include "typedefs.h" | ||||||
| #include "file_recorder.h" | #include "video_engine/main/interface/vie_file.h" | ||||||
| #include "vie_file.h" | #include "voice_engine/main/interface/voe_file.h" | ||||||
| #include "voe_file.h" |  | ||||||
|  |  | ||||||
| namespace webrtc { | namespace webrtc { | ||||||
|  |  | ||||||
| class CriticalSectionWrapper; | class CriticalSectionWrapper; | ||||||
|  |  | ||||||
| class ViEFileRecorder: protected webrtc::OutStream // for audio | class ViEFileRecorder : protected OutStream { | ||||||
| { |  public: | ||||||
| public: |   explicit ViEFileRecorder(int channel_id); | ||||||
|     ViEFileRecorder(int channelId); |   ~ViEFileRecorder(); | ||||||
|     ~ViEFileRecorder(); |  | ||||||
|  |  | ||||||
|     int StartRecording(const char* fileNameUTF8, |   int StartRecording(const char* file_nameUTF8, | ||||||
|                        const webrtc::VideoCodec& codecInst, |                      const VideoCodec& codec_inst, | ||||||
|                        AudioSource audioSource, int audioChannel, |                      AudioSource audio_source, int audio_channel, | ||||||
|                        const webrtc::CodecInst audioCodecInst, |                      const CodecInst audio_codec_inst, | ||||||
|                        VoiceEngine* vePtr, |                      VoiceEngine* voe_ptr, | ||||||
|                        const webrtc::FileFormats fileFormat = webrtc::kFileFormatAviFile); |                      const FileFormats file_format = kFileFormatAviFile); | ||||||
|     int StopRecording(); |   int StopRecording(); | ||||||
|     void SetFrameDelay(int frameDelay); |  | ||||||
|     bool RecordingStarted(); |  | ||||||
|     void RecordVideoFrame(const VideoFrame& videoFrame); |  | ||||||
|  |  | ||||||
| protected: |   void SetFrameDelay(int frame_delay); | ||||||
|     bool FirstFrameRecorded(); |   bool RecordingStarted(); | ||||||
|     bool IsRecordingFileFormat(const webrtc::FileFormats fileFormat); |  | ||||||
|     // From webrtc::OutStream |  | ||||||
|     bool Write(const void* buf, int len); |  | ||||||
|     int Rewind(); |  | ||||||
|  |  | ||||||
| private: |   // Records incoming decoded video frame to file. | ||||||
|     CriticalSectionWrapper* _ptrCritSec; |   void RecordVideoFrame(const VideoFrame& video_frame); | ||||||
|  |  | ||||||
|     FileRecorder* _fileRecorder; |  protected: | ||||||
|     bool _isFirstFrameRecorded; |   bool FirstFrameRecorded(); | ||||||
|     bool _isOutStreamStarted; |   bool IsRecordingFileFormat(const FileFormats file_format); | ||||||
|     int _instanceID; |  | ||||||
|     int _frameDelay; |   // Implements OutStream. | ||||||
|     int _audioChannel; |   bool Write(const void* buf, int len); | ||||||
|     AudioSource _audioSource; |   int Rewind(); | ||||||
|     VoEFile* _veFileInterface; |  | ||||||
|  |  private: | ||||||
|  |   CriticalSectionWrapper* recorder_cs_; | ||||||
|  |  | ||||||
|  |   FileRecorder* file_recorder_; | ||||||
|  |   bool is_first_frame_recorded_; | ||||||
|  |   bool is_out_stream_started_; | ||||||
|  |   int instance_id_; | ||||||
|  |   int frame_delay_; | ||||||
|  |   int audio_channel_; | ||||||
|  |   AudioSource audio_source_; | ||||||
|  |   VoEFile* voe_file_interface_; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| } // namespace webrtc | }  // namespace webrtc | ||||||
| #endif // WEBRTC_VIDEO_ENGINE_MAIN_SOURCE_VIE_FILE_RECORDER_H_ |  | ||||||
|  | #endif  // WEBRTC_VIDEO_ENGINE_VIE_FILE_RECORDER_H_ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 mflodman@webrtc.org
					mflodman@webrtc.org