2016-08-15 19:38:47 +00:00
|
|
|
""" Configurable hooks in the build system. Can be used by various platforms
|
|
|
|
to customize the build process.
|
|
|
|
"""
|
2013-08-30 09:19:08 +00:00
|
|
|
|
|
|
|
################################################################################
|
|
|
|
# Hooks for the various parts of the build process
|
|
|
|
|
|
|
|
# Internal mapping of hooks per tool
|
2016-08-15 19:38:47 +00:00
|
|
|
_HOOKS = {}
|
2013-08-30 09:19:08 +00:00
|
|
|
|
|
|
|
# Internal mapping of running hooks
|
2016-08-15 19:38:47 +00:00
|
|
|
_RUNNING_HOOKS = {}
|
2013-08-30 09:19:08 +00:00
|
|
|
|
|
|
|
# Available hook types
|
2016-08-15 19:38:47 +00:00
|
|
|
_HOOK_TYPES = ["binary", "compile", "link", "assemble"]
|
2013-08-30 09:19:08 +00:00
|
|
|
|
|
|
|
# Available hook steps
|
2016-08-15 19:38:47 +00:00
|
|
|
_HOOK_STEPS = ["pre", "replace", "post"]
|
2013-08-30 09:19:08 +00:00
|
|
|
|
|
|
|
# Hook the given function. Use this function as a decorator
|
|
|
|
def hook_tool(function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Decorate a function as a tool that may be hooked"""
|
2013-08-30 09:19:08 +00:00
|
|
|
tool = function.__name__
|
|
|
|
tool_flag = "_" + tool + "_done"
|
|
|
|
def wrapper(t_self, *args, **kwargs):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""The hooked function itself"""
|
2013-08-30 09:19:08 +00:00
|
|
|
# if a hook for this tool is already running, it's most likely
|
|
|
|
# coming from a derived class, so don't hook the super class version
|
2016-08-15 19:38:47 +00:00
|
|
|
if _RUNNING_HOOKS.get(tool, False):
|
2013-08-30 09:19:08 +00:00
|
|
|
return function(t_self, *args, **kwargs)
|
2016-08-15 19:38:47 +00:00
|
|
|
_RUNNING_HOOKS[tool] = True
|
2013-08-30 09:19:08 +00:00
|
|
|
# If this tool isn't hooked, return original function
|
2018-01-11 21:54:47 +00:00
|
|
|
if tool not in _HOOKS:
|
2013-08-30 09:19:08 +00:00
|
|
|
res = function(t_self, *args, **kwargs)
|
2016-08-15 19:38:47 +00:00
|
|
|
_RUNNING_HOOKS[tool] = False
|
2013-08-30 09:19:08 +00:00
|
|
|
return res
|
2016-08-15 19:38:47 +00:00
|
|
|
tooldesc = _HOOKS[tool]
|
2013-08-30 09:19:08 +00:00
|
|
|
setattr(t_self, tool_flag, False)
|
|
|
|
# If there is a replace hook, execute the replacement instead
|
2018-01-11 21:54:47 +00:00
|
|
|
if "replace" in tooldesc:
|
2013-08-30 09:19:08 +00:00
|
|
|
res = tooldesc["replace"](t_self, *args, **kwargs)
|
|
|
|
# If the replacement has set the "done" flag, exit now
|
|
|
|
# Otherwise continue as usual
|
|
|
|
if getattr(t_self, tool_flag, False):
|
2016-08-15 19:38:47 +00:00
|
|
|
_RUNNING_HOOKS[tool] = False
|
2013-08-30 09:19:08 +00:00
|
|
|
return res
|
|
|
|
# Execute pre-function before main function if specified
|
2018-01-11 21:54:47 +00:00
|
|
|
if "pre" in tooldesc:
|
2013-08-30 09:19:08 +00:00
|
|
|
tooldesc["pre"](t_self, *args, **kwargs)
|
|
|
|
# Execute the main function now
|
|
|
|
res = function(t_self, *args, **kwargs)
|
|
|
|
# Execute post-function after main function if specified
|
2018-01-11 21:54:47 +00:00
|
|
|
if "post" in tooldesc:
|
2013-08-30 09:19:08 +00:00
|
|
|
post_res = tooldesc["post"](t_self, *args, **kwargs)
|
2016-08-15 19:38:47 +00:00
|
|
|
_RUNNING_HOOKS[tool] = False
|
2013-08-30 09:19:08 +00:00
|
|
|
return post_res or res
|
|
|
|
else:
|
2016-08-15 19:38:47 +00:00
|
|
|
_RUNNING_HOOKS[tool] = False
|
2013-08-30 09:19:08 +00:00
|
|
|
return res
|
|
|
|
return wrapper
|
|
|
|
|
2016-08-15 19:38:47 +00:00
|
|
|
class Hook(object):
|
|
|
|
"""A compiler class that may be hooked"""
|
2013-08-30 09:19:08 +00:00
|
|
|
def __init__(self, target, toolchain):
|
2016-08-15 19:38:47 +00:00
|
|
|
_HOOKS.clear()
|
2013-12-05 11:48:42 +00:00
|
|
|
self._cmdline_hooks = {}
|
2013-08-30 09:19:08 +00:00
|
|
|
self.toolchain = toolchain
|
2017-09-13 16:08:03 +00:00
|
|
|
target.init_hooks(self, toolchain)
|
2013-08-30 09:19:08 +00:00
|
|
|
|
2013-12-05 11:48:42 +00:00
|
|
|
# Hook various functions directly
|
2016-08-15 19:38:47 +00:00
|
|
|
@staticmethod
|
|
|
|
def _hook_add(hook_type, hook_step, function):
|
|
|
|
"""Add a hook to a compile function
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
hook_type - one of the _HOOK_TYPES
|
|
|
|
hook_step - one of the _HOOK_STEPS
|
|
|
|
function - the function to add to the list of hooks
|
|
|
|
"""
|
|
|
|
if hook_type not in _HOOK_TYPES or hook_step not in _HOOK_STEPS:
|
2013-08-30 09:19:08 +00:00
|
|
|
return False
|
2016-08-15 19:38:47 +00:00
|
|
|
if hook_type not in _HOOKS:
|
|
|
|
_HOOKS[hook_type] = {}
|
|
|
|
_HOOKS[hook_type][hook_step] = function
|
2013-08-30 09:19:08 +00:00
|
|
|
return True
|
|
|
|
|
2013-12-05 11:48:42 +00:00
|
|
|
def hook_add_compiler(self, hook_step, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the compiler
|
|
|
|
|
|
|
|
Positional Arguments:
|
|
|
|
hook_step - one of the _HOOK_STEPS
|
|
|
|
function - the function to add to the list of hooks
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_add("compile", hook_step, function)
|
|
|
|
|
|
|
|
def hook_add_linker(self, hook_step, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the linker
|
|
|
|
|
|
|
|
Positional Arguments:
|
|
|
|
hook_step - one of the _HOOK_STEPS
|
|
|
|
function - the function to add to the list of hooks
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_add("link", hook_step, function)
|
|
|
|
|
|
|
|
def hook_add_assembler(self, hook_step, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the assemble
|
|
|
|
|
|
|
|
Positional Arguments:
|
|
|
|
hook_step - one of the _HOOK_STEPS
|
|
|
|
function - the function to add to the list of hooks
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_add("assemble", hook_step, function)
|
|
|
|
|
2013-08-30 09:19:08 +00:00
|
|
|
def hook_add_binary(self, hook_step, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the elf to binary tool
|
|
|
|
|
|
|
|
Positional Arguments:
|
|
|
|
hook_step - one of the _HOOK_STEPS
|
|
|
|
function - the function to add to the list of hooks
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_add("binary", hook_step, function)
|
|
|
|
|
|
|
|
# Hook command lines
|
|
|
|
def _hook_cmdline(self, hook_type, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to a command line function
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
hook_type - one of the _HOOK_TYPES
|
|
|
|
function - the function to add to the list of hooks
|
|
|
|
"""
|
|
|
|
if hook_type not in _HOOK_TYPES:
|
2013-12-05 11:48:42 +00:00
|
|
|
return False
|
|
|
|
self._cmdline_hooks[hook_type] = function
|
|
|
|
return True
|
|
|
|
|
|
|
|
def hook_cmdline_compiler(self, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the compiler command line
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
function - the function to call
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_cmdline("compile", function)
|
|
|
|
|
|
|
|
def hook_cmdline_linker(self, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the linker command line
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
function - the function to call
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_cmdline("link", function)
|
|
|
|
|
|
|
|
def hook_cmdline_assembler(self, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the assembler command line
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
function - the function to call
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_cmdline("assemble", function)
|
|
|
|
|
|
|
|
def hook_cmdline_binary(self, function):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Add a hook to the elf to bin tool command line
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
function - the function to call
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._hook_cmdline("binary", function)
|
|
|
|
|
|
|
|
# Return the command line after applying the hook
|
|
|
|
def _get_cmdline(self, hook_type, cmdline):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Get the command line after running all hooks
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
hook_type - one of the _HOOK_TYPES
|
|
|
|
cmdline - the initial command line
|
|
|
|
"""
|
2018-01-11 21:54:47 +00:00
|
|
|
if hook_type in self._cmdline_hooks:
|
2016-08-15 19:38:47 +00:00
|
|
|
cmdline = self._cmdline_hooks[hook_type](
|
|
|
|
self.toolchain.__class__.__name__, cmdline)
|
2013-12-05 11:48:42 +00:00
|
|
|
return cmdline
|
|
|
|
|
|
|
|
def get_cmdline_compiler(self, cmdline):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Get the compiler command line after running all hooks
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
cmdline - the initial command line
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._get_cmdline("compile", cmdline)
|
|
|
|
|
|
|
|
def get_cmdline_linker(self, cmdline):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Get the linker command line after running all hooks
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
cmdline - the initial command line
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._get_cmdline("link", cmdline)
|
|
|
|
|
|
|
|
def get_cmdline_assembler(self, cmdline):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Get the assmebler command line after running all hooks
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
cmdline - the initial command line
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._get_cmdline("assemble", cmdline)
|
|
|
|
|
|
|
|
def get_cmdline_binary(self, cmdline):
|
2016-08-15 19:38:47 +00:00
|
|
|
"""Get the binary command line after running all hooks
|
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
cmdline - the initial command line
|
|
|
|
"""
|
2013-12-05 11:48:42 +00:00
|
|
|
return self._get_cmdline("binary", cmdline)
|
2013-08-30 09:19:08 +00:00
|
|
|
|
|
|
|
################################################################################
|
|
|
|
|