-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrun_predict.py
141 lines (119 loc) · 6.74 KB
/
run_predict.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
import os
import torch
import torchvision
import numpy as np
import argparse
from models.poseMF_shapeGaussian_net import PoseMFShapeGaussianNet
from models.smpl_official import SMPL
from models.pose2D_hrnet import PoseHighResolutionNet
from models.canny_edge_detector import CannyEdgeDetector
from models.parametric.tn import ParametricModel
from configs.poseMF_shapeGaussian_net_config import get_cfg_defaults
from configs.pose2D_hrnet_config import get_pose2D_hrnet_cfg_defaults
from configs import paths
from predict.predict_poseMF_shapeGaussian_net import predict_poseMF_shapeGaussian_net
from utils.garment_classes import GarmentClasses
def run_predict(device,
image_dir,
save_dir,
pose_shape_weights_path,
pose2D_hrnet_weights_path,
pose_shape_cfg_path=None,
already_cropped_images=False,
visualise_samples=False,
visualise_uncropped=False,
joints2Dvisib_threshold=0.75,
gender='neutral'):
# ------------------------- Models -------------------------
# Configs
pose2D_hrnet_cfg = get_pose2D_hrnet_cfg_defaults()
pose_shape_cfg = get_cfg_defaults()
if pose_shape_cfg_path is not None:
pose_shape_cfg.merge_from_file(pose_shape_cfg_path)
print('\nLoaded Distribution Predictor config from', pose_shape_cfg_path)
else:
print('\nUsing default Distribution Predictor config.')
# Bounding box / Object detection model
if not already_cropped_images:
object_detect_model = torchvision.models.detection.maskrcnn_resnet50_fpn(pretrained=True).to(device)
else:
object_detect_model = None
# HRNet model for 2D joint detection
hrnet_model = PoseHighResolutionNet(pose2D_hrnet_cfg).to(device)
hrnet_checkpoint = torch.load(pose2D_hrnet_weights_path, map_location=device)
hrnet_model.load_state_dict(hrnet_checkpoint, strict=False)
print('\nLoaded HRNet weights from', pose2D_hrnet_weights_path)
# Edge detector
edge_detect_model = CannyEdgeDetector(non_max_suppression=pose_shape_cfg.DATA.EDGE_NMS,
gaussian_filter_std=pose_shape_cfg.DATA.EDGE_GAUSSIAN_STD,
gaussian_filter_size=pose_shape_cfg.DATA.EDGE_GAUSSIAN_SIZE,
threshold=pose_shape_cfg.DATA.EDGE_THRESHOLD).to(device)
# SMPL model
print('\nUsing {} SMPL model with {} shape parameters.'.format(gender, str(pose_shape_cfg.MODEL.NUM_SMPL_BETAS)))
smpl_model = SMPL(paths.SMPL,
batch_size=1,
gender=gender,
num_betas=pose_shape_cfg.MODEL.NUM_SMPL_BETAS).to(device)
smpl_immediate_parents = smpl_model.parents.tolist()
upper_class='t-shirt'
lower_class='pant'
parametric_model = ParametricModel(gender=gender,
garment_classes=GarmentClasses(
upper_class=upper_class,
lower_class=lower_class
),
eval=True)
# 3D shape and pose distribution predictor
pose_shape_dist_model = PoseMFShapeGaussianNet(smpl_parents=smpl_immediate_parents,
config=pose_shape_cfg).to(device)
checkpoint = torch.load(pose_shape_weights_path, map_location=device)
pose_shape_dist_model.load_state_dict(checkpoint['best_model_state_dict'])
print('\nLoaded Distribution Predictor weights from', pose_shape_weights_path)
# ------------------------- Predict -------------------------
torch.manual_seed(0)
np.random.seed(0)
predict_poseMF_shapeGaussian_net(pose_shape_model=pose_shape_dist_model,
pose_shape_cfg=pose_shape_cfg,
smpl_model=smpl_model,
parametric_model=parametric_model,
hrnet_model=hrnet_model,
hrnet_cfg=pose2D_hrnet_cfg,
edge_detect_model=edge_detect_model,
device=device,
image_dir=image_dir,
save_dir=save_dir,
object_detect_model=object_detect_model,
joints2Dvisib_threshold=joints2Dvisib_threshold,
visualise_uncropped=visualise_uncropped,
visualise_samples=visualise_samples)
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument('--image_dir', '-I', type=str, help='Path to directory of test images.')
parser.add_argument('--save_dir', '-S', type=str, help='Path to directory where test outputs will be saved.')
parser.add_argument('--pose_shape_weights', '-W3D', type=str, default='/data/hierprob3d/poseMF_shapeGaussian_net_weights.tar')
parser.add_argument('--pose_shape_cfg', type=str, default=None)
parser.add_argument('--pose2D_hrnet_weights', '-W2D', type=str, default='/data/hierprob3d/pose_hrnet_w48_384x288.pth')
parser.add_argument('--cropped_images', '-C', action='store_true', help='Images already cropped and centred.')
parser.add_argument('--visualise_samples', '-VS', action='store_true')
parser.add_argument('--visualise_uncropped', '-VU', action='store_true')
parser.add_argument('--joints2Dvisib_threshold', '-T', type=float, default=0.75)
parser.add_argument('--gender', '-G', type=str, default='neutral', choices=['neutral', 'male', 'female'], help='Gendered SMPL models may be used.')
parser.add_argument('--gpu', type=int, default=0)
args = parser.parse_args()
os.environ["CUDA_DEVICE_ORDER"] = "PCI_BUS_ID" # see issue #152
os.environ["CUDA_VISIBLE_DEVICES"] = str(args.gpu)
device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")
print('\nDevice: {}'.format(device))
if not os.path.exists(args.save_dir):
os.makedirs(args.save_dir)
run_predict(device=device,
image_dir=args.image_dir,
save_dir=args.save_dir,
pose_shape_weights_path=args.pose_shape_weights,
pose_shape_cfg_path=args.pose_shape_cfg,
pose2D_hrnet_weights_path=args.pose2D_hrnet_weights,
already_cropped_images=args.cropped_images,
visualise_samples=args.visualise_samples,
visualise_uncropped=args.visualise_uncropped,
joints2Dvisib_threshold=args.joints2Dvisib_threshold,
gender=args.gender)