make_cffi.py
154 lines
| 4.1 KiB
| text/x-python
|
PythonLexer
Gregory Szorc
|
r30435 | # Copyright (c) 2016-present, Gregory Szorc | ||
# All rights reserved. | ||||
# | ||||
# This software may be modified and distributed under the terms | ||||
# of the BSD license. See the LICENSE file for details. | ||||
from __future__ import absolute_import | ||||
import cffi | ||||
Gregory Szorc
|
r30822 | import distutils.ccompiler | ||
Gregory Szorc
|
r30435 | import os | ||
Gregory Szorc
|
r30895 | import re | ||
Gregory Szorc
|
r30822 | import subprocess | ||
import tempfile | ||||
Gregory Szorc
|
r30435 | |||
HERE = os.path.abspath(os.path.dirname(__file__)) | ||||
SOURCES = ['zstd/%s' % p for p in ( | ||||
'common/entropy_common.c', | ||||
'common/error_private.c', | ||||
'common/fse_decompress.c', | ||||
Gregory Szorc
|
r30895 | 'common/pool.c', | ||
'common/threading.c', | ||||
Gregory Szorc
|
r30435 | 'common/xxhash.c', | ||
'common/zstd_common.c', | ||||
'compress/fse_compress.c', | ||||
'compress/huf_compress.c', | ||||
'compress/zstd_compress.c', | ||||
'decompress/huf_decompress.c', | ||||
'decompress/zstd_decompress.c', | ||||
Gregory Szorc
|
r30895 | 'dictBuilder/cover.c', | ||
Gregory Szorc
|
r30435 | 'dictBuilder/divsufsort.c', | ||
'dictBuilder/zdict.c', | ||||
)] | ||||
Gregory Szorc
|
r30895 | HEADERS = [os.path.join(HERE, 'zstd', *p) for p in ( | ||
('zstd.h',), | ||||
('common', 'pool.h'), | ||||
('dictBuilder', 'zdict.h'), | ||||
)] | ||||
Gregory Szorc
|
r30435 | INCLUDE_DIRS = [os.path.join(HERE, d) for d in ( | ||
'zstd', | ||||
'zstd/common', | ||||
'zstd/compress', | ||||
'zstd/decompress', | ||||
'zstd/dictBuilder', | ||||
)] | ||||
Gregory Szorc
|
r30822 | # cffi can't parse some of the primitives in zstd.h. So we invoke the | ||
# preprocessor and feed its output into cffi. | ||||
compiler = distutils.ccompiler.new_compiler() | ||||
# Needed for MSVC. | ||||
if hasattr(compiler, 'initialize'): | ||||
compiler.initialize() | ||||
# Distutils doesn't set compiler.preprocessor, so invoke the preprocessor | ||||
# manually. | ||||
if compiler.compiler_type == 'unix': | ||||
args = list(compiler.executables['compiler']) | ||||
args.extend([ | ||||
'-E', | ||||
'-DZSTD_STATIC_LINKING_ONLY', | ||||
Gregory Szorc
|
r30895 | '-DZDICT_STATIC_LINKING_ONLY', | ||
Gregory Szorc
|
r30822 | ]) | ||
elif compiler.compiler_type == 'msvc': | ||||
args = [compiler.cc] | ||||
args.extend([ | ||||
'/EP', | ||||
'/DZSTD_STATIC_LINKING_ONLY', | ||||
Gregory Szorc
|
r30895 | '/DZDICT_STATIC_LINKING_ONLY', | ||
Gregory Szorc
|
r30822 | ]) | ||
else: | ||||
raise Exception('unsupported compiler type: %s' % compiler.compiler_type) | ||||
Gregory Szorc
|
r30895 | def preprocess(path): | ||
# zstd.h includes <stddef.h>, which is also included by cffi's boilerplate. | ||||
# This can lead to duplicate declarations. So we strip this include from the | ||||
# preprocessor invocation. | ||||
with open(path, 'rb') as fh: | ||||
lines = [l for l in fh if not l.startswith(b'#include <stddef.h>')] | ||||
Gregory Szorc
|
r30822 | |||
Gregory Szorc
|
r30895 | fd, input_file = tempfile.mkstemp(suffix='.h') | ||
os.write(fd, b''.join(lines)) | ||||
os.close(fd) | ||||
Gregory Szorc
|
r30822 | |||
Gregory Szorc
|
r30895 | try: | ||
process = subprocess.Popen(args + [input_file], stdout=subprocess.PIPE) | ||||
output = process.communicate()[0] | ||||
ret = process.poll() | ||||
if ret: | ||||
raise Exception('preprocessor exited with error') | ||||
Gregory Szorc
|
r30822 | |||
Gregory Szorc
|
r30895 | return output | ||
finally: | ||||
os.unlink(input_file) | ||||
Gregory Szorc
|
r30822 | |||
Gregory Szorc
|
r30895 | |||
def normalize_output(output): | ||||
Gregory Szorc
|
r30822 | lines = [] | ||
for line in output.splitlines(): | ||||
# CFFI's parser doesn't like __attribute__ on UNIX compilers. | ||||
if line.startswith(b'__attribute__ ((visibility ("default"))) '): | ||||
line = line[len(b'__attribute__ ((visibility ("default"))) '):] | ||||
Gregory Szorc
|
r30895 | if line.startswith(b'__attribute__((deprecated('): | ||
continue | ||||
elif b'__declspec(deprecated(' in line: | ||||
continue | ||||
Gregory Szorc
|
r30822 | lines.append(line) | ||
return b'\n'.join(lines) | ||||
Gregory Szorc
|
r30435 | |||
Gregory Szorc
|
r30895 | |||
Gregory Szorc
|
r30435 | ffi = cffi.FFI() | ||
ffi.set_source('_zstd_cffi', ''' | ||||
Gregory Szorc
|
r30895 | #include "mem.h" | ||
Gregory Szorc
|
r30435 | #define ZSTD_STATIC_LINKING_ONLY | ||
#include "zstd.h" | ||||
Gregory Szorc
|
r30895 | #define ZDICT_STATIC_LINKING_ONLY | ||
#include "pool.h" | ||||
#include "zdict.h" | ||||
Gregory Szorc
|
r30822 | ''', sources=SOURCES, include_dirs=INCLUDE_DIRS) | ||
Gregory Szorc
|
r30435 | |||
Gregory Szorc
|
r30895 | DEFINE = re.compile(b'^\\#define ([a-zA-Z0-9_]+) ') | ||
sources = [] | ||||
for header in HEADERS: | ||||
preprocessed = preprocess(header) | ||||
sources.append(normalize_output(preprocessed)) | ||||
# Do another pass over source and find constants that were preprocessed | ||||
# away. | ||||
with open(header, 'rb') as fh: | ||||
for line in fh: | ||||
line = line.strip() | ||||
m = DEFINE.match(line) | ||||
if not m: | ||||
continue | ||||
# The parser doesn't like some constants with complex values. | ||||
if m.group(1) in (b'ZSTD_LIB_VERSION', b'ZSTD_VERSION_STRING'): | ||||
continue | ||||
sources.append(m.group(0) + b' ...') | ||||
ffi.cdef(u'\n'.join(s.decode('latin1') for s in sources)) | ||||
Gregory Szorc
|
r30435 | |||
if __name__ == '__main__': | ||||
ffi.compile() | ||||