#! /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()