1 // Copyright 2012 The Chromium Authors
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "media/filters/blocking_url_protocol.h"
10 #include "base/synchronization/waitable_event.h"
11 #include "base/threading/thread_restrictions.h"
12 #include "media/base/data_source.h"
13 #include "media/ffmpeg/ffmpeg_common.h"
17 BlockingUrlProtocol::BlockingUrlProtocol(DataSource* data_source,
18 const base::RepeatingClosure& error_cb)
19 : data_source_(data_source),
21 is_streaming_(data_source_->IsStreaming()),
22 aborted_(base::WaitableEvent::ResetPolicy::MANUAL,
23 base::WaitableEvent::InitialState::NOT_SIGNALED), // We never
27 read_complete_(base::WaitableEvent::ResetPolicy::AUTOMATIC,
28 base::WaitableEvent::InitialState::NOT_SIGNALED),
32 BlockingUrlProtocol::~BlockingUrlProtocol() = default;
34 void BlockingUrlProtocol::Abort() {
36 base::AutoLock lock(data_source_lock_);
37 data_source_ = nullptr;
40 int BlockingUrlProtocol::Read(int size, uint8_t* data) {
42 // Read errors are unrecoverable.
43 base::AutoLock lock(data_source_lock_);
45 DCHECK(aborted_.IsSignaled());
49 // Not sure this can happen, but it's unclear from the ffmpeg code, so guard
57 if (data_source_->GetSize(&file_size) && read_position_ >= file_size)
60 // Blocking read from data source until either:
61 // 1) |last_read_bytes_| is set and |read_complete_| is signalled
62 // 2) |aborted_| is signalled
63 data_source_->Read(read_position_, size, data,
64 base::BindOnce(&BlockingUrlProtocol::SignalReadCompleted,
65 base::Unretained(this)));
68 base::WaitableEvent* events[] = { &aborted_, &read_complete_ };
71 base::ScopedAllowBaseSyncPrimitives allow_base_sync_primitives;
72 index = base::WaitableEvent::WaitMany(events, std::size(events));
75 if (events[index] == &aborted_)
78 if (last_read_bytes_ == DataSource::kReadError) {
84 if (last_read_bytes_ == DataSource::kAborted)
87 read_position_ += last_read_bytes_;
88 return last_read_bytes_;
91 bool BlockingUrlProtocol::GetPosition(int64_t* position_out) {
92 *position_out = read_position_;
96 bool BlockingUrlProtocol::SetPosition(int64_t position) {
97 base::AutoLock lock(data_source_lock_);
100 (data_source_->GetSize(&file_size) && position > file_size) ||
105 read_position_ = position;
109 bool BlockingUrlProtocol::GetSize(int64_t* size_out) {
110 base::AutoLock lock(data_source_lock_);
111 return data_source_ ? data_source_->GetSize(size_out) : false;
114 bool BlockingUrlProtocol::IsStreaming() {
115 return is_streaming_;
118 void BlockingUrlProtocol::SignalReadCompleted(int size) {
119 last_read_bytes_ = size;
120 read_complete_.Signal();