Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(16)

Side by Side Diff: content/browser/renderer_host/media/midi_host.cc

Issue 68353002: Use MIDIMessageQueue/IsValidWebMIDIData for MIDI byte stream validation (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: rebase Created 7 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved. 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 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "content/browser/renderer_host/media/midi_host.h" 5 #include "content/browser/renderer_host/media/midi_host.h"
6 6
7 #include "base/bind.h" 7 #include "base/bind.h"
8 #include "base/bind_helpers.h" 8 #include "base/bind_helpers.h"
9 #include "base/debug/trace_event.h" 9 #include "base/debug/trace_event.h"
10 #include "base/process/process.h" 10 #include "base/process/process.h"
11 #include "content/browser/browser_main_loop.h" 11 #include "content/browser/browser_main_loop.h"
12 #include "content/browser/child_process_security_policy_impl.h" 12 #include "content/browser/child_process_security_policy_impl.h"
13 #include "content/browser/media/media_internals.h" 13 #include "content/browser/media/media_internals.h"
14 #include "content/common/media/midi_messages.h" 14 #include "content/common/media/midi_messages.h"
15 #include "content/public/browser/content_browser_client.h" 15 #include "content/public/browser/content_browser_client.h"
16 #include "content/public/browser/media_observer.h" 16 #include "content/public/browser/media_observer.h"
17 #include "content/public/browser/user_metrics.h" 17 #include "content/public/browser/user_metrics.h"
18 #include "media/midi/midi_manager.h" 18 #include "media/midi/midi_manager.h"
19 #include "media/midi/midi_message_queue.h"
20 #include "media/midi/midi_message_util.h"
19 21
20 using media::MIDIManager; 22 using media::MIDIManager;
21 using media::MIDIPortInfoList; 23 using media::MIDIPortInfoList;
22 24
25 namespace content {
26 namespace {
27
23 // The total number of bytes which we're allowed to send to the OS 28 // The total number of bytes which we're allowed to send to the OS
24 // before knowing that they have been successfully sent. 29 // before knowing that they have been successfully sent.
25 static const size_t kMaxInFlightBytes = 10 * 1024 * 1024; // 10 MB. 30 const size_t kMaxInFlightBytes = 10 * 1024 * 1024; // 10 MB.
26 31
27 // We keep track of the number of bytes successfully sent to 32 // We keep track of the number of bytes successfully sent to
28 // the hardware. Every once in a while we report back to the renderer 33 // the hardware. Every once in a while we report back to the renderer
29 // the number of bytes sent since the last report. This threshold determines 34 // the number of bytes sent since the last report. This threshold determines
30 // how many bytes will be sent before reporting back to the renderer. 35 // how many bytes will be sent before reporting back to the renderer.
31 static const size_t kAcknowledgementThresholdBytes = 1024 * 1024; // 1 MB. 36 const size_t kAcknowledgementThresholdBytes = 1024 * 1024; // 1 MB.
32 37
33 static const uint8 kSysExMessage = 0xf0; 38 const uint8 kSysExMessage = 0xf0;
39 const uint8 kEndOfSysExMessage = 0xf7;
34 40
35 namespace content { 41 bool IsDataByte(uint8 data) {
42 return (data & 0x80) == 0;
43 }
44
45 bool IsSystemRealTimeMessage(uint8 data) {
46 return 0xf8 <= data && data <= 0xff;
47 }
48
49 } // namespace
36 50
37 MIDIHost::MIDIHost(int renderer_process_id, media::MIDIManager* midi_manager) 51 MIDIHost::MIDIHost(int renderer_process_id, media::MIDIManager* midi_manager)
38 : renderer_process_id_(renderer_process_id), 52 : renderer_process_id_(renderer_process_id),
53 has_sys_ex_permission_(false),
39 midi_manager_(midi_manager), 54 midi_manager_(midi_manager),
40 sent_bytes_in_flight_(0), 55 sent_bytes_in_flight_(0),
41 bytes_sent_since_last_acknowledgement_(0) { 56 bytes_sent_since_last_acknowledgement_(0) {
42 } 57 }
43 58
44 MIDIHost::~MIDIHost() { 59 MIDIHost::~MIDIHost() {
45 if (midi_manager_) 60 if (midi_manager_)
46 midi_manager_->EndSession(this); 61 midi_manager_->EndSession(this);
47 } 62 }
48 63
(...skipping 19 matching lines...) Expand all
68 MIDIPortInfoList input_ports; 83 MIDIPortInfoList input_ports;
69 MIDIPortInfoList output_ports; 84 MIDIPortInfoList output_ports;
70 85
71 // Initialize devices and register to receive MIDI data. 86 // Initialize devices and register to receive MIDI data.
72 bool success = false; 87 bool success = false;
73 if (midi_manager_) { 88 if (midi_manager_) {
74 success = midi_manager_->StartSession(this); 89 success = midi_manager_->StartSession(this);
75 if (success) { 90 if (success) {
76 input_ports = midi_manager_->input_ports(); 91 input_ports = midi_manager_->input_ports();
77 output_ports = midi_manager_->output_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_);
78 } 99 }
79 } 100 }
80 101
81 Send(new MIDIMsg_SessionStarted( 102 Send(new MIDIMsg_SessionStarted(
82 client_id, 103 client_id,
83 success, 104 success,
84 input_ports, 105 input_ports,
85 output_ports)); 106 output_ports));
86 } 107 }
87 108
88 void MIDIHost::OnSendData(uint32 port, 109 void MIDIHost::OnSendData(uint32 port,
89 const std::vector<uint8>& data, 110 const std::vector<uint8>& data,
90 double timestamp) { 111 double timestamp) {
91 if (!midi_manager_) 112 if (!midi_manager_)
92 return; 113 return;
93 114
94 if (data.empty()) 115 if (data.empty())
95 return; 116 return;
96 117
97 base::AutoLock auto_lock(in_flight_lock_); 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();
125 return;
126 }
98 127
99 // Sanity check that we won't send too much. 128 if (!IsValidWebMIDIData(data))
100 if (sent_bytes_in_flight_ > kMaxInFlightBytes ||
101 data.size() > kMaxInFlightBytes ||
102 data.size() + sent_bytes_in_flight_ > kMaxInFlightBytes)
103 return; 129 return;
104 130
105 if (data[0] >= kSysExMessage) { 131 base::AutoLock auto_lock(in_flight_lock_);
106 // Blink running in a renderer checks permission to raise a SecurityError in 132 // Sanity check that we won't send too much data.
107 // JavaScript. The actual permission check for security perposes happens 133 // TODO(yukawa): Consider to send an error event back to the renderer
108 // here in the browser process. 134 // after some future discussion in W3C.
109 if (!ChildProcessSecurityPolicyImpl::GetInstance()->CanSendMIDISysExMessage( 135 if (data.size() + sent_bytes_in_flight_ > kMaxInFlightBytes)
110 renderer_process_id_)) { 136 return;
111 RecordAction(UserMetricsAction("BadMessageTerminate_MIDI")); 137 midi_manager_->DispatchSendMIDIData(this, port, data, timestamp);
112 BadMessageReceived();
113 return;
114 }
115 }
116
117 midi_manager_->DispatchSendMIDIData(
118 this,
119 port,
120 data,
121 timestamp);
122
123 sent_bytes_in_flight_ += data.size(); 138 sent_bytes_in_flight_ += data.size();
124 } 139 }
125 140
126 void MIDIHost::ReceiveMIDIData( 141 void MIDIHost::ReceiveMIDIData(
127 uint32 port, 142 uint32 port,
128 const uint8* data, 143 const uint8* data,
129 size_t length, 144 size_t length,
130 double timestamp) { 145 double timestamp) {
131 TRACE_EVENT0("midi", "MIDIHost::ReceiveMIDIData"); 146 TRACE_EVENT0("midi", "MIDIHost::ReceiveMIDIData");
132 147
133 // Check a process security policy to receive a system exclusive message. 148 if (received_messages_queues_.size() <= port)
134 if (length > 0 && data[0] >= kSysExMessage) { 149 return;
135 if (!ChildProcessSecurityPolicyImpl::GetInstance()->CanSendMIDISysExMessage( 150
136 renderer_process_id_)) { 151 // Lazy initialization
137 // MIDI devices may send a system exclusive messages even if the renderer 152 if (received_messages_queues_[port] == NULL)
138 // doesn't have a permission to receive it. Don't kill the renderer as 153 received_messages_queues_[port] = new media::MIDIMessageQueue(true);
139 // OnSendData() does. 154
140 return; 155 received_messages_queues_[port]->Add(data, length);
141 } 156 std::vector<uint8> message;
157 while (true) {
158 received_messages_queues_[port]->Get(&message);
159 if (message.empty())
160 break;
161
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_)
166 continue;
167
168 // Send to the renderer.
169 Send(new MIDIMsg_DataReceived(port, message, timestamp));
142 } 170 }
143
144 // Send to the renderer.
145 std::vector<uint8> v(data, data + length);
146 Send(new MIDIMsg_DataReceived(port, v, timestamp));
147 } 171 }
148 172
149 void MIDIHost::AccumulateMIDIBytesSent(size_t n) { 173 void MIDIHost::AccumulateMIDIBytesSent(size_t n) {
150 { 174 {
151 base::AutoLock auto_lock(in_flight_lock_); 175 base::AutoLock auto_lock(in_flight_lock_);
152 if (n <= sent_bytes_in_flight_) 176 if (n <= sent_bytes_in_flight_)
153 sent_bytes_in_flight_ -= n; 177 sent_bytes_in_flight_ -= n;
154 } 178 }
155 179
156 if (bytes_sent_since_last_acknowledgement_ + n >= 180 if (bytes_sent_since_last_acknowledgement_ + n >=
157 bytes_sent_since_last_acknowledgement_) 181 bytes_sent_since_last_acknowledgement_)
158 bytes_sent_since_last_acknowledgement_ += n; 182 bytes_sent_since_last_acknowledgement_ += n;
159 183
160 if (bytes_sent_since_last_acknowledgement_ >= 184 if (bytes_sent_since_last_acknowledgement_ >=
161 kAcknowledgementThresholdBytes) { 185 kAcknowledgementThresholdBytes) {
162 Send(new MIDIMsg_AcknowledgeSentData( 186 Send(new MIDIMsg_AcknowledgeSentData(
163 bytes_sent_since_last_acknowledgement_)); 187 bytes_sent_since_last_acknowledgement_));
164 bytes_sent_since_last_acknowledgement_ = 0; 188 bytes_sent_since_last_acknowledgement_ = 0;
165 } 189 }
166 } 190 }
167 191
192 // static
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.
205 }
206 if (in_sysex) {
207 if (data[i] == kEndOfSysExMessage)
208 in_sysex = false;
209 else if (!IsDataByte(current))
210 return false; // Error: |current| should have been data byte.
211 continue; // Found data byte as expected.
212 }
213 if (current == kSysExMessage) {
214 in_sysex = true;
215 continue; // Found SysEX
216 }
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
221 }
222 return waiting_data_length == 0 && !in_sysex;
223 }
224
168 } // namespace content 225 } // namespace content
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698