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