Home | History | Annotate | Download | only in PCbuild
      1 #! /usr/bin/env python3
      2 # Script for preparing OpenSSL for building on Windows.
      3 # Uses Perl to create nmake makefiles and otherwise prepare the way
      4 # for building on 32 or 64 bit platforms.
      5 
      6 # Script originally authored by Mark Hammond.
      7 # Major revisions by:
      8 #   Martin v. Lwis
      9 #   Christian Heimes
     10 #   Zachary Ware
     11 
     12 # THEORETICALLY, you can:
     13 # * Unpack the latest OpenSSL release where $(opensslDir) in
     14 #   PCbuild\pyproject.props expects it to be.
     15 # * Install ActivePerl and ensure it is somewhere on your path.
     16 # * Run this script with the OpenSSL source dir as the only argument.
     17 #
     18 # it should configure OpenSSL such that it is ready to be built by
     19 # ssl.vcxproj on 32 or 64 bit platforms.
     20 
     21 from __future__ import print_function
     22 
     23 import os
     24 import re
     25 import sys
     26 import subprocess
     27 from shutil import copy
     28 
     29 # Find all "foo.exe" files on the PATH.
     30 def find_all_on_path(filename, extras=None):
     31     entries = os.environ["PATH"].split(os.pathsep)
     32     ret = []
     33     for p in entries:
     34         fname = os.path.abspath(os.path.join(p, filename))
     35         if os.path.isfile(fname) and fname not in ret:
     36             ret.append(fname)
     37     if extras:
     38         for p in extras:
     39             fname = os.path.abspath(os.path.join(p, filename))
     40             if os.path.isfile(fname) and fname not in ret:
     41                 ret.append(fname)
     42     return ret
     43 
     44 
     45 # Find a suitable Perl installation for OpenSSL.
     46 # cygwin perl does *not* work.  ActivePerl does.
     47 # Being a Perl dummy, the simplest way I can check is if the "Win32" package
     48 # is available.
     49 def find_working_perl(perls):
     50     for perl in perls:
     51         try:
     52             subprocess.check_output([perl, "-e", "use Win32;"])
     53         except subprocess.CalledProcessError:
     54             continue
     55         else:
     56             return perl
     57 
     58     if perls:
     59         print("The following perl interpreters were found:")
     60         for p in perls:
     61             print(" ", p)
     62         print(" None of these versions appear suitable for building OpenSSL")
     63     else:
     64         print("NO perl interpreters were found on this machine at all!")
     65     print(" Please install ActivePerl and ensure it appears on your path")
     66 
     67 
     68 def copy_includes(makefile, suffix):
     69     dir = 'inc'+suffix+'\\openssl'
     70     try:
     71         os.makedirs(dir)
     72     except OSError:
     73         pass
     74     copy_if_different = r'$(PERL) $(SRC_D)\util\copy-if-different.pl'
     75     with open(makefile) as fin:
     76         for line in fin:
     77             if copy_if_different in line:
     78                 perl, script, src, dest = line.split()
     79                 if not '$(INCO_D)' in dest:
     80                     continue
     81                 # We're in the root of the source tree
     82                 src = src.replace('$(SRC_D)', '.').strip('"')
     83                 dest = dest.strip('"').replace('$(INCO_D)', dir)
     84                 print('copying', src, 'to', dest)
     85                 copy(src, dest)
     86 
     87 
     88 def run_configure(configure, do_script):
     89     print("perl Configure "+configure+" no-idea no-mdc2")
     90     os.system("perl Configure "+configure+" no-idea no-mdc2")
     91     print(do_script)
     92     os.system(do_script)
     93 
     94 def fix_uplink():
     95     # uplink.c tries to find the OPENSSL_Applink function exported from the current
     96     # executable. However, we export it from _ssl[_d].pyd instead. So we update the
     97     # module name here before building.
     98     with open('ms\\uplink.c', 'r', encoding='utf-8') as f1:
     99         code = list(f1)
    100     os.replace('ms\\uplink.c', 'ms\\uplink.c.orig')
    101     already_patched = False
    102     with open('ms\\uplink.c', 'w', encoding='utf-8') as f2:
    103         for line in code:
    104             if not already_patched:
    105                 if re.search('MODIFIED FOR CPYTHON _ssl MODULE', line):
    106                     already_patched = True
    107                 elif re.match(r'^\s+if\s*\(\(h\s*=\s*GetModuleHandle[AW]?\(NULL\)\)\s*==\s*NULL\)', line):
    108                     f2.write("/* MODIFIED FOR CPYTHON _ssl MODULE */\n")
    109                     f2.write('if ((h = GetModuleHandleW(L"_ssl.pyd")) == NULL) if ((h = GetModuleHandleW(L"_ssl_d.pyd")) == NULL)\n')
    110                     already_patched = True
    111             f2.write(line)
    112     if not already_patched:
    113         print("WARN: failed to patch ms\\uplink.c")
    114 
    115 def prep(arch):
    116     makefile_template = "ms\\ntdll{}.mak"
    117     generated_makefile = makefile_template.format('')
    118     if arch == "x86":
    119         configure = "VC-WIN32"
    120         do_script = "ms\\do_nasm"
    121         suffix = "32"
    122     elif arch == "amd64":
    123         configure = "VC-WIN64A"
    124         do_script = "ms\\do_win64a"
    125         suffix = "64"
    126     else:
    127         raise ValueError('Unrecognized platform: %s' % arch)
    128 
    129     print("Creating the makefiles...")
    130     sys.stdout.flush()
    131     # run configure, copy includes, patch files
    132     run_configure(configure, do_script)
    133     makefile = makefile_template.format(suffix)
    134     try:
    135         os.unlink(makefile)
    136     except FileNotFoundError:
    137         pass
    138     os.rename(generated_makefile, makefile)
    139     copy_includes(makefile, suffix)
    140 
    141     print('patching ms\\uplink.c...')
    142     fix_uplink()
    143 
    144 def main():
    145     if len(sys.argv) == 1:
    146         print("Not enough arguments: directory containing OpenSSL",
    147               "sources must be supplied")
    148         sys.exit(1)
    149 
    150     if len(sys.argv) == 3 and sys.argv[2] not in ('x86', 'amd64'):
    151         print("Second argument must be x86 or amd64")
    152         sys.exit(1)
    153 
    154     if len(sys.argv) > 3:
    155         print("Too many arguments supplied, all we need is the directory",
    156               "containing OpenSSL sources and optionally the architecture")
    157         sys.exit(1)
    158 
    159     ssl_dir = sys.argv[1]
    160     arch = sys.argv[2] if len(sys.argv) >= 3 else None
    161 
    162     if not os.path.isdir(ssl_dir):
    163         print(ssl_dir, "is not an existing directory!")
    164         sys.exit(1)
    165 
    166     # perl should be on the path, but we also look in "\perl" and "c:\\perl"
    167     # as "well known" locations
    168     perls = find_all_on_path("perl.exe", [r"\perl\bin",
    169                                           r"C:\perl\bin",
    170                                           r"\perl64\bin",
    171                                           r"C:\perl64\bin",
    172                                          ])
    173     perl = find_working_perl(perls)
    174     if perl:
    175         print("Found a working perl at '%s'" % (perl,))
    176     else:
    177         sys.exit(1)
    178     if not find_all_on_path('nmake.exe'):
    179         print('Could not find nmake.exe, try running env.bat')
    180         sys.exit(1)
    181     if not find_all_on_path('nasm.exe'):
    182         print('Could not find nasm.exe, please add to PATH')
    183         sys.exit(1)
    184     sys.stdout.flush()
    185 
    186     # Put our working Perl at the front of our path
    187     os.environ["PATH"] = os.path.dirname(perl) + \
    188                                 os.pathsep + \
    189                                 os.environ["PATH"]
    190 
    191     old_cwd = os.getcwd()
    192     try:
    193         os.chdir(ssl_dir)
    194         if arch:
    195             prep(arch)
    196         else:
    197             for arch in ['amd64', 'x86']:
    198                 prep(arch)
    199     finally:
    200         os.chdir(old_cwd)
    201 
    202 if __name__=='__main__':
    203     main()
    204