2018-07-01 18:17:46 +00:00
|
|
|
'''
|
|
|
|
PROJECT: ReactOS baseaddress updater
|
|
|
|
LICENSE: MIT (https://spdx.org/licenses/MIT)
|
|
|
|
PURPOSE: Update baseaddresses of all modules
|
|
|
|
COPYRIGHT: Copyright 2017,2018 Mark Jansen (mark.jansen@reactos.org)
|
|
|
|
'''
|
2019-12-27 18:51:20 +00:00
|
|
|
|
2018-09-08 09:10:51 +00:00
|
|
|
from __future__ import print_function, absolute_import, division
|
2019-12-27 18:51:20 +00:00
|
|
|
|
|
|
|
USAGE = """
|
|
|
|
This script will update the baseaddresses of all modules, based on the build output.
|
|
|
|
|
|
|
|
Specify the build output dir as commandline argument to the script:
|
|
|
|
`python gen_baseaddress.py C:\\Users\\Mark\\reactos\\output-MinGW-i386`
|
|
|
|
|
|
|
|
Multiple directories can be specified:
|
|
|
|
`python gen_baseaddress r:/build/msvc r:/build/gcc`
|
2023-10-29 20:28:37 +00:00
|
|
|
|
|
|
|
Specify -64 as first argument to use 64-bit addresses:
|
|
|
|
`python gen_baseaddress -64 r:\\build\\msvc-x64`
|
2019-12-27 18:51:20 +00:00
|
|
|
"""
|
|
|
|
|
2018-07-01 18:17:46 +00:00
|
|
|
import os
|
|
|
|
import struct
|
|
|
|
import sys
|
2018-07-02 20:32:03 +00:00
|
|
|
|
2018-07-01 18:17:46 +00:00
|
|
|
try:
|
|
|
|
import pefile
|
|
|
|
except ImportError:
|
2018-09-08 09:10:51 +00:00
|
|
|
print('# Please install pefile from pip or https://github.com/erocarrera/pefile')
|
2019-12-14 17:28:03 +00:00
|
|
|
sys.exit(-1)
|
2018-07-01 18:17:46 +00:00
|
|
|
|
2019-12-27 18:51:20 +00:00
|
|
|
|
2018-07-02 20:32:03 +00:00
|
|
|
ALL_EXTENSIONS = (
|
|
|
|
'.dll', '.acm', '.ax', '.cpl', '.drv', '.ocx'
|
2018-07-01 18:17:46 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
PRIORITIES = (
|
|
|
|
'ntdll.dll',
|
|
|
|
'kernel32.dll',
|
|
|
|
'msvcrt.dll',
|
|
|
|
'advapi32.dll',
|
|
|
|
'gdi32.dll',
|
|
|
|
'user32.dll',
|
|
|
|
'dhcpcsvc.dll',
|
|
|
|
'dnsapi.dll',
|
|
|
|
'icmp.dll',
|
|
|
|
'iphlpapi.dll',
|
|
|
|
'ws2_32.dll',
|
|
|
|
'ws2help.dll',
|
|
|
|
'shlwapi.dll',
|
|
|
|
'rpcrt4.dll',
|
|
|
|
'comctl32.dll',
|
|
|
|
'ole32.dll',
|
|
|
|
'winspool.drv',
|
|
|
|
'winmm.dll',
|
|
|
|
'comdlg32.dll',
|
|
|
|
'shell32.dll',
|
|
|
|
'lz32.dll',
|
|
|
|
'version.dll',
|
|
|
|
'oleaut32.dll',
|
|
|
|
'setupapi.dll',
|
|
|
|
'mpr.dll',
|
|
|
|
'crypt32.dll',
|
|
|
|
'wininet.dll',
|
|
|
|
'urlmon.dll',
|
|
|
|
'psapi.dll',
|
|
|
|
'imm32.dll',
|
|
|
|
'msvfw32.dll',
|
|
|
|
'dbghelp.dll',
|
|
|
|
'devmgr.dll',
|
|
|
|
'msacm32.dll',
|
|
|
|
'netapi32.dll',
|
|
|
|
'powrprof.dll',
|
|
|
|
'secur32.dll',
|
|
|
|
'wintrust.dll',
|
|
|
|
'avicap32.dll',
|
|
|
|
'cabinet.dll',
|
|
|
|
'dsound.dll',
|
|
|
|
'glu32.dll',
|
|
|
|
'opengl32.dll',
|
|
|
|
'riched20.dll',
|
|
|
|
'userenv.dll',
|
|
|
|
'uxtheme.dll',
|
|
|
|
'cryptui.dll',
|
|
|
|
'csrsrv.dll',
|
|
|
|
'basesrv.dll',
|
|
|
|
'winsrv.dll',
|
|
|
|
'dplayx.dll',
|
|
|
|
'gdiplus.dll',
|
|
|
|
'msimg32.dll',
|
|
|
|
'mswsock.dll',
|
|
|
|
'oledlg.dll',
|
|
|
|
'rasapi32.dll',
|
|
|
|
'rsaenh.dll',
|
|
|
|
'samlib.dll',
|
|
|
|
'sensapi.dll',
|
|
|
|
'sfc_os.dll',
|
|
|
|
'snmpapi.dll',
|
|
|
|
'spoolss.dll',
|
|
|
|
'usp10.dll',
|
|
|
|
)
|
|
|
|
|
|
|
|
EXCLUDE = (
|
|
|
|
'bmfd.dll',
|
|
|
|
'bootvid.dll',
|
2019-05-01 07:44:33 +00:00
|
|
|
'framebuf.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'ftfd.dll',
|
2019-05-01 07:44:33 +00:00
|
|
|
'genincdata.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'hal.dll',
|
|
|
|
'halaacpi.dll',
|
|
|
|
'halacpi.dll',
|
|
|
|
'halapic.dll',
|
|
|
|
'kbda1.dll',
|
|
|
|
'kbda2.dll',
|
|
|
|
'kbda3.dll',
|
|
|
|
'kbdal.dll',
|
|
|
|
'kbdarme.dll',
|
|
|
|
'kbdarmw.dll',
|
|
|
|
'kbdaze.dll',
|
|
|
|
'kbdazel.dll',
|
|
|
|
'kbdbe.dll',
|
|
|
|
'kbdbga.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdbgm.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdbgt.dll',
|
|
|
|
'kbdblr.dll',
|
|
|
|
'kbdbr.dll',
|
2022-08-05 23:03:46 +00:00
|
|
|
'kbdbu.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdbur.dll',
|
|
|
|
'kbdcan.dll',
|
|
|
|
'kbdcr.dll',
|
|
|
|
'kbdcz.dll',
|
|
|
|
'kbdcz1.dll',
|
|
|
|
'kbdda.dll',
|
|
|
|
'kbddv.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdeo.dll',
|
|
|
|
'kbdes.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdest.dll',
|
|
|
|
'kbdfc.dll',
|
|
|
|
'kbdfi.dll',
|
|
|
|
'kbdfr.dll',
|
|
|
|
'kbdgeo.dll',
|
|
|
|
'kbdgerg.dll',
|
|
|
|
'kbdgneo.dll',
|
|
|
|
'kbdgr.dll',
|
2022-08-05 23:03:46 +00:00
|
|
|
'kbdgr1.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdgrist.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdhe.dll',
|
|
|
|
'kbdheb.dll',
|
|
|
|
'kbdhu.dll',
|
|
|
|
'kbdic.dll',
|
|
|
|
'kbdinasa.dll',
|
|
|
|
'kbdinben.dll',
|
|
|
|
'kbdindev.dll',
|
|
|
|
'kbdinguj.dll',
|
|
|
|
'kbdinmal.dll',
|
|
|
|
'kbdir.dll',
|
|
|
|
'kbdit.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdja.dll',
|
2022-08-05 23:03:46 +00:00
|
|
|
'kbdjpn.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdkaz.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdko.dll',
|
2022-08-05 23:03:46 +00:00
|
|
|
'kbdkor.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdla.dll',
|
|
|
|
'kbdlt1.dll',
|
|
|
|
'kbdlv.dll',
|
|
|
|
'kbdmac.dll',
|
|
|
|
'kbdne.dll',
|
|
|
|
'kbdno.dll',
|
|
|
|
'kbdpl.dll',
|
|
|
|
'kbdpl1.dll',
|
|
|
|
'kbdpo.dll',
|
|
|
|
'kbdro.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdrost.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdru.dll',
|
|
|
|
'kbdru1.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdsf.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdsg.dll',
|
2023-03-04 10:45:06 +00:00
|
|
|
'kbdsk.dll',
|
|
|
|
'kbdsk1.dll',
|
2022-08-05 23:03:46 +00:00
|
|
|
'kbdsl.dll',
|
|
|
|
'kbdsl1.dll',
|
|
|
|
'kbdsp.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'kbdsw.dll',
|
|
|
|
'kbdtat.dll',
|
|
|
|
'kbdth0.dll',
|
|
|
|
'kbdth1.dll',
|
|
|
|
'kbdth2.dll',
|
|
|
|
'kbdth3.dll',
|
|
|
|
'kbdtuf.dll',
|
|
|
|
'kbdtuq.dll',
|
|
|
|
'kbduk.dll',
|
|
|
|
'kbdur.dll',
|
|
|
|
'kbdurs.dll',
|
|
|
|
'kbdus.dll',
|
|
|
|
'kbdusa.dll',
|
|
|
|
'kbdusl.dll',
|
|
|
|
'kbdusr.dll',
|
|
|
|
'kbdusx.dll',
|
|
|
|
'kbduzb.dll',
|
|
|
|
'kbdvntc.dll',
|
|
|
|
'kbdycc.dll',
|
|
|
|
'kbdycl.dll',
|
|
|
|
'kdcom.dll',
|
|
|
|
'kdvbox.dll',
|
|
|
|
'vgaddi.dll',
|
|
|
|
'dllexport_test_dll1.dll',
|
|
|
|
'dllexport_test_dll2.dll',
|
|
|
|
'dllimport_test.dll',
|
2023-11-07 16:50:52 +00:00
|
|
|
'localspl_apitest.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
'MyEventProvider.dll',
|
2023-11-07 16:50:52 +00:00
|
|
|
'redirtest1.dll',
|
|
|
|
'redirtest2.dll',
|
2023-10-29 20:28:37 +00:00
|
|
|
'win32u_2k3sp2.dll',
|
|
|
|
'win32u_vista.dll',
|
|
|
|
'win32u_xpsp2.dll',
|
|
|
|
'testvdd.dll',
|
2018-07-01 18:17:46 +00:00
|
|
|
)
|
|
|
|
|
2018-09-01 19:09:18 +00:00
|
|
|
IMAGE_NT_OPTIONAL_HDR32_MAGIC = 0x10b
|
|
|
|
IMAGE_NT_OPTIONAL_HDR64_MAGIC = 0x20b
|
|
|
|
|
|
|
|
IMAGE_TYPES = {
|
|
|
|
IMAGE_NT_OPTIONAL_HDR32_MAGIC: 0,
|
|
|
|
IMAGE_NT_OPTIONAL_HDR64_MAGIC: 0
|
|
|
|
}
|
|
|
|
|
2023-10-29 20:28:37 +00:00
|
|
|
IS_64_BIT = False
|
2018-07-01 18:17:46 +00:00
|
|
|
|
2019-12-14 17:28:03 +00:00
|
|
|
def size_of_image(filename):
|
2023-10-29 20:28:37 +00:00
|
|
|
if IS_64_BIT:
|
|
|
|
return 0xFE0000 # This results in 0x1000000 / 16MB space
|
2018-07-01 18:17:46 +00:00
|
|
|
with open(filename, 'rb') as fin:
|
2019-04-29 15:00:18 +00:00
|
|
|
if fin.read(2) != b'MZ':
|
2018-09-08 09:10:51 +00:00
|
|
|
print(filename, 'No dos header found!')
|
2018-07-01 18:17:46 +00:00
|
|
|
return 0
|
|
|
|
fin.seek(0x3C)
|
|
|
|
e_lfanew = struct.unpack('i', fin.read(4))[0]
|
|
|
|
fin.seek(e_lfanew)
|
2019-04-29 15:00:18 +00:00
|
|
|
if fin.read(4) != b'PE\0\0':
|
2018-09-08 09:10:51 +00:00
|
|
|
print(filename, 'No PE header found!')
|
2018-07-01 18:17:46 +00:00
|
|
|
return 0
|
|
|
|
fin.seek(e_lfanew + 0x18)
|
|
|
|
pe_magic = struct.unpack('h', fin.read(2))[0]
|
2018-09-01 19:09:18 +00:00
|
|
|
if pe_magic in IMAGE_TYPES.keys():
|
|
|
|
IMAGE_TYPES[pe_magic] += 1
|
|
|
|
fin.seek(e_lfanew + 0x50)
|
|
|
|
pe_size_of_image = struct.unpack('i', fin.read(4))[0]
|
|
|
|
return pe_size_of_image
|
|
|
|
print(filename, 'Unknown executable format!')
|
|
|
|
return 0
|
2018-07-01 18:17:46 +00:00
|
|
|
|
|
|
|
|
2018-07-02 20:32:03 +00:00
|
|
|
class Module(object):
|
2018-09-01 19:09:18 +00:00
|
|
|
def __init__(self, name, address, size, filename):
|
2018-07-02 20:32:03 +00:00
|
|
|
self._name = name
|
|
|
|
self.address = address
|
|
|
|
self.size = size
|
|
|
|
self._reserved = address != 0
|
2018-09-01 19:09:18 +00:00
|
|
|
self.filename = filename
|
2018-07-02 20:32:03 +00:00
|
|
|
|
2019-12-14 17:28:03 +00:00
|
|
|
def gen_baseaddress(self, output_file):
|
2018-07-02 20:32:03 +00:00
|
|
|
name, ext = os.path.splitext(self._name)
|
|
|
|
postfix = ''
|
|
|
|
if ext in('.acm', '.drv') and self._name != 'winspool.drv':
|
|
|
|
name = self._name
|
|
|
|
if name == 'ntdll':
|
|
|
|
postfix = ' # should be above 0x%08x' % self.address
|
|
|
|
elif self._reserved:
|
|
|
|
postfix = ' # reserved'
|
2019-12-14 17:28:03 +00:00
|
|
|
output_file.write('set(baseaddress_%-30s 0x%08x)%s\n' % (name, self.address, postfix))
|
2018-07-02 20:32:03 +00:00
|
|
|
|
|
|
|
def end(self):
|
|
|
|
return self.address + self.size
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return '%s (0x%08x - 0x%08x)' % (self._name, self.address, self.end())
|
|
|
|
|
|
|
|
class MemoryLayout(object):
|
|
|
|
def __init__(self, startaddress):
|
|
|
|
self.addresses = []
|
|
|
|
self.found = {}
|
|
|
|
self.reserved = {}
|
|
|
|
self.initial = startaddress
|
|
|
|
self.start_at = 0
|
|
|
|
self.module_padding = 0x2000
|
|
|
|
|
|
|
|
def add_reserved(self, name, address):
|
|
|
|
self.reserved[name] = (address, 0)
|
|
|
|
|
|
|
|
def add(self, filename, name):
|
2019-12-14 17:28:03 +00:00
|
|
|
size = size_of_image(filename)
|
2018-07-02 20:32:03 +00:00
|
|
|
addr = 0
|
|
|
|
if name in self.found:
|
|
|
|
return # Assume duplicate files (rshell, ...) are 1:1 copies
|
|
|
|
if name in self.reserved:
|
|
|
|
addr = self.reserved[name][0]
|
|
|
|
self.reserved[name] = (addr, size)
|
2018-09-01 19:09:18 +00:00
|
|
|
self.found[name] = Module(name, addr, size, filename)
|
2018-07-02 20:32:03 +00:00
|
|
|
|
|
|
|
def _next_address(self, size):
|
|
|
|
if self.start_at:
|
2023-10-29 20:28:37 +00:00
|
|
|
addr = (self.start_at - size - self.module_padding - 0xffff) & 0xffffffffffff0000
|
2018-07-02 20:32:03 +00:00
|
|
|
self.start_at = addr
|
|
|
|
else:
|
|
|
|
addr = self.start_at = self.initial
|
|
|
|
return addr
|
|
|
|
|
|
|
|
def next_address(self, size):
|
|
|
|
while True:
|
|
|
|
current_start = self._next_address(size)
|
|
|
|
current_end = current_start + size + self.module_padding
|
|
|
|
# Is there overlap with reserved modules?
|
2018-09-08 09:10:51 +00:00
|
|
|
for key, reserved in self.reserved.items():
|
2018-07-02 20:32:03 +00:00
|
|
|
res_start = reserved[0]
|
|
|
|
res_end = res_start + reserved[1] + self.module_padding
|
|
|
|
if (res_start <= current_start <= res_end) or \
|
|
|
|
(res_start <= current_end <= res_end) or \
|
|
|
|
(current_start < res_start and current_end > res_end):
|
|
|
|
# We passed this reserved item, we can remove it now
|
|
|
|
self.start_at = min(res_start, current_start)
|
|
|
|
del self.reserved[key]
|
|
|
|
current_start = 0
|
|
|
|
break
|
|
|
|
# No overlap with a reserved module?
|
|
|
|
if current_start:
|
|
|
|
return current_start
|
|
|
|
|
|
|
|
def update(self, priorities):
|
|
|
|
# sort addresses, should only contain reserved modules at this point!
|
2018-09-08 09:10:51 +00:00
|
|
|
for key, reserved in self.reserved.items():
|
2018-07-02 20:32:03 +00:00
|
|
|
assert reserved[1] != 0, key
|
|
|
|
for curr in priorities:
|
|
|
|
if not curr in self.found:
|
2018-09-08 09:10:51 +00:00
|
|
|
print('# Did not find', curr, '!')
|
2018-07-02 20:32:03 +00:00
|
|
|
else:
|
|
|
|
obj = self.found[curr]
|
|
|
|
del self.found[curr]
|
|
|
|
if not obj.address:
|
|
|
|
obj.address = self.next_address(obj.size)
|
|
|
|
self.addresses.append(obj)
|
|
|
|
# We handled all known modules now, run over the rest we found
|
|
|
|
for key in sorted(self.found):
|
|
|
|
obj = self.found[key]
|
|
|
|
obj.address = self.next_address(obj.size)
|
|
|
|
self.addresses.append(obj)
|
|
|
|
|
2019-12-14 17:28:03 +00:00
|
|
|
def gen_baseaddress(self, output_file):
|
2018-07-02 20:32:03 +00:00
|
|
|
for obj in self.addresses:
|
2019-12-14 17:28:03 +00:00
|
|
|
obj.gen_baseaddress(output_file)
|
2018-07-01 18:17:46 +00:00
|
|
|
|
2019-12-14 17:28:03 +00:00
|
|
|
def get_target_file(ntdll_path):
|
2018-09-01 19:09:18 +00:00
|
|
|
if 'pefile' in globals():
|
|
|
|
ntdll_pe = pefile.PE(ntdll_path, fast_load=True)
|
2019-04-29 15:00:18 +00:00
|
|
|
names = [sect.Name.strip(b'\0') for sect in ntdll_pe.sections]
|
|
|
|
count = b'|'.join(names).count(b'/')
|
2023-10-29 20:28:37 +00:00
|
|
|
if IS_64_BIT:
|
|
|
|
return 'baseaddress64.cmake'
|
|
|
|
elif b'.rossym' in names:
|
2019-12-14 17:28:03 +00:00
|
|
|
return 'baseaddress.cmake'
|
2018-09-01 19:09:18 +00:00
|
|
|
elif count == 0:
|
2019-12-14 17:28:03 +00:00
|
|
|
return 'baseaddress_msvc.cmake'
|
2018-09-01 19:09:18 +00:00
|
|
|
elif count > 3:
|
2019-12-14 17:28:03 +00:00
|
|
|
return 'baseaddress_dwarf.cmake'
|
2018-09-01 19:09:18 +00:00
|
|
|
else:
|
2019-12-14 17:28:03 +00:00
|
|
|
assert False, "Unknown"
|
|
|
|
return None
|
2018-09-01 19:09:18 +00:00
|
|
|
|
2018-07-01 18:17:46 +00:00
|
|
|
def run_dir(target):
|
2023-10-29 20:28:37 +00:00
|
|
|
if IS_64_BIT:
|
|
|
|
layout = MemoryLayout(0x7FFB7000000)
|
|
|
|
else:
|
|
|
|
layout = MemoryLayout(0x7c920000)
|
|
|
|
layout.add_reserved('user32.dll', 0x77a20000)
|
2019-12-14 17:28:03 +00:00
|
|
|
IMAGE_TYPES[IMAGE_NT_OPTIONAL_HDR64_MAGIC] = 0
|
|
|
|
IMAGE_TYPES[IMAGE_NT_OPTIONAL_HDR32_MAGIC] = 0
|
2018-07-01 18:17:46 +00:00
|
|
|
for root, _, files in os.walk(target):
|
2018-07-02 20:32:03 +00:00
|
|
|
for dll in [filename for filename in files if filename.endswith(ALL_EXTENSIONS)]:
|
2018-07-01 18:17:46 +00:00
|
|
|
if not dll in EXCLUDE and not dll.startswith('api-ms-win-'):
|
2018-07-02 20:32:03 +00:00
|
|
|
layout.add(os.path.join(root, dll), dll)
|
2018-09-01 19:09:18 +00:00
|
|
|
ntdll_path = layout.found['ntdll.dll'].filename
|
2019-12-14 17:28:03 +00:00
|
|
|
target_file = get_target_file(ntdll_path)
|
|
|
|
if target_file:
|
|
|
|
target_dir = os.path.realpath(os.path.dirname(os.path.dirname(__file__)))
|
|
|
|
target_path = os.path.join(target_dir, 'cmake', target_file)
|
|
|
|
output_file = open(target_path, "w")
|
|
|
|
else:
|
|
|
|
output_file = sys.stdout
|
|
|
|
with output_file:
|
|
|
|
output_file.write('# Generated from {}\n'.format(target))
|
|
|
|
output_file.write('# Generated by sdk/tools/gen_baseaddress.py\n\n')
|
|
|
|
layout.update(PRIORITIES)
|
|
|
|
layout.gen_baseaddress(output_file)
|
2018-07-01 18:17:46 +00:00
|
|
|
|
2018-09-01 19:09:18 +00:00
|
|
|
def main():
|
|
|
|
dirs = sys.argv[1:]
|
2023-10-29 20:28:37 +00:00
|
|
|
|
|
|
|
if len(dirs) > 0 and dirs[0] == '-64':
|
|
|
|
print('Using 64-bit addresses')
|
|
|
|
global IS_64_BIT
|
|
|
|
IS_64_BIT = True
|
|
|
|
dirs = sys.argv[2:]
|
2018-07-01 18:17:46 +00:00
|
|
|
if len(dirs) < 1:
|
|
|
|
trydir = os.getcwd()
|
2019-12-27 18:51:20 +00:00
|
|
|
print(USAGE)
|
|
|
|
print('No path specified, trying the working directory: ', trydir)
|
2018-07-01 18:17:46 +00:00
|
|
|
dirs = [trydir]
|
|
|
|
for onedir in dirs:
|
2019-12-27 18:51:20 +00:00
|
|
|
if onedir.lower() in ['-help', '/help', '/h', '-h', '/?', '-?']:
|
|
|
|
print(USAGE)
|
|
|
|
else:
|
|
|
|
run_dir(onedir)
|
2018-07-01 18:17:46 +00:00
|
|
|
|
2018-09-01 19:09:18 +00:00
|
|
|
|
2018-07-01 18:17:46 +00:00
|
|
|
if __name__ == '__main__':
|
2018-09-01 19:09:18 +00:00
|
|
|
main()
|