-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlibfreenect.hpp
253 lines (246 loc) · 9.18 KB
/
libfreenect.hpp
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
/*
* This file is part of the OpenKinect Project. http://www.openkinect.org
*
* Copyright (c) 2010 individual OpenKinect contributors. See the CONTRIB file
* for details.
*
* This code is licensed to you under the terms of the Apache License, version
* 2.0, or, at your option, the terms of the GNU General Public License,
* version 2.0. See the APACHE20 and GPL2 files for the text of the licenses,
* or the following URLs:
* http://www.apache.org/licenses/LICENSE-2.0
* http://www.gnu.org/licenses/gpl-2.0.txt
*
* If you redistribute this file in source form, modified or unmodified, you
* may:
* 1) Leave this header intact and distribute it under the same terms,
* accompanying it with the APACHE20 and GPL20 files, or
* 2) Delete the Apache 2.0 clause and accompany it with the GPL2 file, or
* 3) Delete the GPL v2 clause and accompany it with the APACHE20 file
* In all cases you must keep the copyright notice intact and include a copy
* of the CONTRIB file.
*
* Binary distributions must follow the binary distribution requirements of
* either License.
*/
#pragma once
#include "libfreenect/libfreenect.h"
#include <stdexcept>
#include <sstream>
#include <map>
#include <pthread.h>
#include "libusb-1.0/libusb.h"
namespace Freenect {
class Noncopyable {
public:
Noncopyable() {}
~Noncopyable() {}
private:
Noncopyable( const Noncopyable& );
const Noncopyable& operator=( const Noncopyable& );
};
class FreenectTiltState {
friend class FreenectDevice;
FreenectTiltState(freenect_raw_tilt_state *_state):
m_code(_state->tilt_status), m_state(_state)
{}
public:
void getAccelerometers(double* x, double* y, double* z) {
freenect_get_mks_accel(m_state, x, y, z);
}
double getTiltDegs() {
return freenect_get_tilt_degs(m_state);
}
public:
freenect_tilt_status_code m_code;
private:
freenect_raw_tilt_state *m_state;
};
class FreenectDevice : Noncopyable {
public:
FreenectDevice(freenect_context *_ctx, int _index)
: m_video_resolution(FREENECT_RESOLUTION_MEDIUM), m_depth_resolution(FREENECT_RESOLUTION_MEDIUM)
{
if(freenect_open_device(_ctx, &m_dev, _index) < 0) throw std::runtime_error("Cannot open Kinect");
freenect_set_user(m_dev, this);
setVideoFormat(FREENECT_VIDEO_RGB, FREENECT_RESOLUTION_MEDIUM);
setDepthFormat(FREENECT_DEPTH_11BIT, FREENECT_RESOLUTION_MEDIUM);
freenect_set_depth_callback(m_dev, freenect_depth_callback);
freenect_set_video_callback(m_dev, freenect_video_callback);
}
virtual ~FreenectDevice() {
if(freenect_close_device(m_dev) < 0){} //FN_WARNING("Device did not shutdown in a clean fashion");
}
void startVideo() {
if(freenect_start_video(m_dev) < 0) throw std::runtime_error("Cannot start RGB callback");
}
void stopVideo() {
if(freenect_stop_video(m_dev) < 0) throw std::runtime_error("Cannot stop RGB callback");
}
void startDepth() {
if(freenect_start_depth(m_dev) < 0) throw std::runtime_error("Cannot start depth callback");
}
void stopDepth() {
if(freenect_stop_depth(m_dev) < 0) throw std::runtime_error("Cannot stop depth callback");
}
void setTiltDegrees(double _angle) {
if(freenect_set_tilt_degs(m_dev, _angle) < 0) throw std::runtime_error("Cannot set angle in degrees");
}
void setLed(freenect_led_options _option) {
if(freenect_set_led(m_dev, _option) < 0) throw std::runtime_error("Cannot set led");
}
void updateState() {
if (freenect_update_tilt_state(m_dev) < 0) throw std::runtime_error("Cannot update device state");
}
FreenectTiltState getState() const {
return FreenectTiltState(freenect_get_tilt_state(m_dev));
}
void setVideoFormat(freenect_video_format requested_format, freenect_resolution requested_resolution = FREENECT_RESOLUTION_MEDIUM) {
if (requested_format != m_video_format || requested_resolution != m_video_resolution) {
bool wasRunning = (freenect_stop_video(m_dev) >= 0);
freenect_frame_mode mode = freenect_find_video_mode(requested_resolution, requested_format);
if (!mode.is_valid) throw std::runtime_error("Cannot set video format: invalid mode");
if (freenect_set_video_mode(m_dev, mode) < 0) throw std::runtime_error("Cannot set video format");
if (wasRunning)
freenect_start_video(m_dev);
m_video_format = requested_format;
m_video_resolution = requested_resolution;
}
}
freenect_video_format getVideoFormat() {
return m_video_format;
}
freenect_resolution getVideoResolution() {
return m_video_resolution;
}
void setDepthFormat(freenect_depth_format requested_format, freenect_resolution requested_resolution = FREENECT_RESOLUTION_MEDIUM) {
if (requested_format != m_depth_format || requested_resolution != m_depth_resolution) {
bool wasRunning = (freenect_stop_depth(m_dev) >= 0);
freenect_frame_mode mode = freenect_find_depth_mode(requested_resolution, requested_format);
if (!mode.is_valid) throw std::runtime_error("Cannot set depth format: invalid mode");
if (freenect_set_depth_mode(m_dev, mode) < 0) throw std::runtime_error("Cannot set depth format");
if (wasRunning)
freenect_start_depth(m_dev);
m_depth_format = requested_format;
m_depth_resolution = requested_resolution;
}
}
freenect_depth_format getDepthFormat() {
return m_depth_format;
}
freenect_resolution getDepthResolution() {
return m_depth_resolution;
}
int setFlag(freenect_flag flag, bool value)
{
return freenect_set_flag(m_dev, flag, value ? FREENECT_ON : FREENECT_OFF);
}
const freenect_device *getDevice() {
return m_dev;
}
// Do not call directly even in child
virtual void VideoCallback(void *video, uint32_t timestamp) { }
// Do not call directly even in child
virtual void DepthCallback(void *depth, uint32_t timestamp) { }
protected:
int getVideoBufferSize(){
switch(m_video_format) {
case FREENECT_VIDEO_RGB:
case FREENECT_VIDEO_BAYER:
case FREENECT_VIDEO_IR_8BIT:
case FREENECT_VIDEO_IR_10BIT:
case FREENECT_VIDEO_IR_10BIT_PACKED:
case FREENECT_VIDEO_YUV_RGB:
case FREENECT_VIDEO_YUV_RAW:
return freenect_find_video_mode(m_video_resolution, m_video_format).bytes;
default:
return 0;
}
}
int getDepthBufferSize(){
return freenect_get_current_depth_mode(m_dev).bytes;
}
private:
freenect_device *m_dev;
freenect_video_format m_video_format;
freenect_depth_format m_depth_format;
freenect_resolution m_video_resolution;
freenect_resolution m_depth_resolution;
static void freenect_depth_callback(freenect_device *dev, void *depth, uint32_t timestamp) {
FreenectDevice* device = static_cast<FreenectDevice*>(freenect_get_user(dev));
device->DepthCallback(depth, timestamp);
}
static void freenect_video_callback(freenect_device *dev, void *video, uint32_t timestamp) {
FreenectDevice* device = static_cast<FreenectDevice*>(freenect_get_user(dev));
device->VideoCallback(video, timestamp);
}
};
class Freenect : Noncopyable {
private:
typedef std::map<int, FreenectDevice*> DeviceMap;
public:
Freenect() : m_stop(false) {
if(freenect_init(&m_ctx, NULL) < 0) throw std::runtime_error("Cannot initialize freenect library");
// We claim both the motor and camera devices, since this class exposes both.
// It does not support audio, so we do not claim it.
freenect_select_subdevices(m_ctx, static_cast<freenect_device_flags>(FREENECT_DEVICE_MOTOR | FREENECT_DEVICE_CAMERA));
if(pthread_create(&m_thread, NULL, pthread_callback, (void*)this) != 0) throw std::runtime_error("Cannot initialize freenect thread");
}
~Freenect() {
m_stop = true;
for(DeviceMap::iterator it = m_devices.begin() ; it != m_devices.end() ; ++it) {
delete it->second;
}
pthread_join(m_thread, NULL);
if(freenect_shutdown(m_ctx) < 0){} //FN_WARNING("Freenect did not shutdown in a clean fashion");
}
template <typename ConcreteDevice>
ConcreteDevice& createDevice(int _index) {
DeviceMap::iterator it = m_devices.find(_index);
if (it != m_devices.end()) delete it->second;
ConcreteDevice * device = new ConcreteDevice(m_ctx, _index);
m_devices[_index] = device;
return *device;
}
void deleteDevice(int _index) {
DeviceMap::iterator it = m_devices.find(_index);
if (it == m_devices.end()) return;
delete it->second;
m_devices.erase(it);
}
int deviceCount() {
return freenect_num_devices(m_ctx);
}
// Do not call directly, thread runs here
void operator()() {
while (!m_stop) {
static timeval timeout = { 1, 0 };
int res = freenect_process_events_timeout(m_ctx, &timeout);
if (res < 0)
{
// libusb signals an error has occurred
if (res == LIBUSB_ERROR_INTERRUPTED)
{
// This happens sometimes, it means that a system call in libusb was interrupted somehow (perhaps due to a signal)
// The simple solution seems to be just ignore it.
continue;
}
std::stringstream ss;
ss << "Cannot process freenect events (libusb error code: " << res << ")";
throw std::runtime_error(ss.str());
}
}
}
static void *pthread_callback(void *user_data) {
Freenect* freenect = static_cast<Freenect*>(user_data);
(*freenect)();
return NULL;
}
protected:
freenect_context *m_ctx;
private:
volatile bool m_stop;
pthread_t m_thread;
DeviceMap m_devices;
};
}