DebugOff Library

WORK IN PROGRESS

Rust anti-analysis library for making static and dynamic (debugging) analysis more difficult.

The library targets Linux environments.

It is currently based on ptrace anti-analysis trick and has the following features:

Overall, this is not the final solution against static and dynamic analysis but for sure it is going to make the reverser/analyst life a little bit more difficult.

to use the crate, add it to your dependencies:

text [dependencies] debugoff = { version = "0.1.0, features = ["obfuscate"] }

Given that the library generates random code at each compilation, be sure to rebuild everything each time. Something like this:

text cargo clean cargo build --release

Also, it would be a good idea to build the project without symbols:

text [profile.release] debug = false strip = "symbols" panic = "abort"

Usage Example

```rust // Include only for Linux and when building in release mode

[cfg(target_os = "linux")]

[cfg(not(debug_assertions))]

use debugoff; use std::time::SystemTime;

fn main() { // Call only for Linux and when building in release mode #[cfg(targetos = "linux")] #[cfg(not(debugassertions))] debugoff::multiptracemeor_die();

println!( "Time: {}", SystemTime::now() .durationsince(SystemTime::UNIXEPOCH) .unwrap() .as_millis() );

// Call only for Linux and when building in release mode #[cfg(targetos = "linux")] #[cfg(not(debugassertions))] debugoff::multiptracemeor_die();

println!("Example complete!"); } ```

See other examples in examples directory.

Obfuscation example

If we build the following code (which does not use DebugOff) in release mode:

```rust use std::time::SystemTime;

fn main() { println!( "Time: {}", SystemTime::now() .durationsince(SystemTime::UNIXEPOCH) .unwrap() .as_millis() );

println!("Example complete!"); } ```

This is the corresponding main function graph:

Executable build without
DebugOff.

If we build the same code but using DebugOff this time:

```rust

[cfg(target_os = "linux")]

[cfg(not(debug_assertions))]

use debugoff; use std::time::SystemTime;

fn main() { #[cfg(targetos = "linux")] #[cfg(not(debugassertions))] debugoff::multiptracemeor_die();

println!( "Time: {}", SystemTime::now() .durationsince(SystemTime::UNIXEPOCH) .unwrap() .as_millis() );

#[cfg(targetos = "linux")] #[cfg(not(debugassertions))] debugoff::multiptracemeor_die();

println!("Example complete!"); } ```

This is the obfuscated main function graph obtained:

Executable build with
DebugOff.

In this particular example, all the code produced by DebugOff was inlided in the main function. this is not always the case, it can depends on many factors like the points where DebugOff is called and the toolchain version. In other cases the resulting function graph could be simpler than the one reported in the example but DebugOff always offer some degree of obfuscation.

TODOs