setup.py 8.42 KB
Newer Older
1
2
import glob
import os
Kai Chen's avatar
Kai Chen committed
3
import platform
Rinat Shigapov's avatar
Rinat Shigapov committed
4
import re
5
import setuptools
Rinat Shigapov's avatar
Rinat Shigapov committed
6
from pkg_resources import DistributionNotFound, get_distribution
7
from setuptools import dist, find_packages, setup
8

9
10
dist.Distribution().fetch_build_eggs(['Cython', 'numpy>=1.11.1'])

lizz's avatar
lizz committed
11
import numpy  # NOQA: E402  # isort:skip
12
13
from Cython.Build import cythonize  # NOQA: E402  # isort:skip
from Cython.Distutils import build_ext as build_cmd  # NOQA: E402  # isort:skip
14

Rinat Shigapov's avatar
Rinat Shigapov committed
15
16
17
18
19
20
21
22
23
24
25
26
27
28

def choose_requirement(primary, secondary):
    """If some version of primary requirement installed, return primary,
    else return secondary.
    """
    try:
        name = re.split(r'[!<>=]', primary)[0]
        get_distribution(name)
    except DistributionNotFound:
        return secondary

    return str(primary)


29
def readme():
30
    with open('README.rst', encoding='utf-8') as f:
31
32
33
34
35
36
        content = f.read()
    return content


def get_version():
    version_file = 'mmcv/version.py'
37
    with open(version_file, 'r', encoding='utf-8') as f:
38
39
40
41
        exec(compile(f.read(), version_file, 'exec'))
    return locals()['__version__']


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
def parse_requirements(fname='requirements.txt', with_version=True):
    """
    Parse the package dependencies listed in a requirements file but strips
    specific versioning information.

    Args:
        fname (str): path to requirements file
        with_version (bool, default=False): if True include version specs

    Returns:
        List[str]: list of requirements items

    CommandLine:
        python -c "import setup; print(setup.parse_requirements())"
    """
    import sys
    from os.path import exists
    import re
    require_fpath = fname

    def parse_line(line):
        """
        Parse information from a line in a requirements text file
        """
        if line.startswith('-r '):
            # Allow specifying requirements in other files
            target = line.split(' ')[1]
            for info in parse_require_file(target):
                yield info
        else:
            info = {'line': line}
            if line.startswith('-e '):
                info['package'] = line.split('#egg=')[1]
            else:
                # Remove versioning from the package
                pat = '(' + '|'.join(['>=', '==', '>']) + ')'
                parts = re.split(pat, line, maxsplit=1)
                parts = [p.strip() for p in parts]

                info['package'] = parts[0]
                if len(parts) > 1:
                    op, rest = parts[1:]
                    if ';' in rest:
                        # Handle platform specific dependencies
                        # http://setuptools.readthedocs.io/en/latest/setuptools.html#declaring-platform-specific-dependencies
                        version, platform_deps = map(str.strip,
                                                     rest.split(';'))
                        info['platform_deps'] = platform_deps
                    else:
                        version = rest  # NOQA
                    info['version'] = (op, version)
            yield info

    def parse_require_file(fpath):
        with open(fpath, 'r') as f:
            for line in f.readlines():
                line = line.strip()
                if line and not line.startswith('#'):
                    for info in parse_line(line):
                        yield info

    def gen_packages_items():
        if exists(require_fpath):
            for info in parse_require_file(require_fpath):
                parts = [info['package']]
                if with_version and 'version' in info:
                    parts.extend(info['version'])
                if not sys.version.startswith('3.4'):
                    # apparently package_deps are broken in 3.4
                    platform_deps = info.get('platform_deps')
                    if platform_deps is not None:
                        parts.append(';' + platform_deps)
                item = ''.join(parts)
                yield item

    packages = list(gen_packages_items())
    return packages


# If first not installed install second package
CHOOSE_INSTALL_REQUIRES = [('opencv-python-headless>=3', 'opencv-python>=3')]

install_requires = parse_requirements()
for main, secondary in CHOOSE_INSTALL_REQUIRES:
    install_requires.append(choose_requirement(main, secondary))

Kai Chen's avatar
Kai Chen committed
128

129
130
131
132
133
134
135
136
137
138
139
140
def get_extensions():
    extensions = []

    if platform.system() == 'Darwin':
        extra_compile_args = ['-stdlib=libc++']
        extra_link_args = ['-stdlib=libc++']
    else:
        extra_compile_args = []
        extra_link_args = []

    ext_flow = setuptools.Extension(
        name='mmcv._flow_warp_ext',
141
142
143
144
        sources=[
            './mmcv/video/optflow_warp/flow_warp.cpp',
            './mmcv/video/optflow_warp/flow_warp_module.pyx'
        ],
145
        include_dirs=[numpy.get_include()],
Kai Chen's avatar
Kai Chen committed
146
147
        language='c++',
        extra_compile_args=extra_compile_args,
148
149
150
151
152
153
154
155
        extra_link_args=extra_link_args)
    extensions.extend(cythonize(ext_flow))

    try:
        import torch
        ext_name = 'mmcv._ext'
        if torch.__version__ == 'parrots':
            from parrots.utils.build_extension import BuildExtension, Extension
zhuyuanhao's avatar
zhuyuanhao committed
156
157
158
159
160
161
162
            cuda_args = [
                '-gencode=arch=compute_60,code=sm_60',
                '-gencode=arch=compute_61,code=sm_61',
                '-gencode=arch=compute_70,code=sm_70',
                '-gencode=arch=compute_70,code=compute_70'
            ]
            define_macros = [('MMCV_USE_PARROTS', None)]
163
164
165
166
167
168
            op_files = glob.glob('./mmcv/ops/csrc/parrots/*')
            include_path = os.path.abspath('./mmcv/ops/csrc')
            ext_ops = Extension(
                name=ext_name,
                sources=op_files,
                include_dirs=[include_path],
zhuyuanhao's avatar
zhuyuanhao committed
169
                define_macros=define_macros,
170
171
172
173
174
175
176
177
178
179
180
                extra_compile_args={
                    'nvcc': cuda_args,
                    'cxx': [],
                },
                cuda=True)
            extensions.append(ext_ops)
        else:
            from torch.utils.cpp_extension import (BuildExtension,
                                                   CUDAExtension, CppExtension)
            # prevent ninja from using too many resources
            os.environ.setdefault('MAX_JOBS', '4')
zhuyuanhao's avatar
zhuyuanhao committed
181
182
183
184
185
186
187
            cuda_args = [
                '-gencode=arch=compute_52,code=sm_52',
                '-gencode=arch=compute_60,code=sm_60',
                '-gencode=arch=compute_61,code=sm_61',
                '-gencode=arch=compute_70,code=sm_70',
                '-gencode=arch=compute_70,code=compute_70'
            ]
188
189
190
191
192
            define_macros = []
            extra_compile_args = {'cxx': []}

            if (torch.cuda.is_available()
                    or os.getenv('FORCE_CUDA', '0') == '1'):
zhuyuanhao's avatar
zhuyuanhao committed
193
                define_macros += [('MMCV_WITH_CUDA', None)]
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
                extra_compile_args['nvcc'] = cuda_args
                op_files = glob.glob('./mmcv/ops/csrc/pytorch/*')
                extension = CUDAExtension
            else:
                print(f'Compiling {ext_name} without CUDA')
                op_files = glob.glob('./mmcv/ops/csrc/pytorch/*.cpp')
                extension = CppExtension

            include_path = os.path.abspath('./mmcv/ops/csrc')
            ext_ops = extension(
                name=ext_name,
                sources=op_files,
                include_dirs=[include_path],
                define_macros=define_macros,
                extra_compile_args=extra_compile_args)
            extensions.append(ext_ops)
        global build_cmd
        build_cmd = BuildExtension
    except ModuleNotFoundError:
        print('Skip building ext ops due to the absence of torch.')
    return extensions

216

217
218
219
setup(
    name='mmcv',
    version=get_version(),
220
    description='OpenMMLab Computer Vision Foundation',
221
222
223
    long_description=readme(),
    keywords='computer vision',
    packages=find_packages(),
224
    include_package_data=True,
225
226
    classifiers=[
        'Development Status :: 4 - Beta',
Kai Chen's avatar
Kai Chen committed
227
        'License :: OSI Approved :: Apache Software License',
228
229
230
        'Operating System :: OS Independent',
        'Programming Language :: Python :: 3',
        'Programming Language :: Python :: 3.6',
231
        'Programming Language :: Python :: 3.7',
232
        'Programming Language :: Python :: 3.8',
233
234
235
        'Topic :: Utilities',
    ],
    url='https://github.com/open-mmlab/mmcv',
236
    author='MMCV Authors',
237
238
239
240
    author_email='chenkaidev@gmail.com',
    setup_requires=['pytest-runner'],
    tests_require=['pytest'],
    install_requires=install_requires,
241
242
    ext_modules=get_extensions(),
    cmdclass={'build_ext': build_cmd},
243
    zip_safe=False)