-
Notifications
You must be signed in to change notification settings - Fork 1
/
vrpn_5DT16.C
366 lines (313 loc) · 12.8 KB
/
vrpn_5DT16.C
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
// This is a driver for the 5dt Data Glove 16 sensors
// Look at www.5dt.com for more informations about this product
// Manuals are avalaible freely from this site
// The original code for 5dt was written by Philippe DAVID with the help of Yves GAUVIN
// This version for the 16 sensor version was written by Tom De Weyer
// naming convention used in this file:
// l_ is the prefixe for local variables
// g_ is the prefixe for global variables
// p_ is the prefixe for parameters
#include <string.h>
#include "vrpn_5DT16.h"
#include "vrpn_Shared.h"
#include "vrpn_Serial.h"
#undef VERBOSE
// Defines the modes in which the device can find itself.
#define STATUS_RESETTING (-1) // Resetting the device
#define STATUS_SYNCING (0) // Looking for the first character of report
#define STATUS_READING (1) // Looking for the rest of the report
#define _5DT_INFO(msg) { send_text_message(msg, timestamp, vrpn_TEXT_NORMAL) ; if (d_connection) d_connection->send_pending_reports(); }
#define _5DT_WARNING(msg) { send_text_message(msg, timestamp, vrpn_TEXT_WARNING) ; if (d_connection) d_connection->send_pending_reports(); }
#define _5DT_ERROR(msg) { send_text_message(msg, timestamp, vrpn_TEXT_ERROR) ; if (d_connection) d_connection->send_pending_reports(); }
#define MAX_TIME_INTERVAL (2000000) // max time between reports (usec)
/******************************************************************************
* NAME : duration
* ROLE :
* ARGUMENTS :
* RETURN :
******************************************************************************/
static unsigned long duration (struct timeval p_t1, struct timeval p_t2)
{
return (p_t1.tv_usec - p_t2.tv_usec) +
1000000L * (p_t1.tv_sec - p_t2.tv_sec);
}
/******************************************************************************
* NAME : vrpn_5dt16::vrpn_5dt16
* ROLE : This creates a vrpn_5dt16 and sets it to reset mode. It opens
* the serial device using the code in the vrpn_Serial_Analog constructor.
* ARGUMENTS :
* RETURN :
******************************************************************************/
vrpn_5dt16::vrpn_5dt16 (const char * p_name, vrpn_Connection * p_c, const char * p_port, int p_baud):
vrpn_Serial_Analog (p_name, p_c, p_port, p_baud, 8, vrpn_SER_PARITY_NONE),
_numchannels (17) // This is an estimate; will change when reports come
{
// Set the parameters in the parent classes
vrpn_Analog::num_channel = _numchannels;
// Set the status of the buttons and analogs to 0 to start
clear_values();
// Set the mode to reset
_status = STATUS_RESETTING;
}
/******************************************************************************
* NAME : vrpn_5dt16::clear_values
* ROLE :
* ARGUMENTS :
* RETURN :
******************************************************************************/
void vrpn_5dt16::clear_values (void)
{
int i;
for (i = 0; i < _numchannels; i++) {
vrpn_Analog::channel[i] = vrpn_Analog::last[i] = 0;
}
}
/******************************************************************************
* NAME : vrpn_5dt16::reset
* ROLE : This routine will reset the 5DT
* ARGUMENTS :
* RETURN : 0 else -1 in case of error
******************************************************************************/
int vrpn_5dt16::reset (void)
{
unsigned char l_inbuf [45];
int l_ret;
bool found=false;
struct timeval start, now;
vrpn_gettimeofday(&start, NULL);
while(!found) // read the begin sequence of an info packet "<I"
{
l_ret = vrpn_read_available_characters (serial_fd, l_inbuf, 1);
if(l_ret!=1) {
vrpn_SleepMsecs (100); //Give it time to send data
} else if(l_inbuf[0] == '<' ) {
while(!vrpn_read_available_characters (serial_fd, l_inbuf, 1)) {
vrpn_SleepMsecs (100); //Give it time to send data
}
if(l_inbuf[0]=='I') {
found=true;
}
}
// See if we've timed out on the reset
vrpn_gettimeofday(&now, NULL);
if (now.tv_sec > start.tv_sec + 2) {
fprintf(stderr,"vrpn_5dt16::reset(): Timeout during reset\n");
return -1;
}
}
vrpn_SleepMsecs (100); //Give it time to send data
l_ret=vrpn_read_available_characters (serial_fd, l_inbuf, 5); // read the rest of the data
char text[20];
sprintf(text,"Hardware Version %i.0%i",l_inbuf[0],l_inbuf[1]); // hardware version
_5DT_WARNING(text);
if (l_inbuf[2] | 1) //right or left glove
{
_5DT_WARNING ("A right glove is ready");
} else {
_5DT_WARNING ("A left glove is ready");
}
if (l_inbuf[3] | 1) //wireless glove or wired
{
_5DT_WARNING ("no wireless glove");
} else {
_5DT_WARNING ("wireless glove");
}
// We're now entering the syncing mode which send the read command to the glove
_status = STATUS_READING;
_bufcount = 0;
vrpn_gettimeofday (×tamp, NULL); // Set watchdog now
return 0;
}
/******************************************************************************
* NAME : vrpn_5dt16::get_report
* ROLE : This function will read characters until it has a full report, then
* put that report into analog fields and call the report methods on these.
* ARGUMENTS : void
* RETURN : void
******************************************************************************/
void vrpn_5dt16::get_report (void)
{
int l_ret; // Return value from function call to be checked
// The official doc (http://www.5dt.com/downloads/5DTDataGlove5Manual.pdf)
// says we should get 36 bytes, but once in a while an info packet arrives and we should
// ignore that packet.
_expected_chars = 36;
//--------------------------------------------------------------------
// Read as many bytes of this report as we can, storing them
// in the buffer. We keep track of how many have been read so far
// and only try to read the rest.
//--------------------------------------------------------------------
l_ret = vrpn_read_available_characters (serial_fd, &_buffer [_bufcount],
_expected_chars - _bufcount);
if (l_ret == -1) {
_5DT_ERROR ("Error reading the glove");
_status = STATUS_RESETTING;
return;
}
#ifdef VERBOSE
if (l_ret != 0) printf("... got %d characters (%d total)\n",l_ret, _bufcount);
#endif
//--------------------------------------------------------------------
// The time of the report is the time at which the first character for
// the report is read.
//--------------------------------------------------------------------
if ( (l_ret > 0) && (_bufcount == 0) ) {
vrpn_gettimeofday(×tamp, NULL);
}
//--------------------------------------------------------------------
// We keep track of how many characters we have received and keep
// going back until we get as many as we expect.
//--------------------------------------------------------------------
_bufcount += l_ret;
if (_bufcount < _expected_chars) { // Not done -- go back for more
return;
}
//--------------------------------------------------------------------
// We now have enough characters to make a full report. First check to
// make sure that the first one is what we expect.
if (!( (_buffer[0] == '<') && (_buffer[1] == 'D') ) ) //first characters need to be <D
{
_5DT_INFO ("Unexpected first character in report, probably info packet (recovering)");
for(int i=0;i<29;i++) {
_buffer[i]=_buffer[i+7];
}
_bufcount=29;
return;
}
#ifdef VERBOSE
printf ("Got a complete report (%d of %d)!\n", _bufcount, _expected_chars);
#endif
//--------------------------------------------------------------------
// Decode the report and store the values in it into the analog values
// if appropriate.
//--------------------------------------------------------------------
for(int i=0;i<16;i++) {
channel[i]=_buffer[i*2+2]*256+_buffer[i*2+3];
}
char text[512];
sprintf(text,"original %f|%f|%f|%f|%f|%f|%f|%f|%f|%f|%f|%f|%f|%f|%f|%f\n",channel[0],channel[1],channel[2],channel[3],channel[4],
channel[5],channel[6],channel[7],channel[8],channel[9],
channel[10],channel[11],channel[12],channel[13],channel[14],
channel[15]);
_5DT_ERROR(text);
//--------------------------------------------------------------------
// Done with the decoding, send the reports and go back to syncing
//--------------------------------------------------------------------
report_changes();
_bufcount =0;
}
/******************************************************************************
* NAME : vrpn_5dt16::report_changes
* ROLE :
* ARGUMENTS :
* RETURN : void
******************************************************************************/
void vrpn_5dt16::report_changes (vrpn_uint32 class_of_service)
{
vrpn_Analog::timestamp = timestamp;
vrpn_Analog::report_changes(class_of_service);
}
/******************************************************************************
* NAME : vrpn_5dt16::report
* ROLE :
* ARGUMENTS :
* RETURN : void
******************************************************************************/
void vrpn_5dt16::report (vrpn_uint32 class_of_service)
{
vrpn_Analog::timestamp = timestamp;
vrpn_Analog::report(class_of_service);
}
/******************************************************************************
* NAME : vrpn_5dt16::mainloop
* ROLE : This routine is called each time through the server's main loop. It will
* take a course of action depending on the current status of the device,
* either trying to reset it or trying to get a reading from it. It will
* try to reset the device if no data has come from it for a couple of
* seconds
* ARGUMENTS :
* RETURN : void
******************************************************************************/
void vrpn_5dt16::mainloop ()
{
char l_errmsg[256];
server_mainloop();
switch (_status) {
case STATUS_RESETTING:
if (reset()== -1)
{
_5DT_ERROR ("vrpn_Analog_5dt: Cannot reset the glove!");
}
break;
case STATUS_READING:
{
// It turns out to be important to get the report before checking
// to see if it has been too long since the last report. This is
// because there is the possibility that some other device running
// in the same server may have taken a long time on its last pass
// through mainloop(). Trackers that are resetting do this. When
// this happens, you can get an infinite loop -- where one tracker
// resets and causes the other to timeout, and then it returns the
// favor. By checking for the report here, we reset the timestamp
// if there is a report ready (ie, if THIS device is still operating).
get_report();
struct timeval current_time;
vrpn_gettimeofday(¤t_time, NULL);
if ( duration(current_time,timestamp) > MAX_TIME_INTERVAL)
{
sprintf (l_errmsg, "vrpn_5dt16::mainloop: Timeout... current_time=%ld:%ld, timestamp=%ld:%ld",
current_time.tv_sec,
static_cast<long>(current_time.tv_usec),
timestamp.tv_sec,
static_cast<long>(timestamp.tv_usec));
_5DT_ERROR (l_errmsg);
_status = STATUS_RESETTING;
}
}
break;
default:
_5DT_ERROR ("vrpn_5dt16::mainloop: Unknown mode (internal error)");
break;
}
}
vrpn_Button_5DT_Server::vrpn_Button_5DT_Server(const char *name, const char *deviceName,vrpn_Connection *c, double threshold[16]) : vrpn_Button_Filter(name, c)
{
num_buttons = 16;
for(int i=0;i<num_buttons;i++) {
m_threshold[i]=threshold[i];
}
d_5dt_button=new vrpn_Analog_Remote(deviceName, d_connection);
#ifdef VERBOSE
printf("vrpn_Button_5DT_Server: Adding local analog %s\n",name);
#endif
// Set up the callback handler for the channel
d_5dt_button->register_change_handler(this, handle_analog_update);
}
vrpn_Button_5DT_Server::~vrpn_Button_5DT_Server(void)
{
}
// This routine handles updates of the analog values. The value coming in is
// adjusted per the parameters in the full axis description, and then used to
// update the value there. The value is used by the matrix-generation code in
// mainloop() to update the transformations; that work is not done here.
void vrpn_Button_5DT_Server::handle_analog_update (void *userdata, const vrpn_ANALOGCB info)
{
vrpn_Button_5DT_Server *full = (vrpn_Button_5DT_Server *)userdata;
for(int i=0;i<16;i++) { // adjust with range and center value
if(info.channel[i]>full->m_threshold[i] ) {
full->buttons[i]=true;
} else {
full->buttons[i]=false;
}
}
}
void vrpn_Button_5DT_Server::mainloop()
{
struct timeval current_time;
// Call the generic server mainloop, since we are a server
server_mainloop();
d_5dt_button->mainloop();
vrpn_gettimeofday(¤t_time, NULL);
// Send reports. Stays the same in a real server.
report_changes();
}