webcl-nodep
Version:
A fork of node-webcl without dependencies other than OpenCL
152 lines (126 loc) • 5.26 kB
JavaScript
// Copyright (c) 2011-2012, Motorola Mobility, Inc.
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
// * Neither the name of the Motorola Mobility, Inc. nor the names of its
// contributors may be used to endorse or promote products derived from this
// software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
// DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
// ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
// THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
var nodejs = (typeof window === 'undefined');
if(nodejs) {
WebCL = require('../webcl');
clu = require('../lib/clUtils');
log=console.log;
}
//First check if the WebCL extension is installed at all
if (WebCL == undefined) {
alert("Unfortunately your system does not support WebCL. " +
"Make sure that you have the WebCL extension installed.");
return;
}
VectorAdd();
function VectorAdd() {
BUFFER_SIZE=10;
var A=new Uint32Array(BUFFER_SIZE);
var B=new Uint32Array(BUFFER_SIZE);
for (var i = 0; i < BUFFER_SIZE; i++) {
A[i] = i;
B[i] = i * 2;
}
//Pick platform
var platformList=WebCL.getPlatforms();
platform=platformList[0];
log('using platform: '+platform.getInfo(WebCL.PLATFORM_NAME));
//Query the set of devices on this platform
devices = platform.getDevices(WebCL.DEVICE_TYPE_DEFAULT);
log('using device: '+devices[0].getInfo(WebCL.DEVICE_NAME));
// create GPU context for this platform
context=WebCL.createContext({
deviceType: WebCL.DEVICE_TYPE_DEFAULT,
platform: platform
});
kernelSourceCode = [
"__kernel void vadd(__global int *a, __global int *b, __global int *c, uint iNumElements) ",
"{ ",
" size_t i = get_global_id(0); ",
" if(i >= iNumElements) return; ",
" c[i] = a[i] + b[i]; ",
"} "
].join("\n");
//Create and program from source
program=context.createProgram(kernelSourceCode);
//Build program
program.build(devices);
size=BUFFER_SIZE*Uint32Array.BYTES_PER_ELEMENT; // size in bytes
// Create buffer for A and B and copy host contents
aBuffer = context.createBuffer(WebCL.MEM_READ_ONLY, size);
bBuffer = context.createBuffer(WebCL.MEM_READ_ONLY, size);
// Create buffer for C to read results
cBuffer = context.createBuffer(WebCL.MEM_WRITE_ONLY, size);
// Create kernel object
try {
kernel= program.createKernel("vadd");
}
catch(err) {
console.log(program.getBuildInfo(devices[0],WebCL.PROGRAM_BUILD_LOG));
}
// Set kernel args
kernel.setArg(0, aBuffer);
kernel.setArg(1, bBuffer);
kernel.setArg(2, cBuffer);
kernel.setArg(3, BUFFER_SIZE, WebCL.type.UINT);
// Create command queue
queue=context.createCommandQueue(devices[0], 0);
// Execute the OpenCL kernel on the list
var localWS = [5]; // process one list at a time
var globalWS = [clu.roundUp(localWS, BUFFER_SIZE)]; // process entire list
log("Global work item size: " + globalWS);
log("Local work item size: " + localWS);
// Do the work
queue.enqueueWriteBuffer (aBuffer, false, 0, A.length*Uint32Array.BYTES_PER_ELEMENT, A);
queue.enqueueWriteBuffer (bBuffer, false, 0, B.length*Uint32Array.BYTES_PER_ELEMENT, B);
// Execute (enqueue) kernel
log("using enqueueNDRangeKernel");
queue.enqueueNDRangeKernel(kernel,
null,
globalWS,
localWS);
// get results and block while getting them
var C=new Uint32Array(BUFFER_SIZE);
queue.enqueueReadBuffer (cBuffer, true, 0, C.length*Uint32Array.BYTES_PER_ELEMENT, C);
// print results
printResults(A,B,C);
}
function printResults(A,B,C) {
// Print input vectors and result vector
var output = "\nA = ";
for (var i = 0; i < BUFFER_SIZE; i++) {
output += A[i] + ", ";
}
output += "\nB = ";
for (var i = 0; i < BUFFER_SIZE; i++) {
output += B[i] + ", ";
}
output += "\nC = ";
for (var i = 0; i < BUFFER_SIZE; i++) {
output += C[i] + ", ";
}
log(output);
}