You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							117 lines
						
					
					
						
							3.5 KiB
						
					
					
				
			
		
		
	
	
							117 lines
						
					
					
						
							3.5 KiB
						
					
					
				import re
 | 
						|
import sys
 | 
						|
import subprocess
 | 
						|
 | 
						|
__doc__ = """This module generates a DEF file from the symbols in
 | 
						|
an MSVC-compiled DLL import library.  It correctly discriminates between
 | 
						|
data and functions.  The data is collected from the output of the program
 | 
						|
nm(1).
 | 
						|
 | 
						|
Usage:
 | 
						|
    python lib2def.py [libname.lib] [output.def]
 | 
						|
or
 | 
						|
    python lib2def.py [libname.lib] > output.def
 | 
						|
 | 
						|
libname.lib defaults to python<py_ver>.lib and output.def defaults to stdout
 | 
						|
 | 
						|
Author: Robert Kern <kernr@mail.ncifcrf.gov>
 | 
						|
Last Update: April 30, 1999
 | 
						|
"""
 | 
						|
 | 
						|
__version__ = '0.1a'
 | 
						|
 | 
						|
py_ver = "%d%d" % tuple(sys.version_info[:2])
 | 
						|
 | 
						|
DEFAULT_NM = ['nm', '-Cs']
 | 
						|
 | 
						|
DEF_HEADER = """LIBRARY         python%s.dll
 | 
						|
;CODE           PRELOAD MOVEABLE DISCARDABLE
 | 
						|
;DATA           PRELOAD SINGLE
 | 
						|
 | 
						|
EXPORTS
 | 
						|
""" % py_ver
 | 
						|
# the header of the DEF file
 | 
						|
 | 
						|
FUNC_RE = re.compile(r"^(.*) in python%s\.dll" % py_ver, re.MULTILINE)
 | 
						|
DATA_RE = re.compile(r"^_imp__(.*) in python%s\.dll" % py_ver, re.MULTILINE)
 | 
						|
 | 
						|
def parse_cmd():
 | 
						|
    """Parses the command-line arguments.
 | 
						|
 | 
						|
libfile, deffile = parse_cmd()"""
 | 
						|
    if len(sys.argv) == 3:
 | 
						|
        if sys.argv[1][-4:] == '.lib' and sys.argv[2][-4:] == '.def':
 | 
						|
            libfile, deffile = sys.argv[1:]
 | 
						|
        elif sys.argv[1][-4:] == '.def' and sys.argv[2][-4:] == '.lib':
 | 
						|
            deffile, libfile = sys.argv[1:]
 | 
						|
        else:
 | 
						|
            print("I'm assuming that your first argument is the library")
 | 
						|
            print("and the second is the DEF file.")
 | 
						|
    elif len(sys.argv) == 2:
 | 
						|
        if sys.argv[1][-4:] == '.def':
 | 
						|
            deffile = sys.argv[1]
 | 
						|
            libfile = 'python%s.lib' % py_ver
 | 
						|
        elif sys.argv[1][-4:] == '.lib':
 | 
						|
            deffile = None
 | 
						|
            libfile = sys.argv[1]
 | 
						|
    else:
 | 
						|
        libfile = 'python%s.lib' % py_ver
 | 
						|
        deffile = None
 | 
						|
    return libfile, deffile
 | 
						|
 | 
						|
def getnm(nm_cmd=['nm', '-Cs', 'python%s.lib' % py_ver], shell=True):
 | 
						|
    """Returns the output of nm_cmd via a pipe.
 | 
						|
 | 
						|
nm_output = getnm(nm_cmd = 'nm -Cs py_lib')"""
 | 
						|
    p = subprocess.Popen(nm_cmd, shell=shell, stdout=subprocess.PIPE,
 | 
						|
                         stderr=subprocess.PIPE, text=True)
 | 
						|
    nm_output, nm_err = p.communicate()
 | 
						|
    if p.returncode != 0:
 | 
						|
        raise RuntimeError('failed to run "%s": "%s"' % (
 | 
						|
                                     ' '.join(nm_cmd), nm_err))
 | 
						|
    return nm_output
 | 
						|
 | 
						|
def parse_nm(nm_output):
 | 
						|
    """Returns a tuple of lists: dlist for the list of data
 | 
						|
symbols and flist for the list of function symbols.
 | 
						|
 | 
						|
dlist, flist = parse_nm(nm_output)"""
 | 
						|
    data = DATA_RE.findall(nm_output)
 | 
						|
    func = FUNC_RE.findall(nm_output)
 | 
						|
 | 
						|
    flist = []
 | 
						|
    for sym in data:
 | 
						|
        if sym in func and (sym[:2] == 'Py' or sym[:3] == '_Py' or sym[:4] == 'init'):
 | 
						|
            flist.append(sym)
 | 
						|
 | 
						|
    dlist = []
 | 
						|
    for sym in data:
 | 
						|
        if sym not in flist and (sym[:2] == 'Py' or sym[:3] == '_Py'):
 | 
						|
            dlist.append(sym)
 | 
						|
 | 
						|
    dlist.sort()
 | 
						|
    flist.sort()
 | 
						|
    return dlist, flist
 | 
						|
 | 
						|
def output_def(dlist, flist, header, file = sys.stdout):
 | 
						|
    """Outputs the final DEF file to a file defaulting to stdout.
 | 
						|
 | 
						|
output_def(dlist, flist, header, file = sys.stdout)"""
 | 
						|
    for data_sym in dlist:
 | 
						|
        header = header + '\t%s DATA\n' % data_sym
 | 
						|
    header = header + '\n' # blank line
 | 
						|
    for func_sym in flist:
 | 
						|
        header = header + '\t%s\n' % func_sym
 | 
						|
    file.write(header)
 | 
						|
 | 
						|
if __name__ == '__main__':
 | 
						|
    libfile, deffile = parse_cmd()
 | 
						|
    if deffile is None:
 | 
						|
        deffile = sys.stdout
 | 
						|
    else:
 | 
						|
        deffile = open(deffile, 'w')
 | 
						|
    nm_cmd = DEFAULT_NM + [str(libfile)]
 | 
						|
    nm_output = getnm(nm_cmd, shell=False)
 | 
						|
    dlist, flist = parse_nm(nm_output)
 | 
						|
    output_def(dlist, flist, DEF_HEADER, deffile)
 |