from fastai.vision.all import *
You already have a trained model and want to run inference over a large dataset of images (in my case over 3kk images), how to do this efficiently and fast. We already have access to fastai
’s Learner.get_preds
method, but you need to be able to fit in memory the full output, for my use case of segmentation masks over large images it is just not possible. Let’s build a solution to save the prediction to file using a dataloader to make inference fast and batched.
def is_gpu(dev=0):
if torch.cuda.is_available():
torch.cuda.set_device(dev)print(torch.cuda.get_device_name(dev))
is_gpu()
Quadro RTX 8000
we already have a model that is working good, for the sake of simplicity I am loading a torchscript
model from file.
= torch.jit.load(PATH/'model_checkpoints/unet_small_stage1.jit').cuda() model
we have a dataframe with our data
df.head()
group | image_path | elevation | azimuth | ghi_cs | kt | ghi | |
---|---|---|---|---|---|---|---|
utc | |||||||
2019-01-08 10:04:16.970 | 0 | 2019-01-08/image19-01-08_10-04-16-97.png | 18.020898 | 1412.8 | 277.376311 | 0.181790 | 50.426096 |
2019-01-08 10:18:17.000 | 1 | 2019-01-08/image19-01-08_10-18-17-00.png | 18.975492 | 1412.8 | 295.647541 | 0.185127 | 54.741370 |
2019-01-08 10:19:16.980 | 1 | 2019-01-08/image19-01-08_10-19-16-98.png | 19.038769 | 1412.8 | 296.875387 | 0.177907 | 52.806228 |
2019-01-08 10:41:16.960 | 2 | 2019-01-08/image19-01-08_10-41-16-96.png | 20.297296 | 1412.8 | 321.138689 | 0.177493 | 57.007973 |
2019-01-08 10:42:16.950 | 2 | 2019-01-08/image19-01-08_10-42-16-95.png | 20.344993 | 1412.8 | 322.050887 | 0.194303 | 62.576990 |
Inference on one image at a time
let’s grab one image:
= df.image_path.sample(1).item()
img_path = load_image(img_path)
pil_img pil_img
We will use fastai transforms, but the same can be done using torchvision.transforms
,
import torchvision.transforms as T
= T.Compose([T.ToTensor(),
img_tfms *WSISEG_STATS)]) T.Normalize(
Here to compose we use the Pipeline
= Pipeline([PILImage.create,
tfms
ToTensor(),
IntToFloatTensor(), *WSISEG_STATS, cuda=False)]) Normalize.from_stats(
these transforms will convert the image to tensor, so we can pass it through the model
= tfms(img_path) tensor_img
into the model
= model(tensor_img.cuda())
raw_out raw_out.shape
torch.Size([1, 4, 192, 256])
this is a segmentation model with 4 classes, so the output has 4 channels. We need to postprocess this to get a 1 channel uint8
image with values in [0,1,2,3]. We will also reconvert this output to PIL
to save it later.
def postprocess(out):
"Transform the output of the model to a uint8 mask"
return PILMask.create(out.squeeze(0).argmax(0).cpu().numpy().astype(np.uint8))
it looks fine
show_images([pil_img, postprocess(raw_out)])
Now, if we want to compute this process on all images (it is going to be slow…) let’s do some refactor:
def predict_one(img_path):
"evaluate `img_path` into model"
= tfms(img_path)
tensor_img with torch.no_grad():
= model(tensor_img.cuda())
raw_out return postprocess(raw_out)
= predict_one(img_path)
mask ; mask.show()
We now want to save this image besides the original one. Let’s leverage some fastcore
’s magic and patch pathlib.Path
to be able to put an arbitrary suffix on our images: :::{.callout-note}
Path.with_suffix
cannot put an arbitrary suffif with the _GT
string before the extension, so we have to patch it.
:::
@patch
def my_suffix(self:Path, suffix=''):
"replace the everything after the dot (including the dot) with `suffix`"
= self.with_suffix('')
path return path.parent/f'{path.name}{suffix}'
this way, our masks will be called *_GT.png
'_GT.png') img_path, img_path.my_suffix(
(Path('2019-01-27/image19-01-27_09-54-01-97.png'),
Path('2019-01-27/image19-01-27_09-54-01-97_GT.png'))
To process the full dataset one would do this: - iterate over all images one by one - compute inference on each - save the postprocessed mask
for img_path in progress_bar(df.image_path.to_list()):
= predict_one(img_path)
mask '_GT.png')) mask.save(img_path.my_suffix(
Batched images to files
From DataLoader to files
Let’s try to make better use of the GPU, we will feed batches of images all at once. We already have DataLoaders
to do this, let’s make a simple TfmdDL
object, to stack our items with the transforms together. Here we need to split the transforms on the ones that are called after getting the item (the image path) and after stacking the tensors into a batch. The latter ones are computer on the GPU.
= df.image_path.to_list()
files = TfmdDL(files,
dl =64,
bs=[PILImage.create, ToTensor()],
after_item=[IntToFloatTensor, Normalize.from_stats(*WSISEG_STATS)],
after_batch='cuda:0') device
we get a nice batch of images (64 in this case) ready to feed the model
= next(iter(dl))
b b.shape
torch.Size([64, 3, 192, 256])
model(b).shape
torch.Size([64, 4, 192, 256])
Here we have performed inference on 64 images all at once, but the posprocessing need to happen imager per image anyway. We have recover the original images filenames to be able to store the masks and make the maping with the original image. We kind of need a DataLoader for the filenames, here comes chunked
function to the recue. :::{.callout-note}
chunked
splits the files on chunks of bs
so we can iterate at the same time on the image paths.
:::
= chunked(files, chunk_sz=64) fnames_bs
In my case, this solution is 10 times faster than doing the images one by one.
for b, fnames in progress_bar(zip(dl, fnames_bs), total=len(files)//bs):
with torch.no_grad():
= model(b)
y for torch_mask, fname in zip(y, fnames):
= postprocess(torch_mask)
mask '_GT.png')) mask.save(fname.my_suffix(
With a DataBlock
We can do the same thing with a data block
= DataBlock(blocks=ImageBlock,
block =ColReader('image_path'),
get_x= [Normalize.from_stats(*WSISEG_STATS)],
batch_tfms=lambda o: ([], range_of(o)),
splitter )
= block.dataloaders(df, bs=64, suffle=False).valid dl
the DataBlock
API generates a DataLoaders
object, that is just a wrapper around a train/valid pair of DataLoaders
. As we passed a dummy split (no split), the train
dataloader is empty, and we will use only the valid one. Also, the batch in this case is composed of a tuple (x,y)
where y
is empty, hence we need to do x, = b
(or b[0]
)
for b, fnames in progress_bar(zip(dl, fnames_bs), total=len(files)//bs):
with torch.no_grad():
= b
x, = model(x)
y for torch_mask, fname in zip(y, fnames):
= postprocess(torch_mask)
mask '_GT.png')) mask.save(fname.my_suffix(
The fastai way
Using the
test_dl
on theLearner
.
If you have loaded in memory a learner (or a DataLoaders) you can use the test_dl
method. This is very handful when you just finished training your model. You can easily construct a dataloader with the exact same transforms used to train the model (in reality it takes the validation transforms, so no augment and no shuffling). Then, you can do just like before using the test_dl
= learn.dls.test_dl(files) test_dl
for b, fnames in progress_bar(zip(test_dl, fnames_bs), total=len(files)//bs):
with torch.no_grad():
= model(b)
y for torch_mask, fname in zip(y, fnames):
= postprocess(torch_mask)
mask '_GT.png')) mask.save(fname.my_suffix(