GetValueFromBuffer.js 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. 'use strict';
  2. var GetIntrinsic = require('get-intrinsic');
  3. var $SyntaxError = require('es-errors/syntax');
  4. var $TypeError = require('es-errors/type');
  5. var $Uint8Array = GetIntrinsic('%Uint8Array%', true);
  6. var callBound = require('call-bind/callBound');
  7. var $slice = callBound('Array.prototype.slice');
  8. var isInteger = require('../helpers/isInteger');
  9. var IsDetachedBuffer = require('./IsDetachedBuffer');
  10. var RawBytesToNumeric = require('./RawBytesToNumeric');
  11. var isArrayBuffer = require('is-array-buffer');
  12. var isSharedArrayBuffer = require('is-shared-array-buffer');
  13. var safeConcat = require('safe-array-concat');
  14. var table61 = {
  15. __proto__: null,
  16. $Int8: 1,
  17. $Uint8: 1,
  18. $Uint8C: 1,
  19. $Int16: 2,
  20. $Uint16: 2,
  21. $Int32: 4,
  22. $Uint32: 4,
  23. $BigInt64: 8,
  24. $BigUint64: 8,
  25. $Float32: 4,
  26. $Float64: 8
  27. };
  28. var defaultEndianness = require('../helpers/defaultEndianness');
  29. // https://262.ecma-international.org/11.0/#sec-getvaluefrombuffer
  30. module.exports = function GetValueFromBuffer(arrayBuffer, byteIndex, type, isTypedArray, order) {
  31. var isSAB = isSharedArrayBuffer(arrayBuffer);
  32. if (!isArrayBuffer(arrayBuffer) && !isSAB) {
  33. throw new $TypeError('Assertion failed: `arrayBuffer` must be an ArrayBuffer or a SharedArrayBuffer');
  34. }
  35. if (!isInteger(byteIndex)) {
  36. throw new $TypeError('Assertion failed: `byteIndex` must be an integer');
  37. }
  38. if (typeof type !== 'string' || typeof table61['$' + type] !== 'number') {
  39. throw new $TypeError('Assertion failed: `type` must be a Typed Array element type');
  40. }
  41. if (typeof isTypedArray !== 'boolean') {
  42. throw new $TypeError('Assertion failed: `isTypedArray` must be a boolean');
  43. }
  44. if (order !== 'SeqCst' && order !== 'Unordered') {
  45. throw new $TypeError('Assertion failed: `order` must be either `SeqCst` or `Unordered`');
  46. }
  47. if (arguments.length > 5 && typeof arguments[5] !== 'boolean') {
  48. throw new $TypeError('Assertion failed: `isLittleEndian` must be a boolean, if present');
  49. }
  50. if (IsDetachedBuffer(arrayBuffer)) {
  51. throw new $TypeError('Assertion failed: `arrayBuffer` is detached'); // step 1
  52. }
  53. // 2. Assert: There are sufficient bytes in arrayBuffer starting at byteIndex to represent a value of type.
  54. if (byteIndex < 0) {
  55. throw new $TypeError('Assertion failed: `byteIndex` must be non-negative'); // step 3
  56. }
  57. // 4. Let block be arrayBuffer.[[ArrayBufferData]].
  58. var elementSize = table61['$' + type]; // step 5
  59. if (!elementSize) {
  60. throw new $TypeError('Assertion failed: `type` must be one of "Int8", "Uint8", "Uint8C", "Int16", "Uint16", "Int32", "Uint32", "BigInt64", "BigUint64", "Float32", or "Float64"');
  61. }
  62. var rawValue;
  63. if (isSAB) { // step 6
  64. /*
  65. a. Let execution be the [[CandidateExecution]] field of the surrounding agent's Agent Record.
  66. b. Let eventList be the [[EventList]] field of the element in execution.[[EventLists]] whose [[AgentSignifier]] is AgentSignifier().
  67. c. If isTypedArray is true and type is "Int8", "Uint8", "Int16", "Uint16", "Int32", or "Uint32", let noTear be true; otherwise let noTear be false.
  68. d. Let rawValue be a List of length elementSize of nondeterministically chosen byte values.
  69. e. NOTE: In implementations, rawValue is the result of a non-atomic or atomic read instruction on the underlying hardware. The nondeterminism is a semantic prescription of the memory model to describe observable behaviour of hardware with weak consistency.
  70. f. Let readEvent be ReadSharedMemory{ [[Order]]: order, [[NoTear]]: noTear, [[Block]]: block, [[ByteIndex]]: byteIndex, [[ElementSize]]: elementSize }.
  71. g. Append readEvent to eventList.
  72. h. Append Chosen Value Record { [[Event]]: readEvent, [[ChosenValue]]: rawValue } to execution.[[ChosenValues]].
  73. */
  74. throw new $SyntaxError('SharedArrayBuffer is not supported by this implementation');
  75. } else {
  76. // 7. Let rawValue be a List of elementSize containing, in order, the elementSize sequence of bytes starting with block[byteIndex].
  77. rawValue = $slice(new $Uint8Array(arrayBuffer, byteIndex), 0, elementSize); // step 6
  78. }
  79. // 8. If isLittleEndian is not present, set isLittleEndian to either true or false. The choice is implementation dependent and should be the alternative that is most efficient for the implementation. An implementation must use the same value each time this step is executed and the same value must be used for the corresponding step in the SetValueInBuffer abstract operation.
  80. var isLittleEndian = arguments.length > 5 ? arguments[5] : defaultEndianness === 'little'; // step 8
  81. var bytes = isLittleEndian
  82. ? $slice(safeConcat([0, 0, 0, 0, 0, 0, 0, 0], rawValue), -elementSize)
  83. : $slice(safeConcat(rawValue, [0, 0, 0, 0, 0, 0, 0, 0]), 0, elementSize);
  84. return RawBytesToNumeric(type, bytes, isLittleEndian);
  85. };