// 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 "media/midi/midi_manager_usb.h" #include "base/callback.h" #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "base/message_loop/message_loop.h" #include "media/midi/usb_midi_descriptor_parser.h" #include "media/midi/usb_midi_device.h" #include "media/midi/usb_midi_input_stream.h" #include "media/midi/usb_midi_jack.h" #include "media/midi/usb_midi_output_stream.h" namespace media { namespace { // Noop callback for (sync) Initialize. // TODO(yhirano): This function should go away when // MidiManager::Initialize() becomes asynchronous. See http://crbug.com/339746. void Noop(bool result) { } } // namespace MidiManagerUsb::MidiManagerUsb(scoped_ptr factory) : device_factory_(factory.Pass()) { } MidiManagerUsb::~MidiManagerUsb() { } bool MidiManagerUsb::Initialize() { Initialize(base::Bind(Noop)); return true; } void MidiManagerUsb::Initialize(base::Callback callback) { initialize_callback_ = callback; // This is safe because EnumerateDevices cancels the operation on destruction. device_factory_->EnumerateDevices( this, base::Bind(&MidiManagerUsb::OnEnumerateDevicesDone, base::Unretained(this))); } void MidiManagerUsb::DispatchSendMidiData(MidiManagerClient* client, uint32_t port_index, const std::vector& data, double timestamp) { DCHECK_LT(port_index, output_streams_.size()); output_streams_[port_index]->Send(data); client->AccumulateMidiBytesSent(data.size()); } void MidiManagerUsb::ReceiveUsbMidiData(UsbMidiDevice* device, int endpoint_number, const uint8* data, size_t size, double timestamp) { if (!input_stream_) return; input_stream_->OnReceivedData(device, endpoint_number, data, size, timestamp); } void MidiManagerUsb::OnReceivedData(size_t jack_index, const uint8* data, size_t size, double timestamp) { ReceiveMidiData(jack_index, data, size, timestamp); } void MidiManagerUsb::OnEnumerateDevicesDone(bool result, UsbMidiDevice::Devices* devices) { if (!result) { initialize_callback_.Run(false); return; } devices->swap(devices_); for (size_t i = 0; i < devices_.size(); ++i) { UsbMidiDescriptorParser parser; std::vector descriptor = devices_[i]->GetDescriptor(); const uint8* data = descriptor.size() > 0 ? &descriptor[0] : NULL; std::vector jacks; bool parse_result = parser.Parse(devices_[i], data, descriptor.size(), &jacks); if (!parse_result) { initialize_callback_.Run(false); return; } std::vector input_jacks; for (size_t j = 0; j < jacks.size(); ++j) { if (jacks[j].direction() == UsbMidiJack::DIRECTION_OUT) { output_streams_.push_back(new UsbMidiOutputStream(jacks[j])); // TODO(yhirano): Set appropriate properties. AddOutputPort(MidiPortInfo()); } else { DCHECK_EQ(jacks[j].direction(), UsbMidiJack::DIRECTION_IN); input_jacks.push_back(jacks[j]); // TODO(yhirano): Set appropriate properties. AddInputPort(MidiPortInfo()); } } input_stream_.reset(new UsbMidiInputStream(input_jacks, this)); } initialize_callback_.Run(true); } } // namespace media