In this exercise, you will learn about the basic structure of a MAMBO plugin and how to add new callbacks to extend the scan time functionality. The specific task of this exercise is:
Extend the functionality of the given plugin by adding two new callbacks before and after basic blocks to print start and end address of those basic blocks.
DBM frameworks work by scanning the code of the application and then passing it to the plugin. The plugin analyses and instruments the code, before returning control back to the DBM framework, which translates the code as necessary to maintain correct execution.
Before continuing the initial code for this exercise has to be copied to $MAMBO_ROOT/plugins
:
cp $MAMBO_ROOT/docs/tutorials/hipeac2024/exercise1/code/tutorial.c $MAMBO_ROOT/plugins/
Now have a look at the copied code. All its parts are explained below.
PLUGINS_NEW
: Informs MAMBO that the plugin uses the current plugin API (this is used to support legacy plugins that were developed before MAMBO API had been re-written).include "../plugins.h"
: The main file that includes all the necessary.h
files related to the interaction of MAMBO and plugins. Functions such as callback-related and code generation functions are listed inplugins.h
.__attribute__((constructor))
: Compiler directive that causes the function to run beforemain
and is used by MAMBO to setup the plugin.
Note
You can find plugins.h
under $MAMBO_ROOT
.
MAMBO functions used in the given plugin are:
mambo_register_plugin()
: The main function for registering a plugin in a MAMBO context. (see below)mambo_register_pre_thread_cb()
: An event hook that runs before each application thread is started. It is a very useful event mainly because it is able to track active threads. Also, this event hook assists users to allocate and initialise thread private resources.mambo_register_post_thread_cb()
: An event hook that runs just before any application thread exits either via thread termination or application exit. Its main purpose is to track active threads, aggregate and output data from thread-level analysis and instrumentation, and release thread-private resources.mambo_get_thread_id()
: A function that returns thethread_id
of the active thread. Useful for multi-threaded program analysis.
Plugins should use an init function with __attribute__((constructor))
to register themselves using mambo_register_plugin()
. Once a plugin is registered, it can install callbacks for various events.
mambo_context * ctx = mambo_register_plugin();
Context provides necessary data structures which users need to analyse/instrument the code. There are various fields such as:
ctx->code.read_address // The untranslated application address of an instruction.
ctx->code.write_p // The current code cache address to place the next instruction.
ctx->code.inst // The enum of the decoded instruction.
The full code can be found under $MAMBO_ROOT/api/plugin_support.h
.
- Copy the given plugin of this exercise (
tutorial.c
) into$MAMBO_ROOT/plugins
. - Run the
test
application under MAMBO.
Note
You should see one thread entered and one exited, since it is a single threaded binary.
The mambo_register_pre_basic_block_cb
event runs just before scanning a single-entry and single-exit code region. See below for the definition of the function and its arguments:
int mambo_register_pre_basic_block_cb(mambo_context *ctx, mambo_callback cb);
The mambo_callback
is simply a pointer to a function with the following signature:
int (*mambo_callback)(mambo_context *ctx);
Tip
This callback can generate basic block-level instrumentation.
The mambo_register_post_basic_block_cb
event runs after scanning a single-entry and single exit code region. See below for the definition of the function and its arguments:
int mambo_register_post_basic_block_cb(mambo_context *ctx, mambo_callback cb);
Tip
This callback can be used to backpatch instrumentation in the basic block based on information not available earlier (e.g. basic block size).
Note
It is important to note that these callbacks enable analysis at scan time.
Finally, once the basic blocks callbacks are added the source address of the beginning and end of the basic blocks should be printed. For that MAMBO provides a helper function:
void *mambo_get_source_addr(mambo_context *ctx);
It takes the MAMBO context and returns the currently scanned source address. For example:
void* source_addr = mambo_get_source_addr(ctx);
Putting it all together:
int tutorial_pre_basic_block_cb(mambo_context* ctx) {
void* source_addr = mambo_get_source_addr(ctx);
printf("Basic block starts at address: %p!\n", source_addr);
}
int tutorial_post_basic_block_cb(mambo_context* ctx) {
void* source_addr = mambo_get_source_addr(ctx);
fprintf(stderr, "Basic block ends at address: %p!\n", source_addr);
}
Remember callbacks have to be registered in the constructor function, for example:
mambo_register_pre_basic_block_cb(ctx, &tutorial_pre_basic_block_cb);
mambo_register_post_basic_block_cb(ctx, &tutorial_post_basic_block_cb);
- Extend the given plugin with
mambo_register_pre_basic_block_cb
,mambo_register_post_basic_block_cb
callbacks to print the start and end address of basic blocks. - Evaluate the
test
application under MAMBO.
Note
Every time the plugin is updated the whole DBM tool has to be recompiled with make
.
You may notice that the number of printed basic blocks is much larger than the expected given that the test binary is a simple loop. This is because MAMBO runs both the binary and libc
start process.
This is the end of Exercise 1. Feel free to ask us any questions or proceed to Exercise 2.