Source code for composer.datasets.cifar

# Copyright 2021 MosaicML. All Rights Reserved.

"""CIFAR image classification dataset.

The CIFAR datasets are a collection of labeled 32x32 colour images. Please refer to the `CIFAR dataset
<https://www.cs.toronto.edu/~kriz/cifar.html>`_ for more details.
"""

import logging
import os
import textwrap
from dataclasses import dataclass
from typing import List

import torch
import yahp as hp
from torchvision import transforms
from torchvision.datasets import CIFAR10

from composer.core.types import DataLoader
from composer.datasets.dataloader import DataLoaderHparams
from composer.datasets.ffcv_utils import write_ffcv_dataset
from composer.datasets.hparams import DatasetHparams, SyntheticHparamsMixin, WebDatasetHparams
from composer.datasets.synthetic import SyntheticBatchPairDataset
from composer.utils import dist

__all__ = [
    "CIFAR10DatasetHparams", "CIFARWebDatasetHparams", "CIFAR10WebDatasetHparams", "CIFAR20WebDatasetHparams",
    "CIFAR100WebDatasetHparams"
]

log = logging.getLogger(__name__)


[docs]@dataclass class CIFAR10DatasetHparams(DatasetHparams, SyntheticHparamsMixin): """Defines an instance of the CIFAR-10 dataset for image classification from a local disk. Args: download (bool): Whether to download the dataset, if needed. Default: ``True``. use_ffcv (bool): Whether to use FFCV dataloaders. Default: ``False``. ffcv_dir (str): A directory containing train/val <file>.ffcv files. If these files don't exist and ``ffcv_write_dataset`` is ``True``, train/val <file>.ffcv files will be created in this dir. Default: ``"/tmp"``. ffcv_dest_train (str): <file>.ffcv file that has training samples. Default: ``"cifar_train.ffcv"``. ffcv_dest_val (str): <file>.ffcv file that has validation samples. Default: ``"cifar_val.ffcv"``. ffcv_write_dataset (std): Whether to create dataset in FFCV format (<file>.ffcv) if it doesn't exist. Default: ``False``. """ download: bool = hp.optional("whether to download the dataset, if needed", default=True) use_ffcv: bool = hp.optional("whether to use ffcv for faster dataloading", default=False) ffcv_dir: str = hp.optional( "A directory containing train/val <file>.ffcv files. If these files don't exist and ffcv_write_dataset is true, train/val <file>.ffcv files will be created in this dir.", default="/tmp") ffcv_dest_train: str = hp.optional("<file>.ffcv file that has training samples", default="cifar_train.ffcv") ffcv_dest_val: str = hp.optional("<file>.ffcv file that has validation samples", default="cifar_val.ffcv") ffcv_write_dataset: bool = hp.optional("Whether to create dataset in FFCV format (<file>.ffcv) if it doesn't exist", default=False)
[docs] def initialize_object(self, batch_size: int, dataloader_hparams: DataLoaderHparams) -> DataLoader: if self.use_synthetic: total_dataset_size = 50_000 if self.is_train else 10_000 dataset = SyntheticBatchPairDataset( total_dataset_size=total_dataset_size, data_shape=[3, 32, 32], num_classes=10, num_unique_samples_to_create=self.synthetic_num_unique_samples, device=self.synthetic_device, memory_format=self.synthetic_memory_format, ) elif self.use_ffcv: try: import ffcv # type: ignore from ffcv.fields.decoders import IntDecoder, SimpleRGBImageDecoder # type: ignore from ffcv.pipeline.operation import Operation # type: ignore except ImportError: raise ImportError( textwrap.dedent("""\ Composer was installed without ffcv support. To use ffcv with Composer, please install ffcv in your environment.""")) dataset_file = self.ffcv_dest_train if self.is_train else self.ffcv_dest_val dataset_filepath = os.path.join(self.ffcv_dir, dataset_file) # always create if ffcv_write_dataset is true if self.ffcv_write_dataset: if dist.get_local_rank() == 0: if self.datadir is None: raise ValueError( "datadir is required if use_synthetic is False and ffcv_write_dataset is True.") ds = CIFAR10( self.datadir, train=self.is_train, download=self.download, ) write_ffcv_dataset(dataset=ds, write_path=dataset_filepath) # Wait for the local rank 0 to be done creating the dataset in ffcv format. dist.barrier() if not os.path.exists(dataset_filepath): raise ValueError( f"Dataset file containing samples not found at {dataset_filepath}. Use ffcv_dir flag to point to a dir containing {dataset_filepath}." ) # Please note that this mean/std is different from the mean/std used for regular PyTorch dataloader as # ToTensor does the normalization for PyTorch dataloaders. cifar10_mean_ffcv = [125.307, 122.961, 113.8575] cifar10_std_ffcv = [51.5865, 50.847, 51.255] label_pipeline: List[Operation] = [IntDecoder(), ffcv.transforms.ToTensor(), ffcv.transforms.Squeeze()] image_pipeline: List[Operation] = [SimpleRGBImageDecoder()] if self.is_train: image_pipeline.extend([ ffcv.transforms.RandomHorizontalFlip(), ffcv.transforms.RandomTranslate(padding=2, fill=tuple(map(int, cifar10_mean_ffcv))), ffcv.transforms.Cutout(4, tuple(map(int, cifar10_mean_ffcv))), ]) # Common transforms for train and test image_pipeline.extend([ ffcv.transforms.ToTensor(), ffcv.transforms.ToTorchImage(channels_last=False, convert_back_int16=False), ffcv.transforms.Convert(torch.float32), transforms.Normalize(cifar10_mean_ffcv, cifar10_std_ffcv), ]) ordering = ffcv.loader.OrderOption.RANDOM if self.is_train else ffcv.loader.OrderOption.SEQUENTIAL return ffcv.Loader( dataset_filepath, batch_size=batch_size, num_workers=dataloader_hparams.num_workers, order=ordering, distributed=False, pipelines={ 'image': image_pipeline, 'label': label_pipeline }, batches_ahead=dataloader_hparams.prefetch_factor, drop_last=self.drop_last, ) else: if self.datadir is None: raise ValueError("datadir is required if use_synthetic is False") cifar10_mean = 0.4914, 0.4822, 0.4465 cifar10_std = 0.247, 0.243, 0.261 if self.is_train: transformation = transforms.Compose([ transforms.RandomCrop(32, padding=4), transforms.RandomHorizontalFlip(), transforms.ToTensor(), transforms.Normalize(cifar10_mean, cifar10_std), ]) else: transformation = transforms.Compose([ transforms.ToTensor(), transforms.Normalize(cifar10_mean, cifar10_std), ]) dataset = CIFAR10( self.datadir, train=self.is_train, download=self.download, transform=transformation, ) sampler = dist.get_sampler(dataset, drop_last=self.drop_last, shuffle=self.shuffle) return dataloader_hparams.initialize_object(dataset, batch_size=batch_size, sampler=sampler, drop_last=self.drop_last)
[docs]@dataclass class CIFARWebDatasetHparams(WebDatasetHparams): """Common functionality for CIFAR WebDatasets. Args: remote (str): S3 bucket or root directory where dataset is stored. name (str): Key used to determine where dataset is cached on local filesystem. n_train_samples (int): Number of training samples. n_val_samples (int): Number of validation samples. height (int): Sample image height in pixels. Default: ``32``. width (int): Sample image width in pixels. Default: ``32``. n_classes (int): Number of output classes. channel_means (list of float): Channel means for normalization. channel_stds (list of float): Channel stds for normalization. """ remote: str = hp.optional('WebDataset S3 bucket name', default='') name: str = hp.optional('WebDataset local cache name', default='') n_train_samples: int = hp.optional('Number of samples in training split', default=0) n_val_samples: int = hp.optional('Number of samples in validation split', default=0) height: int = hp.optional('Image height', default=32) width: int = hp.optional('Image width', default=32) n_classes: int = hp.optional('Number of output classes', default=0) channel_means: List[float] = hp.optional('Mean per image channel', default=(0, 0, 0)) channel_stds: List[float] = hp.optional('Std per image channel', default=(0, 0, 0))
[docs] def initialize_object(self, batch_size: int, dataloader_hparams: DataLoaderHparams) -> DataLoader: from composer.datasets.webdataset_utils import load_webdataset if self.is_train: split = 'train' transform = transforms.Compose([ transforms.RandomCrop((self.height, self.width), (self.height // 8, self.width // 8)), transforms.RandomHorizontalFlip(), transforms.ToTensor(), transforms.Normalize(self.channel_means, self.channel_stds), ]) else: split = 'val' transform = transforms.Compose([ transforms.ToTensor(), transforms.Normalize(self.channel_means, self.channel_stds), ]) preprocess = lambda dataset: dataset.decode('pil').map_dict(jpg=transform).to_tuple('jpg', 'cls') dataset = load_webdataset(self.remote, self.name, split, self.webdataset_cache_dir, self.webdataset_cache_verbose, self.shuffle, self.shuffle_buffer, preprocess, dist.get_world_size(), dataloader_hparams.num_workers, batch_size, self.drop_last) return dataloader_hparams.initialize_object(dataset, batch_size=batch_size, sampler=None, drop_last=self.drop_last)
[docs]@dataclass class CIFAR10WebDatasetHparams(CIFARWebDatasetHparams): """Defines an instance of the CIFAR-10 WebDataset for image classification. Args: remote (str): S3 bucket or root directory where dataset is stored. Default: ``'s3://mosaicml-internal-dataset-cifar10'``. name (str): Key used to determine where dataset is cached on local filesystem. Default: ``'cifar10'``. n_train_samples (int): Number of training samples. Default: ``50000``. n_val_samples (int): Number of validation samples. Default: ``10000``. n_classes (int): Number of output classes. Default: ``10``. channel_means (list of float): Channel means for normalization. Default: ``(0.4914, 0.4822, 0.4465)``. channel_stds (list of float): Channel stds for normalization. Default: ``(0.247, 0.243, 0.261)``. """ remote: str = hp.optional('WebDataset S3 bucket name', default='s3://mosaicml-internal-dataset-cifar10') name: str = hp.optional('WebDataset local cache name', default='cifar10') n_train_samples: int = hp.optional('Number of samples in training split', default=50_000) n_val_samples: int = hp.optional('Number of samples in validation split', default=10_000) n_classes: int = hp.optional('Number of output classes', default=10) channel_means: List[float] = hp.optional('Mean per image channel', default=(0.4914, 0.4822, 0.4465)) channel_stds: List[float] = hp.optional('Std per image channel', default=(0.247, 0.243, 0.261))
[docs]@dataclass class CIFAR20WebDatasetHparams(CIFARWebDatasetHparams): """Defines an instance of the CIFAR-20 WebDataset for image classification. Args: remote (str): S3 bucket or root directory where dataset is stored. Default: ``'s3://mosaicml-internal-dataset-cifar20'``. name (str): Key used to determine where dataset is cached on local filesystem. Default: ``'cifar20'``. n_train_samples (int): Number of training samples. Default: ``50000``. n_val_samples (int): Number of validation samples. Default: ``10000``. n_classes (int): Number of output classes. Default: ``20``. channel_means (list of float): Channel means for normalization. Default: ``(0.5071, 0.4867, 0.4408)``. channel_stds (list of float): Channel stds for normalization. Default: ``(0.2675, 0.2565, 0.2761)``. """ remote: str = hp.optional('WebDataset S3 bucket name', default='s3://mosaicml-internal-dataset-cifar20') name: str = hp.optional('WebDataset local cache name', default='cifar20') n_train_samples: int = hp.optional('Number of samples in training split', default=50_000) n_val_samples: int = hp.optional('Number of samples in validation split', default=10_000) n_classes: int = hp.optional('Number of output classes', default=20) channel_means: List[float] = hp.optional('Mean per image channel', default=(0.5071, 0.4867, 0.4408)) channel_stds: List[float] = hp.optional('Std per image channel', default=(0.2675, 0.2565, 0.2761))
[docs]@dataclass class CIFAR100WebDatasetHparams(CIFARWebDatasetHparams): """Defines an instance of the CIFAR-100 WebDataset for image classification. Args: remote (str): S3 bucket or root directory where dataset is stored. Default: ``'s3://mosaicml-internal-dataset-cifar100'``. name (str): Key used to determine where dataset is cached on local filesystem. Default: ``'cifar100'``. n_train_samples (int): Number of training samples. Default: ``50000``. n_val_samples (int): Number of validation samples. Default: ``10000``. n_classes (int): Number of output classes. Default: ``100``. channel_means (list of float): Channel means for normalization. Default: ``(0.5071, 0.4867, 0.4408)``. channel_stds (list of float): Channel stds for normalization. Default: ``(0.2675, 0.2565, 0.2761)``. """ remote: str = hp.optional('WebDataset S3 bucket name', default='s3://mosaicml-internal-dataset-cifar100') name: str = hp.optional('WebDataset local cache name', default='cifar100') n_train_samples: int = hp.optional('Number of samples in training split', default=50_000) n_val_samples: int = hp.optional('Number of samples in validation split', default=10_000) n_classes: int = hp.optional('Number of output classes', default=100) channel_means: List[float] = hp.optional('Mean per image channel', default=(0.5071, 0.4867, 0.4408)) channel_stds: List[float] = hp.optional('Std per image channel', default=(0.2675, 0.2565, 0.2761))