forked from YBIGTA/pytorch-hair-segmentation
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathevaluate.py
142 lines (114 loc) · 4.58 KB
/
evaluate.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
import cv2
import numpy as np
import torch
import time
import os
import sys
import argparse
from PIL import Image
sys.path.append(os.path.dirname(os.path.dirname(__file__)))
from networks import get_network
from data import get_loader
import torchvision.transforms as std_trnsf
from utils import joint_transforms as jnt_trnsf
from utils.metrics import MultiThresholdMeasures
def str2bool(s):
return s.lower() in ('t', 'true', 1)
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument('--ckpt_dir', help='path to ckpt file',type=str,
default='./models/pspnet_resnet101_sgd_lr_0.002_epoch_100_test_iou_0.918.pth')
parser.add_argument('--dataset', type=str, default='figaro',
help='Name of dataset you want to use default is "figaro"')
parser.add_argument('--data_dir', help='path to Figaro1k folder', type=str, default='./data/Figaro1k')
parser.add_argument('--networks', help='name of neural network', type=str, default='pspnet_resnet101')
parser.add_argument('--save_dir', default='./overlay',
help='path to save overlay images, default=None and do not save images in this case')
parser.add_argument('--use_gpu', type=str2bool, default=True,
help='True if using gpu during inference')
args = parser.parse_args()
ckpt_dir = args.ckpt_dir
data_dir = args.data_dir
img_dir = os.path.join(data_dir, 'Original', 'Testing')
network = args.networks.lower()
save_dir = args.save_dir
device = 'cuda' if args.use_gpu else 'cpu'
assert os.path.exists(ckpt_dir)
assert os.path.exists(data_dir)
assert os.path.exists(os.path.split(save_dir)[0])
if not os.path.exists(save_dir):
os.mkdir(save_dir)
# prepare network with trained parameters
net = get_network(network).to(device)
state = torch.load(ckpt_dir)
net.load_state_dict(state['weight'])
# this is the default setting for train_verbose.py
test_joint_transforms = jnt_trnsf.Compose([
jnt_trnsf.Safe32Padding()
])
test_image_transforms = std_trnsf.Compose([
std_trnsf.ToTensor(),
std_trnsf.Normalize([0.485, 0.456, 0.406], [0.229, 0.224, 0.225])
])
# transforms only on mask
mask_transforms = std_trnsf.Compose([
std_trnsf.ToTensor()
])
test_loader = get_loader(dataset=args.dataset,
data_dir=data_dir,
train=False,
joint_transforms=test_joint_transforms,
image_transforms=test_image_transforms,
mask_transforms=mask_transforms,
batch_size=1,
shuffle=False,
num_workers=4)
# prepare measurements
metric = MultiThresholdMeasures()
metric.reset()
durations = list()
# prepare images
imgs = [os.path.join(img_dir, k) for k in sorted(os.listdir(img_dir)) if k.endswith('.jpg')]
with torch.no_grad():
for i, (data, label) in enumerate(test_loader):
print('[{:3d}/{:3d}] processing image... '.format(i+1, len(test_loader)))
net.eval()
data, label = data.to(device), label.to(device)
# inference
start = time.time()
logit = net(data)
duration = time.time() - start
# prepare mask
pred = torch.sigmoid(logit.cpu())[0][0].data.numpy()
mh, mw = data.size(2), data.size(3)
mask = pred >= 0.5
mask_n = np.zeros((mh, mw, 3))
mask_n[:,:,0] = 255
mask_n[:,:,0] *= mask
path = os.path.join(save_dir, "figaro_img_%04d.png" % i)
image_n = cv2.imread(imgs[i])
# discard padded area
ih, iw, _ = image_n.shape
delta_h = mh - ih
delta_w = mw - iw
top = delta_h // 2
bottom = mh - (delta_h - top)
left = delta_w // 2
right = mw - (delta_w - left)
mask_n = mask_n[top:bottom, left:right, :]
# addWeighted
image_n = image_n * 0.5 + mask_n * 0.5
# log measurements
metric.update((logit, label))
durations.append(duration)
# write overlay image
cv2.imwrite(path,image_n)
# compute measurements
iou = metric.compute_iou()
f = metric.compute_f1()
acc = metric.compute_accuracy()
avg_fps = len(durations)/sum(durations)
print('Avg-FPS:', avg_fps)
print('Pixel-acc:', acc)
print('F1-score:', f)
print('IOU:', iou)