src/hbcore/svgext/hbnvgdecoder/hbnvgcsicon.cpp
author hgs
Mon, 18 Oct 2010 18:23:13 +0300
changeset 34 ed14f46c0e55
parent 5 627c4a0fd0e7
permissions -rw-r--r--
201041

/****************************************************************************
**
** Copyright (C) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (developer.feedback@nokia.com)
**
** This file is part of the HbCore module of the UI Extensions for Mobile.
**
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser General Public
** License version 2.1 as published by the Free Software Foundation and
** appearing in the file LICENSE.LGPL included in the packaging of this file.
** Please review the following information to ensure the GNU Lesser General
** Public License version 2.1 requirements will be met:
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights.  These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you have questions regarding the use of this file, please contact
** Nokia at developer.feedback@nokia.com.
**
****************************************************************************/

#include "hbnvgcsicon_p.h"
#include "hbnvgfittoviewbox_p.h"
#include "hbnvgutil_p.h"
#include "hbnvgexception_p.h"
#include "hbdereferencer_p.h"

#include <QScopedPointer>
#include <QScopedArrayPointer>

/*!
  NVG-CS version
 */
const qint32 Version2   = 2;

/*!
  File offsets
 */
const qint32 NVG_VIEWBOX_WIDTH_OFS = 44;
const qint32 NVG_VIEWBOX_HEIGHT_OFS = 48;
const qint32 NVG_VIEWBOX_X_OFS = 36;
const qint32 NVG_VIEWBOX_Y_OFS = 40;
const qint32 NVG_PATHDATATYPE_OFS = 26;
const qint32 NVG_SCALE_OFS = 28;
const qint32 NVG_BIAS_OFS = 32;
const qint32 NVG_COMMANDSECTION_OFS = 2;
const qint32 NVG_RGBA_OFS = 4;
const qint32 NVG_HEADERSIZE_OFS = 4;
const qint32 NVG_PAINTSECTION_LINEARGRAD_TRANSFORM_OFFSET = 20;
const qint32 NVG_PAINTSECTION_RADIALGRAD_TRANSFORM_OFFSET = 24;
const qint32 NVG_VERSION_OFS         = 3;

/*!
  NVG-CS commands
 */
const qint32 CMD_SET_FILL_PAINT           = 4  << 24;
const qint32 CMD_SET_COLOR_RAMP           = 6  << 24;
const qint32 CMD_DRAW_PATH                = 7  << 24;
const qint32 CMD_SET_TRANSFORMATION       = 8  << 24;
const qint32 CMD_SET_STROKE_PAINT         = 5  << 24;
const qint32 CMD_SET_STROKE_WIDTH         = 9  << 24;
const qint32 CMD_SET_STROKE_LINE_JOIN_CAP = 10 << 24;
const qint32 CMD_SET_STROKE_MITER_LIMIT   = 11 << 24;

/*!
 Stroke cap style
 */
const qint32 CAP_BUTT        = 1;
const qint32 CAP_SQUARE      = 2;
const qint32 CAP_ROUND       = 3;

/*!
 Stroke join style
 */
const qint32 LINE_JOIN_BEVEL = 1;
const qint32 LINE_JOIN_MITER = 2;
const qint32 LINE_JOIN_ROUND = 3;

/*!
 Fill paint type
 */
const qint32 PAINT_FLAT      = 1;
const qint32 PAINT_LGRAD     = 2;
const qint32 PAINT_RGRAD     = 3;

/*!
 Stroke paint type
 */
const qint32 STROKE_LGRAD        = 2;
const qint32 STROKE_RGRAD        = 3;
const qint32 STROKE_COLOR_RAMP   = 4;

/*!
 Transform encoding values
 */
const qint32 TRANSFORM_COMPLETE    = 0;
const qint32 TRANSFORM_SCALING     = 2;
const qint32 TRANSFORM_SHEARING    = 4;
const qint32 TRANSFORM_ROTATION    = 8;
const qint32 TRANSFORM_TRANSLATION = 16;

const VGfloat identityMatrix[] = {
    1.0f, 0.0f, 0.0f,
    0.0f, 1.0f, 0.0f,
    0.0f, 0.0f , 1.0f
};

/**
 * @fn      Constructor
 * @Version
 * @parameter: buffer - None
 *
 * @return: None
 */

HbNvgCsIcon::HbNvgCsIcon()
        :mPaintFill(VG_INVALID_HANDLE),
        mPaintStroke(VG_INVALID_HANDLE),
        mVgPath(VG_INVALID_HANDLE),
        mLastPathDataType(0),
        mDoFill(VG_FALSE),
        mDoStroke(VG_FALSE),
        mPreserveAspectSetting(HbNvgEngine::NvgPreserveAspectRatioXmidYmid),
        mSmilFitSetting(HbNvgEngine::NvgMeet),
        mMirrored(false)
{
}

