Определите, является ли исполняемый файл (или библиотека) 32-разрядным или 64-разрядным (в Windows) - PullRequest
8 голосов
/ 28 августа 2009

Я пытаюсь выяснить, скомпилирован ли данный исполняемый файл (или библиотека) для 32-битных или 64-битных из Python. Я использую 64-битную Vista и хотел бы определить, скомпилировано ли определенное приложение в каталоге для 32-битных или 64-битных.

Есть ли простой способ сделать это, используя только стандартные библиотеки Python (в настоящее время используется 2.5.4)?

Ответы [ 4 ]

20 голосов
/ 28 августа 2009

Windows API для этого GetBinaryType. Вы можете вызвать это из Python, используя pywin32 :

import win32file
type=GetBinaryType("myfile.exe")
if type==win32file.SCS_32BIT_BINARY:
    print "32 bit"
# And so on

Если вы хотите сделать это без pywin32, вам придется прочитать PE header самостоятельно. Вот пример в C #, и вот быстрый порт на Python:

import struct

IMAGE_FILE_MACHINE_I386=332
IMAGE_FILE_MACHINE_IA64=512
IMAGE_FILE_MACHINE_AMD64=34404

f=open("c:\windows\explorer.exe", "rb")

s=f.read(2)
if s!="MZ":
    print "Not an EXE file"
else:
    f.seek(60)
    s=f.read(4)
    header_offset=struct.unpack("<L", s)[0]
    f.seek(header_offset+4)
    s=f.read(2)
    machine=struct.unpack("<H", s)[0]

    if machine==IMAGE_FILE_MACHINE_I386:
        print "IA-32 (32-bit x86)"
    elif machine==IMAGE_FILE_MACHINE_IA64:
        print "IA-64 (Itanium)"
    elif machine==IMAGE_FILE_MACHINE_AMD64:
        print "AMD64 (64-bit x86)"
    else:
        print "Unknown architecture"

f.close()
4 голосов
/ 29 августа 2009

Если вы используете Python 2.5 или более позднюю версию в Windows, вы также можете использовать Windows API без pywin32, используя ctypes.

from ctypes import windll, POINTER
from ctypes.wintypes import LPWSTR, DWORD, BOOL

SCS_32BIT_BINARY = 0 # A 32-bit Windows-based application
SCS_64BIT_BINARY = 6 # A 64-bit Windows-based application
SCS_DOS_BINARY = 1 # An MS-DOS-based application
SCS_OS216_BINARY = 5 # A 16-bit OS/2-based application
SCS_PIF_BINARY = 3 # A PIF file that executes an MS-DOS-based application
SCS_POSIX_BINARY = 4 # A POSIX-based application
SCS_WOW_BINARY = 2 # A 16-bit Windows-based application

_GetBinaryType = windll.kernel32.GetBinaryTypeW
_GetBinaryType.argtypes = (LPWSTR, POINTER(DWORD))
_GetBinaryType.restype = BOOL

def GetBinaryType(filepath):
    res = DWORD()
    handle_nonzero_success(_GetBinaryType(filepath, res))
    return res

Затем используйте GetBinaryType, как если бы вы использовали win32file.GetBinaryType.

Обратите внимание, что вам нужно реализовать handle_nonzero_success, который в основном выдает исключение, если возвращаемое значение равно 0.

1 голос
/ 29 сентября 2016

Я смог успешно использовать ответ Мартина Б в программе Python 3.5 после выполнения этой корректировки:

s=f.read(2).decode(encoding="utf-8", errors="strict")

Первоначально она работала просто отлично с моей программой на Python 2.7, но после внесения других необходимых изменений я обнаружил, что получаю b'MZ ', и расшифровка, кажется, исправляет это.

0 голосов
/ 19 декабря 2018
  1. При использовании Python 3.7, 32-битной на 64-битной Win 7, первый фрагмент кода в верхнем ответе не работает для меня. Сбой, потому что GetBinaryType является неизвестным символом. Решение заключается в использовании win32file.GetBinaryType.
  2. Кроме того, запуск его на .pyd-файле не работает, даже если он переименован в .dll. Смотри дальше:

    import shutil
    
    import win32file
    from pathlib import Path
    
    myDir = Path("C:\\Users\\rdboylan\\AppData\\Roaming\\Python\\Python37\\site-packages\\pythonwin")
    for fn in ("Pythonwin.exe", "win32ui.pyd"):
        print(fn, end=": ")
        myf = myDir / fn
        if myf.suffix == ".pyd":
            mytemp = myf.with_suffix(".dll")
            if mytemp.exists():
                raise "Can not create temporary dll since {} exists".format(mytemp)
            shutil.copyfile(myf, mytemp)
            type = win32file.GetBinaryType(str(mytemp))
            mytemp.unlink()
        else:
            type=win32file.GetBinaryType(str(myf))
        if type==win32file.SCS_32BIT_BINARY:
            print("32 bit")
        else:
            print("Something else")
        # And so on 
    

    Результаты в

    Pythonwin.exe: 32 bit
    win32ui.pyd: Traceback (most recent call last):
      File "C:/Users/rdboylan/Documents/Wk devel/bitness.py", line 14, in <module>
        type = win32file.GetBinaryType(str(mytemp))
    pywintypes.error: (193, 'GetBinaryType', '%1 is not a valid Win32 application.')
    
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...