kitti_dataset.py 6.48 KB
Newer Older
dingchang's avatar
dingchang committed
1
# Copyright (c) OpenMMLab. All rights reserved.
2
from typing import Callable, List, Union
3
4

import numpy as np
zhangwenwei's avatar
zhangwenwei committed
5

jshilong's avatar
jshilong committed
6
from mmdet3d.datasets import DATASETS
zhangshilong's avatar
zhangshilong committed
7
from mmdet3d.structures import CameraInstance3DBoxes
jshilong's avatar
jshilong committed
8
from .det3d_dataset import Det3DDataset
zhangwenwei's avatar
zhangwenwei committed
9
10


11
@DATASETS.register_module()
jshilong's avatar
jshilong committed
12
class KittiDataset(Det3DDataset):
zhangwenwei's avatar
zhangwenwei committed
13
    r"""KITTI Dataset.
wangtai's avatar
wangtai committed
14

zhangwenwei's avatar
zhangwenwei committed
15
16
    This class serves as the API for experiments on the `KITTI Dataset
    <http://www.cvlibs.net/datasets/kitti/eval_object.php?obj_benchmark=3d>`_.
wangtai's avatar
wangtai committed
17
18
19
20

    Args:
        data_root (str): Path of dataset root.
        ann_file (str): Path of annotation file.
21
22
23
24
25
        pipeline (list[dict]): Pipeline used for data processing.
            Defaults to [].
        modality (dict): Modality to specify the sensor data used as input.
            Defaults to `dict(use_lidar=True)`.
        default_cam_key (str): The default camera name adopted.
26
            Defaults to 'CAM2'.
27
        box_type_3d (str): Type of 3D box of this dataset.
wangtai's avatar
wangtai committed
28
29
            Based on the `box_type_3d`, the dataset will encapsulate the box
            to its original format then converted them to `box_type_3d`.
30
            Defaults to 'LiDAR' in this dataset. Available options includes:
wangtai's avatar
wangtai committed
31

wangtai's avatar
wangtai committed
32
33
34
            - 'LiDAR': Box in LiDAR coordinates.
            - 'Depth': Box in depth coordinates, usually for indoor dataset.
            - 'Camera': Box in camera coordinates.
35
36
37
38
39
        filter_empty_gt (bool): Whether to filter the data with empty GT.
            If it's set to be True, the example with empty annotations after
            data pipeline will be dropped and a random example will be chosen
            in `__getitem__`. Defaults to True.
        test_mode (bool): Whether the dataset is in test mode.
wangtai's avatar
wangtai committed
40
            Defaults to False.
41
42
        pcd_limit_range (list[float]): The range of point cloud used to filter
            invalid predicted boxes.
43
            Defaults to [0, -40, -3, 70.4, 40, 0.0].
wangtai's avatar
wangtai committed
44
    """
jshilong's avatar
jshilong committed
45
    # TODO: use full classes of kitti
VVsssssk's avatar
VVsssssk committed
46
47
48
49
    METAINFO = {
        'CLASSES': ('Pedestrian', 'Cyclist', 'Car', 'Van', 'Truck',
                    'Person_sitting', 'Tram', 'Misc')
    }
zhangwenwei's avatar
zhangwenwei committed
50
51

    def __init__(self,
jshilong's avatar
jshilong committed
52
53
54
                 data_root: str,
                 ann_file: str,
                 pipeline: List[Union[dict, Callable]] = [],
55
                 modality: dict = dict(use_lidar=True),
56
                 default_cam_key: str = 'CAM2',
57
                 task: str = 'lidar_det',
jshilong's avatar
jshilong committed
58
59
60
61
                 box_type_3d: str = 'LiDAR',
                 filter_empty_gt: bool = True,
                 test_mode: bool = False,
                 pcd_limit_range: List[float] = [0, -40, -3, 70.4, 40, 0.0],
62
                 **kwargs) -> None:
jshilong's avatar
jshilong committed
63
64

        self.pcd_limit_range = pcd_limit_range
65
66
        assert task in ('lidar_det', 'mono_det')
        self.task = task
zhangwenwei's avatar
zhangwenwei committed
67
68
69
70
71
        super().__init__(
            data_root=data_root,
            ann_file=ann_file,
            pipeline=pipeline,
            modality=modality,
jshilong's avatar
jshilong committed
72
            default_cam_key=default_cam_key,
73
74
            box_type_3d=box_type_3d,
            filter_empty_gt=filter_empty_gt,
75
76
            test_mode=test_mode,
            **kwargs)
zhangwenwei's avatar
zhangwenwei committed
77
        assert self.modality is not None
jshilong's avatar
jshilong committed
78
        assert box_type_3d.lower() in ('lidar', 'camera')
zhangwenwei's avatar
zhangwenwei committed
79

jshilong's avatar
jshilong committed
80
81
    def parse_data_info(self, info: dict) -> dict:
        """Process the raw data info.
zhangwenwei's avatar
zhangwenwei committed
82

jshilong's avatar
jshilong committed
83
84
        The only difference with it in `Det3DDataset`
        is the specific process for `plane`.
85
86

        Args:
jshilong's avatar
jshilong committed
87
            info (dict): Raw info dict.
88
89

        Returns:
jshilong's avatar
jshilong committed
90
91
            dict: Has `ann_info` in training stage. And
            all path has been converted to absolute path.
92
        """
jshilong's avatar
jshilong committed
93
94
95
96
        if self.modality['use_lidar']:
            if 'plane' in info:
                # convert ground plane to velodyne coordinates
                plane = np.array(info['plane'])
zhangshilong's avatar
zhangshilong committed
97
98
                lidar2cam = np.array(
                    info['images']['CAM2']['lidar2cam'], dtype=np.float32)
jshilong's avatar
jshilong committed
99
100
101
102
103
104
105
106
107
108
109
110
111
112
                reverse = np.linalg.inv(lidar2cam)

                (plane_norm_cam, plane_off_cam) = (plane[:3],
                                                   -plane[:3] * plane[3])
                plane_norm_lidar = \
                    (reverse[:3, :3] @ plane_norm_cam[:, None])[:, 0]
                plane_off_lidar = (
                    reverse[:3, :3] @ plane_off_cam[:, None][:, 0] +
                    reverse[:3, 3])
                plane_lidar = np.zeros_like(plane_norm_lidar, shape=(4, ))
                plane_lidar[:3] = plane_norm_lidar
                plane_lidar[3] = -plane_norm_lidar.T @ plane_off_lidar
            else:
                plane_lidar = None
zhangwenwei's avatar
zhangwenwei committed
113

jshilong's avatar
jshilong committed
114
            info['plane'] = plane_lidar
zhangwenwei's avatar
zhangwenwei committed
115

116
117
118
        if self.task == 'mono_det':
            info['instances'] = info['cam_instances'][self.default_cam_key]

jshilong's avatar
jshilong committed
119
        info = super().parse_data_info(info)
zhangwenwei's avatar
zhangwenwei committed
120

jshilong's avatar
jshilong committed
121
        return info
zhangwenwei's avatar
zhangwenwei committed
122

123
    def parse_ann_info(self, info: dict) -> dict:
124
        """Process the `instances` in data info to `ann_info`.
125
126

        Args:
jshilong's avatar
jshilong committed
127
            info (dict): Data information of single data sample.
128
129

        Returns:
130
            dict: Annotation information consists of the following keys:
131

zhangshilong's avatar
zhangshilong committed
132
                - gt_bboxes_3d (:obj:`LiDARInstance3DBoxes`):
133
                  3D ground truth bboxes.
jshilong's avatar
jshilong committed
134
                - bbox_labels_3d (np.ndarray): Labels of ground truths.
wangtai's avatar
wangtai committed
135
136
                - gt_bboxes (np.ndarray): 2D ground truth bboxes.
                - gt_labels (np.ndarray): Labels of ground truths.
137
                - difficulty (int): Difficulty defined by KITTI.
138
                  0, 1, 2 represent xxxxx respectively.
139
        """
jshilong's avatar
jshilong committed
140
        ann_info = super().parse_ann_info(info)
141
        if ann_info is None:
jshilong's avatar
jshilong committed
142
            ann_info = dict()
143
144
145
            # empty instance
            ann_info['gt_bboxes_3d'] = np.zeros((0, 7), dtype=np.float32)
            ann_info['gt_labels_3d'] = np.zeros(0, dtype=np.int64)
146

147
148
149
150
151
152
            if self.task == 'mono_det':
                ann_info['gt_bboxes'] = np.zeros((0, 4), dtype=np.float32)
                ann_info['gt_bboxes_labels'] = np.array(0, dtype=np.int64)
                ann_info['centers_2d'] = np.zeros((0, 2), dtype=np.float32)
                ann_info['depths'] = np.zeros((0), dtype=np.float32)

jshilong's avatar
jshilong committed
153
154
155
156
157
158
159
160
161
        ann_info = self._remove_dontcare(ann_info)
        # in kitti, lidar2cam = R0_rect @ Tr_velo_to_cam
        lidar2cam = np.array(info['images']['CAM2']['lidar2cam'])
        # convert gt_bboxes_3d to velodyne coordinates with `lidar2cam`
        gt_bboxes_3d = CameraInstance3DBoxes(
            ann_info['gt_bboxes_3d']).convert_to(self.box_mode_3d,
                                                 np.linalg.inv(lidar2cam))
        ann_info['gt_bboxes_3d'] = gt_bboxes_3d
        return ann_info