Roblox Reflection - The Basics (Part 1)

windowsx64c++roblox

Introduction

According to Wikipedia (opens in a new tab), reflective programming is the ability of a process to examine, introspect, and modify its own structure and behavior. In the case of Roblox, its reflection system enables Lua scripts running in the engine to interact with internal classes written in C++.

In this blog post, we will use Binary Ninja's intermediate language to examine and reconstruct Roblox's reflection classes. By doing so, we can gain a deeper understanding of how Roblox's reflection system works and how we can interact with it.

Getting Started

To get started, you will need a copy of Binary Ninja, a decrypted Roblox executable, and a basic understanding of C++ and Luau. If you don't have Binary Ninja, you can download a free trial from their website (opens in a new tab). To decrypt the Roblox Client executable, you can use a tool like vulkan (opens in a new tab), which dumps the decrypted image from memory onto your disk.

I'll be performing all my analysis on the version-eadc3c90bb1a4267 version of Roblox and have used the ClassyPP (opens in a new tab) plugin to resolve RTTI information. If you are using a different version of Roblox, you may get different results.

The base address of my dumped image is 0x17ff7a2f70000.

Understanding the Roblox Engine

Before we begin reverse engineering the binary, it's important to have a high-level overview of the Roblox engine. The Roblox engine is written in C++, but many of its classes are exposed to Lua scripts, as documented here (opens in a new tab). The engine consists of numerous classes, each responsible for different aspects of the game. For instance, the Workspace class manages the game world, while the Player class handles the player's character.

Although these classes differ in functionality, they all share a common base class called Instance. This base class forms the root of the class hierarchy and is responsible for managing the lifecycle of objects within the engine. By understanding how the Instance class operates, we can gain insight into how the rest of the engine functions.

Inspecting the Binary

We can begin our reverse engineering once Binary Ninja has finished analyzing the binary. After resolving RTTI information with ClassyPP, we can see several virtual tables under the RBX::Reflection namespace. These virtual tables manage the reflection system and are crucial for interacting with the engine.

There are a lot of classes, each having hundreds of cross-references in code and data. Finding the Instance class is like finding a needle in a haystack. To make things easier, let's find a class derived from Instance that has a unique name.

Finding a Unique Class

After browsing Roblox's documentation, I found DistortionSoundEffect (opens in a new tab), which only had a few cross-references in code. I noticed that it was being used in a call to the RBX::Reflection::Descriptor constructor. The C-like pseudocode looks like this:

DistortionSoundEffect Descriptor

This appears to be an initializer for the DistortionSoundEffect reflection class. Let's look at the cross-references to this function.

Exploring Properties

The first cross-reference brings us to another subroutine that seems responsible for registering the Level property of the DistortionSoundEffect class. For the sake of space, I won't post the entire subroutine but instead focus on the relevant parts.

At the beginning of the subroutine, a GetSet object is created. This object defines the getter and setter functions for the Level property. In this case, the virtual table used is 0x7ff7a6c9cc10, or with renamed symbols:

class RBX::Reflection::PropDescriptor<class RBX::Soundscape::DistortionSoundEffect, float>::GetSetImpl<float (__cdecl RBX::Soundscape::DistortionSoundEffectProp::*)(void) const, void (__cdecl RBX::Soundscape::DistortionSoundEffectProp::*)(float)>::vfTable

This can be seen in the following snippet of Binary Ninja's MLIL:

DistortionSoundEffect Level Property

The getter function pointer is stored at rax + 0x8, and the setter function pointer is stored at rax + 0x10, where rax is the GetSet object. The getter function is defined as float (__cdecl RBX::Soundscape::DistortionSoundEffectProp::*)(void) const, and the setter function is defined as void (__cdecl RBX::Soundscape::DistortionSoundEffectProp::*)(float).

Let's examine the getter subroutine (sub_7ff7a3adf3a0). Its MLIL is shown below:

DistortionSoundEffect Level Getter

As indicated by the virtual table, arg1 is a pointer to the DistortionSoundEffect object. The getter function returns the value located at arg1 + 0x8, which is the Level property of the DistortionSoundEffect object.

For the sake of brevity, I will not cover the setter function for this property, as it simply writes the value to the same location.

Exploring Inheritance

According to the Roblox documentation (opens in a new tab), the DistortionSoundEffect class is derived from the SoundEffect class. Let's go back to the DistortionSoundEffect descriptor and see if we can find any references to the SoundEffect class.

Looking back at the MLIL for the DistortionSoundEffect descriptor, we can see a subroutine being called, and its result is stored in rdx, which is also passed to the RBX::Reflection::Descriptor constructor. This subroutine is likely responsible for initializing the SoundEffect class:

DistortionSoundEffect SoundEffect Inheritance

Let's examine the code located at sub_7ff7a42e43b0 (only the relevant part is shown):

SoundEffect Descriptor

As you can see, we end up at the SoundEffect class descriptor initializer, just as expected. After repeating this process a few more times, we can reconstruct the class hierarchy of the DistortionSoundEffect class:

DistortionSoundEffect : SoundEffect : Instance

This hierarchy matches the one described in the Roblox documentation.

Conclusion

In this blog post, we used Binary Ninja's intermediate language to examine Roblox's reflection classes. By reconstructing the class hierarchy of the DistortionSoundEffect class, we gained a basic understanding of how Roblox's reflection system works.

In the next part of this series, we will explore the reflection system in more depth and reconstruct some of the structures used to interact with the engine. Stay tuned!