Skip to main content
Email Password   helpLost your password?

Introduction

As you know, traditional linking of object code is no longer necessary in .NET. A .NET program will usually consist of multiple parts. A typical .NET application consists of an executable assembly, a few assemblies in the program directory, and a few assemblies in the global assembly cache. When the program is run, the runtime combines all these parts to a program. Linking at compile time is no longer necessary.

But sometimes, it is nevertheless useful to combine all parts a program needs to execute into a single assembly. For example, you might want to simplify the deployment of your application by combining the program, all required libraries, and all resources, into a single .exe file.

A single project

If all parts of your program are written by yourself in the same language, you can obviously just add all source files to a single project. The result will be a single DLL or EXE containing all dependencies.

csc /target:winexe /out:Program.exe 
      MainProgram.cs ClassLibrary1.cs ClassLibrary2.cs

However, if your program is written in multiple languages or if you are using binary third party libraries, you are out of luck.

.NET Modules

The .NET compilers already contain options for exactly this. If you compile a project, there is an option to create a module, which is similar to an assembly but without a manifest file. You can then use the al.exe tool to combine some of these modules to a single assembly. This feature makes it possible to create a single assembly that contains multiple languages.

First, you would compile the program and the class libraries to netmodules using the module target. Then you can use the assembly linker al.exe to combine these modules to a single assembly.

csc /target:module /out:ClassLibrary1.netmodule ClassLibrary1.cs
vbc /target:module /out:ClassLibrary2.netmodule ClassLibrary2.vb
vbc /target:module /out:Program.netmodule Program.vb
al /target:winexe /out:Program.exe ClassLibrary1.netmodule 
                 ClassLibrary2.netmodule Program.netmodule

But unfortunately, this method only works if you have all the required parts of your program either as source code or as .NET modules. If you are useing a third party class library in assembly form, you are again out of luck.

ILMerge

Since a .NET module is basically just an assembly without an assembly manifest, it should be possible to convert an assembly to a .NET module, at least that is what I thought. When researching this on Google, I found a tremendously useful tool on Microsoft research called ILMerge. This little gem makes it possible to link multiple assemblies to a single one.

First, you would compile your libraries to DLLs and your program to an EXE referencing the DLLs. This is exactly what Visual Studio would do if you had multiple libraries and a program referencing these libraries, so there is no need to do this on the command line.

csc /target:library /out:ClassLibrary1.dll ClassLibrary1.cs
vbc /target:library /out:ClassLibrary2.dll ClassLibrary2.vb
vbc /target:winexe /out:Program.exe 
    /reference:ClassLibrary1.dll,ClassLibrary2.dll Program.vb

This will produce a normal .exe that requires the two DLLs in the program directory or in the global assembly cache to run.

Now you can link these parts to a single self-contained EXE, using ILMerge:

ilmerge /target:winexe /out:SelfContainedProgram.exe 
        Program.exe ClassLibrary1.dll ClassLibrary2.dll

The nice thing about this is that you can also merge third party assemblies like commercial class libraries into your program. And you do not have to modify your build process. All you have to do is to merge the assemblies to a single EXE before deploying.

Conclusion

I found ILMerge tremendously useful, and I think that something like this should be a part of the .NET framework SDK. Maybe just enhance al.exe so that it can also link DLLs.

I have only scratched the surface of the .NET build process and the capabilities of ILMerge, and this article might contain many inaccuracies or even errors. But I found ilmerge.exe so useful that I just had to write about it.

Resources

You must Sign In to use this message board.
 
 
Per page   
 FirstPrevNext
GeneralILMerge GUI Pin
Lee.
2:26 17 Oct '09  
GeneralMerged DLL com registration Pin
William F
6:17 8 Jun '09  
GeneralHow do you merge all the assemblies in the bin folder using ILMerge using command line Pin
azamsharp
6:07 24 Jul '08  
GeneralWindows Mobile 6.0 Compatibility Pin
Mandeep Singh Sidhu
2:31 23 Apr '08  
GeneralCOM Pin
GAbirkan
9:33 22 Dec '07  
GeneralProblem after merging Pin
padhalni
1:56 19 Feb '07  
QuestionRe: Problem after merging Pin
a8584
8:56 10 Apr '07  
AnswerRe: Problem after merging Pin
a8584
9:04 10 Apr '07  
Questionmerging a bunch of DLLs of an ASP.NET project Pin
weareborg
7:32 2 Dec '06  
GeneralLocalisation DLLs Pin
Douba
1:49 29 Jun '06  
GeneralGUI for ILMerge - Gilma Pin
Tomer Shalev
11:27 2 Dec '05  
GeneralThere is a product out there with UI for ILMerger Pin
Samar Aarkotti
11:21 14 Oct '05  
GeneralRe: There is a product out there with UI for ILMerger Pin
Frank Hileman
7:58 29 Nov '05  
Generalsimple question. Pin
Samar Aarkotti
4:43 13 Oct '05  
General3rd party control licenses after merge Pin
Drew Noakes
5:29 6 Sep '05  
GeneralRe: 3rd party control licenses after merge Pin
weiser_98@hotmail.com
23:14 6 Feb '06  
GeneralRe: 3rd party control licenses after merge Pin
Bertrand Jobert
11:13 5 Nov '07  
GeneralDotfuscator and ILMerge Pin
mceranski
8:09 29 Aug '05  
GeneralRe: Dotfuscator and ILMerge [modified] Pin
qjhsert
23:27 14 Oct '06  
Generalno output Pin
NewYoda
23:21 12 Aug '05  
GeneralRe: no output Pin
Drew Noakes
5:35 6 Sep '05  
GeneralHow to use this Pin
allenmpcx
21:46 12 Aug '05  
GeneralRe: How to use this Pin
Drew Noakes
5:38 6 Sep '05  
GeneralVB.NET and Setup Projects Pin
AndrewChapman
14:49 9 Aug '05  
GeneralRe: VB.NET and Setup Projects Pin
dcoolidge
13:39 12 Aug '05  


Last Updated 21 Jan 2005 | Advertise | Privacy | Terms of Use | Copyright © CodeProject, 1999-2009