HbNvgCsIcon::~HbNvgCsIcon()
{
    if (mPaintFill) {
        vgDestroyPaint(mPaintFill);
    }
    if (mPaintStroke) {
        vgDestroyPaint(mPaintStroke);
    }
    if (mVgPath) {
        vgDestroyPath(mVgPath);
    }

    vgSetPaint(VG_INVALID_HANDLE, VG_FILL_PATH);
    vgSetPaint(VG_INVALID_HANDLE, VG_STROKE_PATH);
}

/*!
    set the aspectRatio \a preserveAspectSetting and \a smilFitSetting
    to be applied on the nvgicon.
*/
void  HbNvgCsIcon::setPreserveAspectRatio(HbNvgEngine::HbNvgAlignType preserveAspectSetting,
        HbNvgEngine::HbNvgMeetType smilFitSetting)
{
    mPreserveAspectSetting = preserveAspectSetting;
    mSmilFitSetting = smilFitSetting;
}

void HbNvgCsIcon::enableMirroring(bool mirroringMode)
{
    mMirrored = mirroringMode;
}

/*!
    Set the \a angle for rotation of the nvgicon at the
    coordiantes  \a x and \a y.
*/
void HbNvgCsIcon::rotate(float angle, float xValue, float yValue)
{
    mRotationAngle  = angle;
    mRotationX      = xValue;
    mRotationY      = yValue;
}

/*!
    Do the direct draw of the nvg graphic data \a buffer of size \a targetSize
    and return the status of the draw.
*/
void HbNvgCsIcon ::directDraw(const QByteArray &buffer, const QSize &targetSize)
{
    drawCommandSection(buffer, targetSize);
}

void HbNvgCsIcon::drawCommandSection(const QByteArray &buffer, const QSize & targetSize)
{ 
    HbDereferencer iconData(buffer);
    qint16 headerSize  = iconData.derefInt16(NVG_HEADERSIZE_OFS);
    quint8 nvgVersion   = iconData.derefInt8(NVG_VERSION_OFS);

    HbNvgEngine::HbNvgErrorType errInit = HbNvgEngine::NvgErrNone;

    errInit = initializeGc();
    if (HbNvgEngine::NvgErrNone != errInit) {
        throw HbNvgException(errInit);
    }

    qint16 pathDataType = iconData.derefInt16(NVG_PATHDATATYPE_OFS);
    float scale         = iconData.derefReal32(NVG_SCALE_OFS);
    float bias          = iconData.derefReal32(NVG_BIAS_OFS);

    errInit = createPathHandle(pathDataType, scale, bias);
    if (HbNvgEngine::NvgErrNone != errInit) {
        throw HbNvgException(errInit);
    }

    VGfloat currentPathMatrix[9];
    vgGetMatrix(currentPathMatrix);
    vgSeti(VG_MATRIX_MODE, VG_MATRIX_PATH_USER_TO_SURFACE);

    // We use the caller's base batrix regardless of which mode the caller was.
    // The caller may have set the matrix in VG_MATRIX_IMAGE_USER_TO_SURFACE mode
    // as it thinks it is drawing images (textures).
    // But even though the texture gets stroked instead, we must use the caller's
    // transformation matrix.
    // Everything gets restored to the original values before we return.
    vgLoadMatrix(currentPathMatrix);

    /*
     * set the rotation angle if available
     */

    setRotation();

    if (mMirrored) {
        vgTranslate((VGfloat)(targetSize.width()), 0);
        vgScale(-1.0f, 1.0f);
    }


#ifdef __MIRROR_
    vgScale(1.0f, -1.0f);
    vgTranslate(0, (VGfloat)(-targetSize.height()));
#endif

    //viewbox parameters
    float viewboxX   = iconData.derefReal32(NVG_VIEWBOX_X_OFS);
    float viewboxY   = iconData.derefReal32(NVG_VIEWBOX_Y_OFS);
    float viewboxW   = iconData.derefReal32(NVG_VIEWBOX_WIDTH_OFS);
    float viewboxH   = iconData.derefReal32(NVG_VIEWBOX_HEIGHT_OFS);

    applyViewboxToViewPortTransformation(targetSize, viewboxX, viewboxY, viewboxW, viewboxH);

    quint32 offsetSectionLength = iconData.getLength() - headerSize;
    quint8 * offsetPtr = iconData.derefInt8Array(offsetSectionLength, headerSize);

    HbDereferencer offsetSection(offsetPtr, offsetSectionLength);
    quint16 offsetVectorCount = offsetSection.derefInt16();

    offsetPtr = iconData.derefInt8Array(offsetSectionLength - sizeof(quint16), headerSize + sizeof(quint16));
    HbDereferencer offsetVector(offsetPtr, offsetSectionLength - sizeof(quint16));

    qint32 commandSectionOffset = offsetVectorCount * sizeof(quint16);
    HbDereferencer commandSection((offsetVector.getPtr() + commandSectionOffset),
                                  iconData.getLength() - commandSectionOffset - headerSize - sizeof(quint16));

    // from version 2 onwards command section will start on word boundary
    if (nvgVersion >= Version2 && ((offsetVectorCount  & 0x01) == 0)) {
        commandSection.skip(2);
    }

    quint16 cmdCount = commandSection.derefInt16();
    commandSection.skip(NVG_COMMANDSECTION_OFS);

    /*
     * from version 2 onwards there will be a padding added
     * after the command count to make it word aligned
     */
    if (nvgVersion >= Version2) {
        commandSection.skip(2);
    }

    executeNvgCsCommandLoop(cmdCount, &iconData, &offsetVector, &commandSection, nvgVersion);
}

