Spaces:
Sleeping
Sleeping
# Modified from https://github.com/junyanz/pytorch-CycleGAN-and-pix2pix/blob/master/models/networks.py | |
import torch | |
import torch.nn as nn | |
import functools | |
class UnetSkipConnectionBlock(nn.Module): | |
"""Defines the Unet submodule with skip connection. | |
X -------------------identity---------------------- | |
|-- downsampling -- |submodule| -- upsampling --| | |
""" | |
def __init__(self, outer_nc, inner_nc, input_nc=None, | |
submodule=None, outermost=False, innermost=False, norm_layer=nn.BatchNorm2d, use_dropout=False): | |
"""Construct a Unet submodule with skip connections. | |
Parameters: | |
outer_nc (int) -- the number of filters in the outer conv layer | |
inner_nc (int) -- the number of filters in the inner conv layer | |
input_nc (int) -- the number of channels in input images/features | |
submodule (UnetSkipConnectionBlock) -- previously defined submodules | |
outermost (bool) -- if this module is the outermost module | |
innermost (bool) -- if this module is the innermost module | |
norm_layer -- normalization layer | |
use_dropout (bool) -- if use dropout layers. | |
""" | |
super(UnetSkipConnectionBlock, self).__init__() | |
self.outermost = outermost | |
if type(norm_layer) == functools.partial: | |
use_bias = norm_layer.func != nn.BatchNorm2d | |
else: | |
use_bias = norm_layer != nn.BatchNorm2d | |
if input_nc is None: | |
input_nc = outer_nc | |
downconv = nn.Conv2d(input_nc, inner_nc, kernel_size=4, | |
stride=2, padding=1, bias=use_bias) | |
downrelu = nn.LeakyReLU(0.2, True) | |
if norm_layer == nn.GroupNorm: | |
downnorm = norm_layer(32, inner_nc) | |
else: downnorm = norm_layer(inner_nc) | |
uprelu = nn.ReLU(True) | |
if norm_layer == nn.GroupNorm: | |
if outer_nc % 32 != 0: | |
upnorm = norm_layer(outer_nc, outer_nc) # Layer Norm | |
else: | |
upnorm = norm_layer(32, outer_nc) | |
else: | |
upnorm = norm_layer(outer_nc) | |
if outermost: | |
upconv = nn.ConvTranspose2d(inner_nc * 2, outer_nc, | |
kernel_size=4, stride=2, | |
padding=1) | |
down = [downconv] | |
up = [uprelu, upconv, nn.Tanh()] | |
model = down + [submodule] + up | |
elif innermost: | |
upconv = nn.ConvTranspose2d(inner_nc, outer_nc, | |
kernel_size=4, stride=2, | |
padding=1, bias=use_bias) | |
down = [downrelu, downconv] | |
up = [uprelu, upconv, upnorm] | |
model = down + up | |
else: | |
upconv = nn.ConvTranspose2d(inner_nc * 2, outer_nc, | |
kernel_size=4, stride=2, | |
padding=1, bias=use_bias) | |
down = [downrelu, downconv, downnorm] | |
up = [uprelu, upconv, upnorm] | |
if use_dropout: | |
model = down + [submodule] + up + [nn.Dropout(0.5)] | |
else: | |
model = down + [submodule] + up | |
self.model = nn.Sequential(*model) | |
def forward(self, x): | |
if self.outermost: | |
return self.model(x) | |
else: # add skip connections | |
return torch.cat([x, self.model(x)], 1) | |
class UnetGenerator(nn.Module): | |
"""Create a Unet-based generator""" | |
def __init__(self, input_nc, output_nc=2, num_downs=8, ngf=64, norm_layer=nn.GroupNorm, use_dropout=True): | |
"""Construct a Unet generator | |
Parameters: | |
input_nc (int) -- the number of channels in input images | |
output_nc (int) -- the number of channels in output images | |
num_downs (int) -- the number of downsamplings in UNet. For example, # if |num_downs| == 7, | |
image of size 128x128 will become of size 1x1 # at the bottleneck | |
ngf (int) -- the number of filters in the last conv layer | |
norm_layer -- normalization layer | |
We construct the U-Net from the innermost layer to the outermost layer. | |
It is a recursive process. | |
""" | |
super(UnetGenerator, self).__init__() | |
# construct unet structure | |
unet_block = UnetSkipConnectionBlock(ngf * 8, ngf * 8, input_nc=None, submodule=None, norm_layer=norm_layer, innermost=True) # add the innermost layer | |
for i in range(num_downs - 5): # add intermediate layers with ngf * 8 filters | |
unet_block = UnetSkipConnectionBlock(ngf * 8, ngf * 8, input_nc=None, submodule=unet_block, norm_layer=norm_layer, use_dropout=use_dropout) | |
# gradually reduce the number of filters from ngf * 8 to ngf | |
unet_block = UnetSkipConnectionBlock(ngf * 4, ngf * 8, input_nc=None, submodule=unet_block, norm_layer=norm_layer) | |
unet_block = UnetSkipConnectionBlock(ngf * 2, ngf * 4, input_nc=None, submodule=unet_block, norm_layer=norm_layer) | |
unet_block = UnetSkipConnectionBlock(ngf, ngf * 2, input_nc=None, submodule=unet_block, norm_layer=norm_layer) | |
self.model = UnetSkipConnectionBlock(output_nc, ngf, input_nc=input_nc, submodule=unet_block, outermost=True, norm_layer=norm_layer) # add the outermost layer | |
def forward(self, input): | |
"""Standard forward""" | |
return self.model(input) | |
class NLayerDiscriminator(nn.Module): | |
"""Defines a PatchGAN discriminator""" | |
def __init__(self, input_nc, ndf=64, n_layers=6, norm_layer=nn.GroupNorm): | |
"""Construct a PatchGAN discriminator | |
Parameters: | |
input_nc (int) -- the number of channels in input images | |
ndf (int) -- the number of filters in the last conv layer | |
n_layers (int) -- the number of conv layers in the discriminator | |
norm_layer -- normalization layer | |
""" | |
super(NLayerDiscriminator, self).__init__() | |
if type(norm_layer) == functools.partial: # no need to use bias as BatchNorm2d has affine parameters | |
use_bias = norm_layer.func != nn.BatchNorm2d | |
else: | |
use_bias = norm_layer != nn.BatchNorm2d | |
kw = 4 | |
padw = 1 | |
sequence = [nn.Conv2d(input_nc, ndf, kernel_size=kw, stride=2, padding=padw), nn.LeakyReLU(0.2, True)] | |
nf_mult = 1 | |
nf_mult_prev = 1 | |
for n in range(1, n_layers): # gradually increase the number of filters | |
nf_mult_prev = nf_mult | |
nf_mult = min(2 ** n, 8) | |
sequence += [ | |
nn.Conv2d(ndf * nf_mult_prev, ndf * nf_mult, kernel_size=kw, stride=2, padding=padw, bias=use_bias), | |
norm_layer(32, ndf * nf_mult) if norm_layer == nn.GroupNorm else norm_layer(ndf * nf_mult), | |
nn.LeakyReLU(0.2, True) | |
] | |
nf_mult_prev = nf_mult | |
nf_mult = min(2 ** n_layers, 8) | |
sequence += [ | |
nn.Conv2d(ndf * nf_mult_prev, ndf * nf_mult, kernel_size=kw, stride=1, padding=padw, bias=use_bias), | |
norm_layer(32, ndf * nf_mult) if norm_layer == nn.GroupNorm else norm_layer(ndf * nf_mult), | |
nn.LeakyReLU(0.2, True) | |
] | |
sequence += [nn.Conv2d(ndf * nf_mult, 1, kernel_size=kw, stride=1, padding=padw)] # output 1 channel prediction map | |
self.model = nn.Sequential(*sequence) | |
def forward(self, input): | |
"""Standard forward.""" | |
return self.model(input) |