How to boot an uncompressed Linux kernel on ARM

This is a quick post to share my experience booting uncompressed Linux kernel images, during the benchmarks of kernel compression options, and no compression at all was one of these options.

It is sometimes useful to boot a kernel image with no compression. Though the kernel image is bigger, and takes more time to copy from storage to RAM, the kernel image no longer has to be decompressed to RAM. This is useful for systems with a very slow CPU, or very little RAM to store both the compressed and uncompressed images during the boot phase. The typical case is booting CPUs emulated by FPGA, during processor development, before the final silicon is out. For example, I saw a Cortex A15 chip boot at 11 MHz during Linaro Connect Q2.11 in Budapest. At this clock frequency, booting a kernel image with no compression saves several minutes of boot time, reducing development and test time. Note that with such hardware emulators, copying the kernel image to RAM is cheap, as it is done by the emulator from a file given by the user, before starting to emulate the system.

Building a kernel image with no compression on ARM is easy, but only once you know where the uncompressed image is and what to do! For people who have never done that before, I’m sharing quick instructions here.

To generate your uncompressed kernel image, all you have to do is run the usual make command. The file that you need is arch/arm/boot/Image.

Depending on the bootloader that you use, this could be sufficient. However, if you use U-boot, you still need to put this image in a uImage container, to let U-boot know about details such as how big the image is, what its entry point is, whether it is compressed or not… The problem is you can’t run make uImage any more to produce this container. That’s because Linux on ARM has no configuration option to keep the kernel uncompressed, and the uImage file would contain a compressed kernel.

Therefore, you have to create the uImage by invoking the mkimage command manually. To do this without having to guess the right mkimage parameters, I recommend to run make V=1 uImage once:

$ make V=1 uImage
  Kernel: arch/arm/boot/zImage is ready
  /bin/bash /home/mike/linux/scripts/ -A arm -O linux -T kernel -C none -a 0x80008000 -e 0x80008000 -n 'Linux-3.3.0-rc6-00164-g4f262ac' -d arch/arm/boot/zImage arch/arm/boot/uImage
Image Name:   Linux-3.3.0-rc6-00164-g4f262ac
Created:      Thu Mar  8 13:54:00 2012
Image Type:   ARM Linux Kernel Image (uncompressed)
Data Size:    3351272 Bytes = 3272.73 kB = 3.20 MB
Load Address: 80008000
Entry Point:  80008000
  Image arch/arm/boot/uImage is ready

Don’t be surprised if the above message says that the kernel is uncompressed (corresponding to -C none). If we told U-boot that the image is already compressed, it would take care of uncompressing it to RAM before starting the kernel image.

Now, you know what mkimage command you need to run. Just invoke this command on the Image file instead of zImage (you can directly replace by mkimage):

$ mkimage -A arm -O linux -T kernel -C none -a 0x80008000 -e 0x80008000 -n 'Linux-3.3.0-rc6-00164-g4f262ac' -d arch/arm/boot/Image arch/arm/boot/uImage
Image Name:   Linux-3.3.0-rc6-00164-g4f262ac
Created:      Thu Mar  8 14:02:27 2012
Image Type:   ARM Linux Kernel Image (uncompressed)
Data Size:    6958068 Bytes = 6794.99 kB = 6.64 MB
Load Address: 80008000
Entry Point:  80008000

Now, you can use your uImage file as usual.

About Michael Opdenacker

Michael Opdenacker is the founder of Free Electrons. He is best known for all the free embedded Linux and kernel training materials that he created together with Thomas Petazzoni. He is always looking for ways to increase performance, reduce size and boot time, and to maximize Linux' world domination. More details...
This entry was posted in Technical and tagged , , , , , . Bookmark the permalink.

9 Responses to How to boot an uncompressed Linux kernel on ARM

  1. Jagan says:

    I have question regarding decompressed kernel booting. [zImage]
    Ignore this if this question is not valid for this forum.

    What exactly this load and entry point address, why were they same?

    As per my knowledge load address is where once the decompression is done the memory [base+offset] where the kernel loads and this is constant address specified on valid machine Makefile.boot.

    and also Can I attach the mkimage header to ramdisk and boot? is this valid scenario?

  2. Matthias says:

    Does anyone know how this could be done with xipImage?

  3. hakansel says:

    Hi Michael Opdenacker, I created the uncompressed image(uImage). But I didn’t boot it within existing root file system. Because there are already existing uncompressed kernel in boot directory. How can configure my existing root file system with new uncompressed image? OR How can properly create new root file system for new uncompressed image?

    Thanks in advance.

  4. Ablak says:

    After I had a UImage file. What are the steps to test it?

    Thank you

  5. Chris says:

    How do you determine the kernel load address and entry point?

  6. dhruv says:


    I had similar issue compiling rock chip kernel. Could you please update me if possible.
    OBJCOPY arch/arm/boot/Image
    Kernel: arch/arm/boot/Image is ready
    /kernel/mkkrnlimg: /kernel/mkkrnlimg: cannot execute binary file
    make: *** [kernel.img] Error 126

  7. Dantali0n says:

    Hi, I just tried the make V=1 uImage command on linux 4.4.39 but it seems to no longer call and display the desired load address and entry point. Would you have happened to come across another way to determine these important parameters?

Leave a Reply

Your email address will not be published. Required fields are marked *