void HbNvgCsIcon::executeNvgCsCommandLoop(quint16 commandCount, HbDereferencer *iconData,
        HbDereferencer * offsetVector, HbDereferencer * commandSection, quint8 nvgVersion)
{
    quint32 transVal;

    VGfloat currentPathMatrix[9];

    vgGetMatrix(currentPathMatrix);

    qint32 offsetIx = 0;
    for (qint32 i = 0; i < commandCount; i++) {
        quint32 currentCommand = commandSection->derefInt32();
        offsetIx = currentCommand & 0x0000ffff;

        switch (currentCommand & 0xff000000) {
        case CMD_SET_FILL_PAINT: {
            mFillAlpha        = (currentCommand & 0x00ff0000) >> 16;
            quint16 offset    = offsetVector->derefInt16(offsetIx * sizeof(quint16));

            HbDereferencer section = getCommandSection(offset, iconData, nvgVersion);

            setFillPaint(&section);
        }
        break;
        case CMD_SET_COLOR_RAMP: {
            quint16 offset = offsetVector->derefInt16(offsetIx * sizeof(quint16));

            HbDereferencer section = getCommandSection(offset, iconData, nvgVersion);

            setColorRamp(&section);
        }
        break;
        case CMD_DRAW_PATH: {
            if ((currentCommand & 0x00010000)) {
                mDoStroke = VG_TRUE;
            }

            if ((currentCommand & 0x00020000)) {
                mDoFill = VG_TRUE;
            }
            quint16 offset = offsetVector->derefInt16(offsetIx * sizeof(quint16));

            HbDereferencer section = getCommandSection(offset, iconData, nvgVersion);

            drawPath(&section);
        }
        break;
        case CMD_SET_TRANSFORMATION: {
            setTransform(commandSection, transVal, currentPathMatrix);
            commandSection->skip(transVal * sizeof(quint32));
        }
        break;
        case CMD_SET_STROKE_PAINT: {
            mStrokeAlpha = (currentCommand & 0x00ff0000) >> 16;
            quint16 offset = offsetVector->derefInt16(offsetIx * sizeof(quint16));

            HbDereferencer section = getCommandSection(offset, iconData, nvgVersion);

            setStrokePaint(&section);
        }
        break;
        case CMD_SET_STROKE_WIDTH: {
            float strokeWidth;
            commandSection->skip(sizeof(quint32));

            /*
             * check for alignment and copy data if not aligned, else directly convert
             * version 2 or above guarantees that is always word aligned
             */
            quint8 * cptr = commandSection->derefInt8Array(sizeof(float), 0);
            if (nvgVersion < Version2 && !isAligned4(cptr)) {

                memcpy(reinterpret_cast<void *>(&strokeWidth),
                       reinterpret_cast<void *>(cptr), sizeof(strokeWidth));
            } else {
                strokeWidth = commandSection->derefReal32();
            }

            vgSetf(VG_STROKE_LINE_WIDTH, strokeWidth);
        }
        break;
        case CMD_SET_STROKE_MITER_LIMIT: {
            float miterLimit;
            commandSection->skip(sizeof(float));

            /*
             * check for alignment and copy data if not aligned, else directly convert
             * version 2 or above guarantees that is always word aligned
             */
            quint8 * cptr = commandSection->derefInt8Array(sizeof(float), 0);

            if (nvgVersion < Version2 && !isAligned4(cptr)) {
                memcpy(reinterpret_cast<void *>(&miterLimit),
                       reinterpret_cast<void *>(cptr), sizeof(miterLimit));
            } else {
                miterLimit = commandSection->derefReal32();
            }

            vgSetf(VG_STROKE_MITER_LIMIT, miterLimit);
        }
        break;
        case CMD_SET_STROKE_LINE_JOIN_CAP: {
            quint8 joinType = (currentCommand & 0x0000ff00) >> 8;
            quint8 capType = (currentCommand & 0x000000ff);

            VGCapStyle capStyle;
            switch (capType) {
            case CAP_SQUARE:
                capStyle = VG_CAP_SQUARE;
                break;
            case CAP_ROUND:
                capStyle = VG_CAP_ROUND;
                break;
            case CAP_BUTT:
            default:
                capStyle = VG_CAP_BUTT;
                break;
            }

            VGJoinStyle lineJoinStyle;
            switch (joinType) {
            case LINE_JOIN_BEVEL:
                lineJoinStyle = VG_JOIN_BEVEL;
                break;
            case LINE_JOIN_ROUND:
                lineJoinStyle = VG_JOIN_ROUND;
                break;
            case LINE_JOIN_MITER:
            default:
                lineJoinStyle = VG_JOIN_MITER;
                break;
            }

            vgSeti(VG_STROKE_CAP_STYLE, capStyle);
            vgSeti(VG_STROKE_JOIN_STYLE, lineJoinStyle);
        }
        break;
        default: {
            throw HbNvgException(HbNvgEngine::NvgErrCorrupt);
        }
        }

        // go to the next command
        commandSection->skip(sizeof(quint32));
    }

}

