Index: media/midi/message_util.cc |
diff --git a/media/midi/message_util.cc b/media/midi/message_util.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..7d58ac3888eff8f0a3d7649d1c86098d5e7c6606 |
--- /dev/null |
+++ b/media/midi/message_util.cc |
@@ -0,0 +1,103 @@ |
+// Copyright 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/message_util.h" |
+ |
+#include "base/logging.h" |
+#include "base/metrics/histogram_macros.h" |
+ |
+namespace midi { |
+ |
+size_t GetMessageLength(uint8_t status_byte) { |
+ if (status_byte < 0x80) |
+ return 0; |
+ if (0x80 <= status_byte && status_byte <= 0xbf) |
+ return 3; |
+ if (0xc0 <= status_byte && status_byte <= 0xdf) |
+ return 2; |
+ if (0xe0 <= status_byte && status_byte <= 0xef) |
+ return 3; |
+ |
+ switch (status_byte) { |
+ case 0xf0: |
+ return 0; |
+ case 0xf1: |
+ return 2; |
+ case 0xf2: |
+ return 3; |
+ case 0xf3: |
+ return 2; |
+ case 0xf4: // Reserved |
+ case 0xf5: // Reserved |
+ return 0; |
+ case 0xf6: |
+ return 1; |
+ case 0xf7: |
+ return 0; |
+ case 0xf8: |
+ case 0xf9: |
+ case 0xfa: |
+ case 0xfb: |
+ case 0xfc: |
+ case 0xfd: |
+ case 0xfe: |
+ case 0xff: |
+ return 1; |
+ default: |
+ NOTREACHED(); |
+ return 0; |
+ } |
+} |
+ |
+bool IsDataByte(uint8_t data) { |
+ return (data & 0x80) == 0; |
+} |
+ |
+bool IsSystemRealTimeMessage(uint8_t data) { |
+ return 0xf8 <= data; |
+} |
+ |
+bool IsSystemMessage(uint8_t data) { |
+ return 0xf0 <= data; |
+} |
+ |
+bool IsValidWebMIDIData(const std::vector<uint8_t>& data) { |
+ bool in_sysex = false; |
+ size_t sysex_start_offset = 0; |
+ size_t waiting_data_length = 0; |
+ for (size_t i = 0; i < data.size(); ++i) { |
+ const uint8_t current = data[i]; |
+ if (IsSystemRealTimeMessage(current)) |
+ continue; // Real time message can be placed at any point. |
+ if (waiting_data_length > 0) { |
+ if (!IsDataByte(current)) |
+ return false; // Error: |current| should have been data byte. |
+ --waiting_data_length; |
+ continue; // Found data byte as expected. |
+ } |
+ if (in_sysex) { |
+ if (data[i] == kEndOfSysExByte) { |
+ in_sysex = false; |
+ UMA_HISTOGRAM_COUNTS("Media.Midi.SysExMessageSizeUpTo1MB", |
+ static_cast<base::HistogramBase::Sample>( |
+ i - sysex_start_offset + 1)); |
+ } else if (!IsDataByte(current)) { |
+ return false; // Error: |current| should have been data byte. |
+ } |
+ continue; // Found data byte as expected. |
+ } |
+ if (current == kSysExByte) { |
+ in_sysex = true; |
+ sysex_start_offset = i; |
+ continue; // Found SysEX |
+ } |
+ waiting_data_length = GetMessageLength(current); |
+ if (waiting_data_length == 0) |
+ return false; // Error: |current| should have been a valid status byte. |
+ --waiting_data_length; // Found status byte |
+ } |
+ return waiting_data_length == 0 && !in_sysex; |
+} |
+ |
+} // namespace midi |