mirror of https://github.com/python/cpython
204 lines
6.7 KiB
Python
204 lines
6.7 KiB
Python
#! /usr/bin/env python3
|
|
# Script for preparing OpenSSL for building on Windows.
|
|
# Uses Perl to create nmake makefiles and otherwise prepare the way
|
|
# for building on 32 or 64 bit platforms.
|
|
|
|
# Script originally authored by Mark Hammond.
|
|
# Major revisions by:
|
|
# Martin v. Löwis
|
|
# Christian Heimes
|
|
# Zachary Ware
|
|
|
|
# THEORETICALLY, you can:
|
|
# * Unpack the latest OpenSSL release where $(opensslDir) in
|
|
# PCbuild\pyproject.props expects it to be.
|
|
# * Install ActivePerl and ensure it is somewhere on your path.
|
|
# * Run this script with the OpenSSL source dir as the only argument.
|
|
#
|
|
# it should configure OpenSSL such that it is ready to be built by
|
|
# ssl.vcxproj on 32 or 64 bit platforms.
|
|
|
|
from __future__ import print_function
|
|
|
|
import os
|
|
import re
|
|
import sys
|
|
import subprocess
|
|
from shutil import copy
|
|
|
|
# Find all "foo.exe" files on the PATH.
|
|
def find_all_on_path(filename, extras=None):
|
|
entries = os.environ["PATH"].split(os.pathsep)
|
|
ret = []
|
|
for p in entries:
|
|
fname = os.path.abspath(os.path.join(p, filename))
|
|
if os.path.isfile(fname) and fname not in ret:
|
|
ret.append(fname)
|
|
if extras:
|
|
for p in extras:
|
|
fname = os.path.abspath(os.path.join(p, filename))
|
|
if os.path.isfile(fname) and fname not in ret:
|
|
ret.append(fname)
|
|
return ret
|
|
|
|
|
|
# Find a suitable Perl installation for OpenSSL.
|
|
# cygwin perl does *not* work. ActivePerl does.
|
|
# Being a Perl dummy, the simplest way I can check is if the "Win32" package
|
|
# is available.
|
|
def find_working_perl(perls):
|
|
for perl in perls:
|
|
try:
|
|
subprocess.check_output([perl, "-e", "use Win32;"])
|
|
except subprocess.CalledProcessError:
|
|
continue
|
|
else:
|
|
return perl
|
|
|
|
if perls:
|
|
print("The following perl interpreters were found:")
|
|
for p in perls:
|
|
print(" ", p)
|
|
print(" None of these versions appear suitable for building OpenSSL")
|
|
else:
|
|
print("NO perl interpreters were found on this machine at all!")
|
|
print(" Please install ActivePerl and ensure it appears on your path")
|
|
|
|
|
|
def copy_includes(makefile, suffix):
|
|
dir = 'inc'+suffix+'\\openssl'
|
|
try:
|
|
os.makedirs(dir)
|
|
except OSError:
|
|
pass
|
|
copy_if_different = r'$(PERL) $(SRC_D)\util\copy-if-different.pl'
|
|
with open(makefile) as fin:
|
|
for line in fin:
|
|
if copy_if_different in line:
|
|
perl, script, src, dest = line.split()
|
|
if not '$(INCO_D)' in dest:
|
|
continue
|
|
# We're in the root of the source tree
|
|
src = src.replace('$(SRC_D)', '.').strip('"')
|
|
dest = dest.strip('"').replace('$(INCO_D)', dir)
|
|
print('copying', src, 'to', dest)
|
|
copy(src, dest)
|
|
|
|
|
|
def run_configure(configure, do_script):
|
|
print("perl Configure "+configure+" no-idea no-mdc2")
|
|
os.system("perl Configure "+configure+" no-idea no-mdc2")
|
|
print(do_script)
|
|
os.system(do_script)
|
|
|
|
def fix_uplink():
|
|
# uplink.c tries to find the OPENSSL_Applink function exported from the current
|
|
# executable. However, we export it from _ssl[_d].pyd instead. So we update the
|
|
# module name here before building.
|
|
with open('ms\\uplink.c', 'r', encoding='utf-8') as f1:
|
|
code = list(f1)
|
|
os.replace('ms\\uplink.c', 'ms\\uplink.c.orig')
|
|
already_patched = False
|
|
with open('ms\\uplink.c', 'w', encoding='utf-8') as f2:
|
|
for line in code:
|
|
if not already_patched:
|
|
if re.search('MODIFIED FOR CPYTHON _ssl MODULE', line):
|
|
already_patched = True
|
|
elif re.match(r'^\s+if\s*\(\(h\s*=\s*GetModuleHandle[AW]?\(NULL\)\)\s*==\s*NULL\)', line):
|
|
f2.write("/* MODIFIED FOR CPYTHON _ssl MODULE */\n")
|
|
f2.write('if ((h = GetModuleHandleW(L"_ssl.pyd")) == NULL) if ((h = GetModuleHandleW(L"_ssl_d.pyd")) == NULL)\n')
|
|
already_patched = True
|
|
f2.write(line)
|
|
if not already_patched:
|
|
print("WARN: failed to patch ms\\uplink.c")
|
|
|
|
def prep(arch):
|
|
makefile_template = "ms\\ntdll{}.mak"
|
|
generated_makefile = makefile_template.format('')
|
|
if arch == "x86":
|
|
configure = "VC-WIN32"
|
|
do_script = "ms\\do_nasm"
|
|
suffix = "32"
|
|
elif arch == "amd64":
|
|
configure = "VC-WIN64A"
|
|
do_script = "ms\\do_win64a"
|
|
suffix = "64"
|
|
else:
|
|
raise ValueError('Unrecognized platform: %s' % arch)
|
|
|
|
print("Creating the makefiles...")
|
|
sys.stdout.flush()
|
|
# run configure, copy includes, patch files
|
|
run_configure(configure, do_script)
|
|
makefile = makefile_template.format(suffix)
|
|
try:
|
|
os.unlink(makefile)
|
|
except FileNotFoundError:
|
|
pass
|
|
os.rename(generated_makefile, makefile)
|
|
copy_includes(makefile, suffix)
|
|
|
|
print('patching ms\\uplink.c...')
|
|
fix_uplink()
|
|
|
|
def main():
|
|
if len(sys.argv) == 1:
|
|
print("Not enough arguments: directory containing OpenSSL",
|
|
"sources must be supplied")
|
|
sys.exit(1)
|
|
|
|
if len(sys.argv) == 3 and sys.argv[2] not in ('x86', 'amd64'):
|
|
print("Second argument must be x86 or amd64")
|
|
sys.exit(1)
|
|
|
|
if len(sys.argv) > 3:
|
|
print("Too many arguments supplied, all we need is the directory",
|
|
"containing OpenSSL sources and optionally the architecture")
|
|
sys.exit(1)
|
|
|
|
ssl_dir = sys.argv[1]
|
|
arch = sys.argv[2] if len(sys.argv) >= 3 else None
|
|
|
|
if not os.path.isdir(ssl_dir):
|
|
print(ssl_dir, "is not an existing directory!")
|
|
sys.exit(1)
|
|
|
|
# perl should be on the path, but we also look in "\perl" and "c:\\perl"
|
|
# as "well known" locations
|
|
perls = find_all_on_path("perl.exe", [r"\perl\bin",
|
|
r"C:\perl\bin",
|
|
r"\perl64\bin",
|
|
r"C:\perl64\bin",
|
|
])
|
|
perl = find_working_perl(perls)
|
|
if perl:
|
|
print("Found a working perl at '%s'" % (perl,))
|
|
else:
|
|
sys.exit(1)
|
|
if not find_all_on_path('nmake.exe'):
|
|
print('Could not find nmake.exe, try running env.bat')
|
|
sys.exit(1)
|
|
if not find_all_on_path('nasm.exe'):
|
|
print('Could not find nasm.exe, please add to PATH')
|
|
sys.exit(1)
|
|
sys.stdout.flush()
|
|
|
|
# Put our working Perl at the front of our path
|
|
os.environ["PATH"] = os.path.dirname(perl) + \
|
|
os.pathsep + \
|
|
os.environ["PATH"]
|
|
|
|
old_cwd = os.getcwd()
|
|
try:
|
|
os.chdir(ssl_dir)
|
|
if arch:
|
|
prep(arch)
|
|
else:
|
|
for arch in ['amd64', 'x86']:
|
|
prep(arch)
|
|
finally:
|
|
os.chdir(old_cwd)
|
|
|
|
if __name__=='__main__':
|
|
main()
|