HbDereferencer HbNvgCsIcon::getCommandSection(quint16 offset, HbDereferencer * iconData, qint32 nvgVersion)
{
    // the max length that the command section can have
    qint32 commandSectionLength = iconData->getLength() - offset;

    if (commandSectionLength <= 0) {
        throw HbNvgException(HbNvgEngine::NvgErrCorrupt);
    }

    HbDereferencer section(iconData->derefInt8Array(commandSectionLength, offset), commandSectionLength);

    /*
     * all the section are expected to be word aligned
     * all of the nvg-cs icon will be version 2 or above
     * the else won't be there as nvg version will always be greater than 2
     */
    if (nvgVersion >= Version2) {
        if (!isAligned4(offset)) {
            throw HbNvgException(HbNvgEngine::NvgErrCorrupt);
        }
    } else {
        /*
         * no need to do anything here as once the nvgdecoder release
         * its version will be always greater than 2
         * infact the check for version will be removed
         */
    }

    return section;
}

void HbNvgCsIcon::applyViewboxToViewPortTransformation(const QSize& targetSize,
        float viewboxX, float viewboxY, float viewboxW, float viewboxH)
{
    HbNvgFitToViewBoxImpl *viewBoxTx = new HbNvgFitToViewBoxImpl();
    Q_CHECK_PTR(viewBoxTx);
    QScopedPointer<HbNvgFitToViewBoxImpl> viewboxTrnsfr(viewBoxTx);

    viewboxTrnsfr->setAllignment(mPreserveAspectSetting);
    viewboxTrnsfr->setScaling(mSmilFitSetting);

    if (viewboxW > 0 && viewboxH > 0) {
        viewboxTrnsfr->setViewBox(viewboxX, viewboxY, viewboxW, viewboxH);
    }

    qint32 width = targetSize.width();
    qint32 height = targetSize.height();

    viewboxTrnsfr->setWindowViewportTrans(QRect(0, 0, width, height), QSize(0, 0));
}

HbNvgEngine::HbNvgErrorType HbNvgCsIcon::initializeGc()
{
    if (mPaintFill == VG_INVALID_HANDLE) {
        mPaintFill = vgCreatePaint();
        if (mPaintFill == VG_INVALID_HANDLE) {
            return openVgErrorToHbNvgError(vgGetError());
        }
    }

    vgSetPaint(mPaintFill, VG_FILL_PATH);

    if (mPaintStroke == VG_INVALID_HANDLE) {
        mPaintStroke = vgCreatePaint();
        if (mPaintStroke == VG_INVALID_HANDLE) {
            return openVgErrorToHbNvgError(vgGetError());
        }
    }

    vgSetPaint(mPaintStroke, VG_STROKE_PATH);

    return HbNvgEngine::NvgErrNone;
}

