other:visual3dserver:documentation:c_client_library
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revision | |||
other:visual3dserver:documentation:c_client_library [2024/07/17 15:43] – removed sgranger | other:visual3dserver:documentation:c_client_library [2024/07/17 15:44] (current) – created sgranger | ||
---|---|---|---|
Line 1: | Line 1: | ||
+ | ====== C Client Library ====== | ||
+ | |||
+ | [[# | ||
+ | The most important functions are: | ||
+ | |||
+ | '' | ||
+ | '' | ||
+ | '' | ||
+ | '' | ||
+ | '' | ||
+ | The remaining functions are mostly for examining and adjusting the internal state of V3DSClientLib. | ||
+ | |||
+ | C and C++ clients have the option of polling for data or being pushed data through an installed C callback function. Matlab and Python clients must poll for data currently. | ||
+ | |||
+ | [[# | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | ==== V3DSClientLib ==== | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_init(...) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | Recommend default values are: | ||
+ | '' | ||
+ | - TCP timeout: 5000 ms | ||
+ | - TCP buffer size: 16383 B | ||
+ | - UDP buffer size: 32767 B | ||
+ | - thread sleep: 1 ms | ||
+ | - thread priority: " | ||
+ | Thread priority can be one of " | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_kill() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getTcpTimeout() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getTcpBufferSize() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | Importnat: the TCP buffer must be large enough to hold the longest possible message. 16384 bytes is the recommended minimum size. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getUdpBufferSize() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | Important: the UDP buffer must be large enough to hold the longest possible data results. 32676 bytes is the recommended minimum size. | ||
+ | Note that the data results length is typically proportional to the length of the pipeline script set in the client. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getPriority() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getSleep() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | A small sleep value makes the data acquisition loop run faster, minimizing the UDP packet size and the possibly of missing frames from Visaul3DServer. However, less CPU time will be available to other programs. | ||
+ | A larger sleep value makes the data acquisition loop run slower, increasing the UDP packet size and the possibly of missing frames from Visual3DServer. However, more CPU time will be available to other programs. | ||
+ | In general, a lower sleep value is appropriate, | ||
+ | Thread sleep is set when the client calls '' | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_connect(...) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | The port number for Visual3DServer is 12099. | ||
+ | Here are examples for the descriptive client strings: | ||
+ | Client name: " | ||
+ | Client version: " | ||
+ | Client date: " | ||
+ | Client environment: | ||
+ | Client protocol: " | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_isConnected() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_disconnect() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getClientName() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getClientVersion() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getClientDate() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getClientEnvironment() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getClientProtocol() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getTCPIPAddress() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getUDPPort() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_setSystems(char*) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | A client can select more than one system and also set the priority order of those systems. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_isSystemsSet() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getSystems() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_setPipeline(char*) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === v3ds_isPipelineSet === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getPipeline() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_setLinkModel(char*) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_isLinkModelSet() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getLinkModel() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_parseXML(char*, | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | If the pdeqRes argument is NOT 0, then the results are stored in the pdeqRes argument and are accessible C++ clients immediately. | ||
+ | If the pdeqRes argument is 0, then the results are stored internally and are accessed by the client using the '' | ||
+ | The client must ensure the mutex is locked before calling any '' | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_lockMutex() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | While the mutex is locked, the client can call any of the '' | ||
+ | The mutex will ONLY be locked if there is new data available when this function is called. Otherwise the mutex will NOT be locked when this function returns. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === void v3ds_unlockMutex() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getResultsCount() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getResultString(const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | |||
+ | '' | ||
+ | |||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getResultType(const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | This function might alter the status string. | ||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getResultFolder(const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | This function might alter the status string. | ||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getResultName(const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | This function might alter the status string. | ||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getResultFramesCount(const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getResultFrameValue(const int, const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | This function might alter the status string. | ||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === int v3ds_getResultFrameValuesCount(const int, const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | This function might alter the status string. | ||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === float v3ds_getResultValue(const int, const int, const int) === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | This function might alter the status string. | ||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | |||
+ | ---- | ||
+ | |||
+ | === void* v3ds_getResultsObject() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | The client must ensure the mutex is locked before calling this function. See [[# | ||
+ | . | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === void v3ds_installCallback(*pCallback, | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === bool v3ds_isCallbackInstalled() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === const char* v3ds_getStatus() === | ||
+ | |||
+ | * ****Arguments: | ||
+ | * ****Return: | ||
+ | * ****Description: | ||
+ | |||
+ | The memory for the status string is owned by the library. The client should never delete this memory. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ==== Example Client Code ==== | ||
+ | |||
+ | Example client code is provided in [[# | ||
+ | |||
+ | Note that the examples generally contain less error checking code and more global variables than a typical real-world program. This was done so that the important details of using V3DSClientLib would stand out. Ideally your clients would be more fully and robustly designed. | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === ANSI C Example Client Code === | ||
+ | |||
+ | < | ||
+ | / | ||
+ | | ANSI C ConsoleClient Example | ||
+ | | Copyright (C) 2014 C-Motion, Inc. | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | /* Copy V3DSClientLib.dll to client application folder. */ | ||
+ | /* Copy Qt5Core.dll and Qt5Network.dll to client application folder. */ | ||
+ | /* Copy Qt platform folder to client application folder. */ | ||
+ | |||
+ | /* Link to V3DSClientLib.lib to client application. */ | ||
+ | |||
+ | /* Include V3DSClientLib header */ | ||
+ | #include " | ||
+ | |||
+ | /* Global Variables */ | ||
+ | static int g_nCount | ||
+ | static bool | ||
+ | |||
+ | /* Global functions */ | ||
+ | static bool | ||
+ | static bool | ||
+ | |||
+ | / | ||
+ | | main | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | int main(int argc, char* argv[]) | ||
+ | { | ||
+ | /* Initialize V3DSClientLib. */ | ||
+ | bool bContinue = v3ds_init(5000, | ||
+ | |||
+ | if(bContinue) | ||
+ | { | ||
+ | /* Connect to Visual3DServer. */ | ||
+ | bContinue = v3ds_connect(" | ||
+ | " | ||
+ | } | ||
+ | |||
+ | /* Initialize count variables for counting results. */ | ||
+ | g_nCount = 0; | ||
+ | int nMaxCount = 5; | ||
+ | |||
+ | if(bContinue && !bPollForData) | ||
+ | { | ||
+ | /* Install callback if bPollForData is false. */ | ||
+ | v3ds_installCallback(dataCallback, | ||
+ | } | ||
+ | |||
+ | if(bContinue) | ||
+ | { | ||
+ | /* Set selected system to C3D File. */ | ||
+ | bContinue = v3ds_setSystems(" | ||
+ | } | ||
+ | |||
+ | if(bContinue) | ||
+ | { | ||
+ | /* Set pipeline script to look for LTOE marker. */ | ||
+ | char* pchPipelineScript = | ||
+ | " | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | bContinue = v3ds_setPipeline(pchPipelineScript); | ||
+ | } | ||
+ | |||
+ | /* Look for nMaxCount results in a fast loop. */ | ||
+ | while(g_nCount < nMaxCount) | ||
+ | { | ||
+ | /* Check if we are polling instead of the callback method. */ | ||
+ | if(bPollForData) | ||
+ | { | ||
+ | /* We are polling. | ||
+ | if(v3ds_lockMutex()) | ||
+ | { | ||
+ | printData(); | ||
+ | v3ds_unlockMutex(); | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | |||
+ | /* Tear down V3DSClientLib before quitting application. */ | ||
+ | v3ds_kill(); | ||
+ | |||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | / | ||
+ | | dataCallback -- invoked when bPollForData is false | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | static bool dataCallback(void* pUserData, char* pXML, int nMsgCount) | ||
+ | { | ||
+ | bool bHaveData = false; | ||
+ | |||
+ | // mutex is automatically locked while callback is invoked. | ||
+ | |||
+ | // 0 for 2nd arg means we are not a C++ client, so we need to use | ||
+ | // v3ds_get..() functions to get data in printData(). | ||
+ | if(v3ds_parseXML(pXML, | ||
+ | { | ||
+ | bHaveData = printData(); | ||
+ | } | ||
+ | |||
+ | // mutex is automatically unlocked when callback returns. | ||
+ | |||
+ | return bHaveData; | ||
+ | } | ||
+ | |||
+ | / | ||
+ | | printData -- called either when polling or using the callback method | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | static bool printData() | ||
+ | { | ||
+ | bool bHaveData = false; | ||
+ | |||
+ | int nNumRes = v3ds_getResultsCount(); | ||
+ | if(0 < nNumRes) | ||
+ | { | ||
+ | const bool kbUseStringFunc = true; | ||
+ | |||
+ | for(int r = 0; r < nNumRes; r++) | ||
+ | { | ||
+ | if(kbUseStringFunc) | ||
+ | { | ||
+ | const int knBufSize = 4096; // make it something big | ||
+ | char chBuf[knBufSize]; | ||
+ | if(0 < v3ds_getResultString(r, | ||
+ | { | ||
+ | printf(" | ||
+ | } | ||
+ | else | ||
+ | { | ||
+ | printf(" | ||
+ | } | ||
+ | } | ||
+ | else // ask for each piece of data separately. | ||
+ | { | ||
+ | printf(" | ||
+ | int nSubFrames = v3ds_getResultFramesCount(r); | ||
+ | if(0 < nSubFrames) | ||
+ | { | ||
+ | int nFrameVal = v3ds_getResultFrameValue(r, | ||
+ | printf(" | ||
+ | nSubFrames, | ||
+ | |||
+ | int nNumCompPerFrame = v3ds_getResultFrameValuesCount(r); | ||
+ | printf(" | ||
+ | |||
+ | for(int c = 0; c < nNumCompPerFrame; | ||
+ | { | ||
+ | printf(" | ||
+ | v3ds_getResultValue(r, | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | |||
+ | g_nCount++; /* main() is examining this variable in a while loop. */ | ||
+ | |||
+ | bHaveData = true; | ||
+ | } | ||
+ | |||
+ | return bHaveData; | ||
+ | } | ||
+ | |||
+ | / | ||
+ | </ | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === C++ Example Client Code === | ||
+ | |||
+ | The C++ example is very similar to the C example. | ||
+ | |||
+ | The notable difference is the optional use of a '' | ||
+ | |||
+ | When using a callback function, C++ clients can pass a deque of results to '' | ||
+ | |||
+ | When polling, C++ clients can call '' | ||
+ | |||
+ | < | ||
+ | / | ||
+ | | C++ ConsoleClient Example application | ||
+ | | Copyright (C) 2014 C-Motion, Inc. | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | // Copy V3DSClientLib.dll to client application folder. | ||
+ | // Copy Qt5Core.dll and Qt5Network.dll to client application folder. | ||
+ | // Copy Qt platform folder to client application folder. | ||
+ | |||
+ | // Link to V3DSClientLib.lib to client application. | ||
+ | |||
+ | // Include V3DSClientLib header | ||
+ | #include " | ||
+ | |||
+ | // Global Variables | ||
+ | static int g_nCount | ||
+ | static bool | ||
+ | |||
+ | // Global functions | ||
+ | static bool | ||
+ | static bool | ||
+ | |||
+ | static std:: | ||
+ | |||
+ | / | ||
+ | | main | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | int main(int argc, char* argv[]) | ||
+ | { | ||
+ | std::cout << " | ||
+ | |||
+ | // Initialize V3DSClientLib. | ||
+ | bool bContinue = v3ds_init(5000, | ||
+ | |||
+ | if(bContinue) | ||
+ | { | ||
+ | // Connect to Visual3DServer. | ||
+ | bContinue = v3ds_connect(" | ||
+ | " | ||
+ | } | ||
+ | |||
+ | // Initialize count variables for counting results. | ||
+ | g_nCount = 0; | ||
+ | int nMaxCount = 5; | ||
+ | |||
+ | if(bContinue && !bPollForData) | ||
+ | { | ||
+ | // Install callback if bPollForData is false. | ||
+ | v3ds_installCallback(dataCallback, | ||
+ | } | ||
+ | |||
+ | if(bContinue) | ||
+ | { | ||
+ | // Set selected system to C3D File. | ||
+ | bContinue = v3ds_setSystems(" | ||
+ | } | ||
+ | |||
+ | if(bContinue) | ||
+ | { | ||
+ | // Set pipeline script to look for LTOE marker. | ||
+ | char* pchPipelineScript = | ||
+ | " | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | "/ | ||
+ | bContinue = v3ds_setPipeline(pchPipelineScript); | ||
+ | } | ||
+ | |||
+ | std:: | ||
+ | |||
+ | // Look for nMaxCount results in a fast loop. | ||
+ | while(g_nCount < nMaxCount) | ||
+ | { | ||
+ | // Check if we are polling instead of the callback method. | ||
+ | if(bPollForData) | ||
+ | { | ||
+ | // We are polling. | ||
+ | if(v3ds_lockMutex()) | ||
+ | { | ||
+ | void* p = v3ds_getResultsObject(); | ||
+ | if(0 != p) | ||
+ | { | ||
+ | // Magic cast to access the CResType memory objects. | ||
+ | pdeqRes = reinterpret_cast< | ||
+ | printData(*pdeqRes); | ||
+ | |||
+ | g_nCount++; | ||
+ | } | ||
+ | v3ds_unlockMutex(); | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | |||
+ | // Tear down V3DSClientLib before quitting application. | ||
+ | v3ds_kill(); | ||
+ | |||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | / | ||
+ | | dataCallback -- invoked when bPollForData is false | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | static bool dataCallback(void* pUserData, char* pXML, int nMsgCount) | ||
+ | { | ||
+ | std::cout << " | ||
+ | |||
+ | bool bHaveData = false; | ||
+ | |||
+ | if(v3ds_parseXML(pXML, | ||
+ | { | ||
+ | bHaveData = printData(g_deqRes); | ||
+ | |||
+ | g_nCount++; // main() is examining this variable in a while loop. | ||
+ | } | ||
+ | |||
+ | return bHaveData; | ||
+ | } | ||
+ | |||
+ | / | ||
+ | | printData -- called either when polling or using the callback method | ||
+ | |******************************************************************************/ | ||
+ | |||
+ | static bool printData(const std:: | ||
+ | { | ||
+ | std::cout << " | ||
+ | |||
+ | bool bHaveData = false; | ||
+ | |||
+ | const size_t knNumRes = deqRes.size(); | ||
+ | if(0 < knNumRes) | ||
+ | { | ||
+ | const bool kbUseStringFunc = true; | ||
+ | |||
+ | std::cout << " | ||
+ | for(size_t r = 0; r < knNumRes; r++) | ||
+ | { | ||
+ | if(kbUseStringFunc) | ||
+ | { | ||
+ | std::cout << deqRes[r]; | ||
+ | } | ||
+ | else // ask for each piece of data separately. | ||
+ | { | ||
+ | std::cout << | ||
+ | "Name: " << deqRes[r].m_strName.c_str() << | ||
+ | ", Folder: " << deqRes[r].m_strFolder.c_str() << | ||
+ | ", Type: " << deqRes[r].m_strType.c_str() << std::endl; | ||
+ | |||
+ | const size_t knSubframes = deqRes[r].m_deqFrame.size(); | ||
+ | std::cout << " | ||
+ | for(size_t f = 0; f < knSubframes; | ||
+ | { | ||
+ | std::cout << " | ||
+ | std::cout << " | ||
+ | const size_t knNumComp = deqRes[r].m_deqFrame[f].m_deqfVal.size(); | ||
+ | std::cout << " | ||
+ | if(0 < knNumComp) | ||
+ | { | ||
+ | std::cout << " | ||
+ | for(size_t c = 0; c < knNumComp; c++) | ||
+ | { | ||
+ | std::cout << deqRes[r].m_deqFrame[f].m_deqfVal[c]; | ||
+ | if(c < knNumComp-1) | ||
+ | { | ||
+ | std::cout << ","; | ||
+ | } | ||
+ | } | ||
+ | std::cout << " | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | |||
+ | bHaveData = true; | ||
+ | } | ||
+ | |||
+ | return bHaveData; | ||
+ | } | ||
+ | |||
+ | / | ||
+ | </ | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === Matlab Example Client Code === | ||
+ | |||
+ | * **Requirements** Matlab R2013b | ||
+ | * **If coding TCP socket and XML parsing (not necessary using V3DSClientLib), | ||
+ | |||
+ | < | ||
+ | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
+ | % | ||
+ | % | ||
+ | % | ||
+ | % | ||
+ | % with the Visual3DServer application. | ||
+ | % | ||
+ | % This matlab file works with data/ | ||
+ | % | ||
+ | % | ||
+ | % | ||
+ | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
+ | |||
+ | % Load Visual3DServer client library, if necessary. | ||
+ | addpath(strcat(getenv(' | ||
+ | addpath(strcat(getenv(' | ||
+ | addpath(strcat(getenv(' | ||
+ | |||
+ | if not(libisloaded(' | ||
+ | loadlibrary(' | ||
+ | end | ||
+ | |||
+ | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
+ | % Script | ||
+ | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
+ | |||
+ | if libisloaded(' | ||
+ | disp(' | ||
+ | % Initial object to represent Visual3DServer (V3DS). | ||
+ | bContinue = calllib(' | ||
+ | 5, ' | ||
+ | disp(calllib(' | ||
+ | |||
+ | if bContinue | ||
+ | disp(' | ||
+ | % Connect to Visual3DServer with IP address, etc. | ||
+ | bContinue = calllib(' | ||
+ | ' | ||
+ | ' | ||
+ | disp(calllib(' | ||
+ | else | ||
+ | bContinue = calllib(' | ||
+ | unloadlibrary(' | ||
+ | return | ||
+ | end | ||
+ | |||
+ | if bContinue | ||
+ | disp(' | ||
+ | % Tell V3DS that we want data from the C3D File data source. | ||
+ | bContinue = calllib(' | ||
+ | disp(calllib(' | ||
+ | else | ||
+ | bContinue = calllib(' | ||
+ | unloadlibrary(' | ||
+ | return | ||
+ | end | ||
+ | |||
+ | if bContinue | ||
+ | disp(' | ||
+ | % Tell V3DS what pipeline script to execute on each frame of data. | ||
+ | % Pipeline script can be inline, as done here, or loaded from a file. | ||
+ | pipelineScript = [' | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | ' | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | '/ | ||
+ | bContinue = calllib(' | ||
+ | pipelineScript(1: | ||
+ | disp(calllib(' | ||
+ | else | ||
+ | bContinue = calllib(' | ||
+ | unloadlibrary(' | ||
+ | return | ||
+ | end | ||
+ | |||
+ | if bContinue | ||
+ | disp(' | ||
+ | startTime = tic; % Note the start time. | ||
+ | |||
+ | % Get pipeline command results now over and over. | ||
+ | nNumFramesRecv = 0; | ||
+ | nTotalNumFrames = 5; | ||
+ | while nNumFramesRecv < nTotalNumFrames | ||
+ | calllib(' | ||
+ | bClientResults = calllib(' | ||
+ | if bClientResults | ||
+ | nNumFramesRecv = nNumFramesRecv + 1; | ||
+ | nNumRes = calllib(' | ||
+ | fprintf(' | ||
+ | for r = 0:nNumRes-1 | ||
+ | strName = calllib(' | ||
+ | fprintf(' | ||
+ | nSubFrames = calllib(' | ||
+ | if 0 < nSubFrames | ||
+ | nFrameVal = calllib(' | ||
+ | fprintf(' | ||
+ | nSubFrames, | ||
+ | nNumComps = calllib(' | ||
+ | fprintf(' | ||
+ | for c = 0: | ||
+ | fVal = calllib(' | ||
+ | fprintf(' | ||
+ | end | ||
+ | end | ||
+ | end | ||
+ | end | ||
+ | calllib(' | ||
+ | end | ||
+ | |||
+ | % Note how long it took to request frames | ||
+ | fprintf(' | ||
+ | |||
+ | % Disconnect from Visual3DServer. | ||
+ | calllib(' | ||
+ | else | ||
+ | disp(calllib(' | ||
+ | bContinue = calllib(' | ||
+ | unloadlibrary(' | ||
+ | return | ||
+ | end | ||
+ | |||
+ | % Tear down object that represents Visual3DServer (V3DS). | ||
+ | bContinue = calllib(' | ||
+ | |||
+ | if libisloaded(' | ||
+ | unloadlibrary(' | ||
+ | end | ||
+ | end | ||
+ | |||
+ | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
+ | </ | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | === Python Example Client Code === | ||
+ | |||
+ | * **Requirements** Python 2.7.x (http:// | ||
+ | * **Recommend** Microsoft Visual Studio (2010 edition)Python Tools for Visual Studio (http:// | ||
+ | |||
+ | < | ||
+ | ################################################################################ | ||
+ | # | ||
+ | # | ||
+ | # | ||
+ | # | ||
+ | # with the Visual3DServer application. | ||
+ | # | ||
+ | # | ||
+ | # | ||
+ | ################################################################################ | ||
+ | |||
+ | ################################################################################ | ||
+ | # Imports | ||
+ | ################################################################################ | ||
+ | |||
+ | # Standard package imports | ||
+ | from ctypes import * | ||
+ | import time | ||
+ | |||
+ | v3dsClientLibPath = ' | ||
+ | v3dsClientLib = cdll.LoadLibrary(v3dsClientLibPath) | ||
+ | |||
+ | v3dsClientLib.v3ds_init.restype = c_bool | ||
+ | v3dsClientLib.v3ds_kill.restype = c_bool | ||
+ | |||
+ | v3dsClientLib.v3ds_getTcpTimeout.restype = c_int | ||
+ | v3dsClientLib.v3ds_getTcpBufferSize.restype = c_int | ||
+ | v3dsClientLib.v3ds_getUdpBufferSize.restype = c_int | ||
+ | |||
+ | v3dsClientLib.v3ds_getPriority.restype = c_char_p | ||
+ | v3dsClientLib.v3ds_getSleep.restype = c_int | ||
+ | |||
+ | v3dsClientLib.v3ds_connect.restype = c_bool | ||
+ | v3dsClientLib.v3ds_isConnected.restype = c_bool | ||
+ | v3dsClientLib.v3ds_disconnect.restype = c_bool | ||
+ | |||
+ | v3dsClientLib.v3ds_getClientName.restype = c_char_p | ||
+ | v3dsClientLib.v3ds_getClientVersion.restype = c_char_p | ||
+ | v3dsClientLib.v3ds_getClientDate.restype = c_char_p | ||
+ | v3dsClientLib.v3ds_getClientEnvironment.restype = c_char_p | ||
+ | v3dsClientLib.v3ds_getClientProtocol.restype = c_char_p | ||
+ | |||
+ | v3dsClientLib.v3ds_setSystems.restype = c_bool | ||
+ | v3dsClientLib.v3ds_isSystemsSet.restype = c_bool | ||
+ | v3dsClientLib.v3ds_getSystems.restype = c_char_p | ||
+ | |||
+ | v3dsClientLib.v3ds_setPipeline.restype = c_bool | ||
+ | v3dsClientLib.v3ds_isPipelineSet.restype = c_bool | ||
+ | v3dsClientLib.v3ds_getPipeline.restype = c_char_p | ||
+ | |||
+ | v3dsClientLib.v3ds_setLinkModel.restype = c_bool | ||
+ | v3dsClientLib.v3ds_isLinkModelSet.restype = c_bool | ||
+ | v3dsClientLib.v3ds_getLinkModel.restype = c_char_p | ||
+ | |||
+ | v3dsClientLib.v3ds_hasNewResults.restype = c_bool | ||
+ | v3dsClientLib.v3ds_getNumResults.restype = c_int | ||
+ | v3dsClientLib.v3ds_getResultString.restype = c_char_p | ||
+ | v3dsClientLib.v3ds_getName.restype = c_char_p | ||
+ | v3dsClientLib.v3ds_getNumFrames.restype = c_int | ||
+ | v3dsClientLib.v3ds_getFrameValue.restype = c_int | ||
+ | v3dsClientLib.v3ds_getNumCompsPerFrame.restype = c_int | ||
+ | v3dsClientLib.v3ds_getValue.restype = c_float | ||
+ | |||
+ | v3dsClientLib.v3ds_getStatus.restype = c_char_p | ||
+ | |||
+ | ################################################################################ | ||
+ | # Script | ||
+ | ################################################################################ | ||
+ | |||
+ | # Create object to represent Visual3DServer (V3DS). | ||
+ | print(' | ||
+ | bContinue = v3dsClientLib.v3ds_init(5000, | ||
+ | print(' | ||
+ | |||
+ | # Connect to Visual3DServer with IP address, etc. | ||
+ | if bContinue: | ||
+ | print(' | ||
+ | bContinue = v3dsClientLib.v3ds_connect(b" | ||
+ | b" | ||
+ | print(' | ||
+ | else: | ||
+ | exit() | ||
+ | |||
+ | if bContinue: | ||
+ | print(' | ||
+ | # Ask V3DS the systems. | ||
+ | strSystems = v3dsClientLib.v3ds_getSystems() | ||
+ | print(' | ||
+ | print(' | ||
+ | else: | ||
+ | exit() | ||
+ | |||
+ | if bContinue: | ||
+ | print(' | ||
+ | # Tell V3DS the systems, in priority order, in which to receive data. | ||
+ | bContinue = v3dsClientLib.v3ds_setSystems(b' | ||
+ | print(' | ||
+ | else: | ||
+ | exit() | ||
+ | |||
+ | if bContinue: | ||
+ | print(' | ||
+ | # Tell V3DS what pipeline script to execute on each frame of data. | ||
+ | # Pipeline script can be inline, as done here, or loaded from a file. | ||
+ | pipelineScript = \ | ||
+ | b' | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b' | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | b'/ | ||
+ | | ||
+ | bContinue = v3dsClientLib.v3ds_setPipeline(pipelineScript) | ||
+ | print(' | ||
+ | else: | ||
+ | exit() | ||
+ | |||
+ | if bContinue: | ||
+ | print(' | ||
+ | |||
+ | startTime = time.clock() | ||
+ | |||
+ | nNumFramesRecv = 0 | ||
+ | nTotalNumFrames = 5 | ||
+ | while nNumFramesRecv < nTotalNumFrames: | ||
+ | v3dsClientLib.v3ds_lockMutex() | ||
+ | clientResults = v3dsClientLib.v3ds_hasNewResults() | ||
+ | if clientResults: | ||
+ | nNumFramesRecv += 1 | ||
+ | nNumRes = v3dsClientLib.v3ds_getNumResults() | ||
+ | print(' | ||
+ | for r in range(0, nNumRes): | ||
+ | strName = str(v3dsClientLib.v3ds_getName(r)) | ||
+ | print(' | ||
+ | nSubFrames = v3dsClientLib.v3ds_getNumFrames(r) | ||
+ | if 0 < nSubFrames: | ||
+ | nFrameVal = v3dsClientLib.v3ds_getFrameValue(r, | ||
+ | print(' | ||
+ | ', subframe 1 value: ' + str(nFrameVal)) | ||
+ | nNumComps = v3dsClientLib.v3ds_getNumCompsPerFrame(r) | ||
+ | print(' | ||
+ | for c in range(0, nNumComps): | ||
+ | fVal = v3dsClientLib.v3ds_getValue(r, | ||
+ | print(' | ||
+ | |||
+ | |||
+ | v3dsClientLib.v3ds_unlockMutex() | ||
+ | |||
+ | # Note how long it took to request frames | ||
+ | print(' | ||
+ | |||
+ | # Disconnect from Visual3DServer. | ||
+ | v3dsClientLib.v3ds_disconnect() | ||
+ | else: | ||
+ | print(v3dsClientLib.v3ds_getStatus()) | ||
+ | exit() | ||
+ | |||
+ | print(' | ||
+ | |||
+ | # Tear down object that represents Visual3DServer (V3DS). | ||
+ | bContinue = v3dsClientLib.v3ds_kill() | ||
+ | |||
+ | ################################################################################ | ||
+ | </ | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | |||
other/visual3dserver/documentation/c_client_library.1721230990.txt.gz · Last modified: 2024/07/17 15:43 by sgranger