Last updated
Last updated
This blog post was written while I was working for Securify B.V. and can be found . The post is copied here (my personal blog) for archiving purposes.
Cobalt Strike beacon object files (BOFs) is a feature that was added to the beacon in order to allow rapid beacon extendibility in a more OPSEC way. The BOF file is a common object file format (COFF) object that will be loaded and executed in the same process as the beacon, and therefore eliminates the need for using OPSEC expensive techniques like fork&run. BOFs are written in C\C++ and can be built using Visual Studio or MinGW. The official documentation on BOFs can be found . TrustedSec provided a well-written , and also provided an for those interested in learning more about how BOFs work under the hood.
Developing Windows applications in Visual Studio has its advantages, mainly the ease of building, debugging, and testing as well as the integration of testing tools like virtual leak detector, application verifier, cppcheck, and so on. However, creating BOFs with Visual Studio is unpleasant because of the syntax of dynamic function resolution and because additional steps must be taken to generate the BOF and strip its debug symbols. Additionally, sometimes BOFs fail in engagements, and it would be handy to know the cause of the failure in production.
I wanted to create a baseline template that can be reused to develop BOFs with Visual Studio without having to worry about dynamic function resolution syntax, stripping symbols, compiler configurations, C++ name mangling, or unexpected runtime errors. Thus, the requirements were
Default debug and release build configurations for debugging and testing for aspects like memory or handle leaks
Custom BOF build configuration that will generate the BOFs, strip their debug symbols, and move them to an output directory
Prebuilt list of function definitions to allow developers to reuse existing code without having to add the dynamic function resolution syntax
Built-in error function that will print, in production, which line and function caused the error
Ability to write simple C++ code without having to worry about name mangling
A first prototype that fulfills these requirements was created and is available on GitHub.
To use the template, download and copy the latest zip file from the into the project templates directory under My Documents. For Visual Studio 2019, the directory would be
The template will be automatically loaded into Visual Studio the next time you run it. After a restart, you can search for โbeaconโ on the new project wizard to find the template.
The structure of the solution is shown in the image below. The Header Files filter contains the two headers needed to develop BOFs. The first header is the beacon header provided by the Cobalt Strike team. The bofdefs header is a modified version of the bofdefs header from TrustedSec. Next, the Resources filter contains a PowerShell script that will strip the debugging symbols. This PowerShell script is based on the work of Matthew Graeber (@mattifestation) on his ObjDump script. Finally, the Source Files filter contains the source code of the BOF.
The template makes it as easy as copying the main code to the go function for BOFs that take no arguments. The image below shows that the code in go is identical to the code in main. The ability to reuse the code without further modification increases the efficiency of porting existing projects into BOFs and eliminates the unnatural syntax of dynamic function resolution. For BOFs that take arguments, only the argument parsing part will be different. A goal for next versions is to create generic argument parsing macros or functions to eliminate the need for any modification.
When the code is ready, it can be built for x64 and x86 BOFs at one go using the predefined build configuration inside the project template. In order to do so, go to Build -> Batch Build and tick the two architectures.
In order to build for only one architecture, select the appropriate BOF build configuration from the dropdown menu and hit build, as shown in the image below.
Regardless of the method used to build the BOF, the result is a stripped BOF ready to be used with Cobalt Strike.
Sometimes, especially when dealing with component object model (COM) objects, it is easier to use C++ instead of C. The template supports that as well. Simply rename the Source.c to Source.cpp and compile. You will be spared from name mangling because the template will automatically add extern โCโ if C++ is used. However, you should not get your hopes up too high with developing C++ BOFs, as this is not yet fully supported by Cobalt Strike.
The template comes with a built-in function to print errors in a meaningful manner for operators. The purpose is to provide the operators with sufficient information to understand why the BOF failed without revisiting the source code. The error message below is a screenshot from the Cobalt Strike console and provides the following information for the operators: the error occurred
Name function causing the error: create_folder
Line number: 164
Description by developer: โfailed to create directoryโ
WIN32 error code: 123
Different teams have different needs, and therefore this template might not be suitable for everyone. Modifying the template is easy. Just unzip the zip file, make your modifications to the files, and zip it again!
To show how this template can be used, I have included two sample BOFs. The first BOF is a port of @am0nsec AppLocker enumerator, which will dump the effective AppLocker policy as beacon output. The second example is a scheduled task persistence that relies heavily on COM.
This template is an initial prototype and many more features can be added to it. For example, the features on my to-do list includes:
Adding generic argument parsing that can be used by main and go without modifications
Internal print that takes variadic arguments
Pull requests and ideas are also most welcome!
Big thanks to the team and for sharing the code of the situational awareness BOFs . This template relied heavily on their work.
Paul () for sharing his code and experience with undocumented COM objects .
Matthew () for sharing his work on PowerShell ObjDump .