HbNvgEngine::HbNvgErrorType HbNvgCsIcon::createPathHandle(qint16 pathDataType, float scale, float bias)
{
    Q_UNUSED(scale);
    Q_UNUSED(bias);

    HbNvgEngine::HbNvgErrorType error = HbNvgEngine::NvgErrNone;

    if (mLastPathDataType != pathDataType) {
        if (mVgPath != VG_INVALID_HANDLE) {
            vgDestroyPath(mVgPath);
            mVgPath = VG_INVALID_HANDLE;
        }
    }

    if (mVgPath == VG_INVALID_HANDLE) {
        switch (pathDataType) {
        case NvgEightBitEncoding: {
            mVgPath = vgCreatePath(VG_PATH_FORMAT_STANDARD,
                                   VG_PATH_DATATYPE_S_16, 1.0f / 2.0f, 0.0f, 0, 0,
                                   VG_PATH_CAPABILITY_APPEND_TO);
            break;
        }

        case NvgSixteenBitEncoding: {
            mVgPath = vgCreatePath(VG_PATH_FORMAT_STANDARD,
                                   VG_PATH_DATATYPE_S_16, 1.0f / 16.0f, 0.0f, 0, 0,
                                   VG_PATH_CAPABILITY_APPEND_TO);
            break;
        }

        case NvgThirtyTwoBitEncoding: {
            mVgPath = vgCreatePath(VG_PATH_FORMAT_STANDARD,
                                   VG_PATH_DATATYPE_S_32, 1.0f / 65536.0f, 0.0f, 0, 0,
                                   VG_PATH_CAPABILITY_APPEND_TO);
            break;
        }

        default: {
            return HbNvgEngine::NvgErrCorrupt;
        }
        }
    }

    if (mVgPath == VG_INVALID_HANDLE) {
        // Get the Symbian error code
        error = openVgErrorToHbNvgError(vgGetError());

        if (error == HbNvgEngine::NvgErrNoMemory) {
            NVG_DEBUGP1("NVG Error OOMs: vgCreatePaint");
            resetNvgState();
        }

        NVG_DEBUGP1("NVG Error: vgCreatePaint");
        return error;
    }

    mLastPathDataType   = pathDataType;

    return error;
}

void HbNvgCsIcon::resetNvgState()
{
    if (mVgPath != VG_INVALID_HANDLE) {
        vgDestroyPath(mVgPath);
        mVgPath = VG_INVALID_HANDLE;
    }

    if (mPaintFill != VG_INVALID_HANDLE) {
        vgSetPaint(VG_INVALID_HANDLE, VG_FILL_PATH);
        vgDestroyPaint(mPaintFill);
        mPaintFill = VG_INVALID_HANDLE;
    }

    if (mPaintStroke != VG_INVALID_HANDLE) {
        vgSetPaint(VG_INVALID_HANDLE, VG_STROKE_PATH);
        vgDestroyPaint(mPaintStroke);
        mPaintStroke = VG_INVALID_HANDLE;
    }
}

/**
 * @fn setFillPaint
 * @brief  SetPaint gradient or solid in OpenVG
 * @version
 * @param   buffer - buffer containing OpenVG data
 * @return  None
 */

