2002-04-03 18:45:09 -07:00
|
|
|
#! /usr/bin/env python
|
2009-08-03 15:14:44 -04:00
|
|
|
#
|
|
|
|
# setup.py : Distutils setup script
|
|
|
|
#
|
|
|
|
# Part of the Python Cryptography Toolkit
|
|
|
|
#
|
|
|
|
# ===================================================================
|
|
|
|
# Portions Copyright (c) 2001, 2002, 2003 Python Software Foundation;
|
|
|
|
# All Rights Reserved
|
|
|
|
#
|
|
|
|
# This file contains code from the Python 2.2 setup.py module (the
|
|
|
|
# "Original Code"), with modifications made after it was incorporated
|
|
|
|
# into PyCrypto (the "Modifications").
|
|
|
|
#
|
|
|
|
# To the best of our knowledge, the Python Software Foundation is the
|
|
|
|
# copyright holder of the Original Code, and has licensed it under the
|
|
|
|
# Python 2.2 license. See the file LEGAL/copy/LICENSE.python-2.2 for
|
|
|
|
# details.
|
|
|
|
#
|
|
|
|
# The Modifications to this file are dedicated to the public domain.
|
|
|
|
# To the extent that dedication to the public domain is not available,
|
|
|
|
# everyone is granted a worldwide, perpetual, royalty-free,
|
|
|
|
# non-exclusive license to exercise all rights associated with the
|
|
|
|
# contents of this file for any purpose whatsoever. No rights are
|
|
|
|
# reserved.
|
|
|
|
#
|
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
|
|
# BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
|
|
# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
# SOFTWARE.
|
|
|
|
# ===================================================================
|
2002-07-11 14:31:19 -07:00
|
|
|
|
2014-05-25 11:03:35 +02:00
|
|
|
longdesc = """
|
|
|
|
This is a self-contained, public domain package of low-level
|
|
|
|
cryptographic primitives.
|
2002-07-11 14:31:19 -07:00
|
|
|
|
2014-05-25 11:03:35 +02:00
|
|
|
It supports Python 2.4 or newer, all Python 3 versions and PyPy.
|
|
|
|
|
|
|
|
It is a (recent) fork of PyCrypto (https://www.dlitz.net/software/pycrypto).
|
|
|
|
"""
|
|
|
|
|
|
|
|
from distutils.core import Extension, Command, setup
|
2003-02-28 17:38:29 -07:00
|
|
|
from distutils.command.build_ext import build_ext
|
2014-05-22 16:29:52 +02:00
|
|
|
from distutils.errors import CCompilerError
|
|
|
|
import distutils
|
|
|
|
import os, sys
|
2002-04-03 18:45:09 -07:00
|
|
|
|
2002-07-18 13:05:37 -07:00
|
|
|
if sys.version[0:1] == '1':
|
2010-12-28 16:26:52 -05:00
|
|
|
raise RuntimeError ("The Python Cryptography Toolkit requires "
|
|
|
|
"Python 2.x or 3.x to build.")
|
2002-07-18 13:05:37 -07:00
|
|
|
|
2010-12-28 16:26:52 -05:00
|
|
|
try:
|
|
|
|
# Python 3
|
|
|
|
from distutils.command.build_py import build_py_2to3 as build_py
|
|
|
|
except ImportError:
|
|
|
|
# Python 2
|
|
|
|
from distutils.command.build_py import build_py
|
2011-10-18 23:20:26 +02:00
|
|
|
|
2014-05-22 16:29:52 +02:00
|
|
|
|
2010-12-29 13:21:05 -05:00
|
|
|
# Work around the print / print() issue with Python 2.x and 3.x. We only need
|
|
|
|
# to print at one point of the code, which makes this easy
|
2010-12-28 16:26:52 -05:00
|
|
|
def PrintErr(*args, **kwd):
|
|
|
|
fout = kwd.get("file", sys.stderr)
|
|
|
|
w = fout.write
|
|
|
|
if args:
|
|
|
|
w(str(args[0]))
|
|
|
|
sep = kwd.get("sep", " ")
|
|
|
|
for a in args[1:]:
|
|
|
|
w(sep)
|
|
|
|
w(str(a))
|
|
|
|
w(kwd.get("end", "\n"))
|
2008-08-13 23:11:05 -04:00
|
|
|
|
2014-05-22 16:29:52 +02:00
|
|
|
def test_compilation(program, extra_cc_options=None, extra_libraries=None):
|
|
|
|
"""Test if a certain C program can be compiled."""
|
|
|
|
|
|
|
|
import tempfile
|
|
|
|
|
|
|
|
# Create a temporary file with the C program
|
|
|
|
fname = tempfile.mktemp(".c")
|
|
|
|
f = open(fname, 'w')
|
|
|
|
f.write(program)
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
# Name for the temporary executable
|
|
|
|
oname = tempfile.mktemp(".out")
|
|
|
|
|
|
|
|
# Mute the compiler and the linker
|
|
|
|
old_stdout = os.dup(sys.stdout.fileno())
|
|
|
|
old_stderr = os.dup(sys.stderr.fileno())
|
|
|
|
dev_null = open(os.devnull, "w")
|
|
|
|
os.dup2(dev_null.fileno(), sys.stdout.fileno())
|
|
|
|
os.dup2(dev_null.fileno(), sys.stderr.fileno())
|
|
|
|
|
|
|
|
objects = []
|
|
|
|
try:
|
|
|
|
compiler = distutils.ccompiler.new_compiler()
|
|
|
|
distutils.sysconfig.customize_compiler(compiler)
|
|
|
|
objects = compiler.compile([fname], extra_postargs=extra_cc_options)
|
|
|
|
compiler.link_executable(objects, oname, libraries=extra_libraries)
|
|
|
|
result = True
|
|
|
|
except CCompilerError:
|
|
|
|
result = False
|
|
|
|
for f in objects + [fname, oname]:
|
|
|
|
try:
|
|
|
|
os.remove(f)
|
|
|
|
except OSError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Restore stdout and stderr
|
|
|
|
if old_stdout is not None:
|
|
|
|
os.dup2(old_stdout, sys.stdout.fileno())
|
|
|
|
if old_stderr is not None:
|
|
|
|
os.dup2(old_stderr, sys.stderr.fileno())
|
|
|
|
if dev_null is not None:
|
|
|
|
dev_null.close()
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
def libgmp_exists():
|
|
|
|
'''Tests if the GMP library is available'''
|
|
|
|
|
|
|
|
source = """
|
|
|
|
#include <gmp.h>
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
mpz_init((void*)0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
|
|
|
|
return test_compilation(source, extra_libraries=('gmp',))
|
|
|
|
|
|
|
|
def aesni_support():
|
|
|
|
|
|
|
|
extra_cc_options = []
|
|
|
|
|
|
|
|
source = """
|
|
|
|
#include <cpuid.h>
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
unsigned int eax, ebx, ecx, edx;
|
|
|
|
__get_cpuid(1, &eax, &ebx, &ecx, &edx);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
if test_compilation(source):
|
|
|
|
extra_cc_options += [ "-DHAVE_CPUID_H" ]
|
|
|
|
else:
|
|
|
|
return (False, None)
|
|
|
|
|
|
|
|
source = """
|
|
|
|
#include <wmmintrin.h>
|
|
|
|
__m128i f(__m128i x, __m128i y) {
|
|
|
|
return _mm_aesenc_si128(x, y);
|
|
|
|
}
|
|
|
|
int main(void) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
|
|
|
|
result = test_compilation(source)
|
|
|
|
if not result:
|
|
|
|
result = test_compilation(source, extra_cc_options=['-maes'])
|
|
|
|
if result:
|
|
|
|
extra_cc_options += ['-maes']
|
|
|
|
return (result, extra_cc_options)
|
|
|
|
|
|
|
|
|
2008-09-15 21:32:58 -04:00
|
|
|
def endianness_macro():
|
2014-05-22 16:29:52 +02:00
|
|
|
import struct
|
2008-09-16 12:48:24 -04:00
|
|
|
s = struct.pack("@I", 0x33221100)
|
2010-12-28 16:26:52 -05:00
|
|
|
if s == "\x00\x11\x22\x33".encode(): # little endian
|
2008-09-15 21:32:58 -04:00
|
|
|
return ('PCT_LITTLE_ENDIAN', 1)
|
2010-12-28 16:26:52 -05:00
|
|
|
elif s == "\x33\x22\x11\x00".encode(): # big endian
|
2008-09-15 21:32:58 -04:00
|
|
|
return ('PCT_BIG_ENDIAN', 1)
|
|
|
|
raise AssertionError("Machine is neither little-endian nor big-endian")
|
|
|
|
|
2003-02-28 17:38:29 -07:00
|
|
|
class PCTBuildExt (build_ext):
|
|
|
|
def build_extensions(self):
|
|
|
|
# Detect which modules should be compiled
|
|
|
|
self.detect_modules()
|
2009-03-10 23:40:47 -04:00
|
|
|
|
2003-02-28 17:38:29 -07:00
|
|
|
build_ext.build_extensions(self)
|
|
|
|
|
|
|
|
def detect_modules (self):
|
2014-05-22 16:29:52 +02:00
|
|
|
# Detect libgmp and don't build _fastmath if it is missing.
|
|
|
|
if libgmp_exists():
|
|
|
|
PrintErr("Compiling _fastmath using the GMP library")
|
2011-10-10 19:14:30 -04:00
|
|
|
else:
|
2014-05-22 16:29:52 +02:00
|
|
|
PrintErr ("warning: GMP library not found; Not building " +
|
2011-10-10 19:14:30 -04:00
|
|
|
"Crypto.PublicKey._fastmath.")
|
|
|
|
self.__remove_extensions(["Crypto.PublicKey._fastmath"])
|
2010-12-29 13:21:05 -05:00
|
|
|
|
2014-05-22 16:29:52 +02:00
|
|
|
# Detect compiler support for AESNI
|
|
|
|
compile_supports_aesni, extra_cc_options = aesni_support()
|
|
|
|
for option in extra_cc_options:
|
|
|
|
self.__add_compiler_option(option)
|
|
|
|
if compile_supports_aesni:
|
|
|
|
PrintErr("Compiling support for Intel AES instructions")
|
|
|
|
else:
|
|
|
|
PrintErr ("warning: no support for Intel AESNI intructions; Not building " +
|
|
|
|
"Crypto.Cipher._AESNI")
|
2013-10-28 02:52:39 +01:00
|
|
|
self.__remove_extensions(["Crypto.Cipher._AESNI"])
|
2009-08-18 19:57:08 -04:00
|
|
|
|
|
|
|
def __remove_extensions(self, names):
|
2014-05-22 16:29:52 +02:00
|
|
|
"""Remove the specified extension from the list of extensions
|
|
|
|
to build"""
|
|
|
|
|
|
|
|
self.extensions = [ x for x in self.extensions if x.name not in names ]
|
2003-02-28 17:38:29 -07:00
|
|
|
|
2009-03-10 23:40:47 -04:00
|
|
|
def __remove_compiler_option(self, option):
|
|
|
|
"""Remove the specified compiler option.
|
|
|
|
|
|
|
|
Return true if the option was found. Return false otherwise.
|
|
|
|
"""
|
|
|
|
found = 0
|
|
|
|
for attrname in ('compiler', 'compiler_so'):
|
|
|
|
compiler = getattr(self.compiler, attrname, None)
|
|
|
|
if compiler is not None:
|
|
|
|
while option in compiler:
|
|
|
|
compiler.remove(option)
|
|
|
|
found += 1
|
|
|
|
return found
|
|
|
|
|
|
|
|
def __add_compiler_option(self, option):
|
|
|
|
for attrname in ('compiler', 'compiler_so'):
|
|
|
|
compiler = getattr(self.compiler, attrname, None)
|
|
|
|
if compiler is not None:
|
|
|
|
compiler.append(option)
|
|
|
|
|
2008-08-13 23:11:05 -04:00
|
|
|
class PCTBuildPy(build_py):
|
|
|
|
def find_package_modules(self, package, package_dir, *args, **kwargs):
|
2010-12-29 13:21:05 -05:00
|
|
|
modules = build_py.find_package_modules(self, package, package_dir,
|
|
|
|
*args, **kwargs)
|
2008-08-13 23:11:05 -04:00
|
|
|
|
|
|
|
# Exclude certain modules
|
2008-09-15 18:16:11 -04:00
|
|
|
retval = []
|
|
|
|
for item in modules:
|
|
|
|
pkg, module = item[:2]
|
|
|
|
retval.append(item)
|
|
|
|
return retval
|
2008-08-13 23:11:05 -04:00
|
|
|
|
2008-11-21 13:04:18 -05:00
|
|
|
class TestCommand(Command):
|
|
|
|
|
|
|
|
description = "Run self-test"
|
|
|
|
|
2011-01-10 18:52:40 +01:00
|
|
|
# Long option name, short option name, description
|
2008-11-21 13:04:18 -05:00
|
|
|
user_options = [
|
|
|
|
('skip-slow-tests', None,
|
2011-01-10 18:52:40 +01:00
|
|
|
'Skip slow tests'),
|
|
|
|
('module=', 'm', 'Test a single module (e.g. Cipher, PublicKey)')
|
2008-11-21 13:04:18 -05:00
|
|
|
]
|
|
|
|
|
|
|
|
def initialize_options(self):
|
|
|
|
self.build_dir = None
|
|
|
|
self.skip_slow_tests = None
|
2011-01-10 18:52:40 +01:00
|
|
|
self.module = None
|
2008-11-21 13:04:18 -05:00
|
|
|
|
|
|
|
def finalize_options(self):
|
|
|
|
self.set_undefined_options('install', ('build_lib', 'build_dir'))
|
|
|
|
self.config = {'slow_tests': not self.skip_slow_tests}
|
|
|
|
|
|
|
|
def run(self):
|
2012-10-14 12:59:59 +02:00
|
|
|
# Run sub commands
|
|
|
|
for cmd_name in self.get_sub_commands():
|
|
|
|
self.run_command(cmd_name)
|
|
|
|
|
2008-11-21 13:04:18 -05:00
|
|
|
# Run SelfTest
|
|
|
|
self.announce("running self-tests")
|
|
|
|
old_path = sys.path[:]
|
|
|
|
try:
|
|
|
|
sys.path.insert(0, self.build_dir)
|
|
|
|
from Crypto import SelfTest
|
2011-01-10 18:52:40 +01:00
|
|
|
moduleObj = None
|
|
|
|
if self.module:
|
|
|
|
if self.module.count('.')==0:
|
|
|
|
# Test a whole a sub-package
|
|
|
|
full_module = "Crypto.SelfTest." + self.module
|
|
|
|
module_name = self.module
|
|
|
|
else:
|
|
|
|
# Test only a module
|
|
|
|
# Assume only one dot is present
|
|
|
|
comps = self.module.split('.')
|
|
|
|
module_name = "test_" + comps[1]
|
|
|
|
full_module = "Crypto.SelfTest." + comps[0] + "." + module_name
|
|
|
|
# Import sub-package or module
|
|
|
|
moduleObj = __import__( full_module, globals(), locals(), module_name )
|
|
|
|
SelfTest.run(module=moduleObj, verbosity=self.verbose, stream=sys.stdout, config=self.config)
|
2008-11-21 13:04:18 -05:00
|
|
|
finally:
|
|
|
|
# Restore sys.path
|
|
|
|
sys.path[:] = old_path
|
|
|
|
|
|
|
|
# Run slower self-tests
|
|
|
|
self.announce("running extended self-tests")
|
|
|
|
|
2012-10-14 12:59:59 +02:00
|
|
|
sub_commands = [ ('build', None) ]
|
|
|
|
|
2014-05-17 21:40:28 +02:00
|
|
|
|
2014-05-25 11:03:35 +02:00
|
|
|
setup(
|
|
|
|
name = "pycryptodome",
|
|
|
|
version = "3.0rc1",
|
|
|
|
description = "Cryptographic library for Python",
|
|
|
|
long_description = longdesc,
|
|
|
|
author = "Legrandin",
|
|
|
|
author_email = "helderijs@gmail.com",
|
|
|
|
url = "https://www.pycryptodome.org",
|
|
|
|
license = "Public Domain",
|
|
|
|
platforms = 'Posix; MacOS X; Windows',
|
|
|
|
classifiers = [
|
2014-05-25 14:33:34 +02:00
|
|
|
'Development Status :: 4 - Beta',
|
2014-05-25 11:03:35 +02:00
|
|
|
'License :: Public Domain',
|
|
|
|
'Intended Audience :: Developers',
|
|
|
|
'Operating System :: Unix',
|
|
|
|
'Operating System :: Microsoft :: Windows',
|
|
|
|
'Operating System :: MacOS :: MacOS X',
|
|
|
|
'Topic :: Security :: Cryptography',
|
|
|
|
'Programming Language :: Python :: 2',
|
|
|
|
'Programming Language :: Python :: 2.4',
|
|
|
|
'Programming Language :: Python :: 3',
|
|
|
|
],
|
|
|
|
packages = [
|
|
|
|
"Crypto",
|
|
|
|
"Crypto.Cipher",
|
|
|
|
"Crypto.Hash",
|
|
|
|
"Crypto.IO",
|
|
|
|
"Crypto.PublicKey",
|
|
|
|
"Crypto.Protocol",
|
|
|
|
"Crypto.Random",
|
|
|
|
"Crypto.Random.Fortuna",
|
|
|
|
"Crypto.Signature",
|
|
|
|
"Crypto.Util",
|
|
|
|
"Crypto.SelfTest",
|
|
|
|
"Crypto.SelfTest.Cipher",
|
|
|
|
"Crypto.SelfTest.Hash",
|
|
|
|
"Crypto.SelfTest.IO",
|
|
|
|
"Crypto.SelfTest.Protocol",
|
|
|
|
"Crypto.SelfTest.PublicKey",
|
|
|
|
"Crypto.SelfTest.Random",
|
|
|
|
"Crypto.SelfTest.Random.Fortuna",
|
|
|
|
"Crypto.SelfTest.Signature",
|
|
|
|
"Crypto.SelfTest.Util",
|
|
|
|
],
|
|
|
|
package_dir = { "Crypto": "lib/Crypto" },
|
|
|
|
package_data = {
|
|
|
|
"Crypto.SelfTest.Hash" : [
|
|
|
|
"test_vectors/SHA3/*.txt" ],
|
|
|
|
"Crypto.SelfTest.Signature" : [
|
|
|
|
"test_vectors/DSA/*.*" ],
|
|
|
|
},
|
|
|
|
cmdclass = {
|
|
|
|
'build_ext':PCTBuildExt,
|
|
|
|
'build_py': PCTBuildPy,
|
|
|
|
'test': TestCommand
|
|
|
|
},
|
|
|
|
ext_modules = [
|
|
|
|
# _fastmath (uses GNU mp library)
|
|
|
|
Extension("Crypto.PublicKey._fastmath",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
libraries=['gmp'],
|
|
|
|
sources=["src/_fastmath.c"]),
|
|
|
|
|
|
|
|
# Hash functions
|
|
|
|
Extension("Crypto.Hash.MD2",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/MD2.c"]),
|
|
|
|
Extension("Crypto.Hash.MD4",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/MD4.c"]),
|
|
|
|
Extension("Crypto.Hash.SHA256",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA256.c"]),
|
|
|
|
Extension("Crypto.Hash.SHA224",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA224.c"]),
|
|
|
|
Extension("Crypto.Hash.SHA384",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA384.c"]),
|
|
|
|
Extension("Crypto.Hash.SHA512",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA512.c"]),
|
|
|
|
Extension("Crypto.Hash.RIPEMD160",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/RIPEMD160.c"],
|
|
|
|
define_macros=[endianness_macro()]),
|
|
|
|
Extension("Crypto.Hash.SHA3_224",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA3_224.c"]),
|
|
|
|
Extension("Crypto.Hash.SHA3_256",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA3_256.c"]),
|
|
|
|
Extension("Crypto.Hash.SHA3_384",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA3_384.c"]),
|
|
|
|
Extension("Crypto.Hash.SHA3_512",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/SHA3_512.c"]),
|
|
|
|
|
|
|
|
# Block encryption algorithms
|
|
|
|
Extension("Crypto.Cipher._AES",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/AES.c"]),
|
|
|
|
Extension("Crypto.Cipher._AESNI",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/AESNI.c"]),
|
|
|
|
Extension("Crypto.Cipher._ARC2",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/ARC2.c"]),
|
|
|
|
Extension("Crypto.Cipher._Blowfish",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/Blowfish.c"]),
|
|
|
|
Extension("Crypto.Cipher._CAST",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/CAST.c"]),
|
|
|
|
Extension("Crypto.Cipher._DES",
|
|
|
|
include_dirs=['src/', 'src/libtom/'],
|
|
|
|
sources=["src/DES.c"]),
|
|
|
|
Extension("Crypto.Cipher._DES3",
|
|
|
|
include_dirs=['src/', 'src/libtom/'],
|
|
|
|
sources=["src/DES3.c"]),
|
|
|
|
Extension("Crypto.Util._galois",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=['src/galois.c']),
|
|
|
|
Extension("Crypto.Util.cpuid",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=['src/cpuid.c']),
|
|
|
|
|
|
|
|
# Stream ciphers
|
|
|
|
Extension("Crypto.Cipher._ARC4",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/ARC4.c"]),
|
|
|
|
Extension("Crypto.Cipher._XOR",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=["src/XOR.c"]),
|
|
|
|
Extension("Crypto.Cipher._Salsa20",
|
|
|
|
include_dirs=['src/', 'src/libtom/'],
|
|
|
|
sources=["src/Salsa20.c"]),
|
|
|
|
|
|
|
|
# Utility modules
|
|
|
|
Extension("Crypto.Util.strxor",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=['src/strxor.c']),
|
|
|
|
|
|
|
|
# Counter modules
|
|
|
|
Extension("Crypto.Util._counter",
|
|
|
|
include_dirs=['src/'],
|
|
|
|
sources=['src/_counter.c']),
|
|
|
|
]
|
|
|
|
)
|