diff options
Diffstat (limited to 'contrib/trackir-client')
| -rw-r--r-- | contrib/trackir-client/client.c | 168 | ||||
| -rw-r--r-- | contrib/trackir-client/client.exe | bin | 0 -> 100864 bytes | 
2 files changed, 168 insertions, 0 deletions
| diff --git a/contrib/trackir-client/client.c b/contrib/trackir-client/client.c new file mode 100644 index 00000000..9d01d5f9 --- /dev/null +++ b/contrib/trackir-client/client.c @@ -0,0 +1,168 @@ +/* This is free and unencumbered software released into the public domain. + + * Anyone is free to copy, modify, publish, use, compile, sell, or + * distribute this software, either in source code form or as a compiled + * binary, for any purpose, commercial or non-commercial, and by any + * means. + *  + * In jurisdictions that recognize copyright laws, the author or authors + * of this software dedicate any and all copyright interest in the + * software to the public domain. We make this dedication for the benefit + * of the public at large and to the detriment of our heirs and + * successors. We intend this dedication to be an overt act of + * relinquishment in perpetuity of all present and future rights to this + * software under copyright law. + *  + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. + * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR + * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, + * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + * OTHER DEALINGS IN THE SOFTWARE. +*/ + +#include <stdlib.h> +#include <stdio.h> +/* this is unfortunately necessary as the API is Windows-only */ +#include <windows.h> + +#ifdef __cplusplus +#  define MY_EXTERN extern "C" +#else +#  define MY_EXTERN +#define MY_EXPORT(ret) MY_EXTERN ret __declspec(dllexport) __cdecl +#endif + +#pragma pack(push, 1) +typedef struct { +  __int16 status; /* zero means success, it's easy to forget! */ +  __int16 frame; /* if idempotent, device paused */ +  __int32 padding; +  float roll, pitch, yaw; /* divide by 8191 when received, in radians */ +  float x, y, z; /* between -16383 and 16383 */ +  float padding2[9]; +} tir_headpose_t; +#pragma pack(pop) + +typedef struct { +	HMODULE hlibrary; +	__int32 (__stdcall* tir_reghwnd)(HWND hwnd); +	__int32 (__stdcall* tir_regprogid)(__int16 progid); +	__int32 (__stdcall* tir_queryver)(__int16* version); +	__int32 (__stdcall* tir_reqdata)(__int16 bitmask); +	__int32 (__stdcall* tir_stopcurs)(void); +	__int32 (__stdcall* tir_startxmit)(void); +	__int32 (__stdcall* tir_getdata)(tir_headpose_t* pose); +	/* there are more functions, especially for cleaning up +	 * but the library is typically called on startup +	 * and unloaded on exit. if you really must add them, +	 * you can either extract them from NPClient.dll or +	 * google for it. for instance I once found symbols +	 * and some constants in Doxygen format... +	 * if you REALLY need them I can finish the damn thing +	 * but it's so much boilerplate :( +	 */ +} tir_state_t; + +MY_EXPORT(tir_state_t*) mytir_init(void) +{ +	HKEY hkey = NULL; +	tir_state_t* ret; +	int i; +	DWORD sz = 1900; +	/* Windows paths are limited to 260 characters */ +	char buf[2048]; + +	ret = malloc(sizeof(tir_state_t)); +	if (ret == NULL) +		return NULL; +	memset(ret, 0, sizeof(tir_state_t)); +	RegOpenKeyExA(HKEY_CURRENT_USER, "Software\\NaturalPoint\\NATURALPOINT\\NPClient Location", 0, KEY_QUERY_VALUE, &hkey); +	if (!hkey) +		goto error; +	buf[0] = '\0'; +	if (RegQueryValueExA(hkey, "Path", NULL, NULL, buf, &sz) != ERROR_SUCCESS) +		goto error; +	(void) RegCloseKey(hkey); +	hkey = NULL; +	buf[2047] = '\0'; +	for (i = 0; buf[i]; i++) +		if (buf[i] == '/') +			buf[i] = '\\'; +	/* in the registry path there's normally a trailing slash already */ +#ifdef MY_BUILD_AMD64 +	strcat(buf, "NPClient64.dll"); +#else +	strcat(buf, "NPClient.dll"); +#endif +	ret->hlibrary = LoadLibraryA(buf); +	if (ret->hlibrary == NULL) +		goto error; +	if ((ret->tir_reghwnd = GetProcAddress(ret->hlibrary, "NP_RegisterWindowHandle")) == NULL) +		goto error; +	if ((ret->tir_regprogid = GetProcAddress(ret->hlibrary, "NP_RegisterProgramProfileID")) == NULL) +		goto error; +	if ((ret->tir_queryver = GetProcAddress(ret->hlibrary, "NP_QueryVersion")) == NULL) +		goto error; +	if ((ret->tir_reqdata = GetProcAddress(ret->hlibrary, "NP_RequestData")) == NULL) +		goto error; +	if ((ret->tir_stopcurs = GetProcAddress(ret->hlibrary, "NP_StopCursor")) == NULL) +		goto error; +	if ((ret->tir_startxmit = GetProcAddress(ret->hlibrary, "NP_StartDataTransmission")) == NULL) +		goto error; +	if ((ret->tir_getdata = GetProcAddress(ret->hlibrary, "NP_GetData")) == NULL) +		goto error; +	return ret; +error: +	if (hkey) +	  (void) RegCloseKey(hkey); +	free(ret); +	/* sorry but I don't know anything about unloading DLLs. unless you do, there's a handle leak */ +	/* if you unload it, no need to touch function pointers, they just become invalid */ +	return NULL; +} + +MY_EXPORT(void) mytir_start(tir_state_t* state, HWND window_handle) +{ +	__int16 tmp; +	/* while you have to call it, can pass anything valid, like desktop handle I suppose? */ +	(void) state->tir_reghwnd(window_handle); +	(void) state->tir_regprogid(1901); /* Falcon 4, an old flight simulation. Need to pass valid Id */ +	(void) state->tir_queryver(&tmp); /* better to call it, even if we don't care about it! */ +	(void) state->tir_reqdata(119); /* specifies which elements of the pose structure to return */ +	(void) state->tir_stopcurs(); /* better to call it */ +	(void) state->tir_startxmit(); /* idem */ +} + +MY_EXPORT(void) mytir_get_pose(tir_state_t* state, tir_headpose_t* ret) +{ +	memset(ret, 0, sizeof(tir_headpose_t)); +	ret->status = 1; /* if the call fails for some reason, at least we shouldn't think we have a valid pose when we don't */ +	(void) state->tir_getdata(ret); +} + +#define MY_DEMO + +#ifdef MY_DEMO +int main(void) +{ +	tir_state_t* state; +	int i; +	tir_headpose_t ret; + +	state = mytir_init(); +	if (!state) +		return 1; +	mytir_start(state, GetDesktopWindow()); +	for (i = 0; i < 3600; i++) +	{ +		mytir_get_pose(state, &ret); +		if (ret.status) +			return 1; +		printf("%f %f %f ~ %f %f %f\n", ret.yaw * 180 / 3.14 / 8191, ret.pitch * 180 / 3.14 / 8191, ret.roll * 180 / 3.14 / 8191, ret.x, ret.y, ret.z); +		Sleep(35); +	} +	return 0; +} +#endif
\ No newline at end of file diff --git a/contrib/trackir-client/client.exe b/contrib/trackir-client/client.exeBinary files differ new file mode 100644 index 00000000..c1ac8ebf --- /dev/null +++ b/contrib/trackir-client/client.exe | 
