From 82832902ccd6683d7821c3c952122b19f7436d4f Mon Sep 17 00:00:00 2001 From: Robert Plummer Date: Fri, 5 Jul 2019 11:03:25 -0400 Subject: [PATCH] fix: Move jellyfish images to actual files fix: added `warnVarUsage` and `Dealing With Transpilation` fix: Test dynamic arguments and provide some fixes for fix: Dynamic arguments for `CPUFunctionNode` fix: Dynamic arguments for `WebGLKernelValueDynamicHTMLImage`, `WebGLKernelValueDynamicNumberTexture`, `WebGLKernelValueDynamicSingleInput`, `WebGL2KernelValueDynamicHTMLImage`, `WebGL2KernelValueDynamicNumberTexture`, and `WebGL2KernelValueDynamicSingleInput` fix: `onRequestFallback` settings in `createKernel` and `createKernel` inclusion of `loopMaxIterations`, `dynamicOutput`, and `dynamicArgument` in settings for when switching kernels fix: `WebGL2Kernel` and `HeadlessGLKernel` typescript definition extensions --- README.md | 17 + dist/gpu-browser-core.js | 102 ++- dist/gpu-browser-core.min.js | 106 ++- dist/gpu-browser.js | 102 ++- dist/gpu-browser.min.js | 106 ++- package.json | 2 +- src/backend/cpu/function-node.js | 8 +- src/backend/function-builder.js | 8 +- src/backend/function-node.js | 2 + src/backend/kernel.js | 15 + src/backend/web-gl/function-node.js | 2 +- .../web-gl/kernel-value/dynamic-html-image.js | 4 +- .../kernel-value/dynamic-number-texture.js | 4 +- .../kernel-value/dynamic-single-input.js | 2 +- .../kernel-value/dynamic-html-image.js | 6 +- .../kernel-value/dynamic-single-array.js | 14 +- .../kernel-value/dynamic-single-input.js | 14 +- src/backend/web-gl2/kernel.js | 8 +- src/gpu.js | 24 +- src/index.d.ts | 6 +- test/all.html | 2 + test/features/constants-image-array.js | 2 +- test/features/constants-image.js | 2 +- test/features/dynamic-arguments.js | 602 ++++++++++++++++++ .../single/arguments/html-image-array.js | 8 +- .../precision/single/arguments/html-image.js | 2 +- .../single/constants/html-image-array.js | 8 +- .../precision/single/constants/html-image.js | 2 +- .../unsigned/arguments/html-image-array.js | 8 +- .../unsigned/arguments/html-image.js | 2 +- .../unsigned/constants/html-image-array.js | 8 +- .../unsigned/constants/html-image.js | 2 +- test/internal/backend/cpu/function-node.js | 28 +- .../backend/web-gl/function-node/index.js | 28 +- test/issues/487-dynamic-arguments.js | 59 ++ test/jellyfish-1.jpeg | Bin 0 -> 3004 bytes test/jellyfish-2.jpeg | Bin 0 -> 1857 bytes test/jellyfish-3.jpeg | Bin 0 -> 1700 bytes test/jellyfish-4.jpeg | Bin 0 -> 1688 bytes test/jellyfish.jpeg | Bin 0 -> 6996 bytes 40 files changed, 1148 insertions(+), 167 deletions(-) create mode 100644 test/features/dynamic-arguments.js create mode 100644 test/issues/487-dynamic-arguments.js create mode 100644 test/jellyfish-1.jpeg create mode 100644 test/jellyfish-2.jpeg create mode 100644 test/jellyfish-3.jpeg create mode 100644 test/jellyfish-4.jpeg create mode 100644 test/jellyfish.jpeg diff --git a/README.md b/README.md index 41fc0dad..a78e9d1e 100644 --- a/README.md +++ b/README.md @@ -92,6 +92,7 @@ NOTE: documentation is slightly out of date for the upcoming release of v2. We * [Supported Math functions](#supported-math-functions) * [How to check what is supported](#how-to-check-what-is-supported) * [Typescript Typings](#typescript-typings) +* [Dealing With Transpilation](#dealing-with-transpilation) * [Full API reference](#full-api-reference) * [Automatically-built Documentation](#automatically-built-documentation) * [Contributors](#contributors) @@ -180,6 +181,7 @@ Settings are an object used to create a `kernel` or `kernelMap`. Example: `gpu. * `immutable` or `kernel.setImmutable(boolean)`: boolean, default = `false` * `strictIntegers` or `kernel.setStrictIntegers(boolean)`: boolean, default = `false` - allows undefined argumentTypes and function return values to use strict integer declarations. * `useLegacyEncoder` or `kernel.setUseLegacyEncoder(boolean)`: boolean, default `false` - more info [here](https://github.com/gpujs/gpu.js/wiki/Encoder-details). +* `warnVarUsage` or `kernel.setWarnVarUsage(boolean)`: turn off var usage warnings, they can be irritating, and in transpiled environments, there is nothing we can do about it. ## Creating and Running Functions @@ -858,6 +860,21 @@ To assist with mostly unit tests, but perhaps in scenarios outside of GPU.js, th ## Typescript Typings Typescript is supported! Typings can be found [here](src/index.d.ts)! +## Dealing With Transpilation +Transpilation doesn't do the best job of keeping code beautiful. To aid in this endeavor GPU.js can handle some scenarios to still aid you harnessing the GPU in less than ideal circumstances. +Here is a list of a few things that GPU.js does to fix transpilation: + +* When a transpiler such as [Babel](https://babeljs.io/) changes `myCall()` to `(0, _myCall.myCall)`, it is gracefully handled. +* Using `var` will have a lot of warnings by default, this can be irritating because sometimes there is nothing we can do about this in transpiled environment. + To aid in the irritation, there is an option to alleviate the irritation. + When `const` and `let` are converted to `var`, and you'r prefer not to see it, use the following: + ```js + const kernel = gpu.createKernel(myKernelFunction) + .setWarnVarUsage(false); + // or + const kernel = gpu.createKernel(myKernelFunction, { output: [1], warnVarUsage: false }); + ``` + ## Full API Reference You can find a [complete API reference here](https://doxdox.org/gpujs/gpu.js/). diff --git a/dist/gpu-browser-core.js b/dist/gpu-browser-core.js index 68169b21..ac0539c7 100644 --- a/dist/gpu-browser-core.js +++ b/dist/gpu-browser-core.js @@ -4,8 +4,8 @@ * * GPU Accelerated JavaScript * - * @version 2.0.0-rc.19 - * @date Tue Jul 02 2019 12:17:44 GMT-0400 (Eastern Daylight Time) + * @version 2.0.0-rc.20 + * @date Fri Jul 05 2019 11:02:54 GMT-0400 (Eastern Daylight Time) * * @license MIT * The MIT License @@ -717,7 +717,7 @@ class CPUFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } retArr.push(`${varDecNode.kind} `); @@ -900,9 +900,9 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(zProperty, retArr); - retArr.push(`*${ size[1] * size[0]})+(`); + retArr.push(`*${ this.dynamicArguments ? '(outputY * outputX)' : size[1] * size[0] })+(`); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -920,7 +920,7 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -1674,7 +1674,9 @@ class FunctionBuilder { functions, leadingReturnStatement, followingReturnStatement, + dynamicArguments, dynamicOutput, + warnVarUsage, } = kernel; const needsArgumentType = (functionName, index) => { @@ -1736,7 +1738,8 @@ class FunctionBuilder { triggerImplyArgumentType, triggerTrackArgumentSynonym, lookupArgumentSynonym, - onFunctionCall + onFunctionCall, + warnVarUsage, })); nestedFunction.traceFunctionAST(ast); functionBuilder.addFunctionNode(nestedFunction); @@ -1764,6 +1767,7 @@ class FunctionBuilder { loopMaxIterations, output, plugins, + dynamicArguments, dynamicOutput, }, extraNodeOptions || {}); @@ -2234,8 +2238,10 @@ class FunctionNode { this.leadingReturnStatement = null; this.followingReturnStatement = null; this.dynamicOutput = null; + this.dynamicArguments = null; this.strictTypingChecking = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; if (settings) { for (const p in settings) { @@ -5465,6 +5471,7 @@ class Kernel { this.optimizeFloatMemory = null; this.strictIntegers = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; } mergeSettings(settings) { @@ -5660,6 +5667,11 @@ class Kernel { return this; } + setWarnVarUsage(flag) { + this.warnVarUsage = flag; + return this; + } + getCanvas() { utils.warnDeprecated('method', 'getCanvas'); return this.canvas; @@ -6784,7 +6796,7 @@ class WebGLFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } const declarations = varDecNode.declarations; @@ -7569,7 +7581,7 @@ module.exports = { const { utils } = require('../../../utils'); const { WebGLKernelValueHTMLImage } = require('./html-image'); -class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { +class WebGLKernelValueDynamicHTMLImage extends WebGLKernelValueHTMLImage { getSource() { return utils.linesToString([ `uniform sampler2D ${this.id}`, @@ -7589,7 +7601,7 @@ class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { } module.exports = { - WebGLKernelValueDynamicInput + WebGLKernelValueDynamicHTMLImage }; },{"../../../utils":89,"./html-image":47}],40:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -7630,8 +7642,8 @@ class WebGLKernelValueDynamicNumberTexture extends WebGLKernelValueNumberTexture } updateValue(value) { - this.dimensions = inputTexture.dimensions; - this.textureSize = inputTexture.size; + this.dimensions = value.dimensions; + this.textureSize = value.size; this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); super.updateValue(value); @@ -7684,7 +7696,7 @@ class WebGLKernelValueDynamicSingleInput extends WebGLKernelValueSingleInput { updateValue(value) { this.dimensions = value.size; this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); - this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * (4 / this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; this.uploadValue = new Float32Array(this.uploadArrayLength); this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); @@ -9957,6 +9969,7 @@ void main(void) { module.exports = { fragmentShader }; + },{}],59:[function(require,module,exports){ const { WebGLFunctionNode } = require('../web-gl/function-node'); @@ -10148,9 +10161,9 @@ module.exports = { }; },{"./html-image-array":71}],63:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicInput } = require('../../web-gl/kernel-value/dynamic-html-image'); +const { WebGLKernelValueDynamicHTMLImage } = require('../../web-gl/kernel-value/dynamic-html-image'); -class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { +class WebGL2KernelValueDynamicHTMLImage extends WebGLKernelValueDynamicHTMLImage { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -10161,7 +10174,7 @@ class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { } module.exports = { - WebGL2KernelValueDynamicInput + WebGL2KernelValueDynamicHTMLImage }; },{"../../../utils":89,"../../web-gl/kernel-value/dynamic-html-image":39}],64:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -10199,9 +10212,9 @@ module.exports = { }; },{"../../../utils":89,"../../web-gl/kernel-value/dynamic-number-texture":41}],66:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleArray } = require('../../web-gl/kernel-value/dynamic-single-array'); +const { WebGL2KernelValueSingleArray } = require('../../web-gl2/kernel-value/single-array'); -class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleArray { +class WebGL2KernelValueDynamicSingleArray extends WebGL2KernelValueSingleArray { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -10209,16 +10222,26 @@ class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleA `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = utils.getDimensions(value, true); + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleArray }; -},{"../../../utils":89,"../../web-gl/kernel-value/dynamic-single-array":42}],67:[function(require,module,exports){ +},{"../../../utils":89,"../../web-gl2/kernel-value/single-array":76}],67:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleInput } = require('../../web-gl/kernel-value/dynamic-single-input'); +const { WebGL2KernelValueSingleInput } = require('../../web-gl2/kernel-value/single-input'); -class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleInput { +class WebGL2KernelValueDynamicSingleInput extends WebGL2KernelValueSingleInput { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -10226,12 +10249,22 @@ class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleI `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = value.size; + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleInput }; -},{"../../../utils":89,"../../web-gl/kernel-value/dynamic-single-input":43}],68:[function(require,module,exports){ +},{"../../../utils":89,"../../web-gl2/kernel-value/single-input":77}],68:[function(require,module,exports){ const { utils } = require('../../../utils'); const { WebGLKernelValueDynamicUnsignedArray } = require('../../web-gl/kernel-value/dynamic-unsigned-array'); @@ -10523,6 +10556,7 @@ let isSupported = null; let testCanvas = null; let testContext = null; let testExtensions = null; + let features = null; class WebGL2Kernel extends WebGLKernel { @@ -11338,7 +11372,25 @@ class GPU { gpu: this, validate, onRequestFallback: (args) => { - const fallbackKernel = new CPUKernel(source, mergedSettings); + const fallbackKernel = new CPUKernel(source, { + graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, + constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, + output: kernel.output, + precision: kernel.precision, + pipeline: kernel.pipeline, + immutable: kernel.immutable, + optimizeFloatMemory: kernel.optimizeFloatMemory, + fixIntegerDivisionAccuracy: kernel.fixIntegerDivisionAccuracy, + functions: kernel.functions, + nativeFunctions: kernel.nativeFunctions, + subKernels: kernel.subKernels, + strictIntegers: kernel.strictIntegers, + debug: kernel.debug, + warnVarUsage: kernel.warnVarUsage, + }); fallbackKernel.build.apply(fallbackKernel, args); const result = fallbackKernel.run.apply(fallbackKernel, args); kernel.replaceKernel(fallbackKernel); @@ -11361,7 +11413,10 @@ class GPU { } const newKernel = switchableKernels[signature] = new this.Kernel(source, { graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, context: kernel.context, canvas: kernel.canvas, output: kernel.output, @@ -11377,6 +11432,7 @@ class GPU { debug: kernel.debug, gpu: this, validate, + warnVarUsage: kernel.warnVarUsage, }); newKernel.build.apply(newKernel, args); newKernel.run.apply(newKernel, args); diff --git a/dist/gpu-browser-core.min.js b/dist/gpu-browser-core.min.js index c47575fc..9d40381f 100644 --- a/dist/gpu-browser-core.min.js +++ b/dist/gpu-browser-core.min.js @@ -4,8 +4,8 @@ * * GPU Accelerated JavaScript * - * @version 2.0.0-rc.19 - * @date Tue Jul 02 2019 12:17:46 GMT-0400 (Eastern Daylight Time) + * @version 2.0.0-rc.20 + * @date Fri Jul 05 2019 11:02:56 GMT-0400 (Eastern Daylight Time) * * @license MIT * The MIT License @@ -17,8 +17,8 @@ * * GPU Accelerated JavaScript * - * @version 2.0.0-rc.19 - * @date Tue Jul 02 2019 12:17:44 GMT-0400 (Eastern Daylight Time) + * @version 2.0.0-rc.20 + * @date Fri Jul 05 2019 11:02:54 GMT-0400 (Eastern Daylight Time) * * @license MIT * The MIT License @@ -730,7 +730,7 @@ class CPUFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } retArr.push(`${varDecNode.kind} `); @@ -913,9 +913,9 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(zProperty, retArr); - retArr.push(`*${ size[1] * size[0]})+(`); + retArr.push(`*${ this.dynamicArguments ? '(outputY * outputX)' : size[1] * size[0] })+(`); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -933,7 +933,7 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -1687,7 +1687,9 @@ class FunctionBuilder { functions, leadingReturnStatement, followingReturnStatement, + dynamicArguments, dynamicOutput, + warnVarUsage, } = kernel; const needsArgumentType = (functionName, index) => { @@ -1749,7 +1751,8 @@ class FunctionBuilder { triggerImplyArgumentType, triggerTrackArgumentSynonym, lookupArgumentSynonym, - onFunctionCall + onFunctionCall, + warnVarUsage, })); nestedFunction.traceFunctionAST(ast); functionBuilder.addFunctionNode(nestedFunction); @@ -1777,6 +1780,7 @@ class FunctionBuilder { loopMaxIterations, output, plugins, + dynamicArguments, dynamicOutput, }, extraNodeOptions || {}); @@ -2247,8 +2251,10 @@ class FunctionNode { this.leadingReturnStatement = null; this.followingReturnStatement = null; this.dynamicOutput = null; + this.dynamicArguments = null; this.strictTypingChecking = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; if (settings) { for (const p in settings) { @@ -5478,6 +5484,7 @@ class Kernel { this.optimizeFloatMemory = null; this.strictIntegers = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; } mergeSettings(settings) { @@ -5673,6 +5680,11 @@ class Kernel { return this; } + setWarnVarUsage(flag) { + this.warnVarUsage = flag; + return this; + } + getCanvas() { utils.warnDeprecated('method', 'getCanvas'); return this.canvas; @@ -6797,7 +6809,7 @@ class WebGLFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } const declarations = varDecNode.declarations; @@ -7582,7 +7594,7 @@ module.exports = { const { utils } = require('../../../utils'); const { WebGLKernelValueHTMLImage } = require('./html-image'); -class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { +class WebGLKernelValueDynamicHTMLImage extends WebGLKernelValueHTMLImage { getSource() { return utils.linesToString([ `uniform sampler2D ${this.id}`, @@ -7602,7 +7614,7 @@ class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { } module.exports = { - WebGLKernelValueDynamicInput + WebGLKernelValueDynamicHTMLImage }; },{"../../../utils":89,"./html-image":47}],40:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -7643,8 +7655,8 @@ class WebGLKernelValueDynamicNumberTexture extends WebGLKernelValueNumberTexture } updateValue(value) { - this.dimensions = inputTexture.dimensions; - this.textureSize = inputTexture.size; + this.dimensions = value.dimensions; + this.textureSize = value.size; this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); super.updateValue(value); @@ -7697,7 +7709,7 @@ class WebGLKernelValueDynamicSingleInput extends WebGLKernelValueSingleInput { updateValue(value) { this.dimensions = value.size; this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); - this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * (4 / this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; this.uploadValue = new Float32Array(this.uploadArrayLength); this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); @@ -9970,6 +9982,7 @@ void main(void) { module.exports = { fragmentShader }; + },{}],59:[function(require,module,exports){ const { WebGLFunctionNode } = require('../web-gl/function-node'); @@ -10161,9 +10174,9 @@ module.exports = { }; },{"./html-image-array":71}],63:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicInput } = require('../../web-gl/kernel-value/dynamic-html-image'); +const { WebGLKernelValueDynamicHTMLImage } = require('../../web-gl/kernel-value/dynamic-html-image'); -class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { +class WebGL2KernelValueDynamicHTMLImage extends WebGLKernelValueDynamicHTMLImage { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -10174,7 +10187,7 @@ class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { } module.exports = { - WebGL2KernelValueDynamicInput + WebGL2KernelValueDynamicHTMLImage }; },{"../../../utils":89,"../../web-gl/kernel-value/dynamic-html-image":39}],64:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -10212,9 +10225,9 @@ module.exports = { }; },{"../../../utils":89,"../../web-gl/kernel-value/dynamic-number-texture":41}],66:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleArray } = require('../../web-gl/kernel-value/dynamic-single-array'); +const { WebGL2KernelValueSingleArray } = require('../../web-gl2/kernel-value/single-array'); -class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleArray { +class WebGL2KernelValueDynamicSingleArray extends WebGL2KernelValueSingleArray { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -10222,16 +10235,26 @@ class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleA `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = utils.getDimensions(value, true); + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleArray }; -},{"../../../utils":89,"../../web-gl/kernel-value/dynamic-single-array":42}],67:[function(require,module,exports){ +},{"../../../utils":89,"../../web-gl2/kernel-value/single-array":76}],67:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleInput } = require('../../web-gl/kernel-value/dynamic-single-input'); +const { WebGL2KernelValueSingleInput } = require('../../web-gl2/kernel-value/single-input'); -class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleInput { +class WebGL2KernelValueDynamicSingleInput extends WebGL2KernelValueSingleInput { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -10239,12 +10262,22 @@ class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleI `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = value.size; + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleInput }; -},{"../../../utils":89,"../../web-gl/kernel-value/dynamic-single-input":43}],68:[function(require,module,exports){ +},{"../../../utils":89,"../../web-gl2/kernel-value/single-input":77}],68:[function(require,module,exports){ const { utils } = require('../../../utils'); const { WebGLKernelValueDynamicUnsignedArray } = require('../../web-gl/kernel-value/dynamic-unsigned-array'); @@ -10536,6 +10569,7 @@ let isSupported = null; let testCanvas = null; let testContext = null; let testExtensions = null; + let features = null; class WebGL2Kernel extends WebGLKernel { @@ -11351,7 +11385,25 @@ class GPU { gpu: this, validate, onRequestFallback: (args) => { - const fallbackKernel = new CPUKernel(source, mergedSettings); + const fallbackKernel = new CPUKernel(source, { + graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, + constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, + output: kernel.output, + precision: kernel.precision, + pipeline: kernel.pipeline, + immutable: kernel.immutable, + optimizeFloatMemory: kernel.optimizeFloatMemory, + fixIntegerDivisionAccuracy: kernel.fixIntegerDivisionAccuracy, + functions: kernel.functions, + nativeFunctions: kernel.nativeFunctions, + subKernels: kernel.subKernels, + strictIntegers: kernel.strictIntegers, + debug: kernel.debug, + warnVarUsage: kernel.warnVarUsage, + }); fallbackKernel.build.apply(fallbackKernel, args); const result = fallbackKernel.run.apply(fallbackKernel, args); kernel.replaceKernel(fallbackKernel); @@ -11374,7 +11426,10 @@ class GPU { } const newKernel = switchableKernels[signature] = new this.Kernel(source, { graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, context: kernel.context, canvas: kernel.canvas, output: kernel.output, @@ -11390,6 +11445,7 @@ class GPU { debug: kernel.debug, gpu: this, validate, + warnVarUsage: kernel.warnVarUsage, }); newKernel.build.apply(newKernel, args); newKernel.run.apply(newKernel, args); diff --git a/dist/gpu-browser.js b/dist/gpu-browser.js index bf890b6f..6e21a38d 100644 --- a/dist/gpu-browser.js +++ b/dist/gpu-browser.js @@ -4,8 +4,8 @@ * * GPU Accelerated JavaScript * - * @version 2.0.0-rc.19 - * @date Tue Jul 02 2019 12:17:44 GMT-0400 (Eastern Daylight Time) + * @version 2.0.0-rc.20 + * @date Fri Jul 05 2019 11:02:54 GMT-0400 (Eastern Daylight Time) * * @license MIT * The MIT License @@ -5481,7 +5481,7 @@ class CPUFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } retArr.push(`${varDecNode.kind} `); @@ -5664,9 +5664,9 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(zProperty, retArr); - retArr.push(`*${ size[1] * size[0]})+(`); + retArr.push(`*${ this.dynamicArguments ? '(outputY * outputX)' : size[1] * size[0] })+(`); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -5684,7 +5684,7 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -6438,7 +6438,9 @@ class FunctionBuilder { functions, leadingReturnStatement, followingReturnStatement, + dynamicArguments, dynamicOutput, + warnVarUsage, } = kernel; const needsArgumentType = (functionName, index) => { @@ -6500,7 +6502,8 @@ class FunctionBuilder { triggerImplyArgumentType, triggerTrackArgumentSynonym, lookupArgumentSynonym, - onFunctionCall + onFunctionCall, + warnVarUsage, })); nestedFunction.traceFunctionAST(ast); functionBuilder.addFunctionNode(nestedFunction); @@ -6528,6 +6531,7 @@ class FunctionBuilder { loopMaxIterations, output, plugins, + dynamicArguments, dynamicOutput, }, extraNodeOptions || {}); @@ -6998,8 +7002,10 @@ class FunctionNode { this.leadingReturnStatement = null; this.followingReturnStatement = null; this.dynamicOutput = null; + this.dynamicArguments = null; this.strictTypingChecking = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; if (settings) { for (const p in settings) { @@ -10229,6 +10235,7 @@ class Kernel { this.optimizeFloatMemory = null; this.strictIntegers = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; } mergeSettings(settings) { @@ -10424,6 +10431,11 @@ class Kernel { return this; } + setWarnVarUsage(flag) { + this.warnVarUsage = flag; + return this; + } + getCanvas() { utils.warnDeprecated('method', 'getCanvas'); return this.canvas; @@ -11548,7 +11560,7 @@ class WebGLFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } const declarations = varDecNode.declarations; @@ -12333,7 +12345,7 @@ module.exports = { const { utils } = require('../../../utils'); const { WebGLKernelValueHTMLImage } = require('./html-image'); -class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { +class WebGLKernelValueDynamicHTMLImage extends WebGLKernelValueHTMLImage { getSource() { return utils.linesToString([ `uniform sampler2D ${this.id}`, @@ -12353,7 +12365,7 @@ class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { } module.exports = { - WebGLKernelValueDynamicInput + WebGLKernelValueDynamicHTMLImage }; },{"../../../utils":90,"./html-image":48}],41:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -12394,8 +12406,8 @@ class WebGLKernelValueDynamicNumberTexture extends WebGLKernelValueNumberTexture } updateValue(value) { - this.dimensions = inputTexture.dimensions; - this.textureSize = inputTexture.size; + this.dimensions = value.dimensions; + this.textureSize = value.size; this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); super.updateValue(value); @@ -12448,7 +12460,7 @@ class WebGLKernelValueDynamicSingleInput extends WebGLKernelValueSingleInput { updateValue(value) { this.dimensions = value.size; this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); - this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * (4 / this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; this.uploadValue = new Float32Array(this.uploadArrayLength); this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); @@ -14721,6 +14733,7 @@ void main(void) { module.exports = { fragmentShader }; + },{}],60:[function(require,module,exports){ const { WebGLFunctionNode } = require('../web-gl/function-node'); @@ -14912,9 +14925,9 @@ module.exports = { }; },{"./html-image-array":72}],64:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicInput } = require('../../web-gl/kernel-value/dynamic-html-image'); +const { WebGLKernelValueDynamicHTMLImage } = require('../../web-gl/kernel-value/dynamic-html-image'); -class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { +class WebGL2KernelValueDynamicHTMLImage extends WebGLKernelValueDynamicHTMLImage { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -14925,7 +14938,7 @@ class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { } module.exports = { - WebGL2KernelValueDynamicInput + WebGL2KernelValueDynamicHTMLImage }; },{"../../../utils":90,"../../web-gl/kernel-value/dynamic-html-image":40}],65:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -14963,9 +14976,9 @@ module.exports = { }; },{"../../../utils":90,"../../web-gl/kernel-value/dynamic-number-texture":42}],67:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleArray } = require('../../web-gl/kernel-value/dynamic-single-array'); +const { WebGL2KernelValueSingleArray } = require('../../web-gl2/kernel-value/single-array'); -class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleArray { +class WebGL2KernelValueDynamicSingleArray extends WebGL2KernelValueSingleArray { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -14973,16 +14986,26 @@ class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleA `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = utils.getDimensions(value, true); + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleArray }; -},{"../../../utils":90,"../../web-gl/kernel-value/dynamic-single-array":43}],68:[function(require,module,exports){ +},{"../../../utils":90,"../../web-gl2/kernel-value/single-array":77}],68:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleInput } = require('../../web-gl/kernel-value/dynamic-single-input'); +const { WebGL2KernelValueSingleInput } = require('../../web-gl2/kernel-value/single-input'); -class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleInput { +class WebGL2KernelValueDynamicSingleInput extends WebGL2KernelValueSingleInput { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -14990,12 +15013,22 @@ class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleI `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = value.size; + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleInput }; -},{"../../../utils":90,"../../web-gl/kernel-value/dynamic-single-input":44}],69:[function(require,module,exports){ +},{"../../../utils":90,"../../web-gl2/kernel-value/single-input":78}],69:[function(require,module,exports){ const { utils } = require('../../../utils'); const { WebGLKernelValueDynamicUnsignedArray } = require('../../web-gl/kernel-value/dynamic-unsigned-array'); @@ -15287,6 +15320,7 @@ let isSupported = null; let testCanvas = null; let testContext = null; let testExtensions = null; + let features = null; class WebGL2Kernel extends WebGLKernel { @@ -16102,7 +16136,25 @@ class GPU { gpu: this, validate, onRequestFallback: (args) => { - const fallbackKernel = new CPUKernel(source, mergedSettings); + const fallbackKernel = new CPUKernel(source, { + graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, + constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, + output: kernel.output, + precision: kernel.precision, + pipeline: kernel.pipeline, + immutable: kernel.immutable, + optimizeFloatMemory: kernel.optimizeFloatMemory, + fixIntegerDivisionAccuracy: kernel.fixIntegerDivisionAccuracy, + functions: kernel.functions, + nativeFunctions: kernel.nativeFunctions, + subKernels: kernel.subKernels, + strictIntegers: kernel.strictIntegers, + debug: kernel.debug, + warnVarUsage: kernel.warnVarUsage, + }); fallbackKernel.build.apply(fallbackKernel, args); const result = fallbackKernel.run.apply(fallbackKernel, args); kernel.replaceKernel(fallbackKernel); @@ -16125,7 +16177,10 @@ class GPU { } const newKernel = switchableKernels[signature] = new this.Kernel(source, { graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, context: kernel.context, canvas: kernel.canvas, output: kernel.output, @@ -16141,6 +16196,7 @@ class GPU { debug: kernel.debug, gpu: this, validate, + warnVarUsage: kernel.warnVarUsage, }); newKernel.build.apply(newKernel, args); newKernel.run.apply(newKernel, args); diff --git a/dist/gpu-browser.min.js b/dist/gpu-browser.min.js index 0ecb60ba..d7f9e6a9 100644 --- a/dist/gpu-browser.min.js +++ b/dist/gpu-browser.min.js @@ -4,8 +4,8 @@ * * GPU Accelerated JavaScript * - * @version 2.0.0-rc.19 - * @date Tue Jul 02 2019 12:17:46 GMT-0400 (Eastern Daylight Time) + * @version 2.0.0-rc.20 + * @date Fri Jul 05 2019 11:02:56 GMT-0400 (Eastern Daylight Time) * * @license MIT * The MIT License @@ -17,8 +17,8 @@ * * GPU Accelerated JavaScript * - * @version 2.0.0-rc.19 - * @date Tue Jul 02 2019 12:17:44 GMT-0400 (Eastern Daylight Time) + * @version 2.0.0-rc.20 + * @date Fri Jul 05 2019 11:02:54 GMT-0400 (Eastern Daylight Time) * * @license MIT * The MIT License @@ -5494,7 +5494,7 @@ class CPUFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } retArr.push(`${varDecNode.kind} `); @@ -5677,9 +5677,9 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(zProperty, retArr); - retArr.push(`*${ size[1] * size[0]})+(`); + retArr.push(`*${ this.dynamicArguments ? '(outputY * outputX)' : size[1] * size[0] })+(`); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -5697,7 +5697,7 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -6451,7 +6451,9 @@ class FunctionBuilder { functions, leadingReturnStatement, followingReturnStatement, + dynamicArguments, dynamicOutput, + warnVarUsage, } = kernel; const needsArgumentType = (functionName, index) => { @@ -6513,7 +6515,8 @@ class FunctionBuilder { triggerImplyArgumentType, triggerTrackArgumentSynonym, lookupArgumentSynonym, - onFunctionCall + onFunctionCall, + warnVarUsage, })); nestedFunction.traceFunctionAST(ast); functionBuilder.addFunctionNode(nestedFunction); @@ -6541,6 +6544,7 @@ class FunctionBuilder { loopMaxIterations, output, plugins, + dynamicArguments, dynamicOutput, }, extraNodeOptions || {}); @@ -7011,8 +7015,10 @@ class FunctionNode { this.leadingReturnStatement = null; this.followingReturnStatement = null; this.dynamicOutput = null; + this.dynamicArguments = null; this.strictTypingChecking = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; if (settings) { for (const p in settings) { @@ -10242,6 +10248,7 @@ class Kernel { this.optimizeFloatMemory = null; this.strictIntegers = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; } mergeSettings(settings) { @@ -10437,6 +10444,11 @@ class Kernel { return this; } + setWarnVarUsage(flag) { + this.warnVarUsage = flag; + return this; + } + getCanvas() { utils.warnDeprecated('method', 'getCanvas'); return this.canvas; @@ -11561,7 +11573,7 @@ class WebGLFunctionNode extends FunctionNode { } astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } const declarations = varDecNode.declarations; @@ -12346,7 +12358,7 @@ module.exports = { const { utils } = require('../../../utils'); const { WebGLKernelValueHTMLImage } = require('./html-image'); -class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { +class WebGLKernelValueDynamicHTMLImage extends WebGLKernelValueHTMLImage { getSource() { return utils.linesToString([ `uniform sampler2D ${this.id}`, @@ -12366,7 +12378,7 @@ class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { } module.exports = { - WebGLKernelValueDynamicInput + WebGLKernelValueDynamicHTMLImage }; },{"../../../utils":90,"./html-image":48}],41:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -12407,8 +12419,8 @@ class WebGLKernelValueDynamicNumberTexture extends WebGLKernelValueNumberTexture } updateValue(value) { - this.dimensions = inputTexture.dimensions; - this.textureSize = inputTexture.size; + this.dimensions = value.dimensions; + this.textureSize = value.size; this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); super.updateValue(value); @@ -12461,7 +12473,7 @@ class WebGLKernelValueDynamicSingleInput extends WebGLKernelValueSingleInput { updateValue(value) { this.dimensions = value.size; this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); - this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * (4 / this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; this.uploadValue = new Float32Array(this.uploadArrayLength); this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); @@ -14734,6 +14746,7 @@ void main(void) { module.exports = { fragmentShader }; + },{}],60:[function(require,module,exports){ const { WebGLFunctionNode } = require('../web-gl/function-node'); @@ -14925,9 +14938,9 @@ module.exports = { }; },{"./html-image-array":72}],64:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicInput } = require('../../web-gl/kernel-value/dynamic-html-image'); +const { WebGLKernelValueDynamicHTMLImage } = require('../../web-gl/kernel-value/dynamic-html-image'); -class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { +class WebGL2KernelValueDynamicHTMLImage extends WebGLKernelValueDynamicHTMLImage { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -14938,7 +14951,7 @@ class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { } module.exports = { - WebGL2KernelValueDynamicInput + WebGL2KernelValueDynamicHTMLImage }; },{"../../../utils":90,"../../web-gl/kernel-value/dynamic-html-image":40}],65:[function(require,module,exports){ const { utils } = require('../../../utils'); @@ -14976,9 +14989,9 @@ module.exports = { }; },{"../../../utils":90,"../../web-gl/kernel-value/dynamic-number-texture":42}],67:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleArray } = require('../../web-gl/kernel-value/dynamic-single-array'); +const { WebGL2KernelValueSingleArray } = require('../../web-gl2/kernel-value/single-array'); -class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleArray { +class WebGL2KernelValueDynamicSingleArray extends WebGL2KernelValueSingleArray { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -14986,16 +14999,26 @@ class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleA `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = utils.getDimensions(value, true); + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleArray }; -},{"../../../utils":90,"../../web-gl/kernel-value/dynamic-single-array":43}],68:[function(require,module,exports){ +},{"../../../utils":90,"../../web-gl2/kernel-value/single-array":77}],68:[function(require,module,exports){ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleInput } = require('../../web-gl/kernel-value/dynamic-single-input'); +const { WebGL2KernelValueSingleInput } = require('../../web-gl2/kernel-value/single-input'); -class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleInput { +class WebGL2KernelValueDynamicSingleInput extends WebGL2KernelValueSingleInput { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -15003,12 +15026,22 @@ class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleI `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = value.size; + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { WebGL2KernelValueDynamicSingleInput }; -},{"../../../utils":90,"../../web-gl/kernel-value/dynamic-single-input":44}],69:[function(require,module,exports){ +},{"../../../utils":90,"../../web-gl2/kernel-value/single-input":78}],69:[function(require,module,exports){ const { utils } = require('../../../utils'); const { WebGLKernelValueDynamicUnsignedArray } = require('../../web-gl/kernel-value/dynamic-unsigned-array'); @@ -15300,6 +15333,7 @@ let isSupported = null; let testCanvas = null; let testContext = null; let testExtensions = null; + let features = null; class WebGL2Kernel extends WebGLKernel { @@ -16115,7 +16149,25 @@ class GPU { gpu: this, validate, onRequestFallback: (args) => { - const fallbackKernel = new CPUKernel(source, mergedSettings); + const fallbackKernel = new CPUKernel(source, { + graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, + constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, + output: kernel.output, + precision: kernel.precision, + pipeline: kernel.pipeline, + immutable: kernel.immutable, + optimizeFloatMemory: kernel.optimizeFloatMemory, + fixIntegerDivisionAccuracy: kernel.fixIntegerDivisionAccuracy, + functions: kernel.functions, + nativeFunctions: kernel.nativeFunctions, + subKernels: kernel.subKernels, + strictIntegers: kernel.strictIntegers, + debug: kernel.debug, + warnVarUsage: kernel.warnVarUsage, + }); fallbackKernel.build.apply(fallbackKernel, args); const result = fallbackKernel.run.apply(fallbackKernel, args); kernel.replaceKernel(fallbackKernel); @@ -16138,7 +16190,10 @@ class GPU { } const newKernel = switchableKernels[signature] = new this.Kernel(source, { graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, context: kernel.context, canvas: kernel.canvas, output: kernel.output, @@ -16154,6 +16209,7 @@ class GPU { debug: kernel.debug, gpu: this, validate, + warnVarUsage: kernel.warnVarUsage, }); newKernel.build.apply(newKernel, args); newKernel.run.apply(newKernel, args); diff --git a/package.json b/package.json index 882866dc..fd4d20e8 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "gpu.js", - "version": "2.0.0-rc.19", + "version": "2.0.0-rc.20", "description": "GPU Accelerated JavaScript", "engines": { "node": ">=10.0.0" diff --git a/src/backend/cpu/function-node.js b/src/backend/cpu/function-node.js index 72db578d..0f3c8acb 100644 --- a/src/backend/cpu/function-node.js +++ b/src/backend/cpu/function-node.js @@ -325,7 +325,7 @@ class CPUFunctionNode extends FunctionNode { * @returns {Array} the append retArr */ astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } retArr.push(`${varDecNode.kind} `); @@ -529,9 +529,9 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(zProperty, retArr); - retArr.push(`*${ size[1] * size[0]})+(`); + retArr.push(`*${ this.dynamicArguments ? '(outputY * outputX)' : size[1] * size[0] })+(`); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { @@ -549,7 +549,7 @@ class CPUFunctionNode extends FunctionNode { if (isInput) { retArr.push('[('); this.astGeneric(yProperty, retArr); - retArr.push(`*${ size[0] })+`); + retArr.push(`*${ this.dynamicArguments ? 'outputX' : size[0] })+`); this.astGeneric(xProperty, retArr); retArr.push(']'); } else { diff --git a/src/backend/function-builder.js b/src/backend/function-builder.js index 088cd04d..598f6916 100644 --- a/src/backend/function-builder.js +++ b/src/backend/function-builder.js @@ -6,7 +6,7 @@ class FunctionBuilder { /** * - * @param {typeof Kernel} kernel + * @param {Kernel} kernel * @param {FunctionNode} FunctionNode * @param {object} [extraNodeOptions] * @returns {FunctionBuilder} @@ -34,7 +34,9 @@ class FunctionBuilder { functions, leadingReturnStatement, followingReturnStatement, + dynamicArguments, dynamicOutput, + warnVarUsage, } = kernel; const needsArgumentType = (functionName, index) => { @@ -96,7 +98,8 @@ class FunctionBuilder { triggerImplyArgumentType, triggerTrackArgumentSynonym, lookupArgumentSynonym, - onFunctionCall + onFunctionCall, + warnVarUsage, })); nestedFunction.traceFunctionAST(ast); functionBuilder.addFunctionNode(nestedFunction); @@ -124,6 +127,7 @@ class FunctionBuilder { loopMaxIterations, output, plugins, + dynamicArguments, dynamicOutput, }, extraNodeOptions || {}); diff --git a/src/backend/function-node.js b/src/backend/function-node.js index c800b39e..a9816366 100644 --- a/src/backend/function-node.js +++ b/src/backend/function-node.js @@ -59,8 +59,10 @@ class FunctionNode { this.leadingReturnStatement = null; this.followingReturnStatement = null; this.dynamicOutput = null; + this.dynamicArguments = null; this.strictTypingChecking = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; if (settings) { for (const p in settings) { diff --git a/src/backend/kernel.js b/src/backend/kernel.js index 3a218217..094529e5 100644 --- a/src/backend/kernel.js +++ b/src/backend/kernel.js @@ -16,6 +16,10 @@ class Kernel { throw new Error(`"isContextMatch" not implemented on ${ this.name }`); } + /** + * @type {IKernelFeatures} + * Used internally to populate the kernel.feature, which is a getter for the output of this value + */ static getFeatures() { throw new Error(`"getFeatures" not implemented on ${ this.name }`); } @@ -181,6 +185,7 @@ class Kernel { this.optimizeFloatMemory = null; this.strictIntegers = false; this.fixIntegerDivisionAccuracy = null; + this.warnVarUsage = true; } mergeSettings(settings) { @@ -486,6 +491,16 @@ class Kernel { return this; } + /** + * + * @param {Boolean} flag + * @return {Kernel} + */ + setWarnVarUsage(flag) { + this.warnVarUsage = flag; + return this; + } + /** * @deprecated * @returns {Object} diff --git a/src/backend/web-gl/function-node.js b/src/backend/web-gl/function-node.js index 44105ba8..4551076d 100644 --- a/src/backend/web-gl/function-node.js +++ b/src/backend/web-gl/function-node.js @@ -767,7 +767,7 @@ class WebGLFunctionNode extends FunctionNode { * @returns {Array} the append retArr */ astVariableDeclaration(varDecNode, retArr) { - if (varDecNode.kind === 'var') { + if (varDecNode.kind === 'var' && this.warnVarUsage) { this.varWarn(); } const declarations = varDecNode.declarations; diff --git a/src/backend/web-gl/kernel-value/dynamic-html-image.js b/src/backend/web-gl/kernel-value/dynamic-html-image.js index 34f7925e..e5653205 100644 --- a/src/backend/web-gl/kernel-value/dynamic-html-image.js +++ b/src/backend/web-gl/kernel-value/dynamic-html-image.js @@ -1,7 +1,7 @@ const { utils } = require('../../../utils'); const { WebGLKernelValueHTMLImage } = require('./html-image'); -class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { +class WebGLKernelValueDynamicHTMLImage extends WebGLKernelValueHTMLImage { getSource() { return utils.linesToString([ `uniform sampler2D ${this.id}`, @@ -21,5 +21,5 @@ class WebGLKernelValueDynamicInput extends WebGLKernelValueHTMLImage { } module.exports = { - WebGLKernelValueDynamicInput + WebGLKernelValueDynamicHTMLImage }; \ No newline at end of file diff --git a/src/backend/web-gl/kernel-value/dynamic-number-texture.js b/src/backend/web-gl/kernel-value/dynamic-number-texture.js index a2d33556..246c47cc 100644 --- a/src/backend/web-gl/kernel-value/dynamic-number-texture.js +++ b/src/backend/web-gl/kernel-value/dynamic-number-texture.js @@ -11,8 +11,8 @@ class WebGLKernelValueDynamicNumberTexture extends WebGLKernelValueNumberTexture } updateValue(value) { - this.dimensions = inputTexture.dimensions; - this.textureSize = inputTexture.size; + this.dimensions = value.dimensions; + this.textureSize = value.size; this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); super.updateValue(value); diff --git a/src/backend/web-gl/kernel-value/dynamic-single-input.js b/src/backend/web-gl/kernel-value/dynamic-single-input.js index fefcd7c7..1db18f17 100644 --- a/src/backend/web-gl/kernel-value/dynamic-single-input.js +++ b/src/backend/web-gl/kernel-value/dynamic-single-input.js @@ -13,7 +13,7 @@ class WebGLKernelValueDynamicSingleInput extends WebGLKernelValueSingleInput { updateValue(value) { this.dimensions = value.size; this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); - this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * (4 / this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; this.uploadValue = new Float32Array(this.uploadArrayLength); this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); this.kernel.setUniform2iv(this.sizeId, this.textureSize); diff --git a/src/backend/web-gl2/kernel-value/dynamic-html-image.js b/src/backend/web-gl2/kernel-value/dynamic-html-image.js index dbcceaac..da5e6eed 100644 --- a/src/backend/web-gl2/kernel-value/dynamic-html-image.js +++ b/src/backend/web-gl2/kernel-value/dynamic-html-image.js @@ -1,7 +1,7 @@ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicInput } = require('../../web-gl/kernel-value/dynamic-html-image'); +const { WebGLKernelValueDynamicHTMLImage } = require('../../web-gl/kernel-value/dynamic-html-image'); -class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { +class WebGL2KernelValueDynamicHTMLImage extends WebGLKernelValueDynamicHTMLImage { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -12,5 +12,5 @@ class WebGL2KernelValueDynamicInput extends WebGLKernelValueDynamicInput { } module.exports = { - WebGL2KernelValueDynamicInput + WebGL2KernelValueDynamicHTMLImage }; \ No newline at end of file diff --git a/src/backend/web-gl2/kernel-value/dynamic-single-array.js b/src/backend/web-gl2/kernel-value/dynamic-single-array.js index a5d25b78..30ef8afe 100644 --- a/src/backend/web-gl2/kernel-value/dynamic-single-array.js +++ b/src/backend/web-gl2/kernel-value/dynamic-single-array.js @@ -1,7 +1,7 @@ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleArray } = require('../../web-gl/kernel-value/dynamic-single-array'); +const { WebGL2KernelValueSingleArray } = require('../../web-gl2/kernel-value/single-array'); -class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleArray { +class WebGL2KernelValueDynamicSingleArray extends WebGL2KernelValueSingleArray { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -9,6 +9,16 @@ class WebGL2KernelValueDynamicSingleArray extends WebGLKernelValueDynamicSingleA `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = utils.getDimensions(value, true); + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { diff --git a/src/backend/web-gl2/kernel-value/dynamic-single-input.js b/src/backend/web-gl2/kernel-value/dynamic-single-input.js index 9c6d616e..5826d04d 100644 --- a/src/backend/web-gl2/kernel-value/dynamic-single-input.js +++ b/src/backend/web-gl2/kernel-value/dynamic-single-input.js @@ -1,7 +1,7 @@ const { utils } = require('../../../utils'); -const { WebGLKernelValueDynamicSingleInput } = require('../../web-gl/kernel-value/dynamic-single-input'); +const { WebGL2KernelValueSingleInput } = require('../../web-gl2/kernel-value/single-input'); -class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleInput { +class WebGL2KernelValueDynamicSingleInput extends WebGL2KernelValueSingleInput { getSource() { return utils.linesToString([ `uniform highp sampler2D ${this.id}`, @@ -9,6 +9,16 @@ class WebGL2KernelValueDynamicSingleInput extends WebGLKernelValueDynamicSingleI `uniform highp ivec3 ${this.dimensionsId}`, ]); } + + updateValue(value) { + this.dimensions = value.size; + this.textureSize = utils.getMemoryOptimizedFloatTextureSize(this.dimensions, this.bitRatio); + this.uploadArrayLength = this.textureSize[0] * this.textureSize[1] * this.bitRatio; + this.uploadValue = new Float32Array(this.uploadArrayLength); + this.kernel.setUniform3iv(this.dimensionsId, this.dimensions); + this.kernel.setUniform2iv(this.sizeId, this.textureSize); + super.updateValue(value); + } } module.exports = { diff --git a/src/backend/web-gl2/kernel.js b/src/backend/web-gl2/kernel.js index c9419c3f..cd466994 100644 --- a/src/backend/web-gl2/kernel.js +++ b/src/backend/web-gl2/kernel.js @@ -10,14 +10,10 @@ let isSupported = null; let testCanvas = null; let testContext = null; let testExtensions = null; + /** * - * @type {{ - * isFloatRead: Boolean, - * isIntegerDivisionAccurate: Boolean, - * kernelMap: Boolean, - * isTextureFloat: Boolean, - * }|null} + * @type {IKernelFeatures} */ let features = null; diff --git a/src/gpu.js b/src/gpu.js index 3bf4aab7..f9150a09 100644 --- a/src/gpu.js +++ b/src/gpu.js @@ -223,7 +223,25 @@ class GPU { gpu: this, validate, onRequestFallback: (args) => { - const fallbackKernel = new CPUKernel(source, mergedSettings); + const fallbackKernel = new CPUKernel(source, { + graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, + constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, + output: kernel.output, + precision: kernel.precision, + pipeline: kernel.pipeline, + immutable: kernel.immutable, + optimizeFloatMemory: kernel.optimizeFloatMemory, + fixIntegerDivisionAccuracy: kernel.fixIntegerDivisionAccuracy, + functions: kernel.functions, + nativeFunctions: kernel.nativeFunctions, + subKernels: kernel.subKernels, + strictIntegers: kernel.strictIntegers, + debug: kernel.debug, + warnVarUsage: kernel.warnVarUsage, + }); fallbackKernel.build.apply(fallbackKernel, args); const result = fallbackKernel.run.apply(fallbackKernel, args); kernel.replaceKernel(fallbackKernel); @@ -246,7 +264,10 @@ class GPU { } const newKernel = switchableKernels[signature] = new this.Kernel(source, { graphical: kernel.graphical, + loopMaxIterations: kernel.loopMaxIterations, constants: kernel.constants, + dynamicOutput: kernel.dynamicOutput, + dynamicArgument: kernel.dynamicArguments, context: kernel.context, canvas: kernel.canvas, output: kernel.output, @@ -262,6 +283,7 @@ class GPU { debug: kernel.debug, gpu: this, validate, + warnVarUsage: kernel.warnVarUsage, }); newKernel.build.apply(newKernel, args); newKernel.run.apply(newKernel, args); diff --git a/src/index.d.ts b/src/index.d.ts index 61bd43b7..ef62b30f 100644 --- a/src/index.d.ts +++ b/src/index.d.ts @@ -83,6 +83,7 @@ export abstract class Kernel { static nativeFunctionReturnType(source: string): string; static destroyContext(context: any): void; static features: IKernelFeatures; + static getFeatures(): IKernelFeatures; source: string | object; Kernel: Kernel; output: number[]; @@ -169,6 +170,7 @@ export abstract class Kernel { ): string; toJSON(): object; setOutput(flag: number[]): this; + setWarnVarUsage(flag: boolean): this; setOptimizeFloatMemory(flag: boolean): this; setArgumentTypes(flag: any): this; setDebug(flag: boolean): this; @@ -197,10 +199,10 @@ export class GLKernel extends Kernel { export class WebGLKernel extends GLKernel { } -export class WebGL2Kernel extends GLKernel { +export class WebGL2Kernel extends WebGLKernel { } -export class HeadlessGLKernel extends GLKernel { +export class HeadlessGLKernel extends WebGLKernel { } diff --git a/test/all.html b/test/all.html index f0505aab..55f4c454 100644 --- a/test/all.html +++ b/test/all.html @@ -159,6 +159,7 @@ + @@ -273,5 +274,6 @@ + diff --git a/test/features/constants-image-array.js b/test/features/constants-image-array.js index efd533cc..f286916b 100644 --- a/test/features/constants-image-array.js +++ b/test/features/constants-image-array.js @@ -34,7 +34,7 @@ function feature(mode, done) { gpu.createKernel(fn, settings)(); } - image.src = ''; + image.src = 'jellyfish-1.jpeg'; image.onload = () => { settings[0] = image.width; settings[1] = image.height; diff --git a/test/features/constants-image.js b/test/features/constants-image.js index f270bbbb..ae70389c 100644 --- a/test/features/constants-image.js +++ b/test/features/constants-image.js @@ -5,7 +5,7 @@ describe('features: constants image'); function imageConstantTest(mode, done) { const gpu = new GPU({ mode }); const image = new Image(); - image.src = ''; + image.src = 'jellyfish-1.jpeg'; image.onload = function() { const width = image.width; const height = image.height; diff --git a/test/features/dynamic-arguments.js b/test/features/dynamic-arguments.js new file mode 100644 index 00000000..afe1555f --- /dev/null +++ b/test/features/dynamic-arguments.js @@ -0,0 +1,602 @@ +const { assert, skip, test, module: describe, only } = require('qunit'); +const { GPU, input } = require('../../src'); + +describe('features: dynamic arguments'); + +function testHTMLImage(done, mode) { + const image1 = new Image(); + const image2 = new Image(); + image1.src = 'jellyfish.jpeg'; + image2.src = 'jellyfish-1.jpeg'; + let loadedCount = 0; + image1.onload = image2.onload = () => { + loadedCount++; + if (loadedCount === 2) loaded(); + }; + + function loaded() { + const gpu = new GPU({ mode }); + const kernel = gpu.createKernel(function(image) { + const pixel = image[this.thread.y][this.thread.x]; + return (pixel[0] + pixel[1] + pixel[2]) / 3; + }) + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([276, 183]); + + const pixels1 = kernel(image1); + kernel.setOutput([138, 91]); + const pixels2 = kernel(image2); + + assert.ok(pixels1[0][0] > .43); + assert.ok(pixels1[0][0] < .45); + assert.equal(pixels1.length, image1.height); + assert.equal(pixels1[0].length, image1.width); + assert.ok(pixels2[0][0] > .82); + assert.ok(pixels2[0][0] < .83); + assert.equal(pixels2.length, image2.height); + assert.equal(pixels2[0].length, image2.width); + + gpu.destroy(); + done(); + } +} + +(typeof Image !== 'undefined' ? test : skip)('HTML Image auto', t => { + testHTMLImage(t.async()); +}); + +(typeof Image !== 'undefined' && GPU.isWebGLSupported ? test : skip)('HTML Image webgl', t => { + testHTMLImage(t.async(), 'webgl'); +}); + +(typeof Image !== 'undefined' && GPU.isWebGL2Supported ? test : skip)('HTML Image webgl2', t => { + testHTMLImage(t.async(), 'webgl2'); +}); + +(typeof Image !== 'undefined' ? test : skip)('HTML Image cpu', t => { + testHTMLImage(t.async(), 'cpu'); +}); + +function testMemoryOptimizedNumberTexture(mode) { + const gpu = new GPU({ mode }); + const matrix4X4 = [ + [1,2,3,4], + [5,6,7,8], + [9,10,11,12], + [13,14,15,16], + ]; + const texture4X4 = ( + gpu.createKernel(function(value) { + return value[this.thread.y][this.thread.x]; + }) + .setOutput([4, 4]) + .setPrecision('single') + .setOptimizeFloatMemory(true) + .setPipeline(true) + )(matrix4X4); + + const matrix3X3 = [ + [1,2,3], + [4,5,6], + [7,8,9] + ]; + const texture3X3 = ( + gpu.createKernel(function(value) { + return value[this.thread.y][this.thread.x]; + }) + .setOutput([3, 3]) + .setPrecision('single') + .setOptimizeFloatMemory(true) + .setPipeline(true) + )(matrix3X3); + + const matrix2X2 = [ + [1,2], + [3,4] + ]; + const texture2X2 = ( + gpu.createKernel(function(value) { + return value[this.thread.y][this.thread.x]; + }) + .setOutput([2, 2]) + .setPrecision('single') + .setOptimizeFloatMemory(true) + .setPipeline(true) + )(matrix2X2); + + const kernel = gpu.createKernel(function(texture) { + return texture[this.thread.y][this.thread.x]; + }) + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([4,4]); + + assert.deepEqual(kernel(texture4X4), [ + new Float32Array([1,2,3,4]), + new Float32Array([5,6,7,8]), + new Float32Array([9,10,11,12]), + new Float32Array([13,14,15,16]), + ]); + + kernel.setOutput([3, 3]); + assert.deepEqual(kernel(texture3X3), [ + new Float32Array([1,2,3]), + new Float32Array([4,5,6]), + new Float32Array([7,8,9]), + ]); + + kernel.setOutput([2, 2]); + assert.deepEqual(kernel(texture2X2), [ + new Float32Array([1,2]), + new Float32Array([3,4]), + ]); + + assert.ok(kernel.kernelArguments[0].constructor.name.match('DynamicMemoryOptimizedNumberTexture')); + + gpu.destroy(); +} + +(GPU.isSinglePrecisionSupported ? test : skip)('MemoryOptimizedNumberTexture (GPU only) auto', () => { + testMemoryOptimizedNumberTexture(); +}); + +(GPU.isSinglePrecisionSupported ? test : skip)('MemoryOptimizedNumberTexture (GPU only) gpu', () => { + testMemoryOptimizedNumberTexture('gpu'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGLSupported ? test : skip)('MemoryOptimizedNumberTexture (GPU only) webgl', () => { + testMemoryOptimizedNumberTexture('webgl'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGL2Supported ? test : skip)('MemoryOptimizedNumberTexture (GPU only) webgl2', () => { + testMemoryOptimizedNumberTexture('webgl2'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isHeadlessGLSupported ? test : skip)('MemoryOptimizedNumberTexture (GPU only) headlessgl', () => { + testMemoryOptimizedNumberTexture('headlessgl'); +}); + +function testNumberTexture(mode) { + const gpu = new GPU({ mode }); + const matrix4X4 = [ + [1,2,3,4], + [5,6,7,8], + [9,10,11,12], + [13,14,15,16], + ]; + const texture4X4 = ( + gpu.createKernel(function(value) { + return value[this.thread.y][this.thread.x]; + }) + .setOutput([4, 4]) + .setPrecision('single') + .setOptimizeFloatMemory(true) + .setPipeline(true) + )(matrix4X4); + + const matrix3X3 = [ + [1,2,3], + [4,5,6], + [7,8,9] + ]; + const texture3X3 = ( + gpu.createKernel(function(value) { + return value[this.thread.y][this.thread.x]; + }) + .setOutput([3, 3]) + .setPrecision('single') + .setOptimizeFloatMemory(true) + .setPipeline(true) + )(matrix3X3); + + const matrix2X2 = [ + [1,2], + [3,4] + ]; + const texture2X2 = ( + gpu.createKernel(function(value) { + return value[this.thread.y][this.thread.x]; + }) + .setOutput([2, 2]) + .setPrecision('single') + .setPipeline(true) + )(matrix2X2); + + const kernel = gpu.createKernel(function(texture) { + return texture[this.thread.y][this.thread.x]; + }) + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([4,4]); + + assert.deepEqual(kernel(texture4X4), [ + new Float32Array([1,2,3,4]), + new Float32Array([5,6,7,8]), + new Float32Array([9,10,11,12]), + new Float32Array([13,14,15,16]), + ]); + + kernel.setOutput([3, 3]); + assert.deepEqual(kernel(texture3X3), [ + new Float32Array([1,2,3]), + new Float32Array([4,5,6]), + new Float32Array([7,8,9]), + ]); + + kernel.setOutput([2, 2]); + assert.deepEqual(kernel(texture2X2), [ + new Float32Array([1,2]), + new Float32Array([3,4]), + ]); + + assert.ok(kernel.kernelArguments[0].constructor.name.match('NumberTexture')); + + gpu.destroy(); +} + +(GPU.isSinglePrecisionSupported ? test : skip)('NumberTexture (GPU only) auto', () => { + testNumberTexture(); +}); + +(GPU.isSinglePrecisionSupported ? test : skip)('NumberTexture (GPU only) gpu', () => { + testNumberTexture('gpu'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGLSupported ? test : skip)('NumberTexture (GPU only) webgl', () => { + testNumberTexture('webgl'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGL2Supported ? test : skip)('NumberTexture (GPU only) webgl2', () => { + testNumberTexture('webgl2'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isHeadlessGLSupported ? test : skip)('NumberTexture (GPU only) headlessgl', () => { + testNumberTexture('headlessgl'); +}); + +function testUnsignedPrecisionArray(mode) { + const gpu = new GPU({ mode }); + const kernel = gpu.createKernel(function(input) { + return input[this.thread.x]; + }) + .setPrecision('unsigned') + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([5]); + + assert.deepEqual(kernel([1,2,3,4,5]), new Float32Array([1,2,3,4,5])); + kernel.setOutput([4]); + assert.deepEqual(kernel([1,2,3,4]), new Float32Array([1,2,3,4])); + kernel.setOutput([3]); + assert.deepEqual(kernel([1,2,3]), new Float32Array([1,2,3])); + kernel.setOutput([2]); + assert.deepEqual(kernel([1,2]), new Float32Array([1,2])); + gpu.destroy(); +} + +test('unsigned precision Array auto', () => { + testUnsignedPrecisionArray(); +}); + +test('unsigned precision Array gpu', () => { + testUnsignedPrecisionArray('gpu'); +}); + +(GPU.isWebGLSupported ? test : skip)('unsigned precision Array webgl', () => { + testUnsignedPrecisionArray('webgl'); +}); + +(GPU.isWebGL2Supported ? test : skip)('unsigned precision Array webgl2', () => { + testUnsignedPrecisionArray('webgl2'); +}); + +(GPU.isHeadlessGLSupported ? test : skip)('unsigned precision Array headlessgl', () => { + testUnsignedPrecisionArray('headlessgl'); +}); + +test('unsigned precision Array cpu', () => { + testUnsignedPrecisionArray('cpu'); +}); + +function testSinglePrecisionArray(mode) { + const gpu = new GPU({ mode }); + const kernel = gpu.createKernel(function(input) { + return input[this.thread.x]; + }) + .setPrecision('single') + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([5]); + + assert.deepEqual(kernel([1,2,3,4,5]), new Float32Array([1,2,3,4,5])); + kernel.setOutput([4]); + assert.deepEqual(kernel([1,2,3,4]), new Float32Array([1,2,3,4])); + kernel.setOutput([3]); + assert.deepEqual(kernel([1,2,3]), new Float32Array([1,2,3])); + kernel.setOutput([2]); + assert.deepEqual(kernel([1,2]), new Float32Array([1,2])); + gpu.destroy(); +} + +(GPU.isSinglePrecisionSupported ? test : skip)('single precision Array auto', () => { + testSinglePrecisionArray(); +}); + +(GPU.isSinglePrecisionSupported ? test : skip)('single precision Array gpu', () => { + testSinglePrecisionArray('gpu'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGLSupported ? test : skip)('single precision Array webgl', () => { + testSinglePrecisionArray('webgl'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGL2Supported ? test : skip)('single precision Array webgl2', () => { + testSinglePrecisionArray('webgl2'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isHeadlessGLSupported ? test : skip)('single precision Array headlessgl', () => { + testSinglePrecisionArray('headlessgl'); +}); + +test('single precision Array cpu', () => { + testSinglePrecisionArray('cpu'); +}); + +function testUnsignedPrecisionMatrix(mode) { + const gpu = new GPU({ mode }); + const kernel = gpu.createKernel(function(input) { + return input[this.thread.y][this.thread.x]; + }) + .setPrecision('unsigned') + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([4,4]); + + let matrix = [ + [1,2,3,4], + [5,6,7,8], + [9,10,11,12], + [13,14,15,16] + ]; + assert.deepEqual(kernel(matrix), matrix.map(row => new Float32Array(row))); + + kernel.setOutput([3,3]); + matrix = [ + [1,2,3], + [4,5,6], + [7,8,9] + ]; + assert.deepEqual(kernel(matrix), matrix.map(row => new Float32Array(row))); + + kernel.setOutput([2,2]); + matrix = [ + [1,2], + [3,4] + ]; + assert.deepEqual(kernel(matrix), matrix.map(row => new Float32Array(row))); + gpu.destroy(); +} + +test('unsigned precision Matrix auto', () => { + testUnsignedPrecisionMatrix(); +}); + +test('unsigned precision Matrix gpu', () => { + testUnsignedPrecisionMatrix('gpu'); +}); + +(GPU.isWebGLSupported ? test : skip)('unsigned precision Matrix webgl', () => { + testUnsignedPrecisionMatrix('webgl'); +}); + +(GPU.isWebGL2Supported ? test : skip)('unsigned precision Matrix webgl2', () => { + testUnsignedPrecisionMatrix('webgl2'); +}); + +(GPU.isHeadlessGLSupported ? test : skip)('unsigned precision Matrix headlessgl', () => { + testUnsignedPrecisionMatrix('headlessgl'); +}); + +test('unsigned precision Matrix cpu', () => { + testUnsignedPrecisionMatrix('cpu'); +}); + +function testSinglePrecisionMatrix(mode) { + const gpu = new GPU({ mode }); + const kernel = gpu.createKernel(function(input) { + return input[this.thread.y][this.thread.x]; + }) + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([4,4]); + + let matrix = [ + [1,2,3,4], + [5,6,7,8], + [9,10,11,12], + [13,14,15,16] + ]; + assert.deepEqual(kernel(matrix), matrix.map(row => new Float32Array(row))); + + kernel.setOutput([3,3]); + matrix = [ + [1,2,3], + [4,5,6], + [7,8,9] + ]; + assert.deepEqual(kernel(matrix), matrix.map(row => new Float32Array(row))); + + kernel.setOutput([2,2]); + matrix = [ + [1,2], + [3,4] + ]; + assert.deepEqual(kernel(matrix), matrix.map(row => new Float32Array(row))); + gpu.destroy(); +} + +(GPU.isSinglePrecisionSupported ? test : skip)('single precision Matrix auto', () => { + testSinglePrecisionMatrix(); +}); + +(GPU.isSinglePrecisionSupported ? test : skip)('single precision Matrix gpu', () => { + testSinglePrecisionMatrix('gpu'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGLSupported ? test : skip)('single precision Matrix webgl', () => { + testSinglePrecisionMatrix('webgl'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGL2Supported ? test : skip)('single precision Matrix webgl2', () => { + testSinglePrecisionMatrix('webgl2'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isHeadlessGLSupported ? test : skip)('single precision Matrix headlessgl', () => { + testSinglePrecisionMatrix('headlessgl'); +}); + +test('single precision Matrix cpu', () => { + testSinglePrecisionMatrix('cpu'); +}); + +function testUnsignedPrecisionInputMatrix(mode) { + const gpu = new GPU({ mode }); + const kernel = gpu.createKernel(function(input) { + return input[this.thread.y][this.thread.x]; + }) + .setPrecision('unsigned') + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([4,4]); + + let matrix = input([ + 1,2,3,4, + 5,6,7,8, + 9,10,11,12, + 13,14,15,16 + ], [4, 4]); + assert.deepEqual(kernel(matrix), [ + new Float32Array([1,2,3,4]), + new Float32Array([5,6,7,8]), + new Float32Array([9,10,11,12]), + new Float32Array([13,14,15,16]), + ]); + + kernel.setOutput([3,3]); + matrix = input([ + 1,2,3, + 4,5,6, + 7,8,9 + ], [3,3]); + assert.deepEqual(kernel(matrix), [ + new Float32Array([1,2,3]), + new Float32Array([4,5,6]), + new Float32Array([7,8,9]) + ]); + + kernel.setOutput([2,2]); + matrix = input([ + 1,2, + 3,4 + ], [2,2]); + assert.deepEqual(kernel(matrix), [ + new Float32Array([1,2]), + new Float32Array([3,4]) + ]); + gpu.destroy(); +} + +test('unsigned precision Input Matrix auto', () => { + testUnsignedPrecisionInputMatrix(); +}); + +test('unsigned precision Input Matrix gpu', () => { + testUnsignedPrecisionInputMatrix('gpu'); +}); + +(GPU.isWebGLSupported ? test : skip)('unsigned precision Input Matrix webgl', () => { + testUnsignedPrecisionInputMatrix('webgl'); +}); + +(GPU.isWebGL2Supported ? test : skip)('unsigned precision Input Matrix webgl2', () => { + testUnsignedPrecisionInputMatrix('webgl2'); +}); + +(GPU.isHeadlessGLSupported ? test : skip)('unsigned precision Input Matrix headlessgl', () => { + testUnsignedPrecisionInputMatrix('headlessgl'); +}); + +test('unsigned precision Input Matrix cpu', () => { + testUnsignedPrecisionInputMatrix('cpu'); +}); + +function testSinglePrecisionInputMatrix(mode) { + const gpu = new GPU({ mode }); + const kernel = gpu.createKernel(function(input) { + return input[this.thread.y][this.thread.x]; + }) + .setDynamicArguments(true) + .setDynamicOutput(true) + .setOutput([4,4]); + + let matrix = input([ + 1,2,3,4, + 5,6,7,8, + 9,10,11,12, + 13,14,15,16 + ], [4, 4]); + assert.deepEqual(kernel(matrix), [ + new Float32Array([1,2,3,4]), + new Float32Array([5,6,7,8]), + new Float32Array([9,10,11,12]), + new Float32Array([13,14,15,16]) + ]); + + kernel.setOutput([3,3]); + matrix = input([ + 1,2,3, + 4,5,6, + 7,8,9 + ], [3, 3]); + assert.deepEqual(kernel(matrix), [ + new Float32Array([1,2,3]), + new Float32Array([4,5,6]), + new Float32Array([7,8,9]) + ]); + + kernel.setOutput([2,2]); + matrix = input([ + 1,2, + 3,4 + ], [2,2]); + assert.deepEqual(kernel(matrix), [ + new Float32Array([1,2]), + new Float32Array([3,4]) + ]); + gpu.destroy(); +} + +(GPU.isSinglePrecisionSupported ? test : skip)('single precision Input Matrix auto', () => { + testSinglePrecisionInputMatrix(); +}); + +(GPU.isSinglePrecisionSupported ? test : skip)('single precision Input Matrix gpu', () => { + testSinglePrecisionInputMatrix('gpu'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGLSupported ? test : skip)('single precision Input Matrix webgl', () => { + testSinglePrecisionInputMatrix('webgl'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isWebGL2Supported ? test : skip)('single precision Input Matrix webgl2', () => { + testSinglePrecisionInputMatrix('webgl2'); +}); + +(GPU.isSinglePrecisionSupported && GPU.isHeadlessGLSupported ? test : skip)('single precision Input Matrix headlessgl', () => { + testSinglePrecisionInputMatrix('headlessgl'); +}); + +test('single precision Input Matrix cpu', () => { + testSinglePrecisionInputMatrix('cpu'); +}); diff --git a/test/features/to-string/precision/single/arguments/html-image-array.js b/test/features/to-string/precision/single/arguments/html-image-array.js index eb042da3..29197d9b 100644 --- a/test/features/to-string/precision/single/arguments/html-image-array.js +++ b/test/features/to-string/precision/single/arguments/html-image-array.js @@ -5,10 +5,10 @@ describe('feature: to-string single precision arguments HTMLImageArray'); function testArgument(mode, done) { const imageSources = [ - '', - '', - '', - '' + 'jellyfish-1.jpeg', + 'jellyfish-2.jpeg', + 'jellyfish-3.jpeg', + 'jellyfish-4.jpeg', ]; let loaded = 0; const images = imageSources.map(imageSource => { diff --git a/test/features/to-string/precision/single/arguments/html-image.js b/test/features/to-string/precision/single/arguments/html-image.js index e6801a0f..07d77c65 100644 --- a/test/features/to-string/precision/single/arguments/html-image.js +++ b/test/features/to-string/precision/single/arguments/html-image.js @@ -5,7 +5,7 @@ describe('feature: to-string single precision arguments HTMLImage'); function testArgument(mode, done) { const image = new Image(); - image.src = ''; + image.src = 'jellyfish.jpeg'; image.onload = () => { const gpu = new GPU({mode}); const originalKernel = gpu.createKernel(function (a) { diff --git a/test/features/to-string/precision/single/constants/html-image-array.js b/test/features/to-string/precision/single/constants/html-image-array.js index 58ce938f..706acd90 100644 --- a/test/features/to-string/precision/single/constants/html-image-array.js +++ b/test/features/to-string/precision/single/constants/html-image-array.js @@ -5,10 +5,10 @@ describe('feature: to-string single precision constants HTMLImageArray'); function testArgument(mode, done) { const imageSources = [ - '', - '', - '', - '' + 'jellyfish-1.jpeg', + 'jellyfish-2.jpeg', + 'jellyfish-3.jpeg', + 'jellyfish-4.jpeg', ]; let loaded = 0; const images = imageSources.map(imageSource => { diff --git a/test/features/to-string/precision/single/constants/html-image.js b/test/features/to-string/precision/single/constants/html-image.js index 5d1f9384..8c14334b 100644 --- a/test/features/to-string/precision/single/constants/html-image.js +++ b/test/features/to-string/precision/single/constants/html-image.js @@ -5,7 +5,7 @@ describe('feature: to-string single precision constants HTMLImage'); function testArgument(mode, done) { const image = new Image(); - image.src = ''; + image.src = 'jellyfish.jpeg'; image.onload = () => { const gpu = new GPU({mode}); const originalKernel = gpu.createKernel(function (a) { diff --git a/test/features/to-string/precision/unsigned/arguments/html-image-array.js b/test/features/to-string/precision/unsigned/arguments/html-image-array.js index 4d2cc398..8085a695 100644 --- a/test/features/to-string/precision/unsigned/arguments/html-image-array.js +++ b/test/features/to-string/precision/unsigned/arguments/html-image-array.js @@ -5,10 +5,10 @@ describe('feature: to-string unsigned precision arguments HTMLImageArray'); function testArgument(mode, done) { const imageSources = [ - '', - '', - '', - '' + 'jellyfish-1.jpeg', + 'jellyfish-2.jpeg', + 'jellyfish-3.jpeg', + 'jellyfish-4.jpeg', ]; let loaded = 0; const images = imageSources.map(imageSource => { diff --git a/test/features/to-string/precision/unsigned/arguments/html-image.js b/test/features/to-string/precision/unsigned/arguments/html-image.js index 8a486482..649df0da 100644 --- a/test/features/to-string/precision/unsigned/arguments/html-image.js +++ b/test/features/to-string/precision/unsigned/arguments/html-image.js @@ -5,7 +5,7 @@ describe('feature: to-string unsigned precision arguments HTMLImage'); function testArgument(mode, done) { const image = new Image(); - image.src = ''; + image.src = 'jellyfish.jpeg'; image.onload = () => { const gpu = new GPU({mode}); const originalKernel = gpu.createKernel(function (a) { diff --git a/test/features/to-string/precision/unsigned/constants/html-image-array.js b/test/features/to-string/precision/unsigned/constants/html-image-array.js index 086d9fc2..db8a7309 100644 --- a/test/features/to-string/precision/unsigned/constants/html-image-array.js +++ b/test/features/to-string/precision/unsigned/constants/html-image-array.js @@ -5,10 +5,10 @@ describe('feature: to-string unsigned precision constants HTMLImageArray'); function testArgument(mode, done) { const imageSources = [ - '', - '', - '', - '' + 'jellyfish-1.jpeg', + 'jellyfish-2.jpeg', + 'jellyfish-3.jpeg', + 'jellyfish-4.jpeg', ]; let loaded = 0; const images = imageSources.map(imageSource => { diff --git a/test/features/to-string/precision/unsigned/constants/html-image.js b/test/features/to-string/precision/unsigned/constants/html-image.js index a6f9b776..536117ef 100644 --- a/test/features/to-string/precision/unsigned/constants/html-image.js +++ b/test/features/to-string/precision/unsigned/constants/html-image.js @@ -5,7 +5,7 @@ describe('feature: to-string unsigned precision constants HTMLImage'); function testArgument(mode, done) { const image = new Image(); - image.src = ''; + image.src = 'jellyfish.jpeg'; image.onload = () => { const gpu = new GPU({mode}); const originalKernel = gpu.createKernel(function (a) { diff --git a/test/internal/backend/cpu/function-node.js b/test/internal/backend/cpu/function-node.js index 55b3a75f..5fe47d99 100644 --- a/test/internal/backend/cpu/function-node.js +++ b/test/internal/backend/cpu/function-node.js @@ -1,4 +1,5 @@ const { assert, skip, test, module: describe, only } = require('qunit'); +const sinon = require('sinon'); const { CPUFunctionNode } = require('../../../../src'); describe('internal: CPUFunctionNode'); @@ -10,12 +11,22 @@ test('should call warn when using var is used in source', () => { argumentTypes: [], name: 'kernel', }); - let called = false; - node.varWarn = () => { - called = true; - }; + node.varWarn = sinon.spy(); node.toString(); - assert.ok(called); + assert.equal(node.varWarn.callCount, 1); +}); + +test('should not call warn when using var is used in source and warnVarUsage is false', () => { + const node = new CPUFunctionNode(`function() { var v = 1; return v; }`, { + output: [1], + precision: 'unsigned', + argumentTypes: [], + name: 'kernel', + warnVarUsage: false, + }); + node.varWarn = sinon.spy(); + node.toString(); + assert.equal(node.varWarn.callCount, 0); }); test('should not call warn when using const or let is used in source', () => { @@ -25,10 +36,7 @@ test('should not call warn when using const or let is used in source', () => { argumentTypes: [], name: 'kernel', }); - let called = false; - node.varWarn = () => { - called = true; - }; + node.varWarn = node.varWarn = sinon.spy(); node.toString(); - assert.notOk(called); + assert.equal(node.varWarn.callCount, 0); }); diff --git a/test/internal/backend/web-gl/function-node/index.js b/test/internal/backend/web-gl/function-node/index.js index 6b2174bf..dc6f1e48 100644 --- a/test/internal/backend/web-gl/function-node/index.js +++ b/test/internal/backend/web-gl/function-node/index.js @@ -1,4 +1,5 @@ const { assert, skip, test, module: describe, only } = require('qunit'); +const sinon = require('sinon'); const { WebGLFunctionNode } = require('../../../../../src'); describe('internal: WebGLFunctionNode'); @@ -10,12 +11,22 @@ test('should call warn when using var is used in source', () => { argumentTypes: [], name: 'kernel', }); - let called = false; - node.varWarn = () => { - called = true; - }; + node.varWarn = sinon.spy(); node.toString(); - assert.ok(called); + assert.equal(node.varWarn.callCount, 1); +}); + +test('should not call warn when using var is used in source and warnVarUsage is false', () => { + const node = new WebGLFunctionNode(`function() { var v = 1; return v; }`, { + output: [1], + precision: 'unsigned', + argumentTypes: [], + name: 'kernel', + warnVarUsage: false, + }); + node.varWarn = sinon.spy(); + node.toString(); + assert.equal(node.varWarn.callCount, 0); }); test('should not call warn when using const or let is used in source', () => { @@ -25,10 +36,7 @@ test('should not call warn when using const or let is used in source', () => { argumentTypes: [], name: 'kernel', }); - let called = false; - node.varWarn = () => { - called = true; - }; + node.varWarn = sinon.spy(); node.toString(); - assert.notOk(called); + assert.equal(node.varWarn.callCount, 0); }); diff --git a/test/issues/487-dynamic-arguments.js b/test/issues/487-dynamic-arguments.js new file mode 100644 index 00000000..3c68bb72 --- /dev/null +++ b/test/issues/487-dynamic-arguments.js @@ -0,0 +1,59 @@ +const { assert, skip, test, module: describe } = require('qunit'); +const { GPU } = require('../../src'); + +describe('issue #487 - pipeline dynamic arguments'); + +function testPipelineDynamicArguments(mode) { + const gpu = new GPU({mode: mode}); + + const kernel = gpu.createKernel(function (w) { + return this.thread.x + this.thread.y * w; + }) + .setPipeline(true) + .setDynamicOutput(true); + + const sumRow = gpu.createKernel(function (texture, w) { + let sum = 0; + for (let i = 0; i < w; i++) + sum = sum + texture[this.thread.x][i]; + return sum; + }) + .setDynamicArguments(true) + .setDynamicOutput(true); + + function doAThing(w, h) { + kernel.setOutput([w, h]); + let intermediate = kernel(w); + const array = intermediate.toArray(); + assert.equal(array.length, h); + assert.equal(array[0].length, w); + sumRow.setOutput([h]); + const result = sumRow(intermediate, w); + assert.equal(result.length, h); + assert.equal(result[0].length, undefined); + } + + doAThing(10, 5); + doAThing(3, 2); + gpu.destroy(); +} + +test('(GPU only) auto', () => { + testPipelineDynamicArguments(); +}); + +test('(GPU only) gpu', () => { + testPipelineDynamicArguments('gpu'); +}); + +(GPU.isWebGLSupported ? test : skip)('(GPU only) webgl', () => { + testPipelineDynamicArguments('webgl'); +}); + +(GPU.isWebGL2Supported ? test : skip)('(GPU only) webgl2', () => { + testPipelineDynamicArguments('webgl2'); +}); + +(GPU.isHeadlessGLSupported ? test : skip)('(GPU only) headlessgl', () => { + testPipelineDynamicArguments('headlessgl'); +}); diff --git a/test/jellyfish-1.jpeg b/test/jellyfish-1.jpeg new file mode 100644 index 0000000000000000000000000000000000000000..3797dc2b0f1894e09a2c6ea829c9eab0f163cb03 GIT binary patch literal 3004 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 G$UgvaY2zmV literal 0 HcmV?d00001 diff --git a/test/jellyfish-2.jpeg b/test/jellyfish-2.jpeg new file mode 100644 index 0000000000000000000000000000000000000000..ca9063d39b1e4714ca78f9d2bb01ba5dda649f71 GIT binary patch literal 1857 zcmbW!X;hPE76x1hr zfX_VL>Fxjo3ILFe0@j}bt^i6+4XK7eA(2Ql8l|p@)zZ|^(A3r5vKhNg&v5%TJp%*0 zvBeHNahH*Sftf97mnDTpqZ#h7ceJBASXj}hABRBDXtbt=rjC}D4wYa)p#I-kZvrqV zAOPUPAVdI)fxs}3^;W=eV^0L+Q-FU30)@d5YDg4XU1Ou6?lS-ifx)0~7yqgJm{W&%&%GhICR)QBsA<;cm$gh8+Yn-d_rQk#54>~%#dgXonih;qQ;g_SzvGIvlld9im-_6Z0 z{PF(7;?hSK1c3d;+PMFM{g(@~;ex{9FgWs~3j&SbI4}$xL7=E<@AE|lpWH&EoZ*fOsdepZX(_Wg)ex!Yp{co_de@pgn*neEZfF=yG@p&)|-~=p$syte* z8kv3nO&Zvn8l5!%pg4&)o!A=6KunZlCzILgat0Ji5;DixLO%vniapOPZ;-3^i{RlE zCQb&u7$!7M;LyW$?d5BLfpW_JZEOg6p>RR;iW@k~FtwJ{Dt$^_tb@3urtZ41+Nr$) zEIFoBcjUE^SZ9@^e-F!M<9}6=vo2zO0}EX8cbnvn?=!>)8bs4O-^f=G$0S`>1IMzm z!_7ovnRpl^s}DCS)L3pWm%+OdSpF8H~A>qf;xjmKy zVjhmoy0#4ZULv z6rwD1X$*64`U2f}^r4>H@(Azj3p1X6KF`T+p@Lf1I{Chxf-RV9Zc;9bt5;J9K|Q|N zyyDnV$+G?Ak)A*<8)uJg*@+K767967F=SUfm}Hk?X%jQGFtt*a-xs#l)*y>%dO%_vtUA@)u|f&F z+!)@uJ7e3ts!J%GWKFLSn(>sL5t<8;fM2`XPOk#=8E0kltfi|akZ z4pj#99xl{W3_jC2;hYei$kJ6h-m`G?9T4BT*H^BL)B54zxgkq5^2zw-#WpXo_Z8ds zmu;BTnFH?!CRVZ+rw;j-R!f&7f@O*EH`iwNq!e(taaD&O&MX#u#q-_Y_~tlmA##3C z0g9|eWerMJX5@fj0q&-(aD?GS?{6AU zF#cY5&2_fY;Y$;8P536Nc0QXXr;+VxBdTbihCuxvZh@7x4c-2_y9 zDR}br%IT=!qcJ|Rb4yH#wLfe4dC}f_t-9xIh&JU=3vzPW*&wUzamov8&Q{_(%S}j!(`!y|ajCy{ryOg& z`jgJ}sy*~6Os(wS~ literal 0 HcmV?d00001 diff --git a/test/jellyfish-3.jpeg b/test/jellyfish-3.jpeg new file mode 100644 index 0000000000000000000000000000000000000000..b8683bb5c92baf00f1d321cd3a3a98b652fb23d5 GIT binary patch literal 1700 zcmbV~dr*>D7{}nchFT(~R)!^(+G;99>Kgl5JG(Rc*X};&HfP>v&N;vHzNS`_7MQ|f z1~CB;0RSM}0Mkpr55S_)7_=D{gTdf%*hv<)mKNsb77mb&mF>*wM8eGJc)TOY&Bf7q z?kqf>vdDGrJP!tgL3Ht2>Ph!@o6n$690KBSI16)&X_l7L=uUVi`hS+G2C&8g5x^lN z=nNpNL8LWkQUXNyoo3)S0e=caAW>#$3>G)Z95&pX0w6#n5`jXRnW0dyTMWkml(iY; z(IjrDsIwL`)Ll-$+KN3u2jZ?g^N7B7W?_H zWCjEVu{hk&u=N|ld0V4nw*9a@mcKiG&)$T@BvIOd^o-1dS%-4-@{birju#dcpDQ_E zD!U+8T)Teb=B?Xz%J1HPpnUkK_VJUtCUvu>pfc%aH&)>oR!DS6|Ay6nJ3Nyh4A`;+%v__dZd7vSmb(p9)8)te7*48)YY}q~B ztob3I?6wG>Pnt?%bdm=r(7uuVJFwLMMfMli-(2m01rmglhqMNk1NwAPqNwB=HdYn>eI)2lYG0zdR{7S^={M))1S`h@zV(OT75Y)+%hCXk6mD~GY#lL&w9fm(Vv*u0_ z(-oNqmG3%SozeJX#VZr=EBB^2s%xuEdFG&>L+j}ZK#Oc8cp2?%jRHnkmrAV!Y51VB zDf${>^YHj$*QWfC{>H6VDH2k&zN(sL0tgbu!=4kXGJ;~pKc;WUC&p7~X*~^0*DM8p zhW*nb55h9G=-o|D!j4Oi(mzv6o8CKuJ2zFqPzO|~<>`q<+F zUQTqas1f2QODpYzN6aO^w|lpr>n8FN3OOI|>jvl&aR{V7Ak~*6bTxXdA$ZjqhF@!684z>_?9|Ar;^3`whh~#eg9(g#jdM|>RR+m#9QXI4P9aXl;<8fc62DQ zkH?Xd3C!R2RaR>WEfIxd_8j~7j9;1_lbDJmV}tlVal+p5PO#-!|@O1p6X71 z{;^j0A+MtZO4fknK>TR4=VYBhlpVxHfIUf zaZhQ1WQDm)C46&IO&7;4DZiK{@8+a6580_Ug}dJCJ)V1#WV|4t>Kf@%lERmjD3Tzq zL|UyGrk=@_s4jFl2)qoktfjr#c^xN7a)CoqwA^0`*NH5PkH-j2z=>7e$r^XML?!QX z*sVIJQb@O-hyNFRI5(pN(y(KNJ38V5j+8zZz%{2?vD(w7kFDLVpnodJ4;tX65qWvD zuL~HVZjn&3S}8RI+w`qPjW!Pz$qSTHy$PVpQ^m10r+E^n`EXR@WzV2cpJg|+s{^Q1 xT^Tu$WK{fIByeewWnU~fP^BMOs6dEgg<^gZyFUU789ByrX_3I)v46j*?GN2C_2~cr literal 0 HcmV?d00001 diff --git a/test/jellyfish-4.jpeg b/test/jellyfish-4.jpeg new file mode 100644 index 0000000000000000000000000000000000000000..65ceb5bea7e61c2b8599830ee9081d801979dc8f GIT binary patch literal 1688 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