How do I uncompress vmlinuz to vmlinux?
Andrew Mclaughlin
I have already tried uncompress, gzip, and all other solutions that come up as google results and these have not worked for me.
To get just the image search for the GZ signature -
1f 8b 08 00.> od -A d -t x1 vmlinuz | grep '1f 8b 08 00' 0024576 24 26 27 00 ae 21 16 00 1f 8b 08 00 7f 2f 6b 45so the image begins at
24576+8 => 24584. Then just copy the image from the point and decompress it -> dd if=vmlinuz bs=1 skip=24584 | zcat > vmlinux 1450414+0 records in 1450414+0 records out 1450414 bytes (1.5 MB) copied, 6.78127 s, 214 kB/sGot these instructions verbatim from a forum online:
This process does not work for me and end up giving errors that states file not found 0024576 and all subsequent numbers.
How do I proceed extracting vmlinux from vmlinuz?
Thank you.
EDITED: This is a reverse engineering question. I have no access to the distro to install any RPM or recompile. I start with nothing but vmlinuz.
37 Answers
Maybe you misunderstood what the author of that post meant.
The
vmlinuzfile contains other things besides the gzipped content, so you need to find out where the gzipped content starts. To do that, use:od -A d -t x1 vmlinuz | grep '1f 8b 08 00'What this does is to show you where in that file you can find the gzip header. The output looks like:
0024576 24 26 27 00 ae 21 16 00 1f 8b 08 00 7f 2f 6b 45This means that at
0024576(at least for the author of the post, yours might be somewhere completely different) in thevmlinuzfile, you will find the binary values "24 26 27 00 ae 21 16 00 1f 8b 08 00 7f 2f 6b 45". You're looking for1f 8b 08 00, which can be found from character 9 onwards, or, at0024576 + 8(start counting from 0)= 24584.Now that you know where the gzipped content starts (at position
24584) you can useddto extract that gzipped content and ungzip it. To do that, use:dd if=vmlinuz bs=1 skip=24584 | zcat > vmlinuxThe first command will seek to that position and copy everything to stdout.
zcatthen will uncompress everything it gets from stdin and will output the uncompressed string to stdout. Then the>will redirectzcat's output to a new file namedvmlinux.
The kernel sources now contain a script for this, extract-vmlinux. You don't need to roll your own.
See this SO answer.
3Actually, before generating the vmlinuz file, most symbols are stripped. So you cannot rebuild a true vmlinux from vmlinuz, the file will not be as useful for debugging.
I ran into simple problem - looking for correct version of vmlinux for crash. Instead of trying to decompress vmlinuz to vmlinux.
The better solution is: install the RPM: kernel-debuginfo, that RPM contains proper vmlinux file.
Pay attention to the rpm name, there are multiple similar (confusing) names. Gotta be: kernel-debuginfo-$(version).rpm
Modern kernels are not always (in fact, not generally) gzip compressed. They may use bzip2 or LZMA. A quick web search didn't help me find the magic strings for those compression methods--you might be better off checking several kernel images to find the invariant header that includes the decompression code.
0The decompression lines above worked for me and, of course, the kernel was stripped. Therefore no good information there.
If you need to make changes to your old kernel, like put it in debug, use uname -r to get the revision of your kernel and get its source:
sudo apt-get source linux-image-\`uname -r\`The source will be in /usr/src/linux... cd to the source tree and:
make oldconfig
makeThis will attempt to find the file containing the config for the currently running kernel -- usually
/boot/config-\`uname -r\` and use it for this build thus recreating the running kernel.
Build that the way you want it; have access to the unstripped kernel as needed.
The shells above probably won't find the gzip magic number due to spacing. Yes they are still compressed the same way, though I'm writing a year after the original discussion. Send the output to less and search for 1f 8b or even 1f. Check the rest of the bytes manually to determine a match and verify you have the first instance. Use the offset found remembering it is decimal.
You might not have readelf installed and need the elfutils package installed with the call being eu-readelf instead.