1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "content/browser/renderer_host/media/midi_host.h"
8 #include "base/bind_helpers.h"
9 #include "base/debug/trace_event.h"
10 #include "base/process/process.h"
11 #include "content/browser/browser_main_loop.h"
12 #include "content/browser/child_process_security_policy_impl.h"
13 #include "content/browser/media/media_internals.h"
14 #include "content/common/media/midi_messages.h"
15 #include "content/public/browser/content_browser_client.h"
16 #include "content/public/browser/media_observer.h"
17 #include "content/public/browser/user_metrics.h"
18 #include "media/midi/midi_manager.h"
19 #include "media/midi/midi_message_queue.h"
20 #include "media/midi/midi_message_util.h"
22 using media::MIDIManager
;
23 using media::MIDIPortInfoList
;
28 // The total number of bytes which we're allowed to send to the OS
29 // before knowing that they have been successfully sent.
30 const size_t kMaxInFlightBytes
= 10 * 1024 * 1024; // 10 MB.
32 // We keep track of the number of bytes successfully sent to
33 // the hardware. Every once in a while we report back to the renderer
34 // the number of bytes sent since the last report. This threshold determines
35 // how many bytes will be sent before reporting back to the renderer.
36 const size_t kAcknowledgementThresholdBytes
= 1024 * 1024; // 1 MB.
38 const uint8 kSysExMessage
= 0xf0;
39 const uint8 kEndOfSysExMessage
= 0xf7;
41 bool IsDataByte(uint8 data
) {
42 return (data
& 0x80) == 0;
45 bool IsSystemRealTimeMessage(uint8 data
) {
46 return 0xf8 <= data
&& data
<= 0xff;
51 MIDIHost::MIDIHost(int renderer_process_id
, media::MIDIManager
* midi_manager
)
52 : renderer_process_id_(renderer_process_id
),
53 has_sys_ex_permission_(false),
54 midi_manager_(midi_manager
),
55 sent_bytes_in_flight_(0),
56 bytes_sent_since_last_acknowledgement_(0) {
59 MIDIHost::~MIDIHost() {
61 midi_manager_
->EndSession(this);
64 void MIDIHost::OnDestruct() const {
65 BrowserThread::DeleteOnIOThread::Destruct(this);
68 ///////////////////////////////////////////////////////////////////////////////
69 // IPC Messages handler
70 bool MIDIHost::OnMessageReceived(const IPC::Message
& message
,
71 bool* message_was_ok
) {
73 IPC_BEGIN_MESSAGE_MAP_EX(MIDIHost
, message
, *message_was_ok
)
74 IPC_MESSAGE_HANDLER(MIDIHostMsg_StartSession
, OnStartSession
)
75 IPC_MESSAGE_HANDLER(MIDIHostMsg_SendData
, OnSendData
)
76 IPC_MESSAGE_UNHANDLED(handled
= false)
77 IPC_END_MESSAGE_MAP_EX()
82 void MIDIHost::OnStartSession(int client_id
) {
83 MIDIPortInfoList input_ports
;
84 MIDIPortInfoList output_ports
;
86 // Initialize devices and register to receive MIDI data.
89 success
= midi_manager_
->StartSession(this);
91 input_ports
= midi_manager_
->input_ports();
92 output_ports
= midi_manager_
->output_ports();
93 received_messages_queues_
.clear();
94 received_messages_queues_
.resize(input_ports
.size());
95 // ChildSecurityPolicy is set just before OnStartSession by
96 // MIDIDispatcherHost. So we can safely cache the policy.
97 has_sys_ex_permission_
= ChildProcessSecurityPolicyImpl::GetInstance()->
98 CanSendMIDISysExMessage(renderer_process_id_
);
102 Send(new MIDIMsg_SessionStarted(
109 void MIDIHost::OnSendData(uint32 port
,
110 const std::vector
<uint8
>& data
,
118 // Blink running in a renderer checks permission to raise a SecurityError
119 // in JavaScript. The actual permission check for security purposes
120 // happens here in the browser process.
121 if (!has_sys_ex_permission_
&&
122 (std::find(data
.begin(), data
.end(), kSysExMessage
) != data
.end())) {
123 RecordAction(UserMetricsAction("BadMessageTerminate_MIDI"));
124 BadMessageReceived();
128 if (!IsValidWebMIDIData(data
))
131 base::AutoLock
auto_lock(in_flight_lock_
);
132 // Sanity check that we won't send too much data.
133 // TODO(yukawa): Consider to send an error event back to the renderer
134 // after some future discussion in W3C.
135 if (data
.size() + sent_bytes_in_flight_
> kMaxInFlightBytes
)
137 midi_manager_
->DispatchSendMIDIData(this, port
, data
, timestamp
);
138 sent_bytes_in_flight_
+= data
.size();
141 void MIDIHost::ReceiveMIDIData(
146 TRACE_EVENT0("midi", "MIDIHost::ReceiveMIDIData");
148 if (received_messages_queues_
.size() <= port
)
151 // Lazy initialization
152 if (received_messages_queues_
[port
] == NULL
)
153 received_messages_queues_
[port
] = new media::MIDIMessageQueue(true);
155 received_messages_queues_
[port
]->Add(data
, length
);
156 std::vector
<uint8
> message
;
158 received_messages_queues_
[port
]->Get(&message
);
162 // MIDI devices may send a system exclusive messages even if the renderer
163 // doesn't have a permission to receive it. Don't kill the renderer as
164 // OnSendData() does.
165 if (message
[0] == kSysExMessage
&& !has_sys_ex_permission_
)
168 // Send to the renderer.
169 Send(new MIDIMsg_DataReceived(port
, message
, timestamp
));
173 void MIDIHost::AccumulateMIDIBytesSent(size_t n
) {
175 base::AutoLock
auto_lock(in_flight_lock_
);
176 if (n
<= sent_bytes_in_flight_
)
177 sent_bytes_in_flight_
-= n
;
180 if (bytes_sent_since_last_acknowledgement_
+ n
>=
181 bytes_sent_since_last_acknowledgement_
)
182 bytes_sent_since_last_acknowledgement_
+= n
;
184 if (bytes_sent_since_last_acknowledgement_
>=
185 kAcknowledgementThresholdBytes
) {
186 Send(new MIDIMsg_AcknowledgeSentData(
187 bytes_sent_since_last_acknowledgement_
));
188 bytes_sent_since_last_acknowledgement_
= 0;
193 bool MIDIHost::IsValidWebMIDIData(const std::vector
<uint8
>& data
) {
194 bool in_sysex
= false;
195 size_t waiting_data_length
= 0;
196 for (size_t i
= 0; i
< data
.size(); ++i
) {
197 const uint8 current
= data
[i
];
198 if (IsSystemRealTimeMessage(current
))
199 continue; // Real time message can be placed at any point.
200 if (waiting_data_length
> 0) {
201 if (!IsDataByte(current
))
202 return false; // Error: |current| should have been data byte.
203 --waiting_data_length
;
204 continue; // Found data byte as expected.
207 if (data
[i
] == kEndOfSysExMessage
)
209 else if (!IsDataByte(current
))
210 return false; // Error: |current| should have been data byte.
211 continue; // Found data byte as expected.
213 if (current
== kSysExMessage
) {
215 continue; // Found SysEX
217 waiting_data_length
= media::GetMIDIMessageLength(current
);
218 if (waiting_data_length
== 0)
219 return false; // Error: |current| should have been a valid status byte.
220 --waiting_data_length
; // Found status byte
222 return waiting_data_length
== 0 && !in_sysex
;
225 } // namespace content