# Copyright (c) 2015-2024 Vector 35 Inc
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to
# deal in the Software without restriction, including without limitation the
# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
# sell copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
# IN THE SOFTWARE.
import atexit
import sys
import ctypes
from time import gmtime, struct_time
import os
from typing import Mapping, Optional
import functools
# Binary Ninja components
import binaryninja._binaryninjacore as core
__version__ = core.BNGetVersionString()
import binaryninja
from .enums import *
from .databuffer import *
from .filemetadata import *
from .fileaccessor import *
from .binaryview import *
from .transform import *
from .architecture import *
from .basicblock import *
from .function import *
from .lowlevelil import *
from .mediumlevelil import *
from .highlevelil import *
from .types import *
from .typelibrary import *
from .functionrecognizer import *
from .update import *
from .plugin import *
from .callingconvention import *
from .platform import *
from .demangle import *
from .mainthread import *
from .interaction import *
from .lineardisassembly import *
from .highlight import *
from .scriptingprovider import *
from .downloadprovider import *
from .pluginmanager import *
from .settings import *
from .metadata import *
from .flowgraph import *
from .datarender import *
from .variable import *
from .websocketprovider import *
from .workflow import *
from .commonil import *
from .database import *
from .secretsprovider import *
from .typeparser import *
from .typeprinter import *
from .component import *
from .typearchive import *
from .typecontainer import *
from .exceptions import *
from .project import *
from .basedetection import *
from .debuginfo import *
from .externallibrary import *
from .undo import *
from .fileaccessor import *
# We import each of these by name to prevent conflicts between
# log.py and the function 'log' which we don't import below
from .log import (
redirect_output_to_log, is_output_redirected_to_log, log_debug, log_info, log_warn, log_error, log_alert,
log_to_stdout, log_to_stderr, log_to_file, close_logs
)
from .log import log as log_at_level
from .deprecation import *
import warnings
# We must alter the filter settings for DeprecatedWarning. Otherwise, it will never show up.
# https://docs.python.org/3/library/warnings.html#default-warning-filter
warnings.filterwarnings('once', '', DeprecatedWarning)
# Only load Enterprise Client support on Ultimate builds
if core.BNGetProduct() == "Binary Ninja Enterprise Client" or core.BNGetProduct() == "Binary Ninja Ultimate":
from .enterprise import *
[docs]
def shutdown():
"""
``shutdown`` cleanly shuts down the core, stopping all workers and closing all log files.
.. note:: This will be called automatically on script exit if you import the binaryninja module.
"""
# Release license if we have one
global _enterprise_license_checkout
_enterprise_license_checkout = None
core.BNShutdown()
atexit.register(shutdown)
[docs]
@functools.total_ordering
@dataclass
class CoreVersionInfo:
"""
Structure representing the Binary Ninja Version.
Use :py:func:`core_version_info` to look up the current version of Binary Ninja loaded.
"""
major: int
"""Major version number, e.g. 4.0.5000-dev would be 4"""
minor: int
"""Minor version number, e.g. 4.0.5000-dev would be 0"""
build: int
"""Build version number, e.g. 4.0.5000-dev would be 5000"""
channel: str
"""Release channel name, e.g. "dev" or "Stable" """
def __init__(self, major, minor = None, build = None, channel = None):
self.major = 0
self.minor = 0
self.build = 0
self.channel = ""
if isinstance(major, str) and minor is None and build is None and channel is None:
core_version_info = core.BNParseVersionString(major)
self.major = core_version_info.major
self.minor = core_version_info.minor
self.build = core_version_info.build
if core_version_info.channel is not None:
self.channel = core_version_info.channel
core.BNFreeString(core_version_info.channel)
else:
self.major = major
if minor is not None:
self.minor = minor
if build is not None:
self.build = build
self.channel = channel
def __str__(self):
if self.channel == "":
return f"{self.major}.{self.minor}.{self.build}"
else:
return f"{self.major}.{self.minor}.{self.build}-{self.channel}"
def __eq__(self, other):
if isinstance(other, CoreVersionInfo):
return self.major == other.major and self.minor == other.minor and self.build == other.build # channel doesn't matter
if isinstance(other, str):
return self == CoreVersionInfo(other)
return False
def _versionCompare(self, other):
a = core.BNVersionInfo()
a.major = self.major
a.minor = self.minor
a.build = self.build
b = core.BNVersionInfo()
b.major = other.major
b.minor = other.minor
b.build = other.build
return core.BNVersionLessThan(a, b)
def __lt__(self, other):
if isinstance(other, CoreVersionInfo):
return self._versionCompare(other)
if isinstance(other, str):
return self._versionCompare(CoreVersionInfo(other))
return False
[docs]
def get_unique_identifier():
"""
Generate a GUID
:return: A GUID string
"""
return core.BNGetUniqueIdentifierString()
[docs]
def get_install_directory():
"""
``get_install_directory`` returns a string pointing to the installed binary currently running
.. warning:: ONLY for use within the Binary Ninja UI, behavior is undefined and unreliable if run headlessly
"""
return core.BNGetInstallDirectory()
class _DestructionCallbackHandler:
def __init__(self):
self._cb = core.BNObjectDestructionCallbacks()
self._cb.context = 0
self._cb.destructBinaryView = self._cb.destructBinaryView.__class__(self.destruct_binary_view)
self._cb.destructFileMetadata = self._cb.destructFileMetadata.__class__(self.destruct_file_metadata)
self._cb.destructFunction = self._cb.destructFunction.__class__(self.destruct_function)
core.BNRegisterObjectDestructionCallbacks(self._cb)
def destruct_binary_view(self, ctxt, view):
binaryninja.binaryview.BinaryView._unregister(view)
def destruct_file_metadata(self, ctxt, f):
binaryninja.filemetadata.FileMetadata._unregister(f)
def destruct_function(self, ctxt, func):
binaryninja.function.Function._unregister(func)
_enable_default_log = True
_plugin_init = False
_enterprise_license_checkout = None
def _init_plugins():
global _enable_default_log
global _plugin_init
global _enterprise_license_checkout
if not _plugin_init:
if not core_ui_enabled() and (core.BNGetProduct() == "Binary Ninja Enterprise Client" or core.BNGetProduct() == "Binary Ninja Ultimate"):
# Enterprise client needs to reserve a license or else BNInitPlugins will fail
_enterprise_license_checkout = enterprise.LicenseCheckout()
_enterprise_license_checkout.acquire()
# The first call to BNInitCorePlugins returns True for successful initialization and True in this context indicates headless operation.
# The result is pulled from BNInitPlugins as that now wraps BNInitCorePlugins.
is_headless_init_once = core.BNInitPlugins(not os.environ.get('BN_DISABLE_USER_PLUGINS'))
min_level = Settings().get_string("python.log.minLevel")
if _enable_default_log and is_headless_init_once and min_level in LogLevel.__members__ and not core_ui_enabled(
) and sys.stderr.isatty():
log_to_stderr(LogLevel[min_level])
core.BNInitRepoPlugins()
if core.BNIsLicenseValidated():
_plugin_init = True
else:
raise RuntimeError("License is not valid. Please supply a valid license.")
_destruct_callbacks = _DestructionCallbackHandler()
[docs]
def disable_default_log() -> None:
'''Disable default logging in headless mode for the current session. By default, logging in headless operation is controlled by the 'python.log.minLevel' settings.'''
global _enable_default_log
_enable_default_log = False
close_logs()
[docs]
def bundled_plugin_path() -> Optional[str]:
"""
``bundled_plugin_path`` returns a string containing the current plugin path inside the `install path <https://docs.binary.ninja/guide/#binary-path>`_
:return: current bundled plugin path
:rtype: str, or None on failure
"""
return core.BNGetBundledPluginDirectory()
[docs]
def user_plugin_path() -> Optional[str]:
"""
``user_plugin_path`` returns a string containing the current plugin path inside the `user directory <https://docs.binary.ninja/guide/#user-folder>`_
:return: current user plugin path
:rtype: str, or None on failure
"""
return core.BNGetUserPluginDirectory()
[docs]
def user_directory() -> Optional[str]:
"""
``user_directory`` returns a string containing the path to the `user directory <https://docs.binary.ninja/guide/#user-folder>`_
:return: current user path
:rtype: str, or None on failure
"""
return core.BNGetUserDirectory()
[docs]
def core_version() -> Optional[str]:
"""
``core_version`` returns a string containing the current version
:return: current version
:rtype: str, or None on failure
"""
return core.BNGetVersionString()
[docs]
def core_version_info() -> CoreVersionInfo:
"""
``core_version_info`` returns a CoreVersionInfo containing the current version information
:return: current version information
:rtype: CoreVersionInfo
"""
handle = core.BNGetVersionInfo()
return CoreVersionInfo(major=handle.major, minor=handle.minor, build=handle.build, channel=handle.channel)
[docs]
def core_build_id() -> int:
"""
``core_build_id`` returns a integer containing the current build id
:return: current build id
:rtype: int
"""
return core.BNGetBuildId()
[docs]
def core_serial() -> Optional[str]:
"""
``core_serial`` returns a string containing the current serial number
:return: current serial
:rtype: str, or None on failure
"""
return core.BNGetSerialNumber()
[docs]
def core_expires() -> struct_time:
'''License Expiration'''
return gmtime(core.BNGetLicenseExpirationTime())
[docs]
def core_product() -> Optional[str]:
'''Product string from the license file'''
return core.BNGetProduct()
[docs]
def core_product_type() -> Optional[str]:
'''Product type from the license file'''
return core.BNGetProductType()
[docs]
def core_license_count() -> int:
'''License count from the license file'''
return core.BNGetLicenseCount()
[docs]
def core_ui_enabled() -> bool:
'''Indicates that a UI exists and the UI has invoked BNInitUI'''
return core.BNIsUIEnabled()
[docs]
def core_set_license(licenseData: str) -> None:
'''
``core_set_license`` is used to initialize the core with a license file that doesn't necessarily reside on a file system. This is especially useful for headless environments such as docker where loading the license file via an environment variable allows for greater security of the license file itself.
:param str licenseData: string containing the full contents of a license file
:rtype: None
:Example:
>>> import os
>>> core_set_license(os.environ['BNLICENSE']) #Do this before creating any BinaryViews
>>> with load("/bin/ls") as bv:
... print(len(list(bv.functions)))
128
'''
core.BNSetLicense(licenseData)
[docs]
def get_memory_usage_info() -> Mapping[str, int]:
"""
Get counts of various Binary Ninja objects in memory.
:return: Dictionary of {class name: count} for objects in memory
"""
count = ctypes.c_ulonglong()
info = core.BNGetMemoryUsageInfo(count)
assert info is not None, "core.BNGetMemoryUsageInfo returned None"
result = {}
for i in range(0, count.value):
result[info[i].name] = info[i].value
core.BNFreeMemoryUsageInfo(info, count.value)
return result
[docs]
def load(*args, **kwargs) -> BinaryView:
"""
Opens a BinaryView object.
:param Union[str, bytes, bytearray, 'databuffer.DataBuffer', 'os.PathLike'] source: a file or byte stream to load into a virtual memory space
:param bool update_analysis: whether or not to run :func:`update_analysis_and_wait` after opening a :py:class:`BinaryView`, defaults to ``True``
:param callback progress_func: optional function to be called with the current progress and total count for BNDB files only
:param dict options: a dictionary in the form {setting identifier string : object value}
:return: returns a :py:class:`BinaryView` object for the given filename
:rtype: :py:class:`BinaryView`
:raises Exception: When a BinaryView could not be created
.. note:: The progress_func callback **must** return True to continue the load operation, False will abort the load operation.
.. warning:: The progress_func will **only** be called for BNDB files, not for any other file format due to a `design limitation <https://docs.binary.ninja/guide/debugger/index.html#navigating-the-binary>`_.
:Example:
>>> from binaryninja import *
>>> with load("/bin/ls") as bv:
... print(len(list(bv.functions)))
...
134
>>> with load(bytes.fromhex('5054ebfe'), options={'loader.platform' : 'x86'}) as bv:
... print(len(list(bv.functions)))
...
1
"""
bv = BinaryView.load(*args, **kwargs)
if bv is None:
raise Exception("Unable to create new BinaryView")
return bv
[docs]
def connect_pycharm_debugger(port=5678):
"""
Connect to PyCharm (Professional Edition) for debugging.
.. note:: See the `user documentation <https://docs.binary.ninja/dev/plugins.html#remote-debugging-with-intellij-pycharm>`_ for step-by-step instructions on how to set up Python debugging.
:param port: Port number for connecting to the debugger.
"""
# Get pip install string from PyCharm's Python Debug Server Configuration
# e.g. for PyCharm 2021.1.1 #PY-7142.13:
# pip install --user pydevd-pycharm~=211.7142.13
import pydevd_pycharm # type: ignore
pydevd_pycharm.settrace('localhost', port=port, stdoutToServer=True, stderrToServer=True, suspend=False)
[docs]
def connect_vscode_debugger(port=5678):
"""
Connect to Visual Studio Code for debugging. This function blocks until the debugger
is connected! Not recommended for use in startup.py
.. note:: See the `user documentation <https://docs.binary.ninja/dev/plugins.html#remote-debugging-with-vscode>`_ for step-by-step instructions on how to set up Python debugging.
:param port: Port number for connecting to the debugger.
"""
# pip install --user debugpy
import debugpy # type: ignore
import sys
if sys.platform == "win32":
debugpy.configure(python=f"{sys.base_exec_prefix}/python", qt="pyside2")
else:
debugpy.configure(python=f"{sys.base_exec_prefix}/bin/python3", qt="pyside2")
debugpy.listen(("127.0.0.1", port))
debugpy.wait_for_client()
execute_on_main_thread(lambda: debugpy.debug_this_thread())
[docs]
class UIPluginInHeadlessError(Exception):
"""
Error thrown when trying to load a UI plugin in a headless Binary Ninja installation.
"""
def __init__(self, *args, **kwargs):
Exception.__init__(self, *args, **kwargs)
[docs]
def fuzzy_match_single(target, query) -> Optional[int]:
"""
Fuzzy match a string against a query string. Returns a score that is higher for
a more confident match, or None if the query does not match the target string.
:param target: Target (larger) string
:param query: Query (smaller) string
:return: Confidence of match, or None if the string doesn't match
"""
result = core.BNFuzzyMatchSingle(target, query)
if result == 0:
return None
return result
# Load Collaboration scripts from Ultimate (they are bundled in shipping builds)
try:
from . import collaboration
except ImportError:
pass