/******************************************************************************** * FaceTrackNoIR This program is a private project of the some enthusiastic * * gamers from Holland, who don't like to pay much for * * head-tracking. * * * * Copyright (C) 2011 Wim Vriend (Developing) * * Ron Hendriks (Researching and Testing) * * * * Homepage * * * * This program is free software; you can redistribute it and/or modify it * * under the terms of the GNU General Public License as published by the * * Free Software Foundation; either version 3 of the License, or (at your * * option) any later version. * * * * This program is distributed in the hope that it will be useful, but * * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * * more details. * * * * You should have received a copy of the GNU General Public License along * * with this program; if not, see . * *********************************************************************************/ /* Modifications (last one on top): 20110501 - WVR: Added some command to be handled from FaceTrackNoIR (settings dialog). 20110322 - WVR: Somehow the video-widget of faceAPI version 3.2.6. does not work with FaceTrackNoIR (Qt issue?!). To be able to use release 3.2.6 of faceAPI anyway, this console-app is used. It exchanges data with FaceTrackNoIR via shared-memory... */ //Precompiled header #include "stdafx.h" //FaceAPI headers #include "sm_api.h" #include "ftnoir_tracker_sm_types.h" #include "utils.h" //local headers #include "build_options.h" //namespaces using namespace std; using namespace sm::faceapi::samplecode; // // global variables // HANDLE hSMMemMap = NULL; SMMemMap *pMemData; HANDLE hSMMutex; smEngineHeadPoseData new_head_pose; bool stopCommand = false; bool ftnoirConnected = false; //enums enum GROUP_ID { GROUP0=0, }; enum EVENT_ID { EVENT_PING=0, EVENT_INIT, }; enum INPUT_ID { INPUT0=0, }; //function definitions void updateHeadPose(smEngineHeadPoseData* temp_head_pose); bool SMCreateMapping(); ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// // //FaceAPI function implementations // ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void STDCALL receiveLogMessage(void *, const char *buf, int /*buf_len*/) { Lock lock(g_mutex); // serialize logging calls from different threads to avoid garbled output. //cout << string(buf); } // Callback function for face-data void STDCALL receiveFaceData(void *, smEngineFaceData face_data, smCameraVideoFrame video_frame) { Lock lock(g_mutex); // Get info including data pointer to original image from camera smImageInfo video_frame_image_info; THROW_ON_ERROR(smImageGetInfo(video_frame.image_handle, &video_frame_image_info)); // reentrant, so ok // video_frame_image_info.plane_addr[*] now point to the image memory planes. // The memory is only valid until the end of this routine unless you call smImageAddRef(video_frame.image_handle). // So you can deep copy the image data here, or use smImageAddRef() and just copy the pointer. // If you use smImageAddRef() you are responsible for calling smImageDestroy() to avoid a memory leak later. // In this callback you will typically want to copy the smEngineFaceData data into your own data-structure. // Since the smEngineFaceData contains multiple pod types copying it is not atomic and // a mutex is required to avoid the race-condition with any thread simultaneously // reading from your data-structure. // Such a race condition will not crash your code but will create weird noise in the tracking data. if (g_do_face_data_printing) { //cout << video_frame << " " << face_data; // Save any face texture to a PNG file if (face_data.texture) { // Create a unique filename std::stringstream filename; filename << "face_" << video_frame.frame_num << ".png"; // Try saving to a file if (saveToPNGFile(filename.str(), face_data.texture->image_info) == SM_API_OK) { cout << "Saved face-texture to " << filename.str() << std::endl; } else { cout << "Error saving face-texture to " << filename.str() << std::endl; } } } } // Callback function for head-pose void STDCALL receiveHeadPose(void *,smEngineHeadPoseData head_pose, smCameraVideoFrame video_frame) { Lock lock(g_mutex); // Get info including data pointer to original image from camera smImageInfo video_frame_image_info; THROW_ON_ERROR(smImageGetInfo(video_frame.image_handle, &video_frame_image_info)); // reentrant, so ok // video_frame_image_info.plane_addr[*] now point to the image memory planes. // The memory is only valid until the end of this routine unless you call smImageAddRef(video_frame.image_handle). // So you can deep copy the image data here, or use smImageAddRef() and just copy the pointer. // If you use smImageAddRef() you are responsible for calling smImageDestroy() to avoid a memory leak later. // In this callback you will typically want to copy the smEngineFaceData data into your own data-structure. // Since the smEngineFaceData contains multiple pod types copying it is not atomic and // a mutex is required to avoid the race-condition with any thread simultaneously // reading from your data-structure. // Such a race condition will not crash your code but will create weird noise in the tracking data. if (g_do_head_pose_printing) { //cout << video_frame << " " << head_pose << std::endl; } //make a copy of the new head pose data and send it to simconnect //when we get a simmconnect frame event the new offset will be applied to the camera updateHeadPose(&head_pose); } // Create the first available camera detected on the system, and return its handle smCameraHandle createFirstCamera() { // Detect cameras smCameraInfoList info_list; THROW_ON_ERROR(smCameraCreateInfoList(&info_list)); if (info_list.num_cameras == 0) { throw runtime_error("No cameras were detected"); } else { cout << "The followings cameras were detected: " << endl; for (int i=0; ihandle,TRUE)); THROW_ON_ERROR(smHTV2SetHeadPoseFilterLevel(engine_handle, pMemData->initial_filter_level)); } else { THROW_ON_ERROR(smVideoDisplayCreate(engine_handle,&video_display_handle,0,TRUE)); } // Setup the VideoDisplay THROW_ON_ERROR(smVideoDisplaySetFlags(video_display_handle,g_overlay_flags)); // Get the handle to the window and change the title to "Hello World" smWindowHandle win_handle = 0; THROW_ON_ERROR(smVideoDisplayGetWindowHandle(video_display_handle,&win_handle)); SetWindowText(win_handle, _T("faceAPI Video-widget")); MoveWindow(win_handle, 0, 0, 250, 180, true); // Loop on the keyboard while (processKeyPress(engine_handle, video_display_handle) && !stopCommand) { // Read and print the current head-pose (if not using the callback mechanism) #if (USE_HEADPOSE_CALLBACK==0) #pragma message("Polling Headpose Manually") if (engine_licensed) { smEngineHeadPoseData head_pose; Lock lock(g_mutex); THROW_ON_ERROR(smHTCurrentHeadPose(engine_handle,&head_pose)); if (g_do_head_pose_printing) { std::cout << head_pose << std::endl; } } #endif // NOTE: If you have a windows event loop in your program you // will not need to call smAPIProcessEvents(). This manually redraws the video window. THROW_ON_ERROR(smAPIProcessEvents()); // Prevent CPU overload in our simple loop. const int frame_period_ms = 10; Sleep(frame_period_ms); // // Process the command sent by FaceTrackNoIR. // if (ftnoirConnected && (pMemData != 0)) { sprintf_s(msg, "Command: %d, \n", pMemData->command, pMemData->par_val_int); OutputDebugStringA(msg); std::cout << msg; // // // Determine the trackers' state and send it to FaceTrackNoIR. // THROW_ON_ERROR(smEngineGetState(engine_handle, &state)); pMemData->state = state; // // Check if a command was issued and do something with it! // switch (pMemData->command) { case FT_SM_START: // // Only execute Start, if the engine is not yet tracking. // if (state != SM_API_ENGINE_STATE_HT_TRACKING) { THROW_ON_ERROR(smEngineStart(engine_handle)); // Start tracking } pMemData->command = 0; // Reset break; case FT_SM_STOP: THROW_ON_ERROR(smEngineStop(engine_handle)); // Stop tracking pMemData->command = 0; // Reset break; case FT_SM_EXIT: THROW_ON_ERROR(smEngineStop(engine_handle)); // Stop tracking stopCommand = TRUE; pMemData->command = 0; // Reset break; case FT_SM_SET_PAR_FILTER: THROW_ON_ERROR(smHTV2SetHeadPoseFilterLevel(engine_handle, pMemData->par_val_int)); pMemData->command = 0; // Reset break; case FT_SM_SHOW_CAM: THROW_ON_ERROR(smEngineShowCameraControlPanel(engine_handle)); pMemData->command = 0; // Reset break; default: pMemData->command = 0; // Reset // should never be reached break; } } } // While(1) // Destroy engine THROW_ON_ERROR(smEngineDestroy(&engine_handle)); // Destroy video display THROW_ON_ERROR(smVideoDisplayDestroy(&video_display_handle)); if (ftnoirConnected) { if ( pMemData != NULL ) { UnmapViewOfFile ( pMemData ); } if (hSMMutex != 0) { CloseHandle( hSMMutex ); } hSMMutex = 0; if (hSMMemMap != 0) { CloseHandle( hSMMemMap ); } hSMMemMap = 0; } } // run() // Application entry point int _tmain(int /*argc*/, _TCHAR** /*argv*/) { OutputDebugString(_T("_tmain() says: Starting Function\n")); try { if (SMCreateMapping()) { run(); } } catch (exception &e) { cerr << e.what() << endl; } return smAPIQuit(); } // // This is called exactly once for each FaceAPI callback and must be within an exclusive lock // void updateHeadPose(smEngineHeadPoseData* temp_head_pose) { // // Check if the pointer is OK and wait for the Mutex. // if ( (pMemData != NULL) && (WaitForSingleObject(hSMMutex, 100) == WAIT_OBJECT_0) ) { // // Copy the Raw measurements directly to the client. // if (temp_head_pose->confidence > 0.0f) { memcpy(&pMemData->data.new_pose,temp_head_pose,sizeof(smEngineHeadPoseData)); } ReleaseMutex(hSMMutex); } }; // // Create a memory-mapping to the faceAPI data. // It contains the tracking data, a command-code from FaceTrackNoIR // // bool SMCreateMapping() { OutputDebugString(_T("FTCreateMapping says: Starting Function\n")); // // A FileMapping is used to create 'shared memory' between the faceAPI and FaceTrackNoIR. // FaceTrackNoIR creates the mapping, this program only opens it. // If it's not there: the program was apparently started by the user instead of FaceTrackNoIR... // // Open an existing FileMapping, Read/Write access // hSMMemMap = OpenFileMappingA( FILE_MAP_ALL_ACCESS , false , (LPCSTR) SM_MM_DATA ); if ( ( hSMMemMap != 0 ) ) { ftnoirConnected = true; OutputDebugString(_T("FTCreateMapping says: FileMapping opened successfully...\n")); pMemData = (SMMemMap *) MapViewOfFile(hSMMemMap, FILE_MAP_ALL_ACCESS, 0, 0, sizeof(TFaceData)); if (pMemData != NULL) { OutputDebugString(_T("FTCreateMapping says: MapViewOfFile OK.\n")); pMemData->state = 0; } hSMMutex = CreateMutexA(NULL, false, SM_MUTEX); } else { OutputDebugString(_T("FTCreateMapping says: FileMapping not opened...FaceTrackNoIR not connected!\n")); ftnoirConnected = false; pMemData = 0; } return true; }