-
Notifications
You must be signed in to change notification settings - Fork 78
/
live_demo.py
208 lines (178 loc) · 8.29 KB
/
live_demo.py
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
import socket
import threading
from articulate.math import *
from datetime import datetime
import torch
import numpy as np
import config
import time
from net import TransPoseNet
from pygame.time import Clock
device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')
inertial_poser = TransPoseNet(num_past_frame=20, num_future_frame=5).to(device)
running = False
start_recording = False
class IMUSet:
r"""
Sensor order: left forearm, right forearm, left lower leg, right lower leg, head, pelvis
"""
def __init__(self, imu_host='127.0.0.1', imu_port=7002, buffer_len=26):
"""
Init an IMUSet for Noitom Perception Legacy IMUs. Please follow the instructions below.
Instructions:
--------
1. Start `Axis Legacy` (Noitom software).
2. Click `File` -> `Settings` -> `Broadcasting`, check `TCP` and `Calculation`. Set `Port` to 7002.
3. Click `File` -> `Settings` -> `Output Format`, change `Calculation Data` to
`Block type = String, Quaternion = Global, Acceleration = Sensor local`
4. Place 1 - 6 IMU on left lower arm, right lower arm, left lower leg, right lower leg, head, root.
5. Connect 1 - 6 IMU to `Axis Legacy` and continue.
:param imu_host: The host that `Axis Legacy` runs on.
:param imu_port: The port that `Axis Legacy` runs on.
:param buffer_len: Max number of frames in the readonly buffer.
"""
self.imu_host = imu_host
self.imu_port = imu_port
self.clock = Clock()
self._imu_socket = None
self._buffer_len = buffer_len
self._quat_buffer = []
self._acc_buffer = []
self._is_reading = False
self._read_thread = None
def _read(self):
"""
The thread that reads imu measurements into the buffer. It is a producer for the buffer.
"""
num_float_one_frame = 21 * 16 + 2
data = ''
while self._is_reading:
data += self._imu_socket.recv(1024).decode('ascii')
strs = data.split(' ', num_float_one_frame)
if len(strs) > num_float_one_frame:
# print(np.array(strs[:-3]).reshape((21, 16))) # full data
d = np.array(strs[:96]).reshape((6, 16)) # first 6 imus
tranc = int(len(self._quat_buffer) == self._buffer_len)
self._quat_buffer = self._quat_buffer[tranc:] + [d[:, 6:10].astype(float)]
self._acc_buffer = self._acc_buffer[tranc:] + [-d[:, 10:13].astype(float) * 9.8]
data = strs[-1]
self.clock.tick()
def start_reading(self):
"""
Start reading imu measurements into the buffer.
"""
if self._read_thread is None:
self._is_reading = True
self._read_thread = threading.Thread(target=self._read)
self._read_thread.setDaemon(True)
self._quat_buffer = []
self._acc_buffer = []
self._imu_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self._imu_socket.connect((self.imu_host, self.imu_port))
self._read_thread.start()
else:
print('Failed to start reading thread: reading is already start.')
def stop_reading(self):
"""
Stop reading imu measurements.
"""
if self._read_thread is not None:
self._is_reading = False
self._read_thread.join()
self._read_thread = None
self._imu_socket.close()
def get_current_buffer(self):
"""
Get a view of current buffer.
:return: Quaternion and acceleration torch.Tensor in shape [buffer_len, 6, 4] and [buffer_len, 6, 3].
"""
q = torch.tensor(self._quat_buffer, dtype=torch.float)
a = torch.tensor(self._acc_buffer, dtype=torch.float)
return q, a
def get_mean_measurement_of_n_second(self, num_seconds=3, buffer_len=120):
"""
Start reading for `num_seconds` seconds and then close the connection. The average of the last
`buffer_len` frames of the measured quaternions and accelerations are returned.
Note that this function is blocking.
:param num_seconds: How many seconds to read.
:param buffer_len: Buffer length. Must be smaller than 60 * `num_seconds`.
:return: The mean quaternion and acceleration torch.Tensor in shape [6, 4] and [6, 3] respectively.
"""
save_buffer_len = self._buffer_len
self._buffer_len = buffer_len
self.start_reading()
time.sleep(num_seconds)
self.stop_reading()
q, a = self.get_current_buffer()
self._buffer_len = save_buffer_len
return q.mean(dim=0), a.mean(dim=0)
def get_input():
global running, start_recording
while running:
c = input()
if c == 'q':
running = False
elif c == 'r':
start_recording = True
elif c == 's':
start_recording = False
if __name__ == '__main__':
imu_set = IMUSet(buffer_len=1)
input('Put imu 1 aligned with your body reference frame (x = Left, y = Up, z = Forward) and then press any key.')
print('Keep for 3 seconds ...', end='')
oris = imu_set.get_mean_measurement_of_n_second(num_seconds=3, buffer_len=200)[0][0]
smpl2imu = quaternion_to_rotation_matrix(oris).view(3, 3).t()
input('\tFinish.\nWear all imus correctly and press any key.')
for i in range(3, 0, -1):
print('\rStand straight in T-pose and be ready. The celebration will begin after %d seconds.' % i, end='')
time.sleep(1)
print('\rStand straight in T-pose. Keep the pose for 3 seconds ...', end='')
oris, accs = imu_set.get_mean_measurement_of_n_second(num_seconds=3, buffer_len=200)
oris = quaternion_to_rotation_matrix(oris)
device2bone = smpl2imu.matmul(oris).transpose(1, 2).matmul(torch.eye(3))
acc_offsets = smpl2imu.matmul(accs.unsqueeze(-1)) # [num_imus, 3, 1], already in global inertial frame
print('\tFinish.\nStart estimating poses. Press q to quit, r to record motion, s to stop recording.')
imu_set.start_reading()
server_for_unity = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_for_unity.bind(('127.0.0.1', 8888))
server_for_unity.listen(1)
print('Server start. Waiting for unity3d to connect.')
conn, addr = server_for_unity.accept()
running = True
clock = Clock()
is_recording = False
record_buffer = None
get_input_thread = threading.Thread(target=get_input)
get_input_thread.setDaemon(True)
get_input_thread.start()
while running:
# calibration
clock.tick(60)
ori_raw, acc_raw = imu_set.get_current_buffer() # [1, 6, 4], get measurements in running fps
ori_raw = quaternion_to_rotation_matrix(ori_raw).view(1, 6, 3, 3)
acc_cal = (smpl2imu.matmul(acc_raw.view(-1, 6, 3, 1)) - acc_offsets).view(1, 6, 3)
ori_cal = smpl2imu.matmul(ori_raw).matmul(device2bone)
# normalization
acc = torch.cat((acc_cal[:, :5] - acc_cal[:, 5:], acc_cal[:, 5:]), dim=1).bmm(ori_cal[:, -1]) / config.acc_scale
ori = torch.cat((ori_cal[:, 5:].transpose(2, 3).matmul(ori_cal[:, :5]), ori_cal[:, 5:]), dim=1)
data_nn = torch.cat((acc.view(-1, 18), ori.view(-1, 54)), dim=1).to(device)
pose, tran = inertial_poser.forward_online(data_nn)
pose = rotation_matrix_to_axis_angle(pose.view(1, 216)).view(72)
# recording
if not is_recording and start_recording:
record_buffer = data_nn.view(1, -1)
is_recording = True
elif is_recording and start_recording:
record_buffer = torch.cat([record_buffer, data_nn.view(1, -1)], dim=0)
elif is_recording and not start_recording:
torch.save(record_buffer, 'data/imu_recordings/r' + datetime.now().strftime('%T').replace(':', '-') + '.pt')
is_recording = False
# send pose
s = ','.join(['%g' % v for v in pose]) + '#' + \
','.join(['%g' % v for v in tran]) + '$'
conn.send(s.encode('utf8')) # I use unity3d to read pose and translation for visualization here
print('\r', '(recording)' if is_recording else '', 'Sensor FPS:', imu_set.clock.get_fps(),
'\tOutput FPS:', clock.get_fps(), end='')
get_input_thread.join()
imu_set.stop_reading()
print('Finish.')