| /********************************************************************* | 
|  * NAN - Native Abstractions for Node.js | 
|  * | 
|  * Copyright (c) 2018 NAN contributors | 
|  * | 
|  * MIT License <https://github.com/nodejs/nan/blob/master/LICENSE.md> | 
|  ********************************************************************/ | 
|   | 
| #ifndef NAN_TYPEDARRAY_CONTENTS_H_ | 
| #define NAN_TYPEDARRAY_CONTENTS_H_ | 
|   | 
| template<typename T> | 
| class TypedArrayContents { | 
|  public: | 
|   inline explicit TypedArrayContents(v8::Local<v8::Value> from) : | 
|       length_(0), data_(NULL) { | 
|     HandleScope scope; | 
|   | 
|     size_t length = 0; | 
|     void*  data = NULL; | 
|   | 
| #if defined(V8_MAJOR_VERSION) && (V8_MAJOR_VERSION > 4 ||                      \ | 
|   (V8_MAJOR_VERSION == 4 && defined(V8_MINOR_VERSION) && V8_MINOR_VERSION >= 3)) | 
|   | 
|     if (from->IsArrayBufferView()) { | 
|       v8::Local<v8::ArrayBufferView> array = | 
|         v8::Local<v8::ArrayBufferView>::Cast(from); | 
|   | 
|       const size_t    byte_length = array->ByteLength(); | 
|       const ptrdiff_t byte_offset = array->ByteOffset(); | 
|       v8::Local<v8::ArrayBuffer> buffer = array->Buffer(); | 
|   | 
|       length = byte_length / sizeof(T); | 
| // Actually it's 7.9 here but this would lead to ABI issues with Node.js 13 | 
| // using 7.8 till 13.2.0. | 
| #if (V8_MAJOR_VERSION >= 8) | 
|       data = static_cast<char*>(buffer->GetBackingStore()->Data()) + byte_offset; | 
| #else | 
|       data = static_cast<char*>(buffer->GetContents().Data()) + byte_offset; | 
| #endif | 
|     } | 
|   | 
| #else | 
|   | 
|     if (from->IsObject() && !from->IsNull()) { | 
|       v8::Local<v8::Object> array = v8::Local<v8::Object>::Cast(from); | 
|   | 
|       MaybeLocal<v8::Value> buffer = Get(array, | 
|         New<v8::String>("buffer").ToLocalChecked()); | 
|       MaybeLocal<v8::Value> byte_length = Get(array, | 
|         New<v8::String>("byteLength").ToLocalChecked()); | 
|       MaybeLocal<v8::Value> byte_offset = Get(array, | 
|         New<v8::String>("byteOffset").ToLocalChecked()); | 
|   | 
|       if (!buffer.IsEmpty() && | 
|           !byte_length.IsEmpty() && byte_length.ToLocalChecked()->IsUint32() && | 
|           !byte_offset.IsEmpty() && byte_offset.ToLocalChecked()->IsUint32()) { | 
|         data = array->GetIndexedPropertiesExternalArrayData(); | 
|         if(data) { | 
|           length = byte_length.ToLocalChecked()->Uint32Value() / sizeof(T); | 
|         } | 
|       } | 
|     } | 
|   | 
| #endif | 
|   | 
| #if defined(_MSC_VER) && _MSC_VER >= 1900 || __cplusplus >= 201103L | 
|     assert(reinterpret_cast<uintptr_t>(data) % alignof (T) == 0); | 
| #elif defined(_MSC_VER) && _MSC_VER >= 1600 || defined(__GNUC__) | 
|     assert(reinterpret_cast<uintptr_t>(data) % __alignof(T) == 0); | 
| #else | 
|     assert(reinterpret_cast<uintptr_t>(data) % sizeof (T) == 0); | 
| #endif | 
|   | 
|     length_ = length; | 
|     data_   = static_cast<T*>(data); | 
|   } | 
|   | 
|   inline size_t length() const      { return length_; } | 
|   inline T* operator*()             { return data_;   } | 
|   inline const T* operator*() const { return data_;   } | 
|   | 
|  private: | 
|   NAN_DISALLOW_ASSIGN_COPY_MOVE(TypedArrayContents) | 
|   | 
|   //Disable heap allocation | 
|   void *operator new(size_t size); | 
|   void operator delete(void *, size_t) { | 
|     abort(); | 
|   } | 
|   | 
|   size_t  length_; | 
|   T*      data_; | 
| }; | 
|   | 
| #endif  // NAN_TYPEDARRAY_CONTENTS_H_ |