-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathevent_to_frame_FE108.py
202 lines (183 loc) · 9.26 KB
/
event_to_frame_FE108.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
import os
import pdb
import shutil
# import csv
import numpy as np
import cv2
import torch
import pandas as pd
from tqdm import tqdm
import scipy.io
from spconv.pytorch.utils import PointToVoxel
from dv import AedatFile
import numpy as np
def transform_points_to_voxels(data_dict={}, voxel_generator=None, device=torch.device("cuda:0")):
"""
将点云转换为voxel,调用spconv的VoxelGeneratorV2
"""
points = data_dict['points']
# 将points打乱
shuffle_idx = np.random.permutation(points.shape[0])
points = points[shuffle_idx]
data_dict['points'] = points
# 使用spconv生成voxel输出
points = torch.as_tensor(data_dict['points']).to(device)
voxel_output = voxel_generator(points)
# 假设一份点云数据是N*4,那么经过pillar生成后会得到三份数据
# voxels代表了每个生成的voxel数据,维度是[M, 5, 4]
# coordinates代表了每个生成的voxel所在的zyx轴坐标,维度是[M,3]
# num_points代表了每个生成的voxel中有多少个有效的点维度是[m,],因为不满5会被0填充
voxels, coordinates, num_points = voxel_output
voxels = voxels.to(device)
coordinates = coordinates.to(device)
num_points = num_points.to(device)
# 选event数量在前5000的voxel 8000 from(4k+,6k+)
# print(torch.where(num_points>=16)[0].shape)
if num_points.shape[0] < save_voxel:
features = voxels[:, :, 3]
coor = coordinates[:, :]
else:
_, voxels_idx = torch.topk(num_points, save_voxel)
# 将每个voxel的1024个p拼接作为voxel初始特征 16
features = voxels[voxels_idx][:, :, 3]
# 前5000个voxel的三维坐标
coor = coordinates[voxels_idx]
# 将y.x.t改为t,x,y
coor[:, [0, 1, 2]] = coor[:, [2, 1, 0]]
return coor, features
if __name__ == '__main__':
os.environ['CUDA_VISIBLE_DEVICES'] = "5"
save_voxel = 10000
device = torch.device("cuda:0")
data_path = r"/home/ioe/xxxxx/EventTracking/FE108/train"
save_path = r"/home/ioe/xxxxx/FE108/train"
# data_path = r"/home/ioe/xxxxx/EventTracking/FE108/train"
# save_path = r"/home/ioe/xxxxx/FE108/train"
video_files = os.listdir(data_path)
dvs_img_interval = 1
voxel_generator = PointToVoxel(
# 给定每个voxel的长宽高 [0.05, 0.05, 0.1]
vsize_xyz=[50, 10, 10], # [0.2, 0.25, 0.16] # [50, 10, 10] [50, 35, 26] 因此坐标范围(20,20,20) (20, 34/35, 26)
# 给定点云的范围 [ 0. -40. -3. 70.4 40. 1. ]
coors_range_xyz=[0, 0, 0, 1000, 345, 259],
# 给定每个点云的特征维度,这里是x,y,z,r 其中r是激光雷达反射强度 # 346x260 t,x,y
num_point_features=4,
# 最多选取多少个voxel,训练16000,推理40000
max_num_voxels=16000, # 16000
# 给定每个pillar中有采样多少个点,不够则补0 因此我将neg voxel改为-1;
max_num_points_per_voxel=16, # 1024
device=device
)
for videoID in range(len(video_files)):
foldName = video_files[videoID]
print("==>> foldName: ", foldName)
fileLIST = os.listdir(os.path.join(data_path, foldName))
if not os.path.exists(os.path.join(save_path, foldName)):
os.mkdir(os.path.join(save_path, foldName))
mat_save = os.path.join(save_path, foldName, 'voxel/')
if not os.path.exists(mat_save):
# shutil.rmtree(mat_save)
os.mkdir(mat_save)
# else:
# continue
save_path_aps = os.path.join(save_path, foldName, 'aps/')
if not os.path.exists(save_path_aps):
# shutil.rmtree(save_path_aps)
os.mkdir(save_path_aps)
# else:
# continue
save_path_dvs = os.path.join(save_path, foldName, 'dvs/')
if not os.path.exists(save_path_dvs):
# shutil.rmtree(save_path_dvs)
os.mkdir(save_path_dvs)
gt_path = os.path.join(data_path, foldName, 'groundtruth_rect.txt')
save_gt_path = os.path.join(save_path, foldName, 'groundtruth_rect.txt')
shutil.copyfile(gt_path, save_gt_path)
aedat4_file = 'events.aedat4'
# print('filename:', aedat4_file)
read_path = os.path.join(data_path, foldName, aedat4_file)
# read aeda4;
frame_all = []
frame_exposure_time = []
frame_interval_time = []
match_file = '/home/ioe/xxxxx/EventTracking/FE108/pair.txt'
pair = {}
with open(match_file, 'r') as f:
for line in f.readlines():
file, start_frame = line.split()
pair[file] = int(start_frame) + 1
start_frame = pair[foldName]
img_path = os.path.join(data_path, foldName, 'img')
frame_end = len(os.listdir(img_path))
with AedatFile(read_path) as f:
# print(f.names)
for frame in f['frames']:
frame_all.append(frame.image)
frame_interval_time.append([frame.timestamp_start_of_frame,
frame.timestamp_end_of_frame]) ## [1607928583387944, 1607928583410285]
frame_timestamp = frame_interval_time
frame_num = len(frame_timestamp)
events = np.hstack([packet for packet in f['events'].numpy()])
t_all = torch.tensor(events['timestamp']).unsqueeze(1).to(device)
x_all = torch.tensor(events['x']).unsqueeze(1).to(device)
y_all = torch.tensor(events['y']).unsqueeze(1).to(device)
p_all = torch.tensor(events['polarity']).unsqueeze(1).to(device)
# for frame_no in range(start_frame, int(frame_num / dvs_img_interval) - 1):
begin_frame = start_frame-1
end_frame = frame_end + start_frame-1
for frame_no in range(begin_frame, end_frame):
start_idx = np.where(events['timestamp'] >= frame_timestamp[frame_no][0])[0][0]
end_idx = np.where(events['timestamp'] >= frame_timestamp[frame_no][1])[0][0]
sub_event = events[start_idx:end_idx]
idx_length = end_idx - start_idx
# t = t_all[start_idx: end_idx]
# if start_idx == end_idx:
# time_length = 0
# t = ((t-t).float() / time_length) * 1000
# scipy.io.savemat(mat_save + 'frame{:0>4d}.mat'.format(frame_no-begin_frame), mdict={'coor': np.zeros([100, 3]),
# 'features': np.zeros([100, 16])}) # coor: numpy.nan; features: numpy.nan
# print('empty event frame ', frame_no)
# continue
# else:
# time_length = t[-1] - t[0]
# # rescale the timestampes to start from 0 up to 1000
# t = ((t-t[0]).float() / time_length) * 1000
# all_idx = np.where(sub_event['polarity'] != -1) # all event
# neg_idx = np.where(sub_event['polarity'] == 0) # neg event
# t = t[all_idx]
# x = x_all[all_idx]
# y = y_all[all_idx]
# p = p_all[all_idx]
# p[neg_idx] = -1 # negtive voxel change from 0 to -1. because after append 0 operation.
# current_events = torch.cat((t, x, y, p), dim=1)
# # if current_events.shape[0] < 10: # remove it
# # continue
# data_dict = {'points': current_events}
# coor, features = transform_points_to_voxels(data_dict=data_dict, voxel_generator=voxel_generator,
# device=device)
# pdb.set_trace()
# coor = coor.cpu().numpy()
# features = features.cpu().numpy()
# # print('coor', coor)
# scipy.io.savemat(mat_save + 'frame{:0>4d}.mat'.format(frame_no-begin_frame), mdict={'coor': coor, 'features': features}) # coor: Nx(t,x,y); features:Nx32 or Nx10024
#
# ## RGB image generate
# this_frame = frame_all[frame_no]
# height, width = f['events'].size
# event_img = np.zeros((height, width))
# cv2.imwrite(os.path.join(save_path_aps, 'frame{:04d}'.format(frame_no-begin_frame)+'.png'), this_frame)
# ## cv2.imshow('APS Image', this_frame)
# ## cv2.waitKey(1)
# event img generate
height, width = f['events'].size
event_frame = 255 * np.ones((height, width, 3), dtype=np.uint8)
start_idx = np.where(events['timestamp'] >= frame_timestamp[frame_no][0])[0][0]
end_idx = np.where(events['timestamp'] >= frame_timestamp[frame_no][1])[0][0]
event = events[start_idx:end_idx]
on_idx = np.where(event['polarity'] == 1) ## (array([ 3, 4, 5, ..., 10633, 10635, 10636]),)
off_idx = np.where(event['polarity'] == 0) ## (array([ 0, 1, 2, ..., 10629, 10632, 10634]),)
event_frame[event['y'][on_idx], event['x'][on_idx], :] = [30, 30, 220] * event['polarity'][on_idx][:, None]
event_frame[event['y'][off_idx], event['x'][off_idx], :] = [200, 30, 30] * (event['polarity'][off_idx] + 1)[
:, None]
cv2.imwrite(os.path.join(save_path_dvs, 'frame{:04d}'.format((frame_no-begin_frame) * dvs_img_interval) + '.png'),
event_frame)