You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

448 lines
13 KiB
C++

#include <tobii/tobii.h>
#include <tobii/tobii_streams.h>
#include <stdio.h>
#include <assert.h>
#include <cstring>
#include <vector>
#include <string>
#include <sstream>
#include <lsl_cpp.h>
#include <sys/types.h> // For uid_t
#include <pwd.h> // For struct passwd and getpwuid()
#include <thread>
#include <atomic>
#include <termios.h>
#include <unistd.h>
//tobii handles
static tobii_api_t *api = nullptr;
static tobii_device_t *device = nullptr;
//LSL outlet streams
static lsl::stream_outlet *gazePointOutlet;
static lsl::stream_outlet *gazeOriginOutlet;
static lsl::stream_outlet *eyePositionNormalizedOutlet;
static lsl::stream_outlet *userPresenceOutlet;
std::atomic<bool> exit_flag(false);
void inputThread()
{
struct termios oldt, newt;
tcgetattr(STDIN_FILENO, &oldt);
newt = oldt;
newt.c_lflag &= ~(ICANON | ECHO);
tcsetattr(STDIN_FILENO, TCSANOW, &newt);
char c;
read(STDIN_FILENO, &c, 1); // Blocks until a key is pressed
exit_flag = true;
tcsetattr(STDIN_FILENO, TCSANOW, &oldt); // Restore original settings
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// tobii callbacks
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
void gaze_point_callback(tobii_gaze_point_t const *gaze_point, void *user_data)
{
(void)user_data; //ignore user_data argument
std::vector<float> result(2, 0.0);
if(gaze_point->validity == TOBII_VALIDITY_VALID)
{
result[0] = gaze_point->position_xy[0];
result[1] = gaze_point->position_xy[1];
// printf("[gaze point]: %f, %f\n",
// gaze_point->position_xy[0],
// gaze_point->position_xy[1]);
if(gazePointOutlet)
{
gazePointOutlet->push_sample(result, lsl::local_clock());
}
}
}
void gaze_origin_callback(tobii_gaze_origin_t const *gaze_origin, void *user_data)
{
(void)user_data; //ignore user_data argument
std::vector<float> result(6, 0.0);
if(gaze_origin->left_validity == TOBII_VALIDITY_VALID)
{
result[0] = gaze_origin->left_xyz[0];
result[1] = gaze_origin->left_xyz[1];
result[2] = gaze_origin->left_xyz[2];
}
else
{
result[0] = -1.0;
result[1] = -1.0;
result[2] = -1.0;
}
if(gaze_origin->right_validity == TOBII_VALIDITY_VALID)
{
result[3] = gaze_origin->right_xyz[0];
result[4] = gaze_origin->right_xyz[1];
result[5] = gaze_origin->right_xyz[2];
}
else
{
result[3] = -1.0;
result[4] = -1.0;
result[5] = -1.0;
}
// printf("[gaze origin] left: %f, %f, %f\n[gaze origin] right: %f, %f, %f\n",
// result[0],
// result[1],
// result[2],
// result[3],
// result[4],
// result[5]);
if(gazeOriginOutlet)
{
gazeOriginOutlet->push_sample(result, lsl::local_clock());
}
}
void eyePositionNormalizedCallback(tobii_eye_position_normalized_t const *eyePosition, void *user_data)
{
(void)user_data; //ignore user_data argument
std::vector<float> result(6, 0.0);
if(eyePosition->left_validity == TOBII_VALIDITY_VALID)
{
result[0] = eyePosition->left_xyz[0];
result[1] = eyePosition->left_xyz[1];
result[2] = eyePosition->left_xyz[2];
}
if(eyePosition->right_validity == TOBII_VALIDITY_VALID)
{
result[3] = eyePosition->right_xyz[0];
result[4] = eyePosition->right_xyz[1];
result[5] = eyePosition->right_xyz[2];
}
// printf("[eye position] left: %f, %f, %f\n[eye position] right: %f, %f, %f\n",
// result[0],
// result[1],
// result[2],
// result[3],
// result[4],
// result[5]);
if(eyePositionNormalizedOutlet)
{
eyePositionNormalizedOutlet->push_sample(result, lsl::local_clock());
}
}
//NOT SUPPORTED :-|
/*
void headPoseCallback(tobii_head_pose_t const *headPose, void *user_data)
{
(void)user_data; //ignore user_data argument
static std::vector<float> sample = {0, 0, 0, 0, 0, 0};
if(headPose->position_validity == TOBII_VALIDITY_VALID)
{
sample[0] = headPose->position_xyz[0];
sample[1] = headPose->position_xyz[1];
sample[2] = headPose->position_xyz[2];
}
if(headPose->rotation_validity_xyz[0] == TOBII_VALIDITY_VALID)
{
sample[3] = headPose->rotation_xyz[0];
}
if(headPose->rotation_validity_xyz[1] == TOBII_VALIDITY_VALID)
{
sample[4] = headPose->rotation_xyz[1];
}
if(headPose->rotation_validity_xyz[2] == TOBII_VALIDITY_VALID)
{
sample[5] = headPose->rotation_xyz[2];
}
// printf("[head pose] left: %f, %f, %f\n[head rotation] right: %f, %f, %f\n",
// sample[0],
// sample[1],
// sample[2],
// sample[3],
// sample[4],
// sample[5]);
}
*/
void userPresenceCallback(tobii_user_presence_status_t status, int64_t timestamp_us, void *user_data)
{
(void)timestamp_us; //ignore timestamp_us argument
(void)user_data; //ignore user_data argument
static std::string msgPrefix("[user presence]");
if(status == TOBII_USER_PRESENCE_STATUS_PRESENT)
{
// printf("%s: present\n", msgPrefix.c_str());
if(userPresenceOutlet)
{
std::string msg("present");
userPresenceOutlet->push_sample(&msg, lsl::local_clock());
}
}
else if(status == TOBII_USER_PRESENCE_STATUS_AWAY)
{
// printf("%s: away\n", msgPrefix.c_str());
if(userPresenceOutlet)
{
std::string msg("away");
userPresenceOutlet->push_sample(&msg, lsl::local_clock());
}
}
else if(status == TOBII_USER_PRESENCE_STATUS_UNKNOWN)
{
// printf("%s: unknown\n", msgPrefix.c_str());
if(userPresenceOutlet)
{
std::string msg("unknown");
userPresenceOutlet->push_sample(&msg, lsl::local_clock());
}
}
}
//--------------------------------------------------------------------
static void url_receiver(char const *url, void *user_data)
{
char *buffer = (char *)user_data;
if(*buffer != '\0') // only keep first value
{
return;
}
if(strlen(url) < 256)
{
strcpy(buffer, url);
}
}
void initTobii()
{
tobii_error_t error = tobii_api_create(&api, nullptr, nullptr);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_api_create() call error: %s\n", tobii_error_message(error));
}
char url[256] = {0x00};
error = tobii_enumerate_local_device_urls(api, url_receiver, url);
if((error != TOBII_ERROR_NO_ERROR) && (*url != '\0'))
{
printf("tobii_enumerate_local_device_urls() call error: %s\n", tobii_error_message(error));
}
error = tobii_device_create(api, url, &device);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_device_create() call error: %s\n", tobii_error_message(error));
}
error = tobii_gaze_point_subscribe(device, gaze_point_callback, 0);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_gaze_point_subscribe() call error: %s\n", tobii_error_message(error));
}
error = tobii_gaze_origin_subscribe(device, gaze_origin_callback, 0);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_gaze_origin_subscribe() call error: %s\n", tobii_error_message(error));
}
error = tobii_eye_position_normalized_subscribe(device, eyePositionNormalizedCallback, 0);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_eye_position_normalized_subscribe() call error: %s\n", tobii_error_message(error));
}
//NOT SUPPORTED :-|
/*
error = tobii_head_pose_subscribe(device, headPoseCallback, 0);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_head_pose_subscribe() call error: %s\n", tobii_error_message(error));
}
*/
error = tobii_user_presence_subscribe(device, userPresenceCallback, 0);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_user_presence_subscribe() call error: %s\n", tobii_error_message(error));
}
}
void deinitTobii()
{
tobii_error_t error = tobii_user_presence_unsubscribe(device);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_user_presence_unsubscribe() call error: %s\n", tobii_error_message(error));
}
//NOT SUPPORTED :-|
// error = tobii_head_pose_unsubscribe(device);
// if(error != TOBII_ERROR_NO_ERROR)
// {
// printf("tobii_head_pose_unsubscribe() call error: %s\n", tobii_error_message(error));
// }
error = tobii_eye_position_normalized_unsubscribe(device);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_eye_position_normalized_unsubscribe() call error: %s\n", tobii_error_message(error));
}
error = tobii_gaze_origin_unsubscribe(device);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_gaze_origin_unsubscribe() call error: %s\n", tobii_error_message(error));
}
error = tobii_gaze_point_unsubscribe(device);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_gaze_point_unsubscribe() call error: %s\n", tobii_error_message(error));
}
error = tobii_device_destroy(device);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_device_destroy() call error: %s\n", tobii_error_message(error));
}
error = tobii_api_destroy(api);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_api_destroy() call error: %s\n", tobii_error_message(error));
}
}
void initLSL()
{
//get username to use as source_id parameter for created LSL streams
uid_t uid = getuid(); // Get the real user ID of the calling process
struct passwd *pw = getpwuid(uid); // Get password structure for the given UID
if(pw != nullptr)
{
printf("Username: %s\n", pw->pw_name);
}
else
{
printf("Error: Could not retrieve username.\n");
}
//gaze point stream
lsl::stream_info gazePointOutletInfo(std::string("TobiiGazePoint"),
std::string("GAZE"),
2,
90,
lsl::cf_float32,
std::string(pw->pw_name));
printf("!!! Create gaze point stream\n");
gazePointOutlet = new lsl::stream_outlet(gazePointOutletInfo, 0);
//gaze origin stream
lsl::stream_info gazeOriginOutletInfo(std::string("TobiiGazeOrigin"),
std::string("GAZE"),
6,
90,
lsl::cf_float32,
std::string(pw->pw_name));
printf("!!! Create gaze origin stream\n");
gazeOriginOutlet = new lsl::stream_outlet(gazeOriginOutletInfo, 0);
//eye position normalized stream
lsl::stream_info eyePositionNormalizedOutletInfo(std::string("TobiiEyePositionNormalized"),
std::string("GAZE"),
6,
90,
lsl::cf_float32,
std::string(pw->pw_name));
printf("!!! Create eye position normalized stream\n");
eyePositionNormalizedOutlet = new lsl::stream_outlet(eyePositionNormalizedOutletInfo, 0);
//eye position normalized stream
lsl::stream_info userPresenceOutletInfo(std::string("TobiiUserPresence"),
std::string("GAZE"),
1,
lsl::IRREGULAR_RATE,
lsl::cf_string,
std::string(pw->pw_name));
printf("!!! Create user presence stream\n");
userPresenceOutlet = new lsl::stream_outlet(userPresenceOutletInfo, 0);
}
void deinitLSL()
{
if(gazePointOutlet)
{
delete gazePointOutlet;
gazePointOutlet = nullptr;
}
if(gazeOriginOutlet)
{
delete gazeOriginOutlet;
gazeOriginOutlet = nullptr;
}
if(eyePositionNormalizedOutlet)
{
delete eyePositionNormalizedOutlet;
eyePositionNormalizedOutlet = nullptr;
}
if(userPresenceOutlet)
{
delete userPresenceOutlet;
userPresenceOutlet = nullptr;
}
}
int main()
{
tobii_error_t error = TOBII_ERROR_NO_ERROR;
std::thread input_handler(inputThread);
//init Tobii and LSL stuff
initLSL();
initTobii();
printf("!!! run callbacks processing\n");
while(!exit_flag)
{
error = tobii_wait_for_callbacks(1, &device);
if((error != TOBII_ERROR_NO_ERROR)&&(error != TOBII_ERROR_TIMED_OUT))
{
printf("tobii_wait_for_callbacks() call error: %s\n", tobii_error_message(error));
}
error = tobii_device_process_callbacks(device);
if(error != TOBII_ERROR_NO_ERROR)
{
printf("tobii_device_process_callbacks() call error: %s\n", tobii_error_message(error));
}
}
input_handler.join(); // Wait for the input thread to finish
//deinit Tobii and LSL stuff
deinitTobii();
deinitLSL();
return 0;
}