LinuxTrack Interface - uglyDwarf/linuxtrack GitHub Wiki

Linuxtrack interface

Linuxtrack provides an interface allowing developers to incorporate head tracking into their applications.

The interface consists of two files: linuxtrack.h and linuxtrack.c; just add them to your project and you get access to the Linuxtrack's functionality.

The reason why the interface is not provided in the usual form of a library is the following: if you'd ship an application directly linked against Linuxtrack's interface library, it wouldn't work on a system where linuxtrack is not installed or configured properly. Technically the Linuxtrack might be provided in a form of a static library, but there might be problems when you'd use it to build a dynamic library (say a plugin).

The interface in its current form looks for the Linuxtrack, and when found it is used; when the Linuxtrack is not found (or not configured properly), it just indicates failure and the main application can work on. Interface overview

int linuxtrack_init(const char *cust_section);
int linuxtrack_shutdown(void);
int linuxtrack_suspend(void);
int linuxtrack_wakeup(void);
int linuxtrack_recenter(void);
typedef enum {INITIALIZING, RUNNING, PAUSED, STOPPED, ERROR} linuxtrack_state_type;
linuxtrack_state_type linuxtrack_get_tracking_state(void);
int linuxtrack_get_pose(float *heading, float *pitch, float *roll,
float *tx, float *ty, float *tz, uint32_t *counter);

Basic Linuxtrack interface is relatively simple: linuxtrack_init initializes the interface provided a string identifying your application (specifies a profile to be used), linuxtrack_shutdown shuts the interface down.

Pair linuxtrack_suspend and linuxtrack_wakeup allow you to pause and resume the tracking temporarily; technically it might seem redundant, but pausing and resuming via these two functions is usually much quicker than shutting the whole tracker down and initializing it again.

Function linuxtrack_recenter allows user to recenter the tracker when needed.

Function linuxtrack_get_tracking_state returns current state of the Linuxtrack's interface; the tracker itself runs in a different process and commands for it are delivered asynchronously. For that reason you might want to query tracker's state to see if your command was performed yet or not.

Function linuxtrack_get_pose returns you the current pose and state info. Normally you should use it once per (your applications) frame to obtain the current pose. The pose data are processed already (filtered, scaled, response curves applied according to the profile you specified upon initialization), so it doesn't make much sense to call it any faster. Interface description

int linuxtrack_init(const char *cust_section);

Requests initialization of the tracker. Looks for Linuxtrack installation and if it finds one, it tries to start the tracking. It is required that Linuxtrack is not only installed, but also configured correctly by the user.

Parameter cust_section indicates a profile name to be used; if it is NULL, default profile ("Default") is used.

Return value of 0 means success. Note however, that it doesn't mean that the tracker started successfully - it means the initialization request was sent.

To check, if the tracker actually started, check the return value of linuxtrack_get_tracking_state function. Just remember, that it can take several seconds for the tracker to start. Note however, that you can start calling linuxtrack_get_pose immediately - if the Linuxtrack was not initialized yet, it will just return "zero" pose (all angles and distances set to zero).

int linuxtrack_shutdown(void);

Requests tracker shutdown. Due to the fact, that there might be several applications using Linuxtrack running in parallel, the tracker actually closes down only when the last client disconnects.

Return value of 0 means success.

int linuxtrack_suspend(void); int linuxtrack_wakeup(void);

Requests the tracking be paused or resumed. Pausing the tracking is especially useful for TrackIR/SmartNav class of devices, because it turns off their IR LEDs, lowering the device's power dissipation.

Return value of 0 means success.

int linuxtrack_recenter(void);

Requests the current pose should be used as a new center pose.

Return value of 0 means success.

typedef enum {INITIALIZING, RUNNING, PAUSED, STOPPED, ERROR} linuxtrack_state_type; linuxtrack_state_type linuxtrack_get_tracking_state(void);

Requests information on the current tracker state.

Return value indicates the tracker state according to the linuxtrack_state_type enum.

int linuxtrack_get_pose(float *heading, float *pitch, float *roll, float *tx, float *ty, float *tz, uint32_t *counter);

Requests the current pose. Heading, pitch and roll values are in degrees, translation values tx, ty, tz are in millimeters. Counter value denotes tracker frame number, from which the current pose was computed; if the number doesn't change in between two calls, the pose didn't change either.

In case of a problem (Linuxtrack not installed, not configured, missconfigured, not initialized yet...), the reported pose consists of all zeros.

