mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 01:19:19 +00:00 
			
		
		
		
	First: Added a DESTDIR option for package building. Second: Change the OpenCL setup. On both linux and windows use CLRun. I completely removed the option here. If CLRun works on MacOSX this should be done there as well, and this change implemented in the scons build also. Then we could remove the HAVE_OPENCL and the new USE_CLRUN definitions. Then we will finally have the dynamic detection of opencl set up cross platform. On a side note, it doesn't seem that the program loaded from TextureDecoder.cl compiles or runs. git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@6366 8ced0084-cf51-0410-be5f-012b33b47a6e
		
			
				
	
	
		
			268 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright (C) 2003 Dolphin Project.
 | |
| 
 | |
| // This program is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU General Public License as published by
 | |
| // the Free Software Foundation, version 2.0.
 | |
| 
 | |
| // This program is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| // GNU General Public License 2.0 for more details.
 | |
| 
 | |
| // A copy of the GPL 2.0 should have been included with the program.
 | |
| // If not, see http://www.gnu.org/licenses/
 | |
| 
 | |
| // Official SVN repository and contact information can be found at
 | |
| // http://code.google.com/p/dolphin-emu/
 | |
| 
 | |
| // TODO: Make a more centralized version of this (for now every plugin that will use it will create its own context, which is weird). An object maybe?
 | |
| 
 | |
| #include "OpenCL.h"
 | |
| #include "Common.h"
 | |
| #include "Timer.h"
 | |
| 
 | |
| #if defined(_WIN32) || defined(USE_CLRUN)
 | |
| #include "clrun.h"
 | |
| #endif
 | |
| 
 | |
| namespace OpenCL
 | |
| {
 | |
| 
 | |
| #if defined(HAVE_OPENCL) && HAVE_OPENCL
 | |
| cl_device_id device_id = NULL;
 | |
| cl_context g_context = NULL;
 | |
| cl_command_queue g_cmdq = NULL;
 | |
| #endif
 | |
| 
 | |
| bool g_bInitialized = false;
 | |
| 
 | |
| bool Initialize()
 | |
| {
 | |
| 	if(g_bInitialized)
 | |
| 		return true;
 | |
| 
 | |
| #if defined(HAVE_OPENCL) && HAVE_OPENCL
 | |
| 	if(g_context)
 | |
| 		return false;
 | |
| 	int err;			// error code returned from api calls
 | |
| 
 | |
| #if defined(_WIN32) || defined(USE_CLRUN)
 | |
| 	clrInit();
 | |
| 	if(!clrHasOpenCL())
 | |
| 		return false;
 | |
| #endif
 | |
| 	// If OpenCL is weakly linked and not found, its symbols will be NULL
 | |
| 	if (clGetPlatformIDs == NULL)
 | |
| 		return false;
 | |
| 
 | |
| 	// Connect to a compute device
 | |
| 	cl_uint numPlatforms;
 | |
| 	cl_platform_id platform = NULL;
 | |
| 	err = clGetPlatformIDs(0, NULL, &numPlatforms);
 | |
| 
 | |
| 	if (err != CL_SUCCESS)
 | |
| 	{
 | |
| 		HandleCLError(err, "clGetPlatformIDs failed.");
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (0 < numPlatforms)
 | |
| 	{
 | |
| 		cl_platform_id* platforms = new cl_platform_id[numPlatforms];
 | |
| 		err = clGetPlatformIDs(numPlatforms, platforms, NULL);
 | |
| 
 | |
| 		if (err != CL_SUCCESS)
 | |
| 		{
 | |
| 			HandleCLError(err, "clGetPlatformIDs failed.");
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		char pbuf[100];
 | |
| 		err = clGetPlatformInfo(platforms[0], CL_PLATFORM_VENDOR, sizeof(pbuf), pbuf, NULL);
 | |
| 
 | |
| 		if (err != CL_SUCCESS)
 | |
| 		{
 | |
| 			HandleCLError(err, "clGetPlatformInfo failed.");
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		platform = platforms[0];
 | |
| 		delete[] platforms;
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		PanicAlert("No OpenCL platform found.");
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	cl_context_properties cps[3] = {CL_CONTEXT_PLATFORM, (cl_context_properties)platform, 0};
 | |
| 
 | |
| 	cl_context_properties* cprops = (NULL == platform) ? NULL : cps;
 | |
| 
 | |
| 	err = clGetDeviceIDs(platform, CL_DEVICE_TYPE_DEFAULT, 1, &device_id, NULL);
 | |
| 	if (err != CL_SUCCESS)
 | |
| 	{
 | |
| 		HandleCLError(err, "Failed to create a device group!");
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	// Create a compute context
 | |
| 	g_context = clCreateContext(cprops, 1, &device_id, NULL, NULL, &err);
 | |
| 	if (!g_context)
 | |
| 	{
 | |
| 		HandleCLError(err, "Failed to create a compute context!");
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	// Create a command commands
 | |
| 	g_cmdq = clCreateCommandQueue(g_context, device_id, 0, &err);
 | |
| 	if (!g_cmdq)
 | |
| 	{
 | |
| 		HandleCLError(err, "Failed to create a command commands!");
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	g_bInitialized = true;
 | |
| 	return true;
 | |
| #else
 | |
| 	return false;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| #if defined(HAVE_OPENCL) && HAVE_OPENCL
 | |
| cl_context GetContext()
 | |
| {
 | |
| 	return g_context;
 | |
| }
 | |
| 
 | |
| cl_command_queue GetCommandQueue()
 | |
| {
 | |
| 	return g_cmdq;
 | |
| }
 | |
| 
 | |
| cl_program CompileProgram(const char *Kernel)
 | |
| {
 | |
| 	u32 compileStart = Common::Timer::GetTimeMs();
 | |
| 	int err;
 | |
| 	cl_program program;
 | |
| 	program = clCreateProgramWithSource(OpenCL::g_context, 1, (const char **) & Kernel, NULL, &err);
 | |
| 	if (!program)
 | |
| 	{
 | |
| 		HandleCLError(err, "Error: Failed to create compute program!");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	// Build the program executable
 | |
| 	err = clBuildProgram(program , 0, NULL, NULL, NULL, NULL);
 | |
| 	if(err != CL_SUCCESS) {
 | |
| 		char *errors[16384] = {0};
 | |
| 		err = clGetProgramBuildInfo(program, OpenCL::device_id, CL_PROGRAM_BUILD_LOG, sizeof(errors),
 | |
| 									errors, NULL);
 | |
| 		ERROR_LOG(COMMON, "Error log:\n%s\n", errors);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	NOTICE_LOG(COMMON, "OpenCL CompileProgram took %.3f seconds", (float)(Common::Timer::GetTimeMs() - compileStart) / 1000.0);
 | |
| 	return program;
 | |
| }
 | |
| 
 | |
| cl_kernel CompileKernel(cl_program program, const char *Function)
 | |
| {
 | |
| 	u32 compileStart = Common::Timer::GetTimeMs();
 | |
| 	int err;
 | |
| 
 | |
| 	// Create the compute kernel in the program we wish to run
 | |
| 	cl_kernel kernel = clCreateKernel(program, Function, &err);
 | |
| 	if (!kernel || err != CL_SUCCESS)
 | |
| 	{
 | |
| 		char buffer[1024];
 | |
| 		sprintf(buffer, "Failed to create compute kernel '%s' !", Function);
 | |
| 		HandleCLError(err, buffer);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	NOTICE_LOG(COMMON, "OpenCL CompileKernel took %.3f seconds", (float)(Common::Timer::GetTimeMs() - compileStart) / 1000.0);
 | |
| 	return kernel;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void Destroy()
 | |
| {
 | |
| #if defined(HAVE_OPENCL) && HAVE_OPENCL
 | |
| 	if (g_cmdq)
 | |
| 	{
 | |
| 		clReleaseCommandQueue(g_cmdq);
 | |
| 		g_cmdq = NULL;
 | |
| 	}
 | |
| 	if (g_context)
 | |
| 	{
 | |
| 		clReleaseContext(g_context);
 | |
| 		g_context = NULL;
 | |
| 	}		
 | |
| 	g_bInitialized = false;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void HandleCLError(cl_int error, const char* str)
 | |
| {
 | |
| #if defined(HAVE_OPENCL) && HAVE_OPENCL
 | |
| 
 | |
| 	const char* name;
 | |
| 	switch(error)
 | |
| 	{
 | |
| #define CL_ERROR(x) case (x): name = #x; break
 | |
| 		CL_ERROR(CL_SUCCESS);
 | |
| 		CL_ERROR(CL_DEVICE_NOT_FOUND);
 | |
| 		CL_ERROR(CL_DEVICE_NOT_AVAILABLE);
 | |
| 		CL_ERROR(CL_COMPILER_NOT_AVAILABLE);
 | |
| 		CL_ERROR(CL_MEM_OBJECT_ALLOCATION_FAILURE);
 | |
| 		CL_ERROR(CL_OUT_OF_RESOURCES);
 | |
| 		CL_ERROR(CL_OUT_OF_HOST_MEMORY);
 | |
| 		CL_ERROR(CL_PROFILING_INFO_NOT_AVAILABLE);
 | |
| 		CL_ERROR(CL_MEM_COPY_OVERLAP);
 | |
| 		CL_ERROR(CL_IMAGE_FORMAT_MISMATCH);
 | |
| 		CL_ERROR(CL_IMAGE_FORMAT_NOT_SUPPORTED);
 | |
| 		CL_ERROR(CL_BUILD_PROGRAM_FAILURE);
 | |
| 		CL_ERROR(CL_MAP_FAILURE);
 | |
| 		CL_ERROR(CL_INVALID_VALUE);
 | |
| 		CL_ERROR(CL_INVALID_DEVICE_TYPE);
 | |
| 		CL_ERROR(CL_INVALID_PLATFORM);
 | |
| 		CL_ERROR(CL_INVALID_DEVICE);
 | |
| 		CL_ERROR(CL_INVALID_CONTEXT);
 | |
| 		CL_ERROR(CL_INVALID_QUEUE_PROPERTIES);
 | |
| 		CL_ERROR(CL_INVALID_COMMAND_QUEUE);
 | |
| 		CL_ERROR(CL_INVALID_HOST_PTR);
 | |
| 		CL_ERROR(CL_INVALID_MEM_OBJECT);
 | |
| 		CL_ERROR(CL_INVALID_IMAGE_FORMAT_DESCRIPTOR);
 | |
| 		CL_ERROR(CL_INVALID_IMAGE_SIZE);
 | |
| 		CL_ERROR(CL_INVALID_SAMPLER);
 | |
| 		CL_ERROR(CL_INVALID_BINARY);
 | |
| 		CL_ERROR(CL_INVALID_BUILD_OPTIONS);
 | |
| 		CL_ERROR(CL_INVALID_PROGRAM);
 | |
| 		CL_ERROR(CL_INVALID_PROGRAM_EXECUTABLE);
 | |
| 		CL_ERROR(CL_INVALID_KERNEL_NAME);
 | |
| 		CL_ERROR(CL_INVALID_KERNEL_DEFINITION);
 | |
| 		CL_ERROR(CL_INVALID_KERNEL);
 | |
| 		CL_ERROR(CL_INVALID_ARG_INDEX);
 | |
| 		CL_ERROR(CL_INVALID_ARG_VALUE);
 | |
| 		CL_ERROR(CL_INVALID_ARG_SIZE);
 | |
| 		CL_ERROR(CL_INVALID_KERNEL_ARGS);
 | |
| 		CL_ERROR(CL_INVALID_WORK_DIMENSION);
 | |
| 		CL_ERROR(CL_INVALID_WORK_GROUP_SIZE);
 | |
| 		CL_ERROR(CL_INVALID_WORK_ITEM_SIZE);
 | |
| 		CL_ERROR(CL_INVALID_GLOBAL_OFFSET);
 | |
| 		CL_ERROR(CL_INVALID_EVENT_WAIT_LIST);
 | |
| 		CL_ERROR(CL_INVALID_EVENT);
 | |
| 		CL_ERROR(CL_INVALID_OPERATION);
 | |
| 		CL_ERROR(CL_INVALID_GL_OBJECT);
 | |
| 		CL_ERROR(CL_INVALID_BUFFER_SIZE);
 | |
| 		CL_ERROR(CL_INVALID_MIP_LEVEL);
 | |
| #undef CL_ERROR
 | |
| 	default:
 | |
| 		name = "Unknown error code";
 | |
| 	}
 | |
| 	if(!str)
 | |
| 		str = "";
 | |
| 	ERROR_LOG(COMMON, "OpenCL error: %s %s (%d)", str, name, error);
 | |
| #endif
 | |
| 	}
 | |
| }
 |