void HbNvgCsIcon::setFillPaint(HbDereferencer *iconData)
{
    quint32 commonData  = iconData->derefInt32();
    quint32 paintType   = commonData & 0x07;
    quint16 specifcData = (commonData >> 16) & 0xff;

    switch (paintType) {
    case PAINT_LGRAD: {
        mGradPaintFill = mPaintFill;
       
        // gradient data, the data will be word aligned
        float* gradData = (float*)iconData->derefInt8Array(4 * sizeof(VGfloat), sizeof(float));

        vgSetParameteri(mGradPaintFill, VG_PAINT_TYPE, VG_PAINT_TYPE_LINEAR_GRADIENT);
        vgSetParameterfv(mGradPaintFill, VG_PAINT_LINEAR_GRADIENT, 4, gradData);
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_FILL_PAINT_TO_USER);

        if (specifcData & 0x1) {
            float* gradMatrix1 = (float*) iconData->derefInt8Array(6 * sizeof(VGfloat),
                                  NVG_PAINTSECTION_LINEARGRAD_TRANSFORM_OFFSET);

            float gradMatrix[9] = {gradMatrix1[0], gradMatrix1[3], 0.0f,
                                    gradMatrix1[1], gradMatrix1[4], 0.0f,
                                    gradMatrix1[2], gradMatrix1[5], 1.0f
                                   };

            vgLoadMatrix(gradMatrix);
            Q_UNUSED(identityMatrix);
        } else {
            vgLoadIdentity();
        }
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_PATH_USER_TO_SURFACE);
    }
    break;
    case PAINT_RGRAD: {
        mGradPaintFill = mPaintFill;
        
        // gradient data, the data will be word aligned
        float* gradData = (float*)iconData->derefInt8Array(4 * sizeof(VGfloat), sizeof(quint32));

        vgSetParameteri(mGradPaintFill, VG_PAINT_TYPE, VG_PAINT_TYPE_RADIAL_GRADIENT);
        vgSetParameterfv(mGradPaintFill, VG_PAINT_RADIAL_GRADIENT, 5, gradData);
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_FILL_PAINT_TO_USER);


        if (specifcData & 0x1) {
            float* gradMatrix1 = (float*)iconData->derefInt8Array(6 * sizeof(VGfloat),
                                  NVG_PAINTSECTION_RADIALGRAD_TRANSFORM_OFFSET);
            float gradMatrix[9] = {gradMatrix1[0], gradMatrix1[3], 0.0f,
                                    gradMatrix1[1], gradMatrix1[4], 0.0f,
                                    gradMatrix1[2], gradMatrix1[5], 1.0f
                                   };

            vgLoadMatrix(gradMatrix);
            Q_UNUSED(identityMatrix);
        } else {
            vgLoadIdentity();
        }
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_PATH_USER_TO_SURFACE);
    }
    break;
    case PAINT_FLAT: {
        quint32 rgba = iconData->derefInt32(NVG_RGBA_OFS);

        rgba = (rgba & 0xffffff00) | mFillAlpha;

        vgSetParameteri(mPaintFill, VG_PAINT_TYPE, VG_PAINT_TYPE_COLOR);
        vgSetColor(mPaintFill, rgba);
    }
    break;
    default: {
        throw HbNvgException(HbNvgEngine::NvgErrCorrupt);
    }
    }
}

void HbNvgCsIcon::setColorRamp(HbDereferencer *iconData)
{
    quint32 commonData = iconData->derefInt32();

    int stopCount = (commonData >> 16) & 0x00ff;
    float* stopData = (float*) iconData->derefInt8Array(stopCount * 5 * sizeof(float), sizeof(quint32));

    VGfloat *crs = new VGfloat[stopCount * 5];
    Q_CHECK_PTR(crs);
    QScopedArrayPointer<VGfloat> colorRamps(crs);

    if (mFillAlpha == 0xff) {
        vgSetParameteri(mGradPaintFill, VG_PAINT_COLOR_RAMP_SPREAD_MODE, VG_COLOR_RAMP_SPREAD_PAD);
        vgSetParameterfv(mGradPaintFill, VG_PAINT_COLOR_RAMP_STOPS, stopCount*5, stopData);
    } else {
        // Copy color ramps and modify alpha
        memcpy(colorRamps.data(), stopData, stopCount*5*sizeof(VGfloat));
        VGfloat alphaInFloat = mFillAlpha * 0.003921568627450f; //(1.0f/255.0f);
        VGfloat* alphaValue = &(colorRamps[4]);
        for (int i = 0; i < stopCount; i++) {
            *alphaValue *= alphaInFloat;
            alphaValue += 5;
        }

        vgSetParameteri(mGradPaintFill, VG_PAINT_COLOR_RAMP_SPREAD_MODE, VG_COLOR_RAMP_SPREAD_PAD);
        vgSetParameterfv(mGradPaintFill, VG_PAINT_COLOR_RAMP_STOPS, stopCount * 5, colorRamps.data());
    }
}

void HbNvgCsIcon::drawPath(HbDereferencer * iconData)
{
    qint32 numSegments = iconData->derefInt16();
    const VGubyte * pathSegments  = iconData->derefInt8Array(numSegments, sizeof(quint16));
    /*
    * verify that what we got is proper data
    * for that calculate the path co-ordinate length
    * and check that the path data does not overflow
    */
    qint32 coordinateCount = 0;
    for (qint32 i = 0; i < numSegments; ++i) {
        switch (pathSegments[i]) {
        case VG_HLINE_TO:
        case VG_VLINE_TO:
            coordinateCount += 1;
            break;
        case VG_MOVE_TO:
        case VG_LINE_TO:
        case VG_SQUAD_TO:
            coordinateCount += 2;
            break;
        case VG_QUAD_TO:
        case VG_SCUBIC_TO:
            coordinateCount += 4;
            break;
        case VG_SCCWARC_TO:
        case VG_SCWARC_TO:
        case VG_LCCWARC_TO:
        case VG_LCWARC_TO:
            coordinateCount += 5;
            break;
        case VG_CUBIC_TO:
            coordinateCount += 6;
            break;
        default:
            break;
        }
    }

    // this one is just to check the alignment
    quint8* pathData = iconData->derefInt8Array(sizeof(float), sizeof(quint16) + numSegments);

    /*
     * path data need to be word aligned
     * alignment are done according to the path format
     */
    quint32 sizeofpathdata = sizeof(float);
    quint32 alignSkip = 0;
    quint8 * alignedPtr = 0;
    if (mLastPathDataType == NvgSixteenBitEncoding) {
        alignedPtr = Align2(pathData);
        sizeofpathdata = sizeof(quint16);
    } else if (mLastPathDataType == NvgThirtyTwoBitEncoding) {
        alignedPtr = Align4(pathData);
    } else {
        throw HbNvgException(HbNvgEngine::NvgErrCorrupt);
    }

    alignSkip = alignedPtr - pathData;

    /*
     * check to see whether we have enough path data
     */
    iconData->assertBound(coordinateCount * sizeofpathdata + alignSkip, sizeof(quint16) + numSegments);

    pathData = alignedPtr;

    VGint paintMode = (mDoFill ? VG_FILL_PATH : 0) | (mDoStroke ? VG_STROKE_PATH : 0);
    if (paintMode == 0) {
        paintMode = VG_FILL_PATH;
    }

    vgClearPath(mVgPath, VG_PATH_CAPABILITY_APPEND_TO);

    vgAppendPathData(mVgPath, numSegments, pathSegments, pathData);
    vgDrawPath(mVgPath, paintMode);

    mDoStroke   = VG_FALSE;
    mDoFill     = VG_FALSE;
}

void HbNvgCsIcon::setTransform(HbDereferencer * iconData, quint32 & counter, const VGfloat* currentMatrix)
{
    vgLoadMatrix(currentMatrix);

    quint32 commonData =  iconData->derefInt32();
    quint32 transformType = (commonData & 0x00ff0000) >> 16 ;

    VGfloat matrixTemp[9] = {
        1.0f, 0.0f, 0.0f,
        0.0f, 1.0f, 0.0f,
        0.0f, 0.0f, 1.0f
    };

    counter = 0;

    if (transformType != 1) {
        if (transformType == TRANSFORM_COMPLETE) {
            matrixTemp[0] = iconData->derefReal32((++counter) * sizeof(VGfloat));
            matrixTemp[4] = iconData->derefReal32((++counter) * sizeof(VGfloat));
            matrixTemp[3] = iconData->derefReal32((++counter) * sizeof(VGfloat));
            matrixTemp[1] = iconData->derefReal32((++counter) * sizeof(VGfloat));
            matrixTemp[6] = iconData->derefReal32((++counter) * sizeof(VGfloat));
            matrixTemp[7] = iconData->derefReal32((++counter) * sizeof(VGfloat));
        } else {
            if (transformType & TRANSFORM_ROTATION) {
                //vgScale
                matrixTemp[0] = iconData->derefReal32((++counter) * sizeof(VGfloat));
                matrixTemp[4] = iconData->derefReal32((++counter) * sizeof(VGfloat));

                //vgShear
                matrixTemp[3] = iconData->derefReal32((++counter) * sizeof(VGfloat));
                matrixTemp[1] = iconData->derefReal32((++counter) * sizeof(VGfloat));
            } else {
                if (transformType & TRANSFORM_SCALING) {
                    //vgScale
                    matrixTemp[0] = iconData->derefReal32((++counter) * sizeof(VGfloat));
                    matrixTemp[4] = iconData->derefReal32((++counter) * sizeof(VGfloat));
                }

                if (transformType & TRANSFORM_SHEARING) {
                    //vgShear
                    matrixTemp[3] = iconData->derefReal32((++counter) * sizeof(VGfloat));
                    matrixTemp[1] = iconData->derefReal32((++counter) * sizeof(VGfloat));;
                }
            }

            if (transformType & TRANSFORM_TRANSLATION) {
                //vgTranslate
                matrixTemp[6] = iconData->derefReal32((++counter) * sizeof(VGfloat));;
                matrixTemp[7] = iconData->derefReal32((++counter) * sizeof(VGfloat));;
            }
        }

        vgMultMatrix(matrixTemp);
    } 
}

void HbNvgCsIcon::setStrokePaint(HbDereferencer * iconData)
{
    quint32 commonData = iconData->derefInt32();
    quint32 strokeType = commonData & 0x07;
    quint16 specifcData = (commonData >> 16) & 0xff;

    switch (strokeType) {
    case STROKE_LGRAD: {
        mGradPaintStroke = mPaintStroke;

        // gradient data, the data will be word aligned
        float* gradData = (float*)iconData->derefInt8Array(4 * sizeof(VGfloat), sizeof(float));

        vgSetParameteri(mGradPaintStroke, VG_PAINT_TYPE, VG_PAINT_TYPE_LINEAR_GRADIENT);
        vgSetParameterfv(mGradPaintStroke, VG_PAINT_LINEAR_GRADIENT, 4, gradData);
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_STROKE_PAINT_TO_USER);

        if (specifcData & 0x1) {
            float* gradMatrix1 = (float*)iconData->derefInt8Array(6 * sizeof(VGfloat),
                                  4 + 4 * sizeof(VGfloat));

            float gradMatrix[9] = {gradMatrix1[0], gradMatrix1[3], 0.0f,
                                    gradMatrix1[1], gradMatrix1[4], 0.0f,
                                    gradMatrix1[2], gradMatrix1[5], 1.0f
                                   };

            vgLoadMatrix(gradMatrix);
            Q_UNUSED(identityMatrix);
        } else {
            vgLoadIdentity();
        }
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_PATH_USER_TO_SURFACE);
    }
    break;
    case STROKE_RGRAD: {
        mGradPaintStroke = mPaintStroke;

        // gradient data, the data will be word aligned
        float* gradData = (float*)iconData->derefInt8Array(5 * sizeof(VGfloat), sizeof(quint32));

        vgSetParameteri(mGradPaintStroke, VG_PAINT_TYPE, VG_PAINT_TYPE_RADIAL_GRADIENT);
        vgSetParameterfv(mGradPaintStroke, VG_PAINT_RADIAL_GRADIENT, 5, gradData);
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_STROKE_PAINT_TO_USER);

        if (specifcData & 0x1) {
            float* gradMatrix1 = (float*)iconData->derefInt8Array(6 * sizeof(VGfloat),
                                  4 + 5 * sizeof(VGfloat));
            float gradMatrix[9] = {gradMatrix1[0], gradMatrix1[3], 0.0f,
                                    gradMatrix1[1], gradMatrix1[4], 0.0f,
                                    gradMatrix1[2], gradMatrix1[5], 1.0f
                                   };

            vgLoadMatrix(gradMatrix);
            Q_UNUSED(identityMatrix);
        } else {
            vgLoadIdentity();
        }
        vgSeti(VG_MATRIX_MODE, VG_MATRIX_PATH_USER_TO_SURFACE);
    }
    break;

    case STROKE_COLOR_RAMP: {
        qint32 stopCount = specifcData;
        float* stopData = (float*) iconData->derefInt8Array(stopCount * 5 * sizeof(VGfloat), 4);

        if (mStrokeAlpha == 0xff) {
            vgSetParameteri(mGradPaintStroke, VG_PAINT_COLOR_RAMP_SPREAD_MODE, VG_COLOR_RAMP_SPREAD_PAD);
            vgSetParameterfv(mGradPaintStroke, VG_PAINT_COLOR_RAMP_STOPS, stopCount*5, stopData);
        } else {
            VGfloat *crs = new VGfloat[stopCount * 5];
            Q_CHECK_PTR(crs);
            QScopedArrayPointer<VGfloat> colorRamps(crs);
            // Copy color ramps and modify alpha
            memcpy(colorRamps.data(), stopData, stopCount*5*sizeof(VGfloat));
            VGfloat alphaInFloat = mStrokeAlpha * (1.0f/255.0f);
            VGfloat* alphaValue = &colorRamps[4];
            for (qint32 i = 0; i < stopCount; i++) {
                *alphaValue *= alphaInFloat;
                alphaValue += 5;
            }

            vgSetParameteri(mGradPaintStroke, VG_PAINT_COLOR_RAMP_SPREAD_MODE, VG_COLOR_RAMP_SPREAD_PAD);
            vgSetParameterfv(mGradPaintStroke, VG_PAINT_COLOR_RAMP_STOPS, stopCount*5, colorRamps.data());
        }
    }
    break;

    default: {
        quint32 rgba = iconData->derefInt32(NVG_RGBA_OFS);
        rgba = (rgba & 0xffffff00) | mStrokeAlpha; // replace alpha

        vgSetParameteri(mPaintStroke, VG_PAINT_TYPE, VG_PAINT_TYPE_COLOR);
        vgSetColor(mPaintStroke, rgba);
    }
    break;
    }
}

void HbNvgCsIcon::setRotation()
{
    if (mRotationAngle) {
        vgTranslate(mRotationX, mRotationY);
        vgRotate(mRotationAngle);
        vgTranslate(-mRotationX, -mRotationY);
    }
}