Return value of 1 means the new pose is available, 0 means no new pose, but everything else ok, -1 signals a problem. Advanced interface

If, for some reason the normal interface is not sufficient for your needs (for normal head tracking applications it should be), there is an advanced interface available. This interface provides additional low level information on current pose, like sensor resolution, raw pose values (before processing) and blob coordinates. This interface is meant mainly for computer vision research and similar applications. Please note, that using raw values completely bypasses Linuxtrack's profile system, application of sensitivity curves and filtration. Advanced Interface overview

typedef struct{
float pitch;
float yaw;
float roll;
float tx;
float ty;
float tz;
uint32_t counter;
uint32_t resolution_x;
uint32_t resolution_y;
float raw_pitch;
float raw_yaw;
float raw_roll; float raw_tx;
float raw_ty;
float raw_tz;
uint8_t status;
} linuxtrack_pose_t;

int linuxtrack_get_pose_full(linuxtrack_pose_t *pose, float blobs[], int num_blobs, int *blobs_read);

On top of the functions available in the basic interface, there is the linuxtrack_get_pose_full function, that provides all the low level information on the current pose. Advanced interface description

int linuxtrack_get_pose_full(linuxtrack_pose_t *pose, float blobs[], int num_blobs, int *blobs_read);

Requests all the low level information on the current pose. These include raw pose values (no sensitivity curves or filtration applied), sensor resolution and blob coordinates.

The blobs array should contain 3 times the num_blobs elements (each blob has the x,y coordinates and a number of pixels value). Value of blobs_read indicates how many blobs were actually read.

At the moment Linuxtrack reports up to 10 blobs; this limit can be increased only by its recompilation (ltlib.h, value of MAX_BLOBS macro). Note however that Wiimote is capable of reporting only 4 blobs (hardware limitation)!

Hello World examples

Linuxtrack provides two "Hello World" examples of the Linuxtrack's interface in external program. The first one called linuxtrack_hello_world.c shows how to use the Linuxtrack's interface, while linuxtrack_hello_world_adv.c provides example of advanced interface usage.

Both programs should be fairly straightforward; all they do is tracking initialization, they track for a bit, pause the tracking, resume it, after a few moments they close the tracking and close down.

There are a few things worth noting. First of all, Linuxtrack's interface is asynchronous - the fact, that a function returned only means that the request was sent, not that it was carried out. For example, running linuxtrack_suspend to pause tracking, does'n stop the tracking immediately; it just sent the tracker a command to pause the tracking, that will be carried out as soon as possible. If you want to be absolutely sure that the command was performed, query the tracker status. Note however, that blocking the application waiting for the status change is not recommended, mainly because it hinders user experience (unresponsive application). It can also cause race condition problems due to the fact, that several applications might issue different commands simultaneously.

This might seem to be a problem, but in fact it is not. Thanks to the fact, that linuxtrack_get_pose and linuxtrack_get_pose_full return value higher than zero only when a now pose is available, you will not receive invalid pose no matter when you call it. This means, that your application can safely start calling those functions right after the initialization is requested (by means of calling linuxtrack_init), provided that you process the pose information only when higher than zero value is returned.

By default, tracking is automatically re-centered when the tracker is started. If user looks away in that instant, he is going to need to recenter the tracking. To do that, simply call linuxtrack_recenter.

The initialize_tracking function used in the examples contains a loop waiting for the tracker initialization. As discussed above, normally there is no need to do that - it would unnecessarily delay the start of your application. In the example it is used to make things simple, there aren't any clean means to close the application in case the tracker doesn't work (except for Ctrl-C, but that is not very nice way). Building

The only thing that Linuxtrack interface needs is that your application is linked against libdl. The reason is, that the interface dynamically loads the Linuxtrack library (if available) and for that it needs libdl. Here is the commandline used to build the example code:

gcc -o linuxtrack_hello_world -g -I. linuxtrack.c linuxtrack_hello_world.c -ldl
gcc -o linuxtrack_hello_world_adv -g -I. linuxtrack.c linuxtrack_hello_world_adv.c -ldl

The "-ldl" is used to link against libdl; if you use some IDE, just add it to linker flags and you should be good to go. The "-I." flag instructs the compiler to look for linuxtrack.h in the current directory (you might need it to point somewhere else, for example if you store all .h files in a single location). It is also expected that you have the linuxtrack.c file in your project directory.