/test/mjsunit/wasm/streaming-error-position.js

http://github.com/v8/v8 · JavaScript · 399 lines · 346 code · 33 blank · 20 comment · 0 complexity · 35869751a6645fc35e680e07750614f0 MD5 · raw file

  1. // Copyright 2017 the V8 project authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style license that can be
  3. // found in the LICENSE file.
  4. // Flags: --wasm-test-streaming --expose-wasm
  5. 'use strict';
  6. load('test/mjsunit/wasm/wasm-module-builder.js');
  7. function testErrorPositionAsyncOnly(bytes, pos, message) {
  8. let buffer = bytes.trunc_buffer();
  9. // Only test the streaming decoder since this kind of error is out of sync
  10. // with the non-streaming decoder, hence errors cannot be compared.
  11. assertThrowsAsync(
  12. WebAssembly.compile(buffer), WebAssembly.CompileError,
  13. new RegExp(message + '.*@\\+' + pos));
  14. }
  15. function testErrorPosition(bytes, pos, message) {
  16. let buffer = bytes.trunc_buffer();
  17. // First check the non-streaming decoder as a reference.
  18. assertThrows(
  19. () => new WebAssembly.Module(buffer), WebAssembly.CompileError,
  20. new RegExp(message + '.*@\\+' + pos));
  21. // Next test the actual streaming decoder.
  22. assertThrowsAsync(
  23. WebAssembly.compile(buffer), WebAssembly.CompileError,
  24. new RegExp(message + '.*@\\+' + pos));
  25. }
  26. (function testInvalidMagic() {
  27. let bytes = new Binary;
  28. bytes.emit_bytes([
  29. kWasmH0, kWasmH1 + 1, kWasmH2, kWasmH3, kWasmV0, kWasmV1, kWasmV2, kWasmV3
  30. ]);
  31. // Error at pos==0 because that's where the magic word is.
  32. testErrorPosition(bytes, 0, 'expected magic word');
  33. })();
  34. (function testInvalidVersion() {
  35. let bytes = new Binary;
  36. bytes.emit_bytes([
  37. kWasmH0, kWasmH1, kWasmH2, kWasmH3, kWasmV0, kWasmV1 + 1, kWasmV2, kWasmV3
  38. ]);
  39. // Error at pos==4 because that's where the version word is.
  40. testErrorPosition(bytes, 4, 'expected version');
  41. })();
  42. (function testSectionLengthInvalidVarint() {
  43. let bytes = new Binary;
  44. bytes.emit_header();
  45. bytes.emit_u8(kTypeSectionCode);
  46. bytes.emit_bytes([0x80, 0x80, 0x80, 0x80, 0x80, 0x00]);
  47. let pos = bytes.length - 1 - 1;
  48. testErrorPosition(bytes, pos, 'expected section length');
  49. })();
  50. (function testSectionLengthTooBig() {
  51. let bytes = new Binary;
  52. bytes.emit_header();
  53. bytes.emit_u8(kTypeSectionCode);
  54. bytes.emit_u32v(0xffffff23);
  55. let pos = bytes.length - 1;
  56. testErrorPositionAsyncOnly(bytes, pos, 'maximum function size');
  57. })();
  58. (function testFunctionsCountInvalidVarint() {
  59. let bytes = new Binary;
  60. bytes.emit_header();
  61. bytes.emit_bytes([
  62. kTypeSectionCode, // section id
  63. 1, // section length
  64. 0 // number of types
  65. ]);
  66. bytes.emit_bytes([
  67. kFunctionSectionCode, // section id
  68. 1, // section length
  69. 0 // number of functions
  70. ]);
  71. bytes.emit_bytes([
  72. kCodeSectionCode, // section id
  73. 20, // section length (arbitrary value > 6)
  74. ]);
  75. // Functions count
  76. bytes.emit_bytes([0x80, 0x80, 0x80, 0x80, 0x80, 0x00]);
  77. let pos = bytes.length - 1 - 1;
  78. testErrorPositionAsyncOnly(bytes, pos, 'expected functions count');
  79. })();
  80. (function testFunctionsCountTooBig() {
  81. let bytes = new Binary;
  82. bytes.emit_header();
  83. bytes.emit_bytes([
  84. kTypeSectionCode, // section id
  85. 1, // section length
  86. 0 // number of types
  87. ]);
  88. bytes.emit_bytes([
  89. kFunctionSectionCode, // section id
  90. 1, // section length
  91. 0 // number of functions
  92. ]);
  93. bytes.emit_bytes([
  94. kCodeSectionCode, // section id
  95. 20, // section length (arbitrary value > 6)
  96. ]);
  97. // Functions count
  98. bytes.emit_u32v(0xffffff23);
  99. let pos = bytes.length - 1;
  100. testErrorPositionAsyncOnly(bytes, pos, 'maximum function size');
  101. })();
  102. (function testFunctionsCountDoesNotMatch() {
  103. let bytes = new Binary;
  104. bytes.emit_header();
  105. bytes.emit_bytes([
  106. kTypeSectionCode, // section id
  107. 1, // section length
  108. 0 // number of types
  109. ]);
  110. bytes.emit_bytes([
  111. kFunctionSectionCode, // section id
  112. 1, // section length
  113. 0 // number of functions
  114. ]);
  115. bytes.emit_bytes([
  116. kCodeSectionCode, // section id
  117. 20, // section length (arbitrary value > 6)
  118. ]);
  119. // Functions count (different than the count in the functions section.
  120. bytes.emit_u32v(5);
  121. let pos = bytes.length - 1;
  122. testErrorPositionAsyncOnly(bytes, pos, 'function body count 5 mismatch');
  123. })();
  124. (function testBodySizeInvalidVarint() {
  125. let bytes = new Binary;
  126. bytes.emit_header();
  127. bytes.emit_bytes([
  128. kTypeSectionCode, // section id
  129. 4, // section length
  130. 1, // number of types
  131. kWasmFunctionTypeForm, // type
  132. 0, // number of parameter
  133. 0 // number of returns
  134. ]);
  135. bytes.emit_bytes([
  136. kFunctionSectionCode, // section id
  137. 2, // section length
  138. 1, // number of functions
  139. 0 // signature index
  140. ]);
  141. bytes.emit_bytes([
  142. kCodeSectionCode, // section id
  143. 20, // section length (arbitrary value > 6)
  144. 1 // functions count
  145. ]);
  146. // Invalid function body size.
  147. bytes.emit_bytes([0x80, 0x80, 0x80, 0x80, 0x80, 0x00]);
  148. let pos = bytes.length - 1 - 1;
  149. testErrorPositionAsyncOnly(bytes, pos, 'expected body size');
  150. })();
  151. (function testBodySizeTooBig() {
  152. let bytes = new Binary;
  153. bytes.emit_header();
  154. bytes.emit_bytes([
  155. kTypeSectionCode, // section id
  156. 4, // section length
  157. 1, // number of types
  158. kWasmFunctionTypeForm, // type
  159. 0, // number of parameter
  160. 0 // number of returns
  161. ]);
  162. bytes.emit_bytes([
  163. kFunctionSectionCode, // section id
  164. 2, // section length
  165. 1, // number of functions
  166. 0 // signature index
  167. ]);
  168. bytes.emit_bytes([
  169. kCodeSectionCode, // section id
  170. 20, // section length (arbitrary value > 6)
  171. 1 // functions count
  172. ]);
  173. // Invalid function body size.
  174. bytes.emit_u32v(0xffffff23);
  175. let pos = bytes.length - 1;
  176. testErrorPositionAsyncOnly(bytes, pos, 'maximum function size');
  177. })();
  178. (function testBodySizeDoesNotFit() {
  179. let bytes = new Binary;
  180. bytes.emit_header();
  181. bytes.emit_bytes([
  182. kTypeSectionCode, // section id
  183. 4, // section length
  184. 1, // number of types
  185. kWasmFunctionTypeForm, // type
  186. 0, // number of parameter
  187. 0 // number of returns
  188. ]);
  189. bytes.emit_bytes([
  190. kFunctionSectionCode, // section id
  191. 2, // section length
  192. 1, // number of functions
  193. 0 // signature index
  194. ]);
  195. bytes.emit_bytes([
  196. kCodeSectionCode, // section id
  197. 20, // section length (arbitrary value > 6)
  198. 1 // functions count
  199. ]);
  200. // Invalid function body size (does not fit into the code section).
  201. bytes.emit_u32v(20);
  202. let pos = bytes.length - 1;
  203. testErrorPositionAsyncOnly(bytes, pos, 'not enough code section bytes');
  204. })();
  205. (function testBodySizeIsZero() {
  206. let bytes = new Binary;
  207. bytes.emit_header();
  208. bytes.emit_bytes([
  209. kTypeSectionCode, // section id
  210. 4, // section length
  211. 1, // number of types
  212. kWasmFunctionTypeForm, // type
  213. 0, // number of parameter
  214. 0 // number of returns
  215. ]);
  216. bytes.emit_bytes([
  217. kFunctionSectionCode, // section id
  218. 2, // section length
  219. 1, // number of functions
  220. 0 // signature index
  221. ]);
  222. bytes.emit_bytes([
  223. kCodeSectionCode, // section id
  224. 20, // section length (arbitrary value > 6)
  225. 1 // functions count
  226. ]);
  227. // Invalid function body size (body size of 0 is invalid).
  228. bytes.emit_u32v(0);
  229. let pos = bytes.length - 1;
  230. testErrorPositionAsyncOnly(bytes, pos, 'invalid function length');
  231. })();
  232. (function testStaleCodeSectionBytes() {
  233. let bytes = new Binary;
  234. bytes.emit_header();
  235. bytes.emit_bytes([
  236. kTypeSectionCode, // section id
  237. 4, // section length
  238. 1, // number of types
  239. kWasmFunctionTypeForm, // type
  240. 0, // number of parameter
  241. 0 // number of returns
  242. ]);
  243. bytes.emit_bytes([
  244. kFunctionSectionCode, // section id
  245. 2, // section length
  246. 1, // number of functions
  247. 0 // signature index
  248. ]);
  249. bytes.emit_bytes([
  250. kCodeSectionCode, // section id
  251. 20, // section length (too big)
  252. 1, // functions count
  253. 2, // body size
  254. 0, // locals count
  255. kExprEnd // body
  256. ]);
  257. let pos = bytes.length - 1;
  258. testErrorPositionAsyncOnly(bytes, pos, 'not all code section bytes were used');
  259. })();
  260. (function testInvalidCode() {
  261. let bytes = new Binary;
  262. bytes.emit_header();
  263. bytes.emit_bytes([
  264. kTypeSectionCode, // section id
  265. 4, // section length
  266. 1, // number of types
  267. kWasmFunctionTypeForm, // type
  268. 0, // number of parameter
  269. 0 // number of returns
  270. ]);
  271. bytes.emit_bytes([
  272. kFunctionSectionCode, // section id
  273. 2, // section length
  274. 1, // number of functions
  275. 0 // signature index
  276. ]);
  277. bytes.emit_bytes([
  278. kCodeSectionCode, // section id
  279. 6, // section length
  280. 1, // functions count
  281. 4, // body size
  282. 0, // locals count
  283. kExprLocalGet, 0, // Access a non-existing local
  284. kExprEnd // --
  285. ]);
  286. // Find error at the index of kExprLocalGet.
  287. let pos = bytes.length - 1 - 1;
  288. testErrorPosition(bytes, pos, 'invalid local index');
  289. })();
  290. (function testCodeSectionRepeats() {
  291. let bytes = new Binary;
  292. bytes.emit_header();
  293. bytes.emit_bytes([
  294. kTypeSectionCode, // section id
  295. 4, // section length
  296. 1, // number of types
  297. kWasmFunctionTypeForm, // type
  298. 0, // number of parameter
  299. 0 // number of returns
  300. ]);
  301. bytes.emit_bytes([
  302. kFunctionSectionCode, // section id
  303. 2, // section length
  304. 1, // number of functions
  305. 0 // signature index
  306. ]);
  307. bytes.emit_bytes([
  308. kCodeSectionCode, // section id
  309. 4, // section length
  310. 1, // functions count
  311. 2, // body size
  312. 0, // locals count
  313. kExprEnd // body
  314. ]);
  315. let pos = bytes.length;
  316. bytes.emit_bytes([
  317. kCodeSectionCode, // section id (repeating)
  318. 4, // section length
  319. 1, // functions count
  320. 2, // body size
  321. 0, // locals count
  322. kExprEnd // body
  323. ]);
  324. // Find error at the second kCodeSectionCode.
  325. testErrorPositionAsyncOnly(bytes, pos, 'code section can only appear once');
  326. })();
  327. (function testCodeSectionSizeZero() {
  328. let bytes = new Binary;
  329. bytes.emit_header();
  330. bytes.emit_bytes([
  331. kTypeSectionCode, // section id
  332. 4, // section length
  333. 1, // number of types
  334. kWasmFunctionTypeForm, // type
  335. 0, // number of parameter
  336. 0 // number of returns
  337. ]);
  338. bytes.emit_bytes([
  339. kFunctionSectionCode, // section id
  340. 2, // section length
  341. 1, // number of functions
  342. 0 // signature index
  343. ]);
  344. bytes.emit_bytes([
  345. kCodeSectionCode, // section id
  346. 0, // section length (empty)
  347. ]);
  348. // Find error at the code section length.
  349. let pos = bytes.length - 1;
  350. testErrorPositionAsyncOnly(bytes, pos, 'code section cannot have size 0');
  351. })();
  352. (function testInvalidSection() {
  353. let bytes = new Binary;
  354. bytes.emit_header();
  355. bytes.emit_bytes([
  356. kTypeSectionCode, // section id
  357. 5, // section length
  358. 1, // number of types
  359. kWasmFunctionTypeForm, // type
  360. 1, // number of parameter
  361. 0x7b, // invalid type
  362. 0 // number of returns
  363. ]);
  364. let pos = bytes.length - 1 - 1;
  365. testErrorPositionAsyncOnly(bytes, pos, 'invalid value type');
  366. })();