gpu.js/test/features/add-custom-native-function.js
2020-01-18 08:42:09 -05:00

516 lines
11 KiB
JavaScript

const { assert, skip, test, module: describe, only } = require('qunit');
const { GPU } = require('../../src');
describe('features: add native');
const glslDivide = `float divide(float a, float b) {
return a / b;
}`;
const jsDivide = `function divide(a, b) {
return a / b;
}`;
function nativeDivide(mode, fn) {
const gpu = new GPU({ mode });
gpu.addNativeFunction('divide', fn, { returnType: 'Number' });
const f = gpu.createKernel(function(a, b) {
return divide(a[this.thread.x], b[this.thread.x]);
}, {
output : [6]
});
assert.ok(f !== null, 'function generated test');
const a = [1, 4, 3, 5, 6, 3];
const b = [4, 2, 6, 1, 2, 3];
const res = f(a,b);
const exp = [0.25, 2, 0.5, 5, 3, 1];
for(let i = 0; i < exp.length; ++i) {
assert.equal(res[i], exp[i], 'Result arr idx: '+i);
}
gpu.destroy();
}
test('nativeDivide auto', () => {
nativeDivide(null, glslDivide);
});
test('nativeDivide gpu', () => {
nativeDivide('gpu', glslDivide);
});
(GPU.isWebGLSupported ? test : skip)('nativeDivide webgl', () => {
nativeDivide('webgl', glslDivide);
});
(GPU.isWebGL2Supported ? test : skip)('nativeDivide webgl2', () => {
nativeDivide('webgl2', glslDivide);
});
(GPU.isHeadlessGLSupported ? test : skip)('nativeDivide headlessgl', () => {
nativeDivide('headlessgl', glslDivide);
});
test('nativeDivide cpu', () => {
nativeDivide('cpu', jsDivide);
});
describe('features: instantiate native and override');
function divideOverride(mode) {
const gpu = new GPU({
mode,
functions: [divide],
nativeFunctions: [{
name: 'divide',
// deliberately add, rather than divide, to ensure native functions are treated as more important than regular ones
source: `float divide(float a, float b) {
return a + b;
}`
}]
});
function divide(a,b) {
return a / b;
}
const kernel = gpu.createKernel(function(a, b) {
return divide(a[this.thread.x], b[this.thread.x]);
}, {
output : [6]
});
const a = [1, 4, 3, 5, 6, 3];
const b = [4, 2, 6, 1, 2, 3];
const res = kernel(a,b);
const exp = [5, 6, 9, 6, 8, 6];
assert.deepEqual(Array.from(res), exp);
gpu.destroy();
}
test('divideOverride (GPU only) auto', () => {
divideOverride(null);
});
test('divideOverride (GPU only) gpu', () => {
divideOverride('gpu');
});
(GPU.isWebGLSupported ? test : skip)('divideOverride (GPU only) webgl', () => {
divideOverride('webgl');
});
(GPU.isWebGL2Supported ? test : skip)('divideOverride (GPU only) webgl2', () => {
divideOverride('webgl2');
});
(GPU.isHeadlessGLSupported ? test : skip)('divideOverride (GPU only) headlessgl', () => {
divideOverride('headlessgl');
});
describe('features: argument casting');
function argumentCasting(mode) {
const gpu = new GPU({
mode,
functions: [divide],
nativeFunctions: [{
// deliberately add, rather than divide, to ensure native functions are treated as more important than regular ones
name: 'divide',
source: `float divide(int a, int b) {
return float(a + b);
}`
}]
});
function divide(a,b) {
return a / b;
}
const kernel = gpu.createKernel(function(a, b) {
return divide(a[this.thread.x], b[this.thread.x]);
}, {
output : [6]
});
const a = [1, 4, 3, 5, 6, 3];
const b = [4, 2, 6, 1, 2, 3];
const res = kernel(a,b);
const exp = [5, 6, 9, 6, 8, 6];
assert.deepEqual(Array.from(res), exp);
gpu.destroy();
}
test('argumentCasting (GPU only) auto', () => {
argumentCasting(null);
});
test('argumentCasting (GPU only) gpu', () => {
argumentCasting('gpu');
});
(GPU.isWebGLSupported ? test : skip)('argumentCasting (GPU only) webgl', () => {
argumentCasting('webgl');
});
(GPU.isWebGL2Supported ? test : skip)('argumentCasting (GPU only) webgl2', () => {
argumentCasting('webgl2');
});
(GPU.isHeadlessGLSupported ? test : skip)('argumentCasting (GPU only) headlessgl', () => {
argumentCasting('headlessgl');
});
describe('features: mixed argument casting');
function mixedArgumentCasting(mode) {
const gpu = new GPU({
mode,
functions: [divide],
nativeFunctions: [{
// deliberately add, rather than divide, to ensure native functions are treated as more important than regular ones
name: 'divide',
source: `float divide(int a, float b) {
return float(a + int(b));
}`
}]
});
function divide(a,b) {
return a / b;
}
const kernel = gpu.createKernel(function(a, b) {
return divide(a[this.thread.x], b[this.thread.x]);
}, {
output : [6]
});
const a = [1, 4, 3, 5, 6, 3];
const b = [4, 2, 6, 1, 2, 3];
const res = kernel(a,b);
const exp = [5, 6, 9, 6, 8, 6];
assert.deepEqual(Array.from(res), exp);
gpu.destroy();
}
test('mixedArgumentCasting (GPU only) auto', () => {
mixedArgumentCasting(null);
});
test('mixedArgumentCasting (GPU only) gpu', () => {
mixedArgumentCasting('gpu');
});
(GPU.isWebGLSupported ? test : skip)('mixedArgumentCasting (GPU only) webgl', () => {
mixedArgumentCasting('webgl');
});
(GPU.isWebGL2Supported ? test : skip)('mixedArgumentCasting (GPU only) webgl2', () => {
mixedArgumentCasting('webgl2');
});
(GPU.isHeadlessGLSupported ? test : skip)('mixedArgumentCasting (GPU only) headlessgl', () => {
mixedArgumentCasting('headlessgl');
});
describe('features: return type casting');
function returnTypeCasting(mode) {
const gpu = new GPU({
mode,
functions: [divide],
nativeFunctions: [{
// deliberately add, rather than divide, to ensure native functions are treated as more important than regular ones
name: 'divide',
source: `int divide(float a, float b) {
return int(a + b);
}`
}]
});
function divide(a,b) {
return a / b;
}
const kernel = gpu.createKernel(function(a, b) {
return divide(a[this.thread.x], b[this.thread.x]);
}, {
output : [6]
});
const a = [1, 4, 3, 5, 6, 3];
const b = [4, 2, 6, 1, 2, 3];
const res = kernel(a,b);
const exp = [5, 6, 9, 6, 8, 6];
assert.deepEqual(Array.from(res), exp);
gpu.destroy();
}
test('returnTypeCasting (GPU only) auto', () => {
returnTypeCasting(null);
});
test('returnTypeCasting (GPU only) gpu', () => {
returnTypeCasting('gpu');
});
(GPU.isWebGLSupported ? test : skip)('returnTypeCasting (GPU only) webgl', () => {
returnTypeCasting('webgl');
});
(GPU.isWebGL2Supported ? test : skip)('returnTypeCasting (GPU only) webgl2', () => {
returnTypeCasting('webgl2');
});
(GPU.isHeadlessGLSupported ? test : skip)('returnTypeCasting (GPU only) headlessgl', () => {
returnTypeCasting('headlessgl');
});
describe('features: Adding nativeFunctions directly on kernel');
function testDirectlyOnKernelViaSettings(nativeFunctions, mode) {
const gpu = new GPU({ mode });
const kernel = gpu.createKernel(function (v) {
return native(v[this.thread.x]);
}, {
output: [1],
nativeFunctions
});
assert.equal(kernel([1])[0], 2);
gpu.destroy();
}
test('via settings auto', () => {
testDirectlyOnKernelViaSettings([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
])
});
test('via settings gpu', () => {
testDirectlyOnKernelViaSettings([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'gpu')
});
(GPU.isWebGLSupported ? test : skip)('via settings webgl', () => {
testDirectlyOnKernelViaSettings([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'webgl')
});
(GPU.isWebGL2Supported ? test : skip)('via settings webgl2', () => {
testDirectlyOnKernelViaSettings([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'webgl2')
});
(GPU.isHeadlessGLSupported ? test : skip)('via settings headlessgl', () => {
testDirectlyOnKernelViaSettings([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'headlessgl')
});
test('via settings cpu', () => {
testDirectlyOnKernelViaSettings([
{
name: 'native',
source: `function native(value) {
return 1.0 + value;
}`,
returnType: 'Float'
}
], 'cpu')
});
describe('features: Adding nativeFunctions directly on kernel');
/**
*
* @param {IGPUNativeFunction[]} nativeFunctions
* @param {GPUMode|GPUInternalMode} [mode]
*/
function testDirectlyOnKernelViaMethod(nativeFunctions, mode) {
const gpu = new GPU({ mode });
const kernel = gpu.createKernel(function (v) {
return native(v[this.thread.x]);
}, {
output: [1]
})
.setNativeFunctions(nativeFunctions);
assert.equal(kernel([1])[0], 2);
gpu.destroy();
}
test('via method auto', () => {
testDirectlyOnKernelViaMethod([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
])
});
test('via method gpu', () => {
testDirectlyOnKernelViaMethod([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'gpu')
});
(GPU.isWebGLSupported ? test : skip)('via method webgl', () => {
testDirectlyOnKernelViaMethod([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'webgl')
});
(GPU.isWebGL2Supported ? test : skip)('via method webgl2', () => {
testDirectlyOnKernelViaMethod([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'webgl2')
});
(GPU.isHeadlessGLSupported ? test : skip)('via method headlessgl', () => {
testDirectlyOnKernelViaMethod([
{
name: 'native',
source: `float native(float value) {
return 1.0 + value;
}`
}
], 'headlessgl')
});
test('via method cpu', () => {
testDirectlyOnKernelViaMethod([
{
name: 'native',
source: `function native(value) {
return 1.0 + value;
}`,
returnType: 'Float'
}
], 'cpu')
});
function testSetNativeFunctionsFromArrayOnGPU(nativeFunction, mode) {
const gpu = new GPU({ mode });
assert.equal(gpu.setNativeFunctions([nativeFunction]), gpu);
const kernel = gpu.createKernel(function() {
return custom();
}, { output: [1] });
assert.equal(kernel()[0], 1);
gpu.destroy();
}
test('auto', () => {
testSetNativeFunctionsFromArrayOnGPU({
name: 'custom',
source: `float custom() {
return 1.0;
}`
});
});
test('gpu', () => {
testSetNativeFunctionsFromArrayOnGPU({
name: 'custom',
source: `float custom() {
return 1.0;
}`
},'gpu');
});
(GPU.isWebGLSupported ? test : skip)('webgl', () => {
testSetNativeFunctionsFromArrayOnGPU({
name: 'custom',
source: `float custom() {
return 1.0;
}`
},'webgl');
});
(GPU.isWebGL2Supported ? test : skip)('webgl2', () => {
testSetNativeFunctionsFromArrayOnGPU({
name: 'custom',
source: `float custom() {
return 1.0;
}`
}, 'webgl2');
});
(GPU.isHeadlessGLSupported ? test : skip)('headlessgl', () => {
testSetNativeFunctionsFromArrayOnGPU({
name: 'custom',
source: `float custom() {
return 1.0;
}`
},'headlessgl');
});
test('cpu', () => {
testSetNativeFunctionsFromArrayOnGPU({
name: 'custom',
source: `function custom() {
return 1.0;
}`,
returnType: 'Number'
},'cpu');
});