Logo
blank Skip to main content

Comparison of User Mode and Kernel Mode Applications for Modifying HTTP Traffic

In this article weโ€™ll look at how to intercept HTTP traffic in order to inject custom code into Windows HTML markup. In order to do this, weโ€™ll look at two completely different approaches: one with Kernel mode, the other with User mode. For the sake of simplicity, we wonโ€™t cover HTTPS traffic.

In this article weโ€™ll look at how to intercept HTTP traffic in order to inject custom code into Windows HTML markup. In order to do this, weโ€™ll look at two completely different approaches to this network management task: one with Kernel mode, the other with User mode. For the sake of simplicity, we wonโ€™t cover HTTPS traffic.

This article is used as a detailed example with code illustrations for a broader topic of User mode vs Kernel mode implementation comparison.

HTTP Response Data Injection Algorithm

Before we start discussing different approaches, letโ€™s look at the algorithm for injecting data into an HTTP response.

The algorithm includes the following data:

  • initial HTTP response
  • body of the response (should not be empty)

All data is compressed by a qzip compression algorithm and located in a single archive. First, we need to analyze the header of the response and search for the following fields:

  1. Transfer-Encoding: chunked
    1. If this header is present, then we need to process the first chunk.
    2. If this header is not found, then we need to look for Content-Length. The value of Content-Length header is necessary to confirm that all data has been received and to hook that data with new values.
  2. Content-Encoding: gzip โ€“ shows whether data has been archived.
  3. โ€œrnrnโ€ string โ€“ shows where the header ends and data begins.

After our search for these fields is complete, we extract data. Next we search for the <!DOCTYPE HTML> tag and inject the following script after it:

HTML
<script language=\"JavaScript\">
    if (confirm(\"Do you want to read more at https:โ„โ„www.apriorit.comโ„dev-blog ?\")) 
    {
         window.open(\"https:โ„โ„www.apriorit.comโ„dev-blog\");
    }
<โ„script>

The modified response is archived.

Next, the new HTTP response is formed using the header from the old response. If the Content-Length header was not found, then we replace its value with the new one based on the new response. Other header fields should not be changed. If the Transfer-Encoding: chunked field was found, then the length of the first chunk is replaced with the new one. The new data is inserted after the dividing row.

An HTTP response with the new data is sent out in place of the original, and the original response is deleted.

If a response doesnโ€™t contain all the necessary data, then the functions to get all the data are called before injection.

Looking for expert software developers?

Whatever challenging tasks your project requires to solve, Aprioritโ€™s engineers are ready to assist you!

User Mode

In this section, weโ€™ll cover how to inject custom JavaScript code into HTML in User mode.

How it works

To inject custom JavaScript in User mode, we need to get access to the process address space. This can be done by injecting a custom dynamic library (DLL) into the process. When access to the process address space has been gained, we can modify the memory in order to hook the standard function with our own. This allows us to modify HTTP traffic containing HTML code.

Ways to inject DLL

There are several ways to inject a DLL:

  • via the AppInit registry value
  • via the SetWindowsHookEx function
  • via remote threads
  • via a trojan

Each approach has its advantages and disadvantages.

Using the register (AppInit)

This approach requires you to use the AppInit registry value, which stores the list of DLLs necessary to load the User32.dll library, which contains the functions to render the graphical interfaces of Windows applications. By adding the path to our own custom DLL to Applnit, we can guarantee that our DLL will be loaded into every graphical application in Windows. You can see more detailed information about this approach here.

Advantages of this approach:

  • This is the simplest approach in terms of implementation
  • Thereโ€™s no need to specify the processes into which the DLL needs to be injected
  • AppInit needs to be modified only once, after which the DLL will be loaded into all graphical applications

Disadvantages of this approach:

  • Doesnโ€™t affect console applications, since they donโ€™t use User32.dll
  • Administrator permissions are necessary to modify the registry

Using the SetWindowsHookEx function

The SetWindowsHookEx function lets you set hook procedures for windowed applications via DLL injection. The injection happens when a message to the process window to which the SetWindowsHookEx function has been applied is intercepted. The type of message is decided when SetWindowsHookEx is called, which allow us to set hook procedures for all graphical applications. You can learn more about SetWindowsHookEx here.

Advantages of this approach:

  • Can cover a single graphical application or all applications

Disadvantages of this approach:

  • Injection occurs only when specific messages are intercepted
  • Applications that execute this function need to be launched by the user
  • Console applications arenโ€™t affected since they donโ€™t use separate windows

Read also

Controlling and Monitoring a Network with User Mode and Driver Mode Techniques: Overview, Pros and Cons, WFP Implementation

Achieve flawless traffic control to block undesired traffic, detect viruses, and perform many other tasks. Read on to learn more about choosing the right techniques for network monitoring and explore helpful insights from Aprioritโ€™s experienced developers.

Learn more
User Mode and Driver Mode for network monitoring

With remote streams

This approach is based on using the CreateRemoteThread function, which allows you to create a remote thread in another process. The signature of the function transferred via a CreateRemoteThread should look like this:

DWORD WINAPI ThreadFunc(PVOID pvParam);

This allows us to use the LoadLibrary function (or more precisely, LoadLibraryA or LoadLibraryW, since LoadLibrary is a macro), which will load the DLL into the process. This approach is hard to implement for two reasons:

  • Transferring the link to LoadLibraryA/LoadLibraryW can result in a memory access violation because the direct link to LoadLibraryA/LoadLibraryW in a CreateRemoteThread call is transformed into a call to the LoadLibraryA gateway in the import section of your module.
  • Transferring the link (with parameters for the path to your DLL) to the string also creates uncertain behavior, since the link will be projected onto the memory of another process, where this address will not contain a string.

Advantages of this approach:

  • The most flexible way to inject a DLL

Disadvantages of this approach:

  • The hardest approach to implement
  • An application that calls the function needs to be running
  • The process in which we want to inject our DLL needs to be clearly specified

Trojan DLL

Thereโ€™s a way to hook an existing DLL with a custom DLL. For this, a custom DLL needs to export the same functions as the initial. This isnโ€™t hard to do if the address modification for DLL functions is used.

If you want to use a trojan DLL for just one app, then you can give your custom DLL a unique name and add it to the import section of the executable module of the application. This, however, requires advanced knowledge of the Portable Executable (PE) format.

You can find information about a similar solution here.

Advantages of this approach:

  • A trojan DLL only needs to be hooked once, after which it will run by itself
  • No need for administrative permissions
  • A trojan DLL can perform two tasks: DLL injection and function hooking

Disadvantages of this approach:

  • Advanced knowledge of the PE format is necessary
  • Problems with system DLL hooking can arise due to digital signatures

Choosing how to inject DLL

Our task requires the support of as many applications that receive and display HTML content as possible. Therefore, we canโ€™t use the CreateRemoteThread function. A trojan DLL injection requires knowledge of PE formats. Thus, weโ€™re left with a choice between the SetWindowsHookEx function and injection via AppInit registry value. The only advantage of the SetWindowsHookEx approach is that it doesnโ€™t require administrative permissions. At the same time, AppInit provides the easiest way to inject DLL and will automatically work with all graphical applications. Since we need to capture network traffic with HTML and JavaScript, we need to cover browsers, all of which use graphical shells. Therefore, in this article weโ€™ll cover DLL injection via the AppInit registry value.

Related project

USB WiFi Driver Development

Unveil the insights of a successful Windows USB WiFi driver development project. Find out how Apriorit helped a US-based cybersecurity company develop a new Windows driver from scratch, providing their end users with a user interface and tools for WiFi network analysis.

Project details
USB WiFi Driver Development

Ways to hook a function

For the purposes of our task, there are two ways to hook a function:

  • Change the PE file import table
  • Change the beginning of the function

Changing the PE file import table

Each PE file has an import table that stores virtual memory addresses for functions imported from the DLL and used by the PE file from the DLL. By having access to the address space, the import table can be modified by changing a function pointer to point to our own custom function. This approach requires extensive knowledge of the PE format, since there are no ready solutions such as libraries or WinAPI functions out there. However, a lot of proof of concept implementations of this approach can be found on the net.

Changing the beginning of a function

This approach is based on modifying the process address space (the beginning of the function, to be exact), which we need to change to the JMP of our function. This approach is implemented in the actively supported open-source MHook library. Moreover, with this approach you still use the original function.

Choosing how to inject the function

Changing the beginning of the function relies on a convenient and actively supported library, and thus itโ€™s the method weโ€™ve chosen to use.

Overview of the implementation method

Weโ€™ll hook the recv function from Ws2_32.dll, since itโ€™s used by all browsers to receive data from the network. First, we need to receive and save in a global variable a pointer to the original function. This can be done the following way:

C
typedef int(WINAPI* _recv)(
    _In_  SOCKET s,
    _Out_ char   *buf,
    _In_  int    len,
    _In_  int    flags
    );
static _recv TrueRecv = (_recv)GetProcAddress(GetModuleHandle(L"Ws2_32.dll"), "recv");

After this, during the loading of a DLL into the process we need to hook recv in DllMain and unhook it during unloading. The DllMain function will thus look as follows:

C
BOOL APIENTRY DllMain( HMODULE hModule,
                       DWORD  ul_reason_for_call,
                       LPVOID lpReserved
                     )
{
    switch (ul_reason_for_call)
    {
    case DLL_PROCESS_ATTACH:
    {
        if (TrueRecv)
            Mhook_SetHook((PVOID*)(&TrueRecv), InjectedRecv);// replacing recv with InjectRecv
        break;
    }
    case DLL_THREAD_ATTACH:
        break;
    case DLL_THREAD_DETACH:
        break;
    case DLL_PROCESS_DETACH:
        if (TrueRecv)
            Mhook_Unhook((PVOID*)(&TrueRecv));// unhooking recv function
        break;
    }
    return TRUE;
}

In the InjectedRecv function, the real recv function is called via TrueRecv and results are processed.

Letโ€™s take another look at the recv function signature:

C
int recv(
  _In_  SOCKET s,
  _Out_ char   *buf,
  _In_  int    len,
  _In_  int    flags
);

Among all the parameters of the recv function, buf and len are the most interesting for us. buf refers to the buffer with the len size that, after the call to the recv function, contains a response with the size len or smaller, received via a socket. The exact size can be determined by the returned value, which can lead to many related problems:

  1. After custom JavaScript code is added to the HTML, the data size can exceed the size of the buf buffer.
  2. recv isnโ€™t required to return the whole response at once, especially if the buffer doesnโ€™t have enough memory.
  3. Many services return responses in the qzip format, since it reduces the size and wait time for the server. In this case, qzip needs to be unpacked, the JavaScript injected into the HTML, and then the HTML compressed back into a qzip again. This scenario can be combined with both problems above. In the first case, the size of the buf buffer will be exceeded. In the second case, the unpacking of an incomplete qzip produces uncertain behavior in the client code, since after compression the single part of the qzip will become the whole qzip, and all remaining parts will appear to the client code as binaries and will not be processed.

To solve this problem, during the recv call by the client code we need to call one or several recv functions under the hood to gather complete data and save it in the container. After this, the algorithm to inject JavaScript in the HTML can be used. The modified buffer is stored in a static container. This container is an std::map containing the following:

typedef std::pair<ByteBuffer, int> BufferInfo;  static std::map<SOCKET, BufferInfo> g_bufferedData; 

ByteBuffer is a typedef std::vector<unsigned char>. BufferInfo is a pair that stores a modified answer and how many bytes have been read by the previous recv. Therefore, our std::map stores a socket as a key and information on the buffer that needs to be transferred to a client as a value.

With each InjectedRecv call, first a check is performed to determine if there is data in the current socket in g_bufferedData. If data exists, then the part with the size len or smaller is returned, depending on how many byte are left. After all data has been transferred to the client code, the buffer and counter for read bytes are cleared.

If there is no data in the current socket, then the call to real recv is performed. We presume that for an HTTP request with HTML, the first part of an answer has all the necessary information to determine the size of the body of the answer. Based on this information, we can call the real recv until we either get all the data or an error occurs. If we get an error, we can save the received data without modification and then follow the steps for the previous case when there is data in g_bufferedData. Then we can modify the HTML, after which the data will be returned fully or partially, depending on the size of data and the len parameter.

Practical example

Below this article, you can find a link to a DLL project for Visual Studio 2013 with C++11. This means that youโ€™ll need Visual Studio 2013 or newer to view it.

You need to build the DLL with two static libraries โ€“ zlib and mHook. mHook is a part of the project and automatically links to the DLL, but youโ€™ll need to add zlib manually for the right configuration. You can find zlib here.

If you have an x64 OS, youโ€™ll need to look into whether the browser whose traffic youโ€™re trying to intercept is using an x64 or x84 instruction set. Itโ€™s important that instruction sets used by the browser and DLL are the same, or else the DLL will not work.

Read also

How to Use WinDbg for Kernel Debugging to Develop a Reliable Windows Driver

Secure your Windows-related projects by efficiently discovering bugs during driver development. Find out the best tricks of debugging in both kernel and user mode using the WinDbg tool.

Learn more
How to Use WinDbg for Kernel Debugging

AppInit

The path to AppInit can differ depending on system architecture (64-bit vs 32-bit OS)

Win x64

For x64 applications, the registry path is the following:

“HKEY_LOCAL_MACHINESOFTWAREMicrosoftWindows NT CurrentVersionWindows”

Here we need to set the LoadAppInit_DLLs parameter to 1 and set the path to the DLL in AppInit.

For x86 applications, the registry path to AppInit is the following:

“HKEY_LOCAL_MACHINESOFTWAREWow6432NodeMicrosoftWindows NTCurrentVersionWindows”

You also need to set the LoadAppInit_DLLs parameter to true (1).

Win x86

“HKEY_LOCAL_MACHINESOFTWAREMicrosoftWindows NT CurrentVersionWindows”

You need to set the LoadAppInit_DLLs parameter to 1 and set the path to the DLL in AppInit.

To test the DLL, we chose the website http://www.unit-conversion.info/. After injecting the DLL, when the webpage is loaded (only if itโ€™s unencrypted) the message will be displayed, as shown in Figure 1 below.

Practical example of User mode HTTP traffic modification

 

The message is displayed when opening a website after DLL injection

Kernel Mode

In this section, weโ€™ll cover how to insert an ad banner via the Kernel driver.

How it works

Windows allows drivers to be embedded into every level of the data transfer protocol. This can be taken advantage of when creating a filter driver. The main platforms for filter drivers are:

1.    NDIS

2.    WFP

Implementation methods

You can also check out our article on network monitoring for more information on this topic.

NDIS

The Network Driver Interface Specification (NDIS) is an interface specification that was developed by Microsoft and 3Com to embed network adapter drivers into an operating system.

Driver initialization is fairly standard โ€“ information about the driver and pointers to functions that will call the NDIS are transferred.

Traffic modification happens within the function on the  FILTER_SEND_NET_BUFFER_LISTS callback. This function is called from the driver each time the NdisSendNetBufferLists function is called from the protocol driver. The modification scenario needs to be set by the developer. All filtering and modification need to be specified manually.

A driver will receive data from the NET_BUFFER and NET_BUFFER_LIST structures, modify this data and, if necessary, send data in these structures to other drivers.

WFP

The Windows Filtering Platform (WFP) is a universal network filtering technology covering all main network layers from the transport layer (TCP/UDP) to the data link layer (Ethernet) and providing a lot of interesting features for developers.

During driver initialization, information about the driver, the pointer to a filter function, and conditions for calling said function are transferred. These conditions include the layer of the data transfer protocol that will be used to process the data, the direction of the connection, the direction of packets, IP address, port, and so on. Since the driver is called by the system, itโ€™s named a callout driver. Data for requests and responses with which the driver interacts is located in the NET_BUFFER and NET_BUFFER_LIST structures. Data necessary for filtration is located in the FWPS_FILTER and FWPS_CLASSIFY_OUT structures. Filtering is done by editing the FWPS_CLASSIFY_OUT structure.

Related project

Developing Drivers for Low Latency Virtual Reality Headsets

Find out how the Apriorit team helped our client reduce time-to-market for their virtual reality headset through cutting-edge driver development and streamlined virtual and physical testing!

Project details
Developing Drivers for Low Latency Virtual Reality Headsets

Why we chose WFP

  • WFP was designed for the development of filtering and modifying drivers.
  • WFP is well documented.

Implementation overview

The thread layer is best suited for modifying HTTP responses. Therefore, we need to register the callout driver at the thread layer.

To get access to all traffic, conditions have been added:

  1. For incoming connections
  2. For outgoing connections

For analyzing and modifying data, we choose only responses from the server (incoming packets). We skip all other traffic, letting other filter drivers handle it.

Next, we need to analyze and inject data as described in the section above.

A common situation when the driver works with data is when all data needs to be gathered from several separate packages. In order to avoid any problems, we set the flag that tells WFP to gather more data in such cases.

To inject new data in a thread, we create a new NET_BUFFER_LIST and MDL. The MDL is injected into the thread in place of the old data, and the old data is blocked. Temporary resources are free.

If the object searched for isnโ€™t found in the extracted data, then the data is sent further down the chain.

Practical example

For the driver to work, you need to build it, install it, run it, and open the browser.

During the build process the driver package is created, which contains the following:

  • Driver security catalog (*.cat)
  • *inf file
  • driver file (*sys)
  • WdfCoinstallerXXXXX.dll

Installing the driver is simple: right-click on the *inf file and choose โ€œinstall.โ€

You can start the driver in two ways:

  1. Call โ€œnet start driverNameโ€ from the console with administrative permissions.
  2. Select โ€œView โ†’ Show hidden devicesโ€ in the device manager, then find the driver with the right name in the โ€œNon-Plug and Play Driversโ€ category and start it.

To test our driver, we used the http://msn.com website.

Figure 2 shows the website before the driver has been started.

After the driver has been started, when loading the page from scratch (and not simply refreshing), a dialog window with an ad is shown before the page is displayed, as seen in Figure 3.

Website page before the HTTP traffic was modified with a custom driver

 

Figure 2: MSN page before the driver has been launched

Website page after the HTTP traffic was modified with a custom driver

 

Figure 3: MSN page after the driver has been launched

Read also

Agile and Hybrid Approaches to Kernel and Driver Development: Pros, Cons, Examples

Driver development projects hide some unobvious project management pitfalls. Read on to find out useful tips and tricks for controlling the driver development process while keeping it flexible and open to changes.

Learn more
v3-1 -blog-article-Low-level-programming-and-Agile-practices-cover

Comparison of Kernel Mode and User Mode Approaches to Modifying HTTP Traffic

User Mode

Advantages:

  • Less knowledge of Windows programming is required than for Kernel mode
  • The worst thing that can happen in User mode is that an application stops working

Disadvantages:

  • More memory is required to save the modified response compared to the Kernel mode approach
  • This approach covers only graphical applications
  • The DLL is injected in all applications, not only network-based
  • Errors inside a DLL can affect all applications in which itโ€™s injected
  • Some antivirus software and other security software can prevent DLL injection or function hooking

Kernel Mode

Advantages:

  • Documented and legal way to capture network requests
  • Requires less memory: additional buffer memory needed only until new data is inserted in the thread

Disadvantages:

  • Greater responsibility on the part of the developer: any single error can result in a BSOD
  • To get all the necessary knowledge, you need to read a lot of documentation
  • Debugging is difficult and requires another computer or a virtual machine
  • Usual static libraries are not applicable, and need to be rebuilt in Kernel mode
  • There are fewer third-party libraries for Kernel mode

In the end, both User mode and Kernel mode work for injecting data in the HTTP response, and thus your choice should be based on the analysis of the pros and cons of each approach and how they relate to the particular task at hand.

Resources

Download the source code for our   Kernel mode and User mode examples of implementing HTTP traffic modification.

Additional reading:

  1. Jeffrey Richter, “Windows via C/C++”
  2. https://en.wikipedia.org/wiki/Chunked_transfer_encoding
  3. Official documentation on driver creation using WFP, https://docs.microsoft.com/en-us/windows-hardware/drivers/network/windows-filtering-platform-callout-drivers2
  4. WFP driver initialization, https://docs.microsoft.com/en-us/windows-hardware/drivers/network/initializing-a-callout-driver
  5. WFP overview, https://msdn.microsoft.com/en-us/library/windows/desktop/aa366510(v=vs.85).aspx
  6. Official documentation on driver creation using NDIS, https://docs.microsoft.com/en-us/windows-hardware/drivers/network/ndis-filter-drivers2
  7. NDIS filter driver initialization, https://docs.microsoft.com/en-us/windows-hardware/drivers/network/initializing-a-filter-driver

Conclusion

Developers often need to intercept and modify HTTP traffic for various purposes, like debugging and troubleshooting, modifying requests and responses, and optimizing software performance. However, your team must do it accurately and securely, knowing tips and tricks of this practice.

Consider contacting Apriorit to access our professional kernel and driver development teams to accelerate your software development process and ensure the top-notch quality of your final solution.

Have a Windows software project in mind?

Fulfill all your technical needs by entrusting engineering tasks to Aprioritโ€™s experts in kernel and driver development and network management.

Have a question?

Ask our expert!

Michael-Teslia
Michael Teslia

Program Manager

Tell us about
your project

...And our team will:

  • Process your request within 1-2 business days.
  • Get back to you with an offer based on your project's scope and requirements.
  • Set a call to discuss your future project in detail and finalize the offer.
  • Sign a contract with you to start working on your project.

Do not have any specific task for us in mind but our skills seem interesting? Get a quick Apriorit intro to better understand our team capabilities.