-
Notifications
You must be signed in to change notification settings - Fork 47
/
Copy pathexport_point_cloud.py
executable file
·128 lines (89 loc) · 4.68 KB
/
export_point_cloud.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
#! /usr/bin/env python3
import pickle
from distutils.util import strtobool
import ace_vis_util as vutil
import argparse
from pathlib import Path
import logging
import torch
from ace_network import Regressor
from dataset import CamLocDataset
from torch.utils.data import DataLoader
logging.basicConfig(level=logging.INFO)
_logger = logging.getLogger(__name__)
def _strtobool(x):
return bool(strtobool(x))
if __name__ == '__main__':
parser = argparse.ArgumentParser(
description='Extract point cloud from network (slow) or visualization buffer file (fast). '
'File ending determines output format where txt and ply are supported.',
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument('output_file', type=Path)
parser.add_argument('--network', type=Path, help='network to extract point cloud from.')
parser.add_argument('--pose_file', type=Path, help='pose file of images that trained the network')
parser.add_argument('--visualization_buffer', type=Path,
help='Vis buffer files that contains a pre-calculated point cloud.')
parser.add_argument('--encoder_path', type=Path, default=Path(__file__).parent / "ace_encoder_pretrained.pt",
help='file containing pre-trained encoder weights')
parser.add_argument('--image_resolution', type=int, default=480, help='base image resolution')
parser.add_argument('--confidence_threshold', type=int, default=500)
parser.add_argument('--convention', type=str, default='opengl', choices=['opengl', 'opencv'],
help='coordinate convention of the point cloud')
parser.add_argument('--dense_point_cloud', type=_strtobool, default=False,
help='do not filter points based on reprojection error, '
'bad for visualisation but good to initialise splats')
opt = parser.parse_args()
if opt.visualization_buffer is None and (opt.network is None or opt.pose_file is None):
parser.error("You must provide either a visualization buffer or network and pose file.")
if opt.dense_point_cloud and opt.visualization_buffer is not None:
parser.error("A dense cloud cannot be extracted from a visualization buffer. "
"Please provide network and pose file.")
device = torch.device("cuda")
if opt.visualization_buffer is None:
_logger.info("Extracting point cloud from network.")
# Load network weights.
encoder_state_dict = torch.load(opt.encoder_path, map_location="cpu")
_logger.info(f"Loaded encoder from: {opt.encoder_path}")
head_state_dict = torch.load(opt.network, map_location="cpu")
_logger.info(f"Loaded head weights from: {opt.network}")
# Create regressor.
network = Regressor.create_from_split_state_dict(encoder_state_dict, head_state_dict)
# Setup for evaluation.
network = network.to(device)
network.eval()
# Setup dataset.
dataset = CamLocDataset(
rgb_files=None,
image_short_size=opt.image_resolution,
ace_pose_file=opt.pose_file,
ace_pose_file_conf_threshold=opt.confidence_threshold
)
_logger.info(f'Images found: {len(dataset)}')
# Setup dataloader. Batch size 1 by default.
data_loader = DataLoader(dataset, shuffle=False, num_workers=6)
pc_xyz, pc_clr = vutil.get_point_cloud_from_network(network, data_loader,
filter_depth=100, dense_cloud=opt.dense_point_cloud)
else:
_logger.info("Extracting point cloud from visualization buffer.")
with open(opt.visualization_buffer, "rb") as file:
state_dict = pickle.load(file)
pc_xyz = state_dict['map_xyz']
pc_clr = state_dict['map_clr']
if opt.convention == 'opencv':
# OpenGL to OpenCV convention
pc_xyz[:, 1] = -pc_xyz[:, 1]
pc_xyz[:, 2] = -pc_xyz[:, 2]
if opt.output_file.suffix == '.txt':
# write as txt file (ascii)
with open(opt.output_file, 'w') as f:
for pt in range(pc_xyz.shape[0]):
f.write(f"{pc_xyz[pt, 0]} {pc_xyz[pt, 1]} {pc_xyz[pt, 2]} "
f"{pc_clr[pt, 0]:.0f} {pc_clr[pt, 1]:.0f} {pc_clr[pt, 2]:.0f}\n")
elif opt.output_file.suffix == '.ply':
# write as ply (binary) via trimesh
import trimesh
cloud = trimesh.PointCloud(pc_xyz, colors=pc_clr)
cloud.export(opt.output_file)
else:
raise ValueError(f"Output file format {opt.output_file.suffix} not supported.")
_logger.info(f"Done. Wrote point cloud to: {opt.output_file}")