forked from anliyuan/Ultralight-Digital-Human
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtrain.py
140 lines (121 loc) · 5.25 KB
/
train.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
import argparse
import os
import cv2
import torch
import numpy as np
import torch.nn as nn
from torch import optim
from tqdm import tqdm
from torch.utils.data import DataLoader
from datasetsss import MyDataset
from syncnet import SyncNet_color
from unet import Model
import random
import torchvision.models as models
def get_args():
parser = argparse.ArgumentParser(description='Train',
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument('--use_syncnet', action='store_true', help="if use syncnet, you need to set 'syncnet_checkpoint'")
parser.add_argument('--syncnet_checkpoint', type=str, default="")
parser.add_argument('--dataset_dir', type=str)
parser.add_argument('--save_dir', type=str, help="trained model save path.")
parser.add_argument('--see_res', action='store_true')
parser.add_argument('--epochs', type=int, default=200)
parser.add_argument('--batchsize', type=int, default=1)
parser.add_argument('--lr', type=float, default=0.001)
parser.add_argument('--asr', type=str, default="hubert")
return parser.parse_args()
args = get_args()
use_syncnet = args.use_syncnet
# Loss functions
class PerceptualLoss():
def contentFunc(self):
conv_3_3_layer = 14
cnn = models.vgg19(pretrained=True).features
cnn = cnn.cuda()
model = nn.Sequential()
model = model.cuda()
for i,layer in enumerate(list(cnn)):
model.add_module(str(i),layer)
if i == conv_3_3_layer:
break
return model
def __init__(self, loss):
self.criterion = loss
self.contentFunc = self.contentFunc()
def get_loss(self, fakeIm, realIm):
f_fake = self.contentFunc.forward(fakeIm)
f_real = self.contentFunc.forward(realIm)
f_real_no_grad = f_real.detach()
loss = self.criterion(f_fake, f_real_no_grad)
return loss
logloss = nn.BCELoss()
def cosine_loss(a, v, y):
d = nn.functional.cosine_similarity(a, v)
loss = logloss(d.unsqueeze(1), y)
return loss
def train(net, epoch, batch_size, lr):
content_loss = PerceptualLoss(torch.nn.MSELoss())
if use_syncnet:
if args.syncnet_checkpoint == "":
raise ValueError("Using syncnet, you need to set 'syncnet_checkpoint'.Please check README")
syncnet = SyncNet_color(args.asr).eval().cuda()
syncnet.load_state_dict(torch.load(args.syncnet_checkpoint))
save_dir= args.save_dir
if not os.path.exists(save_dir):
os.mkdir(save_dir)
dataloader_list = []
dataset_list = []
dataset_dir_list = [args.dataset_dir]
for dataset_dir in dataset_dir_list:
dataset = MyDataset(dataset_dir, args.asr)
train_dataloader = DataLoader(dataset, batch_size=16, shuffle=True, drop_last=False, num_workers=4)
dataloader_list.append(train_dataloader)
dataset_list.append(dataset)
optimizer = optim.Adam(net.parameters(), lr=lr)
criterion = nn.L1Loss()
for e in range(epoch):
net.train()
random_i = random.randint(0, len(dataset_dir_list)-1)
dataset = dataset_list[random_i]
train_dataloader = dataloader_list[random_i]
with tqdm(total=len(dataset), desc=f'Epoch {e + 1}/{epoch}', unit='img') as p:
for batch in train_dataloader:
imgs, labels, audio_feat = batch
imgs = imgs.cuda()
labels = labels.cuda()
audio_feat = audio_feat.cuda()
preds = net(imgs, audio_feat)
if use_syncnet:
y = torch.ones([preds.shape[0],1]).float().cuda()
a, v = syncnet(preds, audio_feat)
sync_loss = cosine_loss(a, v, y)
loss_PerceptualLoss = content_loss.get_loss(preds, labels)
loss_pixel = criterion(preds, labels)
if use_syncnet:
loss = loss_pixel + loss_PerceptualLoss*0.01 + 10*sync_loss
else:
loss = loss_pixel + loss_PerceptualLoss*0.01
p.set_postfix(**{'loss (batch)': loss.item()})
optimizer.zero_grad(set_to_none=True)
loss.backward()
optimizer.step()
p.update(imgs.shape[0])
if e % 5 == 0:
torch.save(net.state_dict(), os.path.join(save_dir, str(e)+'.pth'))
if args.see_res:
net.eval()
img_concat_T, img_real_T, audio_feat = dataset.__getitem__(random.randint(0, dataset.__len__()))
img_concat_T = img_concat_T[None].cuda()
audio_feat = audio_feat[None].cuda()
with torch.no_grad():
pred = net(img_concat_T, audio_feat)[0]
pred = pred.cpu().numpy().transpose(1,2,0)*255
pred = np.array(pred, dtype=np.uint8)
img_real = img_real_T.numpy().transpose(1,2,0)*255
img_real = np.array(img_real, dtype=np.uint8)
cv2.imwrite("./train_tmp_img/epoch_"+str(e)+".jpg", pred)
cv2.imwrite("./train_tmp_img/epoch_"+str(e)+"_real.jpg", img_real)
if __name__ == '__main__':
net = Model(6, args.asr).cuda()
train(net, args.epochs, args.batchsize, args.lr)