mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
211 lines
6.7 KiB
C++
211 lines
6.7 KiB
C++
/*
|
|
* Copyright (C) 2009 Apple Inc. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
|
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
|
|
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#ifndef SKY_ENGINE_WTF_ARRAYBUFFERVIEW_H_
|
|
#define SKY_ENGINE_WTF_ARRAYBUFFERVIEW_H_
|
|
|
|
#include "sky/engine/wtf/ArrayBuffer.h"
|
|
|
|
#include <limits.h>
|
|
#include <algorithm>
|
|
#include "sky/engine/wtf/PassRefPtr.h"
|
|
#include "sky/engine/wtf/RefCounted.h"
|
|
#include "sky/engine/wtf/RefPtr.h"
|
|
#include "sky/engine/wtf/WTFExport.h"
|
|
|
|
namespace WTF {
|
|
|
|
class WTF_EXPORT ArrayBufferView : public RefCounted<ArrayBufferView> {
|
|
public:
|
|
enum ViewType {
|
|
TypeInt8,
|
|
TypeUint8,
|
|
TypeUint8Clamped,
|
|
TypeInt16,
|
|
TypeUint16,
|
|
TypeInt32,
|
|
TypeUint32,
|
|
TypeFloat32,
|
|
TypeFloat64,
|
|
TypeDataView
|
|
};
|
|
virtual ViewType type() const = 0;
|
|
const char* typeName();
|
|
|
|
PassRefPtr<ArrayBuffer> buffer() const
|
|
{
|
|
return m_buffer;
|
|
}
|
|
|
|
void* baseAddress() const
|
|
{
|
|
return m_baseAddress;
|
|
}
|
|
|
|
unsigned byteOffset() const
|
|
{
|
|
return m_byteOffset;
|
|
}
|
|
|
|
virtual unsigned byteLength() const = 0;
|
|
|
|
void setNeuterable(bool flag) { m_isNeuterable = flag; }
|
|
bool isNeuterable() const { return m_isNeuterable; }
|
|
|
|
virtual ~ArrayBufferView();
|
|
|
|
protected:
|
|
ArrayBufferView(PassRefPtr<ArrayBuffer>, unsigned byteOffset);
|
|
|
|
inline bool setImpl(ArrayBufferView*, unsigned byteOffset);
|
|
|
|
inline bool setRangeImpl(const char* data, size_t dataByteLength, unsigned byteOffset);
|
|
|
|
inline bool zeroRangeImpl(unsigned byteOffset, size_t rangeByteLength);
|
|
|
|
static inline void calculateOffsetAndLength(int start, int end, unsigned arraySize,
|
|
unsigned* offset, unsigned* length);
|
|
|
|
// Helper to verify that a given sub-range of an ArrayBuffer is
|
|
// within range.
|
|
template <typename T>
|
|
static bool verifySubRange(PassRefPtr<ArrayBuffer> buffer,
|
|
unsigned byteOffset,
|
|
unsigned numElements)
|
|
{
|
|
if (!buffer)
|
|
return false;
|
|
if (sizeof(T) > 1 && byteOffset % sizeof(T))
|
|
return false;
|
|
if (byteOffset > buffer->byteLength())
|
|
return false;
|
|
unsigned remainingElements = (buffer->byteLength() - byteOffset) / sizeof(T);
|
|
if (numElements > remainingElements)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
// Input offset is in number of elements from this array's view;
|
|
// output offset is in number of bytes from the underlying buffer's view.
|
|
template <typename T>
|
|
static void clampOffsetAndNumElements(PassRefPtr<ArrayBuffer> buffer,
|
|
unsigned arrayByteOffset,
|
|
unsigned *offset,
|
|
unsigned *numElements)
|
|
{
|
|
unsigned maxOffset = (UINT_MAX - arrayByteOffset) / sizeof(T);
|
|
if (*offset > maxOffset) {
|
|
*offset = buffer->byteLength();
|
|
*numElements = 0;
|
|
return;
|
|
}
|
|
*offset = arrayByteOffset + *offset * sizeof(T);
|
|
*offset = std::min(buffer->byteLength(), *offset);
|
|
unsigned remainingElements = (buffer->byteLength() - *offset) / sizeof(T);
|
|
*numElements = std::min(remainingElements, *numElements);
|
|
}
|
|
|
|
virtual void neuter();
|
|
|
|
// This is the address of the ArrayBuffer's storage, plus the byte offset.
|
|
void* m_baseAddress;
|
|
|
|
unsigned m_byteOffset : 31;
|
|
bool m_isNeuterable : 1;
|
|
|
|
private:
|
|
friend class ArrayBuffer;
|
|
RefPtr<ArrayBuffer> m_buffer;
|
|
ArrayBufferView* m_prevView;
|
|
ArrayBufferView* m_nextView;
|
|
};
|
|
|
|
bool ArrayBufferView::setImpl(ArrayBufferView* array, unsigned byteOffset)
|
|
{
|
|
if (byteOffset > byteLength()
|
|
|| byteOffset + array->byteLength() > byteLength()
|
|
|| byteOffset + array->byteLength() < byteOffset) {
|
|
// Out of range offset or overflow
|
|
return false;
|
|
}
|
|
|
|
char* base = static_cast<char*>(baseAddress());
|
|
memmove(base + byteOffset, array->baseAddress(), array->byteLength());
|
|
return true;
|
|
}
|
|
|
|
bool ArrayBufferView::setRangeImpl(const char* data, size_t dataByteLength, unsigned byteOffset)
|
|
{
|
|
if (byteOffset > byteLength()
|
|
|| byteOffset + dataByteLength > byteLength()
|
|
|| byteOffset + dataByteLength < byteOffset) {
|
|
// Out of range offset or overflow
|
|
return false;
|
|
}
|
|
|
|
char* base = static_cast<char*>(baseAddress());
|
|
memmove(base + byteOffset, data, dataByteLength);
|
|
return true;
|
|
}
|
|
|
|
bool ArrayBufferView::zeroRangeImpl(unsigned byteOffset, size_t rangeByteLength)
|
|
{
|
|
if (byteOffset > byteLength()
|
|
|| byteOffset + rangeByteLength > byteLength()
|
|
|| byteOffset + rangeByteLength < byteOffset) {
|
|
// Out of range offset or overflow
|
|
return false;
|
|
}
|
|
|
|
char* base = static_cast<char*>(baseAddress());
|
|
memset(base + byteOffset, 0, rangeByteLength);
|
|
return true;
|
|
}
|
|
|
|
void ArrayBufferView::calculateOffsetAndLength(int start, int end, unsigned arraySize,
|
|
unsigned* offset, unsigned* length)
|
|
{
|
|
if (start < 0)
|
|
start += arraySize;
|
|
if (start < 0)
|
|
start = 0;
|
|
if (end < 0)
|
|
end += arraySize;
|
|
if (end < 0)
|
|
end = 0;
|
|
if (static_cast<unsigned>(end) > arraySize)
|
|
end = arraySize;
|
|
if (end < start)
|
|
end = start;
|
|
*offset = static_cast<unsigned>(start);
|
|
*length = static_cast<unsigned>(end - start);
|
|
}
|
|
|
|
} // namespace WTF
|
|
|
|
using WTF::ArrayBufferView;
|
|
|
|
#endif // SKY_ENGINE_WTF_ARRAYBUFFERVIEW_H_
|