Warning: file_get_contents(/data/phpspider/zhask/data//catemap/4/string/5.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
用于字符串连接的OpenCL内核_C_String_Algorithm_Opencl_Gpgpu - Fatal编程技术网

用于字符串连接的OpenCL内核

用于字符串连接的OpenCL内核,c,string,algorithm,opencl,gpgpu,C,String,Algorithm,Opencl,Gpgpu,我还没有找到太多关于使用GPU对字符串执行操作的文献或示例。具体来说,我有两个字符串数组,我需要将第二个数组的元素连接到第一个数组的相应元素。我不知道如何为此编写内核 C中的串联示例如下: #include <stdio.h> void concatenate_string(char*, char*, char*); int main() { char original[100], add[100], result[100]; printf("Enter source

我还没有找到太多关于使用GPU对字符串执行操作的文献或示例。具体来说,我有两个字符串数组,我需要将第二个数组的元素连接到第一个数组的相应元素。我不知道如何为此编写内核

C中的串联示例如下:

#include <stdio.h>

void concatenate_string(char*, char*, char*);

int main()
{
   char original[100], add[100], result[100];
   printf("Enter source string\n");
   scanf("%s", original);
   printf("Enter string to concatenate\n");
   scanf("%s", add);
   concatenate_string(original, add, result);

   printf("String after concatenation is \"%s\"\n", result);

   return 0;
}

void concatenate_string(char *original, char *add, char *result)
{
    while(*original)
    {
       *result = *original;
       original++;
       result++;
    }
    while(*add)
    {
       *result = *add;
       add++;
       result++;
    }
    *result = '\0';
}
#包括
空连接_字符串(char*,char*,char*);
int main()
{
字符原始[100],添加[100],结果[100];
printf(“输入源字符串\n”);
scanf(“%s”,原件);
printf(“输入要连接的字符串\n”);
scanf(“%s”,添加);
连接字符串(原始、添加、结果);
printf(“连接后的字符串是\%s\”\n,结果);
返回0;
}
无效连接字符串(字符*原始、字符*添加、字符*结果)
{
而(*原件)
{
*结果=*原件;
原创++;
结果++;
}
while(*添加)
{
*结果=*相加;
添加++;
结果++;
}
*结果='\0';
}
下面是包含内核的OpenCL主机代码。内核遵循与上面的concatenate_string函数相同的流程。程序执行成功,但没有输出

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifdef __APPLE__
#include <OpenCL/cl.h>
#else
#include <CL/cl.h>
#endif
#include <ocl_macros.h>
#include <iostream>
#include <string>
//Common defines 
#define VENDOR_NAME "AMD"
#define DEVICE_TYPE CL_DEVICE_TYPE_GPU
#define VECTOR_SIZE 1024 
using namespace std;

//OpenCL kernel which is run for every work item created.
//The below const char string is compiled by the runtime complier
//when a program object is created with clCreateProgramWithSource 
//and built with clBuildProgram.
const char *concat_kernel =
"__kernel                                       \n"
"void concat_kernel(                            \n"
"                  __global uchar *D,           \n"
"                  __global uchar *E,           \n"
"                  __global uchar *F)           \n"
"{                                              \n"
"    //Get the index of the work-item           \n"
"    int index = get_global_id(0);              \n"
"    while(D[index])                            \n"
"    {                                          \n"
"        *F[index] = *D[index];                 \n"
"        D[index]++;                            \n"
"        F[index]++;                            \n"
"    }                                          \n"
"    while(E[index])                            \n"
"    {                                          \n"
"        *F[index] = *E[index];                 \n"
"        E[index]++;                            \n"
"        F[index]++;                            \n"
"    }                                          \n"
"    *F[index] = '\0';                          \n"
"}                                              \n";

int main(void) {

    cl_int clStatus; //Keeps track of the error values returned. 

    // Get platform and device information
    cl_platform_id * platforms = NULL;

    // Set up the Platform. Take a look at the MACROs used in this file. 
    // These are defined in common/ocl_macros.h
    OCL_CREATE_PLATFORMS( platforms );

    // Get the devices list and choose the type of device you want to run on
    cl_device_id *device_list = NULL;
    OCL_CREATE_DEVICE( platforms[0], DEVICE_TYPE, device_list);

    // Create OpenCL context for devices in device_list
    cl_context context;
    cl_context_properties props[3] =
    {
        CL_CONTEXT_PLATFORM,
        (cl_context_properties)platforms[0],
        0
    };
    // An OpenCL context can be associated to multiple devices, either CPU or GPU
    // based on the value of DEVICE_TYPE defined above.
    context = clCreateContext( NULL, num_devices, device_list, NULL, NULL, &clStatus);
    LOG_OCL_ERROR(clStatus, "clCreateContext Failed..." );

    // Create a command queue for the first device in device_list
    cl_command_queue command_queue = clCreateCommandQueue(context, device_list[0], 0, &clStatus);
    LOG_OCL_ERROR(clStatus, "clCreateCommandQueue Failed..." );

    // Allocate space for vectors D, E, and F 
    string *D = (string*)malloc(sizeof(string)*VECTOR_SIZE);
    string *E = (string*)malloc(sizeof(string)*VECTOR_SIZE);
    string *F = (string*)malloc(sizeof(string)*VECTOR_SIZE);
    for(int i = 0; i < VECTOR_SIZE; i++)
    {
        D[i] = ".25_numstring";
    }
    for(int i = 0; i < VECTOR_SIZE; i++)
    {
        E[i] = "string_2";
        F[i] = "0";
    }
    // Create memory buffers on the device for each vector
    cl_mem D_clmem = clCreateBuffer(context, CL_MEM_READ_ONLY,
            VECTOR_SIZE * sizeof(string), NULL, &clStatus);
    cl_mem E_clmem = clCreateBuffer(context, CL_MEM_READ_ONLY,
            VECTOR_SIZE * sizeof(string), NULL, &clStatus);
    cl_mem F_clmem = clCreateBuffer(context, CL_MEM_WRITE_ONLY,
            VECTOR_SIZE * sizeof(string), NULL, &clStatus);

    // Copy the Buffer D and E to the device. We do a blocking write to the device buffer.
    clStatus = clEnqueueWriteBuffer(command_queue, D_clmem, CL_TRUE, 0,
            VECTOR_SIZE * sizeof(string), D, 0, NULL, NULL);
    LOG_OCL_ERROR(clStatus, "clEnqueueWriteBuffer Failed..." );
    clStatus = clEnqueueWriteBuffer(command_queue, E_clmem, CL_TRUE, 0,
            VECTOR_SIZE * sizeof(string), E, 0, NULL, NULL);
    LOG_OCL_ERROR(clStatus, "clEnqueueWriteBuffer Failed..." );

    // Create a program from the kernel source
    cl_program program = clCreateProgramWithSource(context, 1,
            (const char **)&concat_kernel, NULL, &clStatus);
    LOG_OCL_ERROR(clStatus, "clCreateProgramWithSource Failed..." );

    // Build the program
    clStatus = clBuildProgram(program, 1, device_list, NULL, NULL, NULL);
    if(clStatus != CL_SUCCESS)
        LOG_OCL_COMPILER_ERROR(program, device_list[0]);

    // Create the OpenCL kernel
    cl_kernel kernel = clCreateKernel(program, "concat_kernel", &clStatus);

    // Set the arguments of the kernel. Take a look at the kernel definition in concat_kernel 
    // variable. First parameter is a constant and the other three are buffers.
    clStatus |= clSetKernelArg(kernel, 0, sizeof(cl_mem), (void *)&D_clmem);
    clStatus |= clSetKernelArg(kernel, 1, sizeof(cl_mem), (void *)&E_clmem);
    clStatus |= clSetKernelArg(kernel, 2, sizeof(cl_mem), (void *)&F_clmem);
    LOG_OCL_ERROR(clStatus, "clSetKernelArg Failed..." );

    // Execute the OpenCL kernel on the list
    size_t global_size = VECTOR_SIZE; // Process one vector element in each work item
    size_t local_size = 64;           // Process in work groups of size 64.
    cl_event concat_event;
    clStatus = clEnqueueNDRangeKernel(command_queue, kernel, 1, NULL,
            &global_size, &local_size, 0, NULL, &concat_event);
    LOG_OCL_ERROR(clStatus, "clEnqueueNDRangeKernel Failed..." );

    // Read the memory buffer F_clmem on the device to the host allocated buffer C
    // This task is invoked only after the completion of the event concat_event
    clStatus = clEnqueueReadBuffer(command_queue, F_clmem, CL_TRUE, 0,
            VECTOR_SIZE * sizeof(string), F, 1, &concat_event, NULL);
    LOG_OCL_ERROR(clStatus, "clEnqueueReadBuffer Failed..." );

    // Clean up and wait for all the comands to complete.
    clStatus = clFinish(command_queue);

    // Display the result to the screen
    for(int i = 0; i < VECTOR_SIZE; i++)
        printf("%s + %s = %s\n", D[i].c_str(), E[i].c_str(), F[i].c_str());

    // Finally release all OpenCL objects and release the host buffers.
    clStatus = clReleaseKernel(kernel);
    clStatus = clReleaseProgram(program);
    clStatus = clReleaseMemObject(D_clmem);
    clStatus = clReleaseMemObject(E_clmem);
    clStatus = clReleaseMemObject(F_clmem);
    clStatus = clReleaseCommandQueue(command_queue);
    clStatus = clReleaseContext(context);
    free(D);
    free(E);
    free(F);
    free(platforms);
    free(device_list);

    return 0;
}
#包括
#包括
#包括
#苹果__
#包括
#否则
#包括
#恩迪夫
#包括
#包括
#包括
//共同定义
#定义供应商名称“AMD”
#定义设备类型CL\U设备类型\U GPU
#定义向量大小为1024
使用名称空间std;
//为创建的每个工作项运行的OpenCL内核。
//下面的const char字符串由运行时编译器编译
//使用clCreateProgramWithSource创建程序对象时
//并使用clBuildProgram构建。
const char*concat_内核=
“\uuu内核\n”
“void concat\u内核(\n”
“\u全局uchar*D\n”
“\u全局uchar*E\n”
“\uu全局uchar*F)\n”
“{\n”
“//获取工作项的索引\n”
“int index=get_global_id(0);\n”
“而(D[索引])\n”
“{\n”
“*F[索引]=*D[索引];\n”
“D[索引]++;\n”
“F[索引]++;\n”
“}\n”
“while(E[索引])\n”
“{\n”
“*F[索引]=*E[索引];\n”
“E[索引]++;\n”
“F[索引]++;\n”
“}\n”
“*F[索引]=”\0';\n”
“}\n”;
内部主(空){
cl_int clStatus;//跟踪返回的错误值。
//获取平台和设备信息
cl_platform_id*platforms=NULL;
//设置平台。查看此文件中使用的宏。
//这些在common/ocl_macros.h中定义
OCL_创建_平台(平台);
//获取设备列表并选择要在其上运行的设备类型
cl_设备_id*设备_列表=空;
OCL_创建_设备(平台[0],设备类型,设备列表);
//为设备列表中的设备创建OpenCL上下文
语境;
cl_上下文_属性道具[3]=
{
CL_环境_平台,
(cl_上下文_属性)平台[0],
0
};
//OpenCL上下文可以关联到多个设备,CPU或GPU
//基于上面定义的设备类型的值。
context=clCreateContext(NULL、num\u设备、设备列表、NULL、NULL和clStatus);
记录OCL错误(clStatus,“clCreateContext Failed…”);
//为设备列表中的第一个设备创建命令队列
cl_命令_队列命令_队列=clCreateCommandQueue(上下文、设备列表[0]、0和clStatus);
日志OCL错误(clStatus,“clCreateCommandQueue失败…”);
//为向量D、E和F分配空间
string*D=(string*)malloc(sizeof(string)*向量大小);
string*E=(string*)malloc(sizeof(string)*向量大小);
string*F=(string*)malloc(sizeof(string)*向量大小);
对于(int i=0;i__kernel void concat_kernel(__global uchar *D,__global uchar *E,__global uchar *F, const int dSize, const int eSize)
{
    int gid = get_global_id(0);
    int globalSize = get_global_size(0);

    int i;
    for(i=gid; i< dSize; i+= globalSize){
        F[i] = D[i];
    }

    for(i=gid; i< eSize; i+= globalSize){
        F[i+dSize] = E[i];
    }

    if(gid == globalSize-1){
        //using the last work item here because it will be
        //idle when (dSize+eSize) % globalSize != 0
        F[dSize + eSize -1] = '\0';
    }
}