blob: 5fe9f2139f0ff3d325b53a3478cec4b3caad3488 [file] [log] [blame]
// Copyright (c) 2013 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 "media/midi/midi_manager.h"
#include "base/bind.h"
#include "base/message_loop/message_loop.h"
#include "base/metrics/histogram_macros.h"
#include "base/trace_event/trace_event.h"
namespace media {
namespace midi {
namespace {
using Sample = base::HistogramBase::Sample;
// If many users have more devices, this number will be increased.
// But the number is expected to be big enough for now.
const Sample kMaxUmaDevices = 31;
// Used to count events for usage histogram.
enum class Usage {
CREATED,
CREATED_ON_UNSUPPORTED_PLATFORMS,
SESSION_STARTED,
SESSION_ENDED,
INITIALIZED,
INPUT_PORT_ADDED,
OUTPUT_PORT_ADDED,
// New items should be inserted here, and |MAX| should point the last item.
MAX = INITIALIZED,
};
// Used in StartSession.
enum class Completion {
COMPLETE_SYNCHRONOUSLY,
INVOKE_INITIALIZATION,
COMPLETE_ASYNCHRONOUSLY,
};
void ReportUsage(Usage usage) {
UMA_HISTOGRAM_ENUMERATION("Media.Midi.Usage",
static_cast<Sample>(usage),
static_cast<Sample>(Usage::MAX) + 1);
}
} // namespace
MidiManager::MidiManager()
: initialized_(false), finalized_(false), result_(Result::NOT_INITIALIZED) {
ReportUsage(Usage::CREATED);
}
MidiManager::~MidiManager() {
// Make sure that Finalize() is called to clean up resources allocated on
// the Chrome_IOThread.
DCHECK(finalized_);
}
#if !defined(OS_MACOSX) && !defined(OS_WIN) && \
!(defined(USE_ALSA) && defined(USE_UDEV)) && !defined(OS_ANDROID)
MidiManager* MidiManager::Create() {
ReportUsage(Usage::CREATED_ON_UNSUPPORTED_PLATFORMS);
return new MidiManager;
}
#endif
void MidiManager::Shutdown() {
UMA_HISTOGRAM_ENUMERATION("Media.Midi.ResultOnShutdown",
static_cast<int>(result_),
static_cast<int>(Result::MAX) + 1);
base::AutoLock auto_lock(lock_);
if (session_thread_runner_) {
session_thread_runner_->PostTask(
FROM_HERE, base::Bind(&MidiManager::ShutdownOnSessionThread,
base::Unretained(this)));
session_thread_runner_ = nullptr;
} else {
finalized_ = true;
}
}
void MidiManager::StartSession(MidiManagerClient* client) {
ReportUsage(Usage::SESSION_STARTED);
Completion completion = Completion::COMPLETE_SYNCHRONOUSLY;
Result result = Result::NOT_INITIALIZED;
{
base::AutoLock auto_lock(lock_);
if (clients_.find(client) != clients_.end() ||
pending_clients_.find(client) != pending_clients_.end()) {
// Should not happen. But just in case the renderer is compromised.
NOTREACHED();
return;
}
if (initialized_) {
// Platform dependent initialization was already finished for previously
// initialized clients.
if (result_ == Result::OK) {
AddInitialPorts(client);
clients_.insert(client);
}
// Complete synchronously with |result_|;
result = result_;
} else {
bool too_many_pending_clients_exist =
pending_clients_.size() >= kMaxPendingClientCount;
// Do not accept a new request if the pending client list contains too
// many clients, or Shutdown() was already called.
if (too_many_pending_clients_exist || finalized_) {
result = Result::INITIALIZATION_ERROR;
} else {
// Call StartInitialization() only for the first request.
if (pending_clients_.empty()) {
completion = Completion::INVOKE_INITIALIZATION;
session_thread_runner_ = base::MessageLoop::current()->task_runner();
} else {
completion = Completion::COMPLETE_ASYNCHRONOUSLY;
}
pending_clients_.insert(client);
}
}
}
if (completion == Completion::COMPLETE_SYNCHRONOUSLY) {
client->CompleteStartSession(result);
} else if (completion == Completion::INVOKE_INITIALIZATION) {
// Lazily initialize the MIDI back-end.
TRACE_EVENT0("midi", "MidiManager::StartInitialization");
// CompleteInitialization() will be called asynchronously when platform
// dependent initialization is finished.
StartInitialization();
}
}
void MidiManager::EndSession(MidiManagerClient* client) {
ReportUsage(Usage::SESSION_ENDED);
// At this point, |client| can be in the destruction process, and calling
// any method of |client| is dangerous.
base::AutoLock auto_lock(lock_);
clients_.erase(client);
pending_clients_.erase(client);
}
void MidiManager::AccumulateMidiBytesSent(MidiManagerClient* client, size_t n) {
base::AutoLock auto_lock(lock_);
if (clients_.find(client) == clients_.end())
return;
client->AccumulateMidiBytesSent(n);
}
void MidiManager::DispatchSendMidiData(MidiManagerClient* client,
uint32 port_index,
const std::vector<uint8>& data,
double timestamp) {
NOTREACHED();
}
void MidiManager::StartInitialization() {
CompleteInitialization(Result::NOT_SUPPORTED);
}
void MidiManager::CompleteInitialization(Result result) {
base::AutoLock auto_lock(lock_);
if (session_thread_runner_) {
session_thread_runner_->PostTask(
FROM_HERE, base::Bind(&MidiManager::CompleteInitializationInternal,
base::Unretained(this), result));
}
}
void MidiManager::AddInputPort(const MidiPortInfo& info) {
ReportUsage(Usage::INPUT_PORT_ADDED);
base::AutoLock auto_lock(lock_);
input_ports_.push_back(info);
for (auto client : clients_)
client->AddInputPort(info);
}
void MidiManager::AddOutputPort(const MidiPortInfo& info) {
ReportUsage(Usage::OUTPUT_PORT_ADDED);
base::AutoLock auto_lock(lock_);
output_ports_.push_back(info);
for (auto client : clients_)
client->AddOutputPort(info);
}
void MidiManager::SetInputPortState(uint32 port_index, MidiPortState state) {
base::AutoLock auto_lock(lock_);
DCHECK_LT(port_index, input_ports_.size());
input_ports_[port_index].state = state;
for (auto client : clients_)
client->SetInputPortState(port_index, state);
}
void MidiManager::SetOutputPortState(uint32 port_index, MidiPortState state) {
base::AutoLock auto_lock(lock_);
DCHECK_LT(port_index, output_ports_.size());
output_ports_[port_index].state = state;
for (auto client : clients_)
client->SetOutputPortState(port_index, state);
}
void MidiManager::ReceiveMidiData(
uint32 port_index,
const uint8* data,
size_t length,
double timestamp) {
base::AutoLock auto_lock(lock_);
for (auto client : clients_)
client->ReceiveMidiData(port_index, data, length, timestamp);
}
void MidiManager::CompleteInitializationInternal(Result result) {
TRACE_EVENT0("midi", "MidiManager::CompleteInitialization");
ReportUsage(Usage::INITIALIZED);
UMA_HISTOGRAM_ENUMERATION("Media.Midi.InputPorts",
static_cast<Sample>(input_ports_.size()),
kMaxUmaDevices + 1);
UMA_HISTOGRAM_ENUMERATION("Media.Midi.OutputPorts",
static_cast<Sample>(output_ports_.size()),
kMaxUmaDevices + 1);
base::AutoLock auto_lock(lock_);
DCHECK(clients_.empty());
DCHECK(!initialized_);
initialized_ = true;
result_ = result;
for (auto client : pending_clients_) {
if (result_ == Result::OK) {
AddInitialPorts(client);
clients_.insert(client);
}
client->CompleteStartSession(result_);
}
pending_clients_.clear();
}
void MidiManager::AddInitialPorts(MidiManagerClient* client) {
lock_.AssertAcquired();
for (const auto& info : input_ports_)
client->AddInputPort(info);
for (const auto& info : output_ports_)
client->AddOutputPort(info);
}
void MidiManager::ShutdownOnSessionThread() {
Finalize();
base::AutoLock auto_lock(lock_);
finalized_ = true;
// Detach all clients so that they do not call MidiManager methods any more.
for (auto client : clients_)
client->Detach();
}
} // namespace midi
} // namespace media