| /* |
| * Copyright (C) 2013 The Android Open Source Project |
| * |
| * 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 "rsCpuIntrinsic.h" |
| #include "rsCpuIntrinsicInlines.h" |
| |
| namespace android { |
| namespace renderscript { |
| |
| |
| class RsdCpuScriptIntrinsicHistogram : public RsdCpuScriptIntrinsic { |
| public: |
| void populateScript(Script *) override; |
| void invokeFreeChildren() override; |
| |
| void setGlobalVar(uint32_t slot, const void *data, size_t dataLength) override; |
| void setGlobalObj(uint32_t slot, ObjectBase *data) override; |
| |
| ~RsdCpuScriptIntrinsicHistogram() override; |
| RsdCpuScriptIntrinsicHistogram(RsdCpuReferenceImpl *ctx, const Script *s, const Element *e); |
| |
| protected: |
| void preLaunch(uint32_t slot, const Allocation ** ains, uint32_t inLen, |
| Allocation * aout, const void * usr, |
| uint32_t usrLen, const RsScriptCall *sc); |
| void postLaunch(uint32_t slot, const Allocation ** ains, uint32_t inLen, |
| Allocation * aout, const void * usr, |
| uint32_t usrLen, const RsScriptCall *sc); |
| |
| |
| float mDot[4]; |
| int mDotI[4]; |
| int *mSums; |
| ObjectBaseRef<Allocation> mAllocOut; |
| |
| static void kernelP1U4(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| static void kernelP1U3(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| static void kernelP1U2(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| static void kernelP1U1(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| |
| static void kernelP1L4(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| static void kernelP1L3(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| static void kernelP1L2(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| static void kernelP1L1(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep); |
| |
| }; |
| |
| void RsdCpuScriptIntrinsicHistogram::setGlobalObj(uint32_t slot, ObjectBase *data) { |
| rsAssert(slot == 1); |
| mAllocOut.set(static_cast<Allocation *>(data)); |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::setGlobalVar(uint32_t slot, const void *data, size_t dataLength) { |
| rsAssert(slot == 0); |
| rsAssert(dataLength == 16); |
| memcpy(mDot, data, 16); |
| mDotI[0] = (int)((mDot[0] * 256.f) + 0.5f); |
| mDotI[1] = (int)((mDot[1] * 256.f) + 0.5f); |
| mDotI[2] = (int)((mDot[2] * 256.f) + 0.5f); |
| mDotI[3] = (int)((mDot[3] * 256.f) + 0.5f); |
| } |
| |
| |
| |
| void |
| RsdCpuScriptIntrinsicHistogram::preLaunch(uint32_t slot, |
| const Allocation ** ains, |
| uint32_t inLen, Allocation * aout, |
| const void * usr, uint32_t usrLen, |
| const RsScriptCall *sc) { |
| |
| const uint32_t threads = mCtx->getThreadCount(); |
| uint32_t vSize = mAllocOut->getType()->getElement()->getVectorSize(); |
| |
| switch (slot) { |
| case 0: |
| switch(vSize) { |
| case 1: |
| mRootPtr = &kernelP1U1; |
| break; |
| case 2: |
| mRootPtr = &kernelP1U2; |
| break; |
| case 3: |
| mRootPtr = &kernelP1U3; |
| vSize = 4; |
| break; |
| case 4: |
| mRootPtr = &kernelP1U4; |
| break; |
| } |
| break; |
| case 1: |
| switch(ains[0]->getType()->getElement()->getVectorSize()) { |
| case 1: |
| mRootPtr = &kernelP1L1; |
| break; |
| case 2: |
| mRootPtr = &kernelP1L2; |
| break; |
| case 3: |
| mRootPtr = &kernelP1L3; |
| break; |
| case 4: |
| mRootPtr = &kernelP1L4; |
| break; |
| } |
| break; |
| } |
| memset(mSums, 0, 256 * sizeof(int32_t) * threads * vSize); |
| } |
| |
| void |
| RsdCpuScriptIntrinsicHistogram::postLaunch(uint32_t slot, |
| const Allocation ** ains, |
| uint32_t inLen, Allocation * aout, |
| const void * usr, uint32_t usrLen, |
| const RsScriptCall *sc) { |
| |
| unsigned int *o = (unsigned int *)mAllocOut->mHal.drvState.lod[0].mallocPtr; |
| uint32_t threads = mCtx->getThreadCount(); |
| uint32_t vSize = mAllocOut->getType()->getElement()->getVectorSize(); |
| |
| if (vSize == 3) vSize = 4; |
| |
| for (uint32_t ct=0; ct < (256 * vSize); ct++) { |
| o[ct] = mSums[ct]; |
| for (uint32_t t=1; t < threads; t++) { |
| o[ct] += mSums[ct + (256 * vSize * t)]; |
| } |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1U4(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * 4 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| sums[(in[0] << 2) ] ++; |
| sums[(in[1] << 2) + 1] ++; |
| sums[(in[2] << 2) + 2] ++; |
| sums[(in[3] << 2) + 3] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1U3(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * 4 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| sums[(in[0] << 2) ] ++; |
| sums[(in[1] << 2) + 1] ++; |
| sums[(in[2] << 2) + 2] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1U2(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * 2 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| sums[(in[0] << 1) ] ++; |
| sums[(in[1] << 1) + 1] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1L4(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| int t = (cp->mDotI[0] * in[0]) + |
| (cp->mDotI[1] * in[1]) + |
| (cp->mDotI[2] * in[2]) + |
| (cp->mDotI[3] * in[3]); |
| sums[(t + 0x7f) >> 8] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1L3(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| int t = (cp->mDotI[0] * in[0]) + |
| (cp->mDotI[1] * in[1]) + |
| (cp->mDotI[2] * in[2]); |
| sums[(t + 0x7f) >> 8] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1L2(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| int t = (cp->mDotI[0] * in[0]) + |
| (cp->mDotI[1] * in[1]); |
| sums[(t + 0x7f) >> 8] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1L1(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| int t = (cp->mDotI[0] * in[0]); |
| sums[(t + 0x7f) >> 8] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::kernelP1U1(const RsExpandKernelDriverInfo *info, |
| uint32_t xstart, uint32_t xend, |
| uint32_t outstep) { |
| |
| RsdCpuScriptIntrinsicHistogram *cp = (RsdCpuScriptIntrinsicHistogram *)info->usr; |
| uchar *in = (uchar *)info->inPtr[0]; |
| int * sums = &cp->mSums[256 * info->lid]; |
| |
| for (uint32_t x = xstart; x < xend; x++) { |
| sums[in[0]] ++; |
| in += info->inStride[0]; |
| } |
| } |
| |
| |
| RsdCpuScriptIntrinsicHistogram::RsdCpuScriptIntrinsicHistogram(RsdCpuReferenceImpl *ctx, |
| const Script *s, const Element *e) |
| : RsdCpuScriptIntrinsic(ctx, s, e, RS_SCRIPT_INTRINSIC_ID_HISTOGRAM) { |
| |
| mRootPtr = nullptr; |
| mSums = new int[256 * 4 * mCtx->getThreadCount()]; |
| mDot[0] = 0.299f; |
| mDot[1] = 0.587f; |
| mDot[2] = 0.114f; |
| mDot[3] = 0; |
| mDotI[0] = (int)((mDot[0] * 256.f) + 0.5f); |
| mDotI[1] = (int)((mDot[1] * 256.f) + 0.5f); |
| mDotI[2] = (int)((mDot[2] * 256.f) + 0.5f); |
| mDotI[3] = (int)((mDot[3] * 256.f) + 0.5f); |
| } |
| |
| RsdCpuScriptIntrinsicHistogram::~RsdCpuScriptIntrinsicHistogram() { |
| if (mSums) { |
| delete []mSums; |
| } |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::populateScript(Script *s) { |
| s->mHal.info.exportedVariableCount = 2; |
| } |
| |
| void RsdCpuScriptIntrinsicHistogram::invokeFreeChildren() { |
| } |
| |
| RsdCpuScriptImpl * rsdIntrinsic_Histogram(RsdCpuReferenceImpl *ctx, const Script *s, const Element *e) { |
| |
| return new RsdCpuScriptIntrinsicHistogram(ctx, s, e); |
| } |
| |
| } // namespace renderscript |
| } // namespace android |