clSPARSE  v0.10.0.0
a software library containing Sparse functions written in OpenCL
sample-cg.cpp
/* ************************************************************************
* Copyright 2015 Vratis, Ltd.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* ************************************************************************ */
#include <iostream>
#include <vector>
#if defined(__APPLE__) || defined(__MACOSX)
#include <OpenCL/cl.hpp>
#else
#include <CL/cl.hpp>
#endif
#include "clSPARSE.h"
#include "clSPARSE-error.h"
int main (int argc, char* argv[])
{
//parse command line
std::string matrix_path;
if (argc < 2)
{
std::cout << "Not enough parameters. "
<< "Please specify path to matrix in mtx format as parameter"
<< std::endl;
return -1;
}
else
{
matrix_path = std::string(argv[1]);
}
std::cout << "Executing sample clSPARSE CG Solver (A*x = b) C++" << std::endl;
std::cout << "Matrix will be read from: " << matrix_path << std::endl;
// Init OpenCL environment;
cl_int cl_status;
// Get OpenCL platforms
std::vector<cl::Platform> platforms;
cl_status = cl::Platform::get(&platforms);
if (cl_status != CL_SUCCESS)
{
std::cout << "Problem with getting OpenCL platforms"
<< " [" << cl_status << "]" << std::endl;
return -2;
}
int platform_id = 0;
for (const auto& p : platforms)
{
std::cout << "Platform ID " << platform_id++ << " : "
<< p.getInfo<CL_PLATFORM_NAME>() << std::endl;
}
// Using first platform
platform_id = 0;
cl::Platform platform = platforms[platform_id];
// Get device from platform
std::vector<cl::Device> devices;
cl_status = platform.getDevices(CL_DEVICE_TYPE_GPU, &devices);
if (cl_status != CL_SUCCESS)
{
std::cout << "Problem with getting devices from platform"
<< " [" << platform_id << "] " << platform.getInfo<CL_PLATFORM_NAME>()
<< " error: [" << cl_status << "]" << std::endl;
}
std::cout << std::endl
<< "Getting devices from platform " << platform_id << std::endl;
cl_int device_id = 0;
for (const auto& device : devices)
{
std::cout << "Device ID " << device_id++ << " : "
<< device.getInfo<CL_DEVICE_NAME>() << std::endl;
}
// Using first device;
device_id = 0;
cl::Device device = devices[device_id];
// Create OpenCL context;
cl::Context context (device);
// Create OpenCL queue;
cl::CommandQueue queue(context, device);
//we will allocate it after matrix will be loaded;
if (status != clsparseSuccess)
{
std::cout << "Problem with executing clsparseSetup()" << std::endl;
return -3;
}
// Create clSPARSE control object it require queue for kernel execution
clsparseCreateResult createResult = clsparseCreateControl( queue( ) );
CLSPARSE_V( createResult.status, "Failed to create clsparse control" );
// Read matrix from file. Calculates the rowBlocks structures as well.
clsparseIdx_t nnz, row, col;
// read MM header to get the size of the matrix;
clsparseStatus fileError
= clsparseHeaderfromFile( &nnz, &row, &col, matrix_path.c_str( ) );
if( fileError != clsparseSuccess )
{
std::cout << "Could not read matrix market header from disk" << std::endl;
return -5;
}
A.num_nonzeros = nnz;
A.num_rows = row;
A.num_cols = col;
// Allocate memory for CSR matrix
A.values = ::clCreateBuffer( context(), CL_MEM_READ_ONLY,
A.num_nonzeros * sizeof( float ), NULL, &cl_status );
A.col_indices = ::clCreateBuffer( context(), CL_MEM_READ_ONLY,
A.num_nonzeros * sizeof( clsparseIdx_t ), NULL, &cl_status );
A.row_pointer = ::clCreateBuffer( context(), CL_MEM_READ_ONLY,
( A.num_rows + 1 ) * sizeof( clsparseIdx_t ), NULL, &cl_status );
// Read matrix market file with explicit zero values included.
fileError = clsparseSCsrMatrixfromFile( &A, matrix_path.c_str( ), createResult.control, true );
// This function allocates memory for rowBlocks structure. If not called
// the structure will not be calculated and clSPARSE will run the vectorized
// version of SpMV instead of adaptive;
clsparseCsrMetaCreate( &A, createResult.control );
if (fileError != clsparseSuccess)
{
std::cout << "Problem with reading matrix from " << matrix_path
<< " Error: " << status << std::endl;
return -6;
}
float one = 1.0f;
float zero = 0.0f;
// Allocate memory for vector of unknowns;
// We will fill it with zeros as a initial guess
x.values = clCreateBuffer(context(), CL_MEM_READ_ONLY, x.num_values * sizeof(float),
NULL, &cl_status);
cl_status = clEnqueueFillBuffer(queue(), x.values, &zero, sizeof(float),
0, x.num_values * sizeof(float), 0, nullptr, nullptr);
// Allocate memory for rhs vector
b.values = clCreateBuffer(context(), CL_MEM_READ_WRITE, b.num_values * sizeof(float),
NULL, &cl_status);
// Fill it with ones.
cl_status = clEnqueueFillBuffer(queue(), b.values, &one, sizeof(float),
0, b.num_values * sizeof(float), 0, nullptr, nullptr);
// Create solver control object. It keeps the informations
// about the used preconditioner, desired relative and absolute tolerances
// and maximal number of iterations to be performed;
/* TODO:: missing getters of properties for solver control */
// We will use:
// preconditioner: diagonal
// relative tolerance: 1e-2
// absolute tolerance: 1e-5
// max iters: 1000
clsparseCreateSolverControl( DIAGONAL, 1000, 1e-2, 1e-5 );
CLSPARSE_V( solverResult.status, "Failed to create clsparse solver control" );
// We can set different print modes of the solver status:
// QUIET - print no messages (default)
// NORMAL - print summary
// VERBOSE - per iteration status;
clsparseSolverPrintMode( solverResult.control, VERBOSE);
/* TODO: provide various solver statuses for different scenarios
* Solver reached max number of iterations is not a failure.
*/
status = clsparseScsrcg(&x, &A, &b, solverResult.control, createResult.control );
//release solver control structure after finishing execution;
status = clsparseReleaseControl( createResult.control );
if (status != clsparseSuccess)
{
std::cout << "Problem with releasing control object."
<< " Error: " << status << std::endl;
}
status = clsparseTeardown();
if (status != clsparseSuccess)
{
std::cout << "Problem with closing clSPARSE library."
<< " Error: " << status << std::endl;
}
//release mem;
clReleaseMemObject ( A.values );
clReleaseMemObject ( A.col_indices );
clReleaseMemObject ( A.row_pointer );
clReleaseMemObject ( x.values );
clReleaseMemObject ( b.values );
std::cout << "Program completed successfully." << std::endl;
return 0;
}