OMXFD.cpp revision 8cb48e582f4f3c93aa25159c1116a7c83bfa388a
1c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev/*
2c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * Copyright (C) Texas Instruments - http://www.ti.com/
3c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev *
4c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * Licensed under the Apache License, Version 2.0 (the "License");
5c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * you may not use this file except in compliance with the License.
6c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * You may obtain a copy of the License at
7c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev *
8c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev *      http://www.apache.org/licenses/LICENSE-2.0
9c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev *
10c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * Unless required by applicable law or agreed to in writing, software
11c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * distributed under the License is distributed on an "AS IS" BASIS,
12c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * See the License for the specific language governing permissions and
14c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev * limitations under the License.
15c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev */
16c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
17c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev/**
18c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev* @file OMXFD.cpp
19c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev*
20c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev* This file contains functionality for handling face detection.
21c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev*
22c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev*/
23c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
24c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev#undef LOG_TAG
25c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
26c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev#define LOG_TAG "CameraHAL"
27c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
28c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev#include "CameraHal.h"
29c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev#include "OMXCameraAdapter.h"
30c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
31c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchevnamespace android {
32c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
33c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchevstatus_t OMXCameraAdapter::setParametersFD(const CameraParameters &params,
34c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                           BaseCameraAdapter::AdapterState state)
35c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev{
36c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    status_t ret = NO_ERROR;
37c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
38c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME;
39c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
40c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME_EXIT;
41c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
42c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    return ret;
43c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev}
44c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
45c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchevstatus_t OMXCameraAdapter::startFaceDetection()
46c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev{
47c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    status_t ret = NO_ERROR;
48c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
49c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    Mutex::Autolock lock(mFaceDetectionLock);
50c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
51c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    ret = setFaceDetection(true, mDeviceOrientation);
52c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    if (ret != NO_ERROR) {
53c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman        goto out;
54c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    }
55c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
56c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    // Set 3A modes to face priority
57c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    ret = setExposureMode(mParameters3A);
58c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    if (ret != NO_ERROR) {
59c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman        goto out;
60c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    }
61c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
62c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
638cb48e582f4f3c93aa25159c1116a7c83bfa388aSundar Raman    //Note: White balance will not be face prioritized, since
648cb48e582f4f3c93aa25159c1116a7c83bfa388aSundar Raman    //the algorithm needs full frame statistics, and not face
658cb48e582f4f3c93aa25159c1116a7c83bfa388aSundar Raman    //regions alone.
668cb48e582f4f3c93aa25159c1116a7c83bfa388aSundar Raman
67c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
68c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman out:
69c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    return ret;
70c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev}
71c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
72c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchevstatus_t OMXCameraAdapter::stopFaceDetection()
73c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev{
74c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    status_t ret = NO_ERROR;
75c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    const char *str = NULL;
76c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    BaseCameraAdapter::AdapterState state;
77c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    BaseCameraAdapter::getState(state);
78c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
79c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    Mutex::Autolock lock(mFaceDetectionLock);
80c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
81c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    ret = setFaceDetection(false, mDeviceOrientation);
82c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    if (ret != NO_ERROR) {
83c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman        goto out;
84c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    }
85c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
86c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    // Reset 3A settings
87c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    ret = setParameters3A(mParams, state);
88c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    if (ret != NO_ERROR) {
89c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman        goto out;
90c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    }
91c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
92c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    if (mPending3Asettings) {
93c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman        apply3Asettings(mParameters3A);
94c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    }
95c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman
96c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman out:
97c0fc811b503ef0c0ccd024f77f9ac7d6972f77c8Sundar Raman    return ret;
98c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev}
99c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
100aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luuvoid OMXCameraAdapter::pauseFaceDetection(bool pause)
101aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu{
102aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu    Mutex::Autolock lock(mFaceDetectionLock);
103aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu    // pausing will only take affect if fd is already running
104aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu    if (mFaceDetectionRunning) {
105aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu        mFaceDetectionPaused = pause;
106aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu    }
107aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu}
108aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu
109c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchevstatus_t OMXCameraAdapter::setFaceDetection(bool enable, OMX_U32 orientation)
110c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev{
111c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    status_t ret = NO_ERROR;
112c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_ERRORTYPE eError = OMX_ErrorNone;
113c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_CONFIG_EXTRADATATYPE extraDataControl;
114c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_CONFIG_OBJDETECTIONTYPE objDetection;
115c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
116c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME;
117c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
118c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( OMX_StateInvalid == mComponentState )
119c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        {
120c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEA("OMX component is in invalid state");
121c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        ret = -EINVAL;
122c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
123c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
124c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NO_ERROR == ret )
125c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        {
126c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if ( orientation < 0 || orientation > 270 ) {
127c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            orientation = 0;
128c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
129c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
130c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        OMX_INIT_STRUCT_PTR (&objDetection, OMX_CONFIG_OBJDETECTIONTYPE);
131c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        objDetection.nPortIndex = mCameraAdapterParameters.mPrevPortIndex;
132c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        objDetection.nDeviceOrientation = orientation;
133c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if  ( enable )
134c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
135c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            objDetection.bEnable = OMX_TRUE;
136c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
137c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        else
138c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
139c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            objDetection.bEnable = OMX_FALSE;
140c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
141c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
142c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        eError =  OMX_SetConfig(mCameraAdapterParameters.mHandleComp,
143c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                ( OMX_INDEXTYPE ) OMX_IndexConfigImageFaceDetection,
144c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                &objDetection);
145c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if ( OMX_ErrorNone != eError )
146c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
147c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            CAMHAL_LOGEB("Error while configuring face detection 0x%x", eError);
148c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            ret = -1;
149c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
150c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        else
151c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
152c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            CAMHAL_LOGDA("Face detection configured successfully");
153c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
154c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
155c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
156c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NO_ERROR == ret )
157c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        {
158c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        OMX_INIT_STRUCT_PTR (&extraDataControl, OMX_CONFIG_EXTRADATATYPE);
159c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        extraDataControl.nPortIndex = mCameraAdapterParameters.mPrevPortIndex;
160c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        extraDataControl.eExtraDataType = OMX_FaceDetection;
161c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        extraDataControl.eCameraView = OMX_2D;
162c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if  ( enable )
163c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
164c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            extraDataControl.bEnable = OMX_TRUE;
165c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
166c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        else
167c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
168c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            extraDataControl.bEnable = OMX_FALSE;
169c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
170c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
171c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        eError =  OMX_SetConfig(mCameraAdapterParameters.mHandleComp,
172c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                ( OMX_INDEXTYPE ) OMX_IndexConfigOtherExtraDataControl,
173c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                &extraDataControl);
174c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if ( OMX_ErrorNone != eError )
175c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
176c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            CAMHAL_LOGEB("Error while configuring face detection extra data 0x%x",
177c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         eError);
178c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            ret = -1;
179c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
180c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        else
181c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
182c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            CAMHAL_LOGDA("Face detection extra data configured successfully");
183c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
184c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
185c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
186c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NO_ERROR == ret )
187c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        {
188c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        mFaceDetectionRunning = enable;
189aa6e62e279cb54ae76c55ba9f8d02da230ce34e7Tyler Luu        mFaceDetectionPaused = !enable;
190c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
191c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
192c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME_EXIT;
193c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
194c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    return ret;
195c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev}
196c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
197c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchevstatus_t OMXCameraAdapter::detectFaces(OMX_BUFFERHEADERTYPE* pBuffHeader,
198c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                       sp<CameraFDResult> &result,
199c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                       size_t previewWidth,
200c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                       size_t previewHeight)
201c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev{
202c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    status_t ret = NO_ERROR;
203c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_ERRORTYPE eError = OMX_ErrorNone;
204c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_TI_FACERESULT *faceResult;
205c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_OTHER_EXTRADATATYPE *extraData;
206c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_FACEDETECTIONTYPE *faceData;
207c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    OMX_TI_PLATFORMPRIVATE *platformPrivate;
208c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    camera_frame_metadata_t *faces;
209c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
210c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME;
211c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
212c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( OMX_StateExecuting != mComponentState ) {
213c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEA("OMX component is not in executing state");
214c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return NO_INIT;
215c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
216c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
217c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NULL == pBuffHeader ) {
218c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEA("Invalid Buffer header");
219c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return-EINVAL;
220c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
221c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
222c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    platformPrivate = (OMX_TI_PLATFORMPRIVATE *) (pBuffHeader->pPlatformPrivate);
223c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NULL != platformPrivate ) {
224c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if ( sizeof(OMX_TI_PLATFORMPRIVATE) == platformPrivate->nSize ) {
225c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            CAMHAL_LOGVB("Size = %d, sizeof = %d, pAuxBuf = 0x%x, pAuxBufSize= %d, pMetaDataBufer = 0x%x, nMetaDataSize = %d",
226c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         platformPrivate->nSize,
227c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         sizeof(OMX_TI_PLATFORMPRIVATE),
228c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         platformPrivate->pAuxBuf1,
229c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         platformPrivate->pAuxBufSize1,
230c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         platformPrivate->pMetaDataBuffer,
231c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         platformPrivate->nMetaDataSize);
232c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        } else {
233c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            CAMHAL_LOGEB("OMX_TI_PLATFORMPRIVATE size mismatch: expected = %d, received = %d",
234c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         ( unsigned int ) sizeof(OMX_TI_PLATFORMPRIVATE),
235c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         ( unsigned int ) platformPrivate->nSize);
236c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            ret = -EINVAL;
237c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
238c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }  else {
239c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEA("Invalid OMX_TI_PLATFORMPRIVATE");
240c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return-EINVAL;
241c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
242c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
243c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
244c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( 0 >= platformPrivate->nMetaDataSize ) {
245c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEB("OMX_TI_PLATFORMPRIVATE nMetaDataSize is size is %d",
246c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                     ( unsigned int ) platformPrivate->nMetaDataSize);
247c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return -EINVAL;
248c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
249c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
250c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    extraData = (OMX_OTHER_EXTRADATATYPE *) (platformPrivate->pMetaDataBuffer);
251c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NULL != extraData ) {
252c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGVB("Size = %d, sizeof = %d, eType = 0x%x, nDataSize= %d, nPortIndex = 0x%x, nVersion = 0x%x",
253c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                     extraData->nSize,
254c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                     sizeof(OMX_OTHER_EXTRADATATYPE),
255c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                     extraData->eType,
256c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                     extraData->nDataSize,
257c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                     extraData->nPortIndex,
258c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                     extraData->nVersion);
259c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    } else {
260c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEA("Invalid OMX_OTHER_EXTRADATATYPE");
261c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return -EINVAL;
262c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
263c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
264c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    faceData = ( OMX_FACEDETECTIONTYPE * ) extraData->data;
265c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NULL != faceData ) {
266c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if ( sizeof(OMX_FACEDETECTIONTYPE) == faceData->nSize ) {
267c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            CAMHAL_LOGVB("Faces detected %d",
268c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         faceData->ulFaceCount,
269c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         faceData->nSize,
270c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         sizeof(OMX_FACEDETECTIONTYPE),
271c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         faceData->eCameraView,
272c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         faceData->nPortIndex,
273c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         faceData->nVersion);
274c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        } else {
275782a084e5fec691d3985b31c3ab7be1584d879b3Iliyan Malchev            CAMHAL_LOGEB("OMX_FACEDETECTIONTYPE size mismatch: expected = %d, received = %d",
276c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         ( unsigned int ) sizeof(OMX_FACEDETECTIONTYPE),
277c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                         ( unsigned int ) faceData->nSize);
278c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            return -EINVAL;
279c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
280c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    } else {
281c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEA("Invalid OMX_FACEDETECTIONTYPE");
282c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return -EINVAL;
283c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
284c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
285c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    ret = encodeFaceCoordinates(faceData, &faces, previewWidth, previewHeight);
286c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
287c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NO_ERROR == ret ) {
288c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        result = new CameraFDResult(faces);
289c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    } else {
290c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        result.clear();
291c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        result = NULL;
292c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
293c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
294c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME_EXIT;
295c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
296c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    return ret;
297c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev}
298c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
299c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchevstatus_t OMXCameraAdapter::encodeFaceCoordinates(const OMX_FACEDETECTIONTYPE *faceData,
300c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                                 camera_frame_metadata_t **pFaces,
301c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                                 size_t previewWidth,
302c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev                                                 size_t previewHeight)
303c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev{
304c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    status_t ret = NO_ERROR;
305c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    camera_face_t *faces;
306c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    camera_frame_metadata_t *faceResult;
307c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    size_t hRange, vRange;
308c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    double tmp;
309c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
310c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME;
311c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
312c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NULL == faceData ) {
313c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        CAMHAL_LOGEA("Invalid OMX_FACEDETECTIONTYPE parameter");
314c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return EINVAL;
315c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
316c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
317c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME
318c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
319c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    hRange = CameraFDResult::RIGHT - CameraFDResult::LEFT;
320c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    vRange = CameraFDResult::BOTTOM - CameraFDResult::TOP;
321c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
322c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    faceResult = ( camera_frame_metadata_t * ) malloc(sizeof(camera_frame_metadata_t));
323c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( NULL == faceResult ) {
324c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        return -ENOMEM;
325c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
326c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
327c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    if ( 0 < faceData->ulFaceCount ) {
328fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        int orient_mult;
329fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        int trans_left, trans_top, trans_right, trans_bot;
330c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
331c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        faces = ( camera_face_t * ) malloc(sizeof(camera_face_t)*faceData->ulFaceCount);
332c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        if ( NULL == faces ) {
333c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            return -ENOMEM;
334c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        }
335c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
336fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        /**
337fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * When device is 180 degrees oriented to the sensor, need to translate
338fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * the output from Ducati to what Android expects
339fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * Ducati always gives face coordinates in this form, irrespective of
340fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * rotation, i.e (l,t) always represents the point towards the left eye
341fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * and top of hair.
342fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * (l, t)
343fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   ---------------
344fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -   ,,,,,,,   -
345fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -  |       |  -
346fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -  |<a   <a|  -
347fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   - (|   ^   |) -
348fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -  |  -=-  |  -
349fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -   \_____/   -
350fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   ---------------
351fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *               (r, b)
352fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *
353fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * However, Android expects the coords to be in respect with what the
354fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * sensor is viewing, i.e Android expects sensor to see this with (l,t)
355fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * and (r,b) like so:
356fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / * (l, t)
357fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   ---------------
358fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -    _____    -
359fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -   /     \   -
360fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -  |  -=-  |  -
361fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   - (|   ^   |) -
362fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -  |a>   a>|  -
363fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -  |       |  -
364fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   -   ,,,,,,,   -
365fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *   ---------------
366fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        / *               (r, b)
367fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu          */
368fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        if (mDeviceOrientation == 180) {
369fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            orient_mult = -1;
370fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_left = 2; // right is now left
371fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_top = 3; // bottom is now top
372fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_right = 0; // left is now right
373fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_bot = 1; // top is not bottom
374fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        } else {
375fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            orient_mult = 1;
376fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_left = 0; // left
377fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_top = 1; // top
378fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_right = 2; // right
379fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            trans_bot = 3; // bottom
380fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu
381fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu        }
382c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        for ( int i = 0  ; i < faceData->ulFaceCount ; i++)
383c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            {
384c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
385c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp = ( double ) faceData->tFacePosition[i].nLeft / ( double ) previewWidth;
386c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp *= hRange;
387c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp -= hRange/2;
388fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            faces[i].rect[trans_left] = tmp;
389c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
390c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp = ( double ) faceData->tFacePosition[i].nTop / ( double )previewHeight;
391c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp *= vRange;
392c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp -= vRange/2;
393fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            faces[i].rect[trans_top] = tmp;
394c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
395c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp = ( double ) faceData->tFacePosition[i].nWidth / ( double ) previewWidth;
396c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp *= hRange;
397fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            tmp *= orient_mult;
398fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            faces[i].rect[trans_right] = faces[i].rect[trans_left] + tmp;
399c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
400c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp = ( double ) faceData->tFacePosition[i].nHeight / ( double ) previewHeight;
401c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            tmp *= vRange;
402fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            tmp *= orient_mult;
403fa28dae0efd774ddd8919e99c0add3e69300ea3aTyler Luu            faces[i].rect[trans_bot] = faces[i].rect[trans_top] + tmp;
404c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
405c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].score = faceData->tFacePosition[i].nScore;
406c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].id = 0;
407c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].left_eye[0] = CameraFDResult::INVALID_DATA;
408c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].left_eye[1] = CameraFDResult::INVALID_DATA;
409c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].right_eye[0] = CameraFDResult::INVALID_DATA;
410c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].right_eye[1] = CameraFDResult::INVALID_DATA;
411c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].mouth[0] = CameraFDResult::INVALID_DATA;
412c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            faces[i].mouth[1] = CameraFDResult::INVALID_DATA;
413c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev            }
414c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
415c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        faceResult->number_of_faces = faceData->ulFaceCount;
416c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        faceResult->faces = faces;
417c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
418c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    } else {
419c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        faceResult->number_of_faces = 0;
420c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev        faceResult->faces = NULL;
421c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    }
422c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
423c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    *pFaces = faceResult;
424c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
425c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    LOG_FUNCTION_NAME_EXIT;
426c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
427c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev    return ret;
428c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev}
429c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev
430c322989ae6ff6769490828de1b5eda12b749cce9Iliyan Malchev};
431