| 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
|
|
|