[Libreoffice-commits] core.git: Branch 'feature/calc-group-interpreter' - sc/source

Jing Xian jingxian at multicorewareinc.com
Thu Jul 11 03:36:19 PDT 2013


 sc/source/core/opencl/formulagroupcl.cxx |   70 +++++++++++++++++--------------
 sc/source/core/opencl/oclkernels.hxx     |   70 +++++++++++++++++++++++--------
 sc/source/core/opencl/openclwrapper.cxx  |   44 +++++++++++++------
 3 files changed, 121 insertions(+), 63 deletions(-)

New commits:
commit 6c89aa8345e57c8c0b68b47bb67ac32b7dc930bc
Author: Jing Xian <jingxian at multicorewareinc.com>
Date:   Thu Jul 11 11:35:54 2013 +0100

    add min/max/delta kernels and misc. cleanup / bug fixing.

diff --git a/sc/source/core/opencl/formulagroupcl.cxx b/sc/source/core/opencl/formulagroupcl.cxx
index dee465e..8c2e236 100755
--- a/sc/source/core/opencl/formulagroupcl.cxx
+++ b/sc/source/core/opencl/formulagroupcl.cxx
@@ -82,16 +82,19 @@ bool FormulaGroupInterpreterOpenCL::interpret(ScDocument& rDoc, const ScAddress&
     memset(rResult,0,rowSize);
     float * fpOclSrcData = NULL; // Point to the input data from CPU
     uint * npOclStartPos = NULL; // The first position for calculation,for example,the A1 in (=MAX(A1:A100))
-    uint * npOclEndPos     = NULL; // The last position for calculation,for example, the A100 in (=MAX(A1:A100))
-    float * fpLeftData     = NULL; // Left input for binary operator(+,-,*,/),for example,(=leftData+rightData)
+    uint * npOclEndPos   = NULL; // The last position for calculation,for example, the A100 in (=MAX(A1:A100))
+    float * fpLeftData   = NULL; // Left input for binary operator(+,-,*,/),for example,(=leftData+rightData)
     float * fpRightData  = NULL; // Right input for binary operator(+,-,*,/),for example,(=leftData/rightData)
                                  // The rightData can't be zero for "/"
     static OclCalc ocl_calc;
-    // Don't know how large the size will be applied previously, so create them as the rowSize or 65536
-    // Don't know which formulae will be used previously, so create buffers for different formulae used probably
-    ocl_calc.CreateBuffer(fpOclSrcData,npOclStartPos,npOclEndPos,rowSize);
-    ocl_calc.CreateBuffer(fpLeftData,fpRightData,rowSize);
-    //printf("pptrr is %d,%d,%d\n",fpOclSrcData,npOclStartPos,npOclEndPos);
+    if(ocl_calc.GetOpenclState())
+    {
+        // Don't know how large the size will be applied previously, so create them as the rowSize or 65536
+        // Don't know which formulae will be used previously, so create buffers for different formulae used probably
+        ocl_calc.CreateBuffer(fpOclSrcData,npOclStartPos,npOclEndPos,rowSize);
+        ocl_calc.CreateBuffer(fpLeftData,fpRightData,rowSize);
+        //printf("pptrr is %d,%d,%d\n",fpOclSrcData,npOclStartPos,npOclEndPos);
+    }
 ///////////////////////////////////////////////////////////////////////////////////////////
 
     // Until we implement group calculation for real, decompose the group into
@@ -123,10 +126,11 @@ bool FormulaGroupInterpreterOpenCL::interpret(ScDocument& rDoc, const ScAddress&
                         nRowEnd += i;
                     size_t nRowSize = nRowEnd - nRowStart + 1;
                     ScMatrixRef pMat(new ScMatrix(nColSize, nRowSize, 0.0));
-
-                    npOclStartPos[i] = nRowStart; // record the start position
-                    npOclEndPos[i]     = nRowEnd;   // record the end position
-
+                    if(ocl_calc.GetOpenclState())
+                    {
+                        npOclStartPos[i] = nRowStart; // record the start position
+                        npOclEndPos[i]   = nRowEnd;   // record the end position
+                    }
                     for (size_t nCol = 0; nCol < nColSize; ++nCol)
                     {
                         const double* pArray = rArrays[nCol];
@@ -135,12 +139,14 @@ bool FormulaGroupInterpreterOpenCL::interpret(ScDocument& rDoc, const ScAddress&
                             fprintf(stderr,"Error: pArray is NULL!\n");
                             return false;
                         }
-
-                        for( size_t u=0; u<rowSize; u++ )
+                        if(ocl_calc.GetOpenclState())
                         {
-                            // Many video cards can't support double type in kernel, so need transfer the double to float
-                            fpOclSrcData[u] = (float)pArray[u];
-                            //fprintf(stderr,"fpOclSrcData[%d] is %f.\n",u,fpOclSrcData[u]);
+                            for( size_t u=nRowStart; u<=nRowEnd; u++ )
+                            {
+                                // Many video cards can't support double type in kernel, so need transfer the double to float
+                                fpOclSrcData[u] = (float)pArray[u];
+                                //fprintf(stderr,"fpOclSrcData[%d] is %f.\n",u,fpOclSrcData[u]);
+                            }
                         }
 
                         for (size_t nRow = 0; nRow < nRowSize; ++nRow)
@@ -165,22 +171,23 @@ bool FormulaGroupInterpreterOpenCL::interpret(ScDocument& rDoc, const ScAddress&
         ScFormulaCell* pDest = rDoc.GetFormulaCell(aTmpPos);
         if (!pDest)
             return false;
-
-        const formula::FormulaToken *pCur = aCode2.First();
-        aCode2.Reset();
-        while( ( pCur = aCode2.Next() ) != NULL )
+        if(ocl_calc.GetOpenclState())
         {
-            OpCode eOp = pCur->GetOpCode();
-            if(eOp==0)
+            const formula::FormulaToken *pCur = aCode2.First();
+            aCode2.Reset();
+            while( ( pCur = aCode2.Next() ) != NULL )
             {
-                 if(nCount3%2==0)
-                     fpLeftData[nCount1++] = (float)pCur->GetDouble();
-                 else
-                     fpRightData[nCount2++] = (float)pCur->GetDouble();
-                 nCount3++;
-            }
-            else if( eOp!=ocOpen && eOp!=ocClose )
-                nOclOp = eOp;
+                OpCode eOp = pCur->GetOpCode();
+                if(eOp==0)
+                {
+                     if(nCount3%2==0)
+                         fpLeftData[nCount1++] = (float)pCur->GetDouble();
+                     else
+                         fpRightData[nCount2++] = (float)pCur->GetDouble();
+                     nCount3++;
+                }
+                else if( eOp!=ocOpen && eOp!=ocClose )
+                    nOclOp = eOp;
 
 //              if(count1>0){//dbg
 //                  fprintf(stderr,"leftData is %f.\n",leftData[count1-1]);
@@ -190,11 +197,12 @@ bool FormulaGroupInterpreterOpenCL::interpret(ScDocument& rDoc, const ScAddress&
 //                  fprintf(stderr,"rightData is %f.\n",rightData[count2-1]);
 //                  count2--;
 //              }
+            }
         }
 
         if(!getenv("SC_GPU")||!ocl_calc.GetOpenclState())
         {
-            fprintf(stderr,"ccCPU flow...\n\n");
+            //fprintf(stderr,"ccCPU flow...\n\n");
             generateRPNCode(rDoc, aTmpPos, aCode2);
             ScInterpreter aInterpreter(pDest, &rDoc, aTmpPos, aCode2);
             aInterpreter.Interpret();
diff --git a/sc/source/core/opencl/oclkernels.hxx b/sc/source/core/opencl/oclkernels.hxx
index e13c24a..c231dbd 100755
--- a/sc/source/core/opencl/oclkernels.hxx
+++ b/sc/source/core/opencl/oclkernels.hxx
@@ -162,32 +162,66 @@ __kernel void oclFormulaMinverse(__global float *data,
 }
 
 // Double precision is a requirement of spreadsheets
-#if 0
-#if defined(cl_khr_fp64)  // Khronos extension
-#pragma OPENCL EXTENSION cl_khr_fp64 : enable
-#elif defined(cl_amd_fp64)  // AMD extension
-#pragma OPENCL EXTENSION cl_amd_fp64 : enable
-#endif
-typedef double fp_t;
-#else
-typedef float fp_t;
-#endif
-
-__kernel void oclAverageDelta(__global fp_t *values, __global fp_t *subtract, __global int start, __global int end, __global fp_t *output)
+// cl_khr_fp64: Khronos extension
+// cl_amd_fp64: AMD extension
+\n#if 0 \n
+\n#if defined(cl_khr_fp64) \n
+\n#pragma OPENCL EXTENSION cl_khr_fp64 : enable \n
+\n#elif defined(cl_amd_fp64) \n
+\n#pragma OPENCL EXTENSION cl_amd_fp64 : enable \n
+\n#endif \n
+\ntypedef double fp_t; \n
+\n#else \n
+\ntypedef float fp_t; \n
+\n#endif \n
+
+__kernel void oclAverageDelta(__global fp_t *values, __global fp_t *subtract, uint start, uint end, __global fp_t *output)
 {
     const unsigned int id = get_global_id(0);
 
     // Average
-    int i;
-    fp_t sum = 0.0;
-    for(i = start; i < end; i++)
-        sum += values[i];
-    fp_t val = sum/(end-start);
+    fp_t fSum = 0.0f;
+    for(int i = start; i < end; i++)
+        fSum += values[i];
+    fp_t fVal = fSum/(end-start);
 
     // Subtract & output
-    output[id] = val - subtract[id];
+    output[id] = fVal - subtract[id];
 }
 
+__kernel void oclMaxDelta(__global fp_t *values, __global fp_t *subtract, uint start, uint end, __global fp_t *output)
+{
+    const unsigned int id = get_global_id(0);
+
+    // Max
+    float fMaxVal = values[start];
+    for(int i=start+1;i < end;i++)
+    {
+        if(values[i]>fMaxVal)
+            fMaxVal = values[i];
+    }
+
+    // Subtract & output
+    output[id] = fMaxVal - subtract[id];
+}
+
+__kernel void oclMinDelta(__global fp_t *values, __global fp_t *subtract, uint start, uint end, __global fp_t *output)
+{
+    const unsigned int id = get_global_id(0);
+
+    // Min
+    float fMinVal = values[start];
+    for(int i=start+1;i < end;i++)
+    {
+        if(values[i]<fMinVal)
+            fMinVal = values[i];
+    }
+
+    // Subtract & output
+    output[id] = fMinVal - subtract[id];
+}
+
+
 );
 
 #endif // USE_EXTERNAL_KERNEL
diff --git a/sc/source/core/opencl/openclwrapper.cxx b/sc/source/core/opencl/openclwrapper.cxx
index 6132ae3..781c838 100755
--- a/sc/source/core/opencl/openclwrapper.cxx
+++ b/sc/source/core/opencl/openclwrapper.cxx
@@ -106,6 +106,8 @@ int OpenclDevice::RegistOpenclKernel()
     AddKernelConfig(11, (const char*) "oclSignedMul");
     AddKernelConfig(12, (const char*) "oclSignedDiv");
     AddKernelConfig(13, (const char*) "oclAverageDelta");
+    AddKernelConfig(14, (const char*) "OclMaxDelta");
+    AddKernelConfig(15, (const char*) "OclMinDelta");
 
     return 0;
 }
@@ -499,18 +501,17 @@ int OpenclDevice::CompileKernelFile(GPUEnv *gpuInfo, const char *buildOption) {
 
     //char options[512];
     // create a cl program executable for all the devices specified
+    printf("BuildProgram.\n");
     if (!gpuInfo->mnIsUserCreated) {
         status = clBuildProgram(gpuInfo->mpArryPrograms[idx], 1, gpuInfo->mpArryDevsID,
                 buildOption, NULL, NULL);
-        CHECK_OPENCL(status)
     } else {
         status = clBuildProgram(gpuInfo->mpArryPrograms[idx], 1, &(gpuInfo->mpDevID),
                 buildOption, NULL, NULL);
-        CHECK_OPENCL(status)
     }
-    printf("BuildProgram.\n");
 
     if (status != CL_SUCCESS) {
+        printf ("BuildProgram error!\n");
         if (!gpuInfo->mnIsUserCreated) {
             status = clGetProgramBuildInfo(gpuInfo->mpArryPrograms[idx],
                     gpuInfo->mpArryDevsID[0], CL_PROGRAM_BUILD_LOG, 0, NULL,
@@ -678,7 +679,7 @@ int OpenclDevice::InitOpenclRunEnv(GPUEnv *gpuInfo)
                                             &numDevices);
 
                     if (status != CL_SUCCESS) {
-                        return 1;
+                        continue;
                     }
 
                     if (numDevices) {
@@ -686,6 +687,8 @@ int OpenclDevice::InitOpenclRunEnv(GPUEnv *gpuInfo)
                     }
                 }
             }
+            if(status!=CL_SUCCESS)
+                return 1;
             free(platforms);
         }
         if (NULL == gpuInfo->mpPlatformID) {
@@ -2102,7 +2105,7 @@ static cl_mem allocateDoubleBuffer(KernelEnv &rEnv, const double *_pValues,
                                     nElements * sizeof(double), NULL, pStatus);
     fp_t *pValues = (fp_t *)clEnqueueMapBuffer(rEnv.mpkCmdQueue,xValues,CL_TRUE,CL_MAP_WRITE,0,
                                                    nElements * sizeof(fp_t),0,NULL,NULL,NULL);
-    for(int i=0;i<nElements;i++)
+    for(int i=0;i<(int)nElements;i++)
         pValues[i] = (fp_t)_pValues[i];
 
     clEnqueueUnmapMemObject(rEnv.mpkCmdQueue,xValues,pValues,0,NULL,NULL);
@@ -2113,15 +2116,23 @@ static cl_mem allocateDoubleBuffer(KernelEnv &rEnv, const double *_pValues,
 double *OclCalc::OclSimpleDeltaOperation(OpCode eOp, const double *pOpArray,
                                          const double *pSubtractSingle, size_t nElements)
 {
-    KernelEnv kEnv;
     SetKernelEnv(&kEnv);
 
     // select a kernel: cut & paste coding is utterly evil.
-    const char *kernelName;
+    const char *kernelName = NULL;
     switch (eOp) {
+    case ocAdd:
+    case ocSub:
+        fprintf(stderr,"ocSub is %d\n",ocSub);
+    case ocMul:
+    case ocDiv:
+        ; // FIXME: fallthrough for now
     case ocMax:
+        kernelName = "oclMaxDelta";
+        break;
     case ocMin:
-        ; // FIXME: fallthrough for now
+        kernelName = "oclMinDelta";
+        break;
     case ocAverage:
         kernelName = "oclAverageDelta";
         break;
@@ -2133,15 +2144,20 @@ double *OclCalc::OclSimpleDeltaOperation(OpCode eOp, const double *pOpArray,
     cl_int clStatus;
     size_t global_work_size[1];
 
-    kEnv.mpkKernel = clCreateKernel(kEnv.mpkProgram,kernelName, &clStatus);
+    kEnv.mpkKernel = clCreateKernel(kEnv.mpkProgram, kernelName, &clStatus);
     CHECK_OPENCL(clStatus);
+    if (!kEnv.mpkKernel)
+    {
+        fprintf(stderr, "could not clCreateKernel '%s'\n", kernelName);
+        return NULL;
+    }
 
     // Ugh - horrible redundant copying ...
     cl_mem valuesCl   = allocateDoubleBuffer(kEnv, pOpArray, nElements, &clStatus);
     cl_mem subtractCl = allocateDoubleBuffer(kEnv, pSubtractSingle, nElements, &clStatus);
 
-    cl_int start = 0;
-    cl_int end = (cl_int) nElements;
+    cl_uint start = 0;
+    cl_uint end = (cl_uint) nElements;
     cl_mem outputCl = clCreateBuffer(kEnv.mpkContext,
                                      CL_MEM_READ_WRITE,
                                      nElements * sizeof(fp_t),
@@ -2155,10 +2171,10 @@ double *OclCalc::OclSimpleDeltaOperation(OpCode eOp, const double *pOpArray,
     clStatus = clSetKernelArg(kEnv.mpkKernel, 1, sizeof(cl_mem),
                               (void *)&subtractCl);
     CHECK_OPENCL(clStatus);
-    clStatus = clSetKernelArg(kEnv.mpkKernel, 2, sizeof(cl_int),
+    clStatus = clSetKernelArg(kEnv.mpkKernel, 2, sizeof(cl_uint),
                               (void *)&start);
     CHECK_OPENCL(clStatus);
-    clStatus = clSetKernelArg(kEnv.mpkKernel, 3, sizeof(cl_int),
+    clStatus = clSetKernelArg(kEnv.mpkKernel, 3, sizeof(cl_uint),
                               (void *)&end);
     CHECK_OPENCL(clStatus);
     clStatus = clSetKernelArg(kEnv.mpkKernel, 4, sizeof(cl_mem),
@@ -2179,7 +2195,7 @@ double *OclCalc::OclSimpleDeltaOperation(OpCode eOp, const double *pOpArray,
     fp_t *pOutput = (fp_t *)clEnqueueMapBuffer(kEnv.mpkCmdQueue,outputCl,CL_TRUE,
                                                CL_MAP_READ,0,nElements*sizeof(fp_t),
                                                0,NULL,NULL,NULL);
-    for(int i = 0; i < nElements; i++)
+    for(int i = 0; i < (int)nElements; i++)
         pResult[i] = (double)pOutput[i];
 
     clEnqueueUnmapMemObject(kEnv.mpkCmdQueue,outputCl,pOutput,0,NULL,NULL);


More information about the Libreoffice-commits mailing list