// Copyright 2014 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/utility/media_galleries/ipc_data_source.h" #include "base/message_loop/message_loop_proxy.h" #include "chrome/common/extensions/chrome_utility_extensions_messages.h" #include "content/public/utility/utility_thread.h" namespace metadata { IPCDataSource::IPCDataSource(int64 total_size) : total_size_(total_size), utility_task_runner_(base::MessageLoopProxy::current()), next_request_id_(0) { data_source_thread_checker_.DetachFromThread(); } IPCDataSource::~IPCDataSource() { DCHECK(utility_thread_checker_.CalledOnValidThread()); } void IPCDataSource::Stop() { DCHECK(data_source_thread_checker_.CalledOnValidThread()); } void IPCDataSource::Read(int64 position, int size, uint8* data, const DataSource::ReadCB& read_cb) { DCHECK(data_source_thread_checker_.CalledOnValidThread()); utility_task_runner_->PostTask( FROM_HERE, base::Bind(&IPCDataSource::ReadOnUtilityThread, base::Unretained(this), position, size, data, read_cb)); } bool IPCDataSource::GetSize(int64* size_out) { DCHECK(data_source_thread_checker_.CalledOnValidThread()); *size_out = total_size_; return true; } bool IPCDataSource::IsStreaming() { DCHECK(data_source_thread_checker_.CalledOnValidThread()); return false; } void IPCDataSource::SetBitrate(int bitrate) { DCHECK(data_source_thread_checker_.CalledOnValidThread()); } bool IPCDataSource::OnMessageReceived(const IPC::Message& message) { DCHECK(utility_thread_checker_.CalledOnValidThread()); bool handled = true; IPC_BEGIN_MESSAGE_MAP(IPCDataSource, message) IPC_MESSAGE_HANDLER(ChromeUtilityMsg_RequestBlobBytes_Finished, OnRequestBlobBytesFinished) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } IPCDataSource::Request::Request() : destination(NULL) { } IPCDataSource::Request::~Request() { } void IPCDataSource::ReadOnUtilityThread(int64 position, int size, uint8* data, const DataSource::ReadCB& read_cb) { DCHECK(utility_thread_checker_.CalledOnValidThread()); CHECK_GE(total_size_, 0); CHECK_GE(position, 0); CHECK_GE(size, 0); // Cap position and size within bounds. position = std::min(position, total_size_); int64 clamped_size = std::min(static_cast(size), total_size_ - position); int64 request_id = ++next_request_id_; Request request; request.destination = data; request.callback = read_cb; pending_requests_[request_id] = request; content::UtilityThread::Get()->Send(new ChromeUtilityHostMsg_RequestBlobBytes( request_id, position, clamped_size)); } void IPCDataSource::OnRequestBlobBytesFinished(int64 request_id, const std::string& bytes) { DCHECK(utility_thread_checker_.CalledOnValidThread()); std::map::iterator it = pending_requests_.find(request_id); if (it == pending_requests_.end()) return; std::copy(bytes.begin(), bytes.end(), it->second.destination); it->second.callback.Run(bytes.size()); pending_requests_.erase(it); } } // namespace metadata