Frédéric Dalleau
March 13, 2017
Reading time:
After setting up a virtual machine using debootstrap, let’s do some more advanced work.
kmemleak can be used to search for memory leaks. Enable it in the kernel and rebuild. Note that we update the “Maximum kmemleak early log entries”. If the value is too low, kmemleak would disable itself at boot time.
-> Memory Debugging [*]Enable kernel memory leak detector. (2000) Maximum kmemleak early log entries |
After booting this kernel, the memory leaks will be detected periodically. When a leak would be detected, traces will be displayed in dmesg. Leak detection can be triggered manually too.
$ echo scan >> /sys/kernel/debug/kmemleak $ cat /sys/kernel/debug/kmemleak |
A scan has a certain duration, and the detected leaks won’t appear immediately. Even after a manual scan, if no leak is detected, nothing is displayed. To ensure that no leaks where detected, it is better to wait a few tenths of seconds, and also to trigger several manual scans.
Full documentation can be found here: https://www.kernel.org/doc/Documentation/kmemleak.txt
Using serial link has some limitations as terminal emulation problems sometime occur. A screen or vi session in the guest sometimes get screwed up and need to be restarted. Hence ssh is a better candidate. To enable networking, the option is : -net nic -net user. You can immediatly notice that the interface is not setup.
auto eth0 iface eth0 inet dhcp |
$ apt-get install openssh-server $ adduser fredo |
To add a redirection from ssh guest port to a host port -net nic -net user,hostfwd=tcp::5555-:22.
$ qemu-system-x86_64 -kernel bzImage -append "root=/dev/sda console=ttyS0 single" -drive file=toto.img,index=0,media=disk,format=raw --enable-kvm --nographic -net nic -net user,hostfwd=tcp::5555-:22 |
On the host, connect with:
$ ssh -p 5555 fredo@localhost |
A realworld USB device can be bridged into QEMU. It takes the following option. -usb -usbdevice host:050d:016a. The cryptic last two values are product and vendor ids. They can be retrieved using lsusb on the host.
We can go even further in debugging the kernel using GDB. Just add the necessary options. Those can be found in “Kernel hacking”.
[*] KGDB: kernel debugger ---> <*> KGDB: use kgdb over the serial console -> Compile-time checks and compiler options [*] Compile the kernel with debug info |
We must add the -s option to qemu. It will instruct qemu to create a serial port to control kernel debugging in the guest. This serial port will be backed to a tcp socket that a GDB running on the host can connect to.
Our command line becomes:
$ qemu-system-x86_64 -kernel arch/x86/boot/bzImage -hda toto.img -append "root=/dev/sda" -s |
qemu will start as usual and run as if nothing happened.
vmlinux is the uncompressed bzImage and the symbols are included in the binary. This is what we use in GDB:
$ gdb vmlinux |
In GDB, attach to the running machine:
(gdb) target remote localhost:1234 |
Note that the virtual machine has been stopped. From here it is now possible to inspect the backtrace and use typical GDB commands: breakpoint, continue… Try that (and delete quickly):
(gdb) breakpoint spin_lock (gdb) continue |
In kernel hacking, an good amount of debugging tools are available: dynamic_debug: You want to enable dynamic debug.
-> printk and dmesg options [*] Enable dynamic printk() support |
ftrace: ftrace can be useful too
-> Tracers (FTRACE [=y]) |
I wont tell more here, since excellent lectures already exists: see lwn here and here.
Only one QEMU at a time can handle -s. Should you really need to debug two machines at a time, it is possible to create a dedicated serial port per process.
Using the same disk image from two machines at same time will generate undefined results and is highly likely to break the disk image. Just copy the image with a new name to use it. QEMU also provide copy-on-write, which helps to solve that problem by writing changes to the original filesystem in a separate file.
03/12/2024
this is a test post
08/10/2024
Having multiple developers work on pre-merge testing distributes the process and ensures that every contribution is rigorously tested before…
15/08/2024
After rigorous debugging, a new unit testing framework was added to the backend compiler for NVK. This is a walkthrough of the steps taken…
01/08/2024
We're reflecting on the steps taken as we continually seek to improve Linux kernel integration. This will include more detail about the…
27/06/2024
With each board running a mainline-first Linux software stack and tested in a CI loop with the LAVA test framework, the Farm showcased Collabora's…
26/06/2024
WirePlumber 0.5 arrived recently with many new and essential features including the Smart Filter Policy, enabling audio filters to automatically…
Comments (4)
lixiaoquan:
Feb 18, 2018 at 08:07 AM
I meet an error when executing this:
auto eth0
iface eth0 inet dhcp
When I execute them, it reports auto can't be found, it seems "auto / iface"are not included in rootfs which is generated by debootstrap
Reply to this comment
Reply to this comment
Vasyl:
Oct 01, 2018 at 08:49 PM
This are standard ifupdown scripts configuration. It should be located in /etc/network/interfaces file for example. Note that eth0 should be replaced with actual network interface name on your machine.
Reply to this comment
Reply to this comment
pavan:
Oct 01, 2018 at 04:43 PM
For using multiple qemu instances, use -gdb tcp:: option.
for using the same image, we can always pass image in snapshot mode.
Reply to this comment
Reply to this comment
Vasyl:
Oct 01, 2018 at 08:53 PM
When using gdb remote debugging via qemu gdb server I can not step in kernel code. Every time I try I end up in some interrupt handling code
(gdb) b vmem_export_single
Breakpoint 1 at 0xffffffffa0000390: file drivers/dma-buf/vmem-exp.c, line 382.
(gdb) c
Continuing.
Breakpoint 1, vmem_export_single (ctx=0xffff880005886d40, req=0xffffc900003f7e80) at drivers/dma-buf/vmem-exp.c:382
382 req->vaddr &= PAGE_MASK;
(gdb) s
lapic_next_event (delta=, evt=) at arch/x86/kernel/apic/apic.c:462
462 }
(gdb) cont
Continuing.
Breakpoint 1, vmem_export_single (ctx=0xffff880005886d40, req=0xffffc900003f7e80) at drivers/dma-buf/vmem-exp.c:382
382 req->vaddr &= PAGE_MASK;
(gdb) s
lapic_next_event (delta=, evt=) at arch/x86/kernel/apic/apic.c:462
462 }
(gdb)
Reply to this comment
Reply to this comment
Add a Comment