1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
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
|
# Copyright 2013-2021 Lawrence Livermore National Security, LLC and other
# Spack Project Developers. See the top-level COPYRIGHT file for details.
#
# SPDX-License-Identifier: (Apache-2.0 OR MIT)
import os
from spack import *
class Faiss(AutotoolsPackage, CudaPackage):
"""Faiss is a library for efficient similarity search and clustering of
dense vectors.
Faiss contains algorithms that search in sets of vectors of any size, up
to ones that possibly do not fit in RAM. It also contains supporting code
for evaluation and parameter tuning. Faiss is written in C++ with
complete wrappers for Python/numpy. Some of the most useful algorithms
are implemented on the GPU. It is developed by Facebook AI Research.
"""
homepage = "https://github.com/facebookresearch/faiss"
url = "https://github.com/facebookresearch/faiss/archive/v1.6.3.tar.gz"
maintainers = ['bhatiaharsh']
version('1.6.3', sha256='e1a41c159f0b896975fbb133e0240a233af5c9286c09a28fde6aefff5336e542')
version('1.5.3', sha256='b24d347b0285d01c2ed663ccc7596cd0ea95071f3dd5ebb573ccfc28f15f043b')
variant('python', default=False, description='Build Python bindings')
variant('tests', default=False, description='Build Tests')
conflicts('+tests', when='~python', msg='+tests must be accompanied by +python')
depends_on('python@3.7:', when='+python', type=('build', 'run'))
depends_on('py-numpy', when='+python', type=('build', 'run'))
depends_on('py-scipy', when='+tests', type=('build', 'run'))
depends_on('blas')
depends_on('py-setuptools', when='+python', type='build')
depends_on('swig', when='+python', type='build')
# patch for v1.5.3
# faiss assumes that the "source directory" will always
# be called "faiss" (not spack-src or faiss-1.5.3)
# so, we will have to create a symlink to self (faiss did that in 1.6.3)
# and add an include path
patch('fixes-in-v1.5.3.patch', when='@1.5.3')
# patch for v1.6.3
# for v1.6.3, GPU build has a bug (two files need to be deleted)
# https://github.com/facebookresearch/faiss/issues/1159
# also, some include paths in gpu/tests/Makefile are missing
patch('fixes-in-v1.6.3.patch', when='@1.6.3')
def configure_args(self):
args = []
args.extend(self.with_or_without('cuda', activation_value='prefix'))
return args
def build(self, spec, prefix):
make()
if '+python' in self.spec:
make('-C', 'python')
# CPU tests
if '+tests' in self.spec:
with working_dir('tests'):
make('gtest')
make('tests')
# GPU tests
if '+tests+cuda' in self.spec:
with working_dir(os.path.join('gpu', 'test')):
make('gtest')
make('build') # target added by the patch
make('demo_ivfpq_indexing_gpu')
def install(self, spec, prefix):
make('install')
if '+python' in self.spec:
with working_dir('python'):
setup_py('install', '--prefix=' + prefix,
'--single-version-externally-managed', '--root=/')
if '+tests' not in self.spec:
return
if not os.path.isdir(self.prefix.bin):
os.makedirs(self.prefix.bin)
def _prefix_and_install(file):
os.rename(file, 'faiss_' + file)
install('faiss_' + file, self.prefix.bin)
# CPU tests
with working_dir('tests'):
# rename the exec to keep consistent with gpu tests
os.rename('tests', 'TestCpu')
_prefix_and_install('TestCpu')
# GPU tests
if '+cuda' in self.spec:
with working_dir(os.path.join('gpu', 'test')):
_prefix_and_install('TestGpuIndexFlat')
_prefix_and_install('TestGpuIndexBinaryFlat')
_prefix_and_install('TestGpuIndexIVFFlat')
_prefix_and_install('TestGpuIndexIVFPQ')
_prefix_and_install('TestGpuMemoryException')
_prefix_and_install('TestGpuSelect')
_prefix_and_install('demo_ivfpq_indexing_gpu')
@run_after('configure')
def _fix_makefile(self):
# spack injects its own optimization flags
makefile = FileFilter('makefile.inc')
makefile.filter('CPUFLAGS = -mavx2 -mf16c',
'#CPUFLAGS = -mavx2 -mf16c')
def setup_run_environment(self, env):
if '+python' in self.spec:
env.prepend_path('PYTHONPATH', site_packages_dir)
|