Free Software Development
Introduction
This post is created for the Free Software Development course taught by Professor Paulo R. M. Meirelles at the Institute of Mathematics and Statistics (IME) at the University of São Paulo (USP). The course is designed and maintained by the research group FLUSP. You can find all tutorials and more detailed information on its official webpage. More information about the course (content, references, etc.) can be found on the official website Janus for the courses, MAC5856 - Desenvolvimento de Software Livre.
The objective of the course is to introduce students to free and open source software. In Portuguese, both terms mean the same. The course covers basic concepts, appropriate languages, ethical and technical aspects, etc., with the goal of promoting and constructing open and collaborative software.
NOTE
The device used for the course is a Lenovo laptop with Ubuntu 24.04.2 LTS.
Tutorial 1: Setting up a test environment for Linux Kernel
The first tutorial of the course provides an introduction to QEMU and libvirt, powerful tools designed to deploy virtual machines (VMs) quickly and efficiently, eliminating the traditional complexities and time-consuming efforts typically associated with VM setup. Both QEMU and libvirt are libraries used for virtualization and resource emulation, enabling users to create and manage virtualized environments with ease.
A detailed guide can be found at: Setting up a test environment for Linux Kernel Dev using QEMU and libvirt.
Summary
The tutorial is divided into the following sections:
-
Preparing testing environment directory and “all-in-one” script
- Create necessary folders and files.
- Give the required permissions for files and folders.
-
Set up and configure a VM running a guest OS
- Create an
activate.sh
file to deploy VMs manager. To enable it, execute the comand/path_to/activate.sh
, this will print some green text and a pink prompt preamble to . - Check image OS properties and partition size, then resize the disk image to 5Gb to give space for more linux modules.
- Extract kernel and initrd images from OS image to create the VM
- Create an
- Configure SSH access from the host to the VM
- Set up host <-> VM file sharing (optional)
To start a created machine use sudo virsh start --console arm64
.
Troubleshooting
SSH Configuration
At point 3, if the /etc/ssh/
folder is missing, it indicates that SSH is not installed on the system. To resolve this issue, follow these steps:
-
Update and upgrade the system packages:
sudo apt update && sudo apt upgrade
-
Install the OpenSSH server package to enable SSH functionality:
sudo apt install openssh-server
This will download and configure the necessary SSH dependencies, allowing you to establish an SSH connection.
Shared Memory
Blog to solve the shared memory issue: https://discourse.nixos.org/t/virt-manager-cannot-find-virtiofsd/26752/9
EDITOR=nano; virsh edit arm64
Parctical excersie: Ubuntu Jammy kernel
As a practical exercise, let’s create a VM using an Ubuntu image. Since the previous image is lightweight, let’s test with a more robust image.
-
Add environment constants and functions to the “all-in-one” script:
++activate.sh export BOOT_DIR_UBUNTU="${VM_DIR}/ubuntu-jammy-arm64" # path to boot artifacts # ---------- ubuntu image ----------- function launch_ubuntu_vm_qemu() { qemu-system-aarch64 \ -M virt,gic-version=3 \ -m 7G -cpu cortex-a57 \ -smp 2 \ -netdev user,id=net0 -device virtio-net-device,netdev=net0 \ -initrd "${BOOT_DIR_UBUNTU}/initrd.img-5.15.0-134-generic-lpae" \ -kernel "${BOOT_DIR_UBUNTU}/vmlinuz-5.15.0-134-generic-lpae" \ -append "loglevel=8 root=/dev/vda2 rootwait" \ -device virtio-blk-pci,drive=hd \ -drive if=none,file="${VM_DIR}/ubuntu-jammy-arm64.qcow2",format=qcow2,id=hd \ -nographic } function create_ubuntu_vm_virsh() { sudo virt-install \ --name "ubuntu-jammy" \ --memory 2048 \ --arch aarch64 --machine virt \ --osinfo detect=on,require=off \ --import \ --features acpi=off \ --disk path="${VM_DIR}/ubuntu-jammy-arm64.qcow2" \ --boot kernel=${BOOT_DIR_UBUNTU}/vmlinuz-5.15.0-134-generic-lpae,initrd=${BOOT_DIR_UBUNTU}/initrd.img-5.15.0-134-generic-lpae,kernel_args="loglevel=8 root=/dev/vda2 rootwait" \ --network bridge:virbr0 \ --graphics none } export -f launch_ubuntu_vm_qemu export -f create_ubuntu_vm_virsh
-
Set up and configure a VM
2.1. Download image
wget --directory-prefix="${VM_DIR}" https://cloud-images.ubuntu.com/jammy/current/jammy-server-cloudimg-armhf.img mv "${VM_DIR}/jammy-server-cloudimg-armhf.img" "${VM_DIR}/ubuntu-jammy-arm64.qcow2" # rename file for legibility
The kernel downloaded is Debian-12-nocloud-arm64-daily-20250217-2026.qcow2, an image for QEMU without cloud settings and arm64 architecture, link to download. As practice excersie let’s download an Ubuntu arm image, Ubuntu 22.04 LTS (Jammy Jellyfish), select any QCow2 image, note that this image contains a cloud setup.
2.2. Resize disk image
du -h "ubuntu-jammy-arm64.qcow2"
Output:
1,2G base-jammy-server-cloudimg-armhf.img
qemu-img info "${VM_DIR}/ubuntu-jammy-arm64.qcow2"
Output:
image: base-jammy-server-cloudimg-armhf.img file format: qcow2 virtual size: 3.5 GiB (3758096384 bytes) disk size: 1.15 GiB cluster_size: 65536 Format specific information: compat: 0.10 compression type: zlib refcount bits: 16 Child node '/file': filename: base-jammy-server-cloudimg-armhf.img protocol type: file file length: 1.15 GiB (1233318400 bytes) disk size: 1.15 GiB
virt-filesystems --long --human-readable --all --add "${VM_DIR}/ubuntu-jammy-arm64.qcow2"
Output:
Name Type VFS Label MBR Size Parent /dev/sda1 filesystem ext4 cloudimg-rootfs - 3,2G - /dev/sda15 filesystem vfat UEFI - 97M - /dev/sda1 partition - - - 3,4G /dev/sda /dev/sda15 partition - - - 99M /dev/sda /dev/sda device - - - 3,5G -
qemu-img create -f qcow2 -o preallocation=metadata "${VM_DIR}/ubuntu-jammy-arm64.qcow2" 7G # creates new QCOW2 image with 5GB virt-resize --expand /dev/sda1 "${VM_DIR}/base-jammy-server-cloudimg-armhf.img" "${VM_DIR}/arm64_img.qcow2" # makes a copy of the image expanding the `rootfs`
Name Type VFS Label MBR Size Parent /dev/sda1 filesystem vfat UEFI - 97M - /dev/sda2 filesystem ext4 cloudimg-rootfs - 6,6G - /dev/sda1 partition - - - 99M /dev/sda /dev/sda2 partition - - - 6,9G /dev/sda /dev/sda device - - - 7,0G -
initrd.img-5.15.0-134-generic-lpae
vmlinuz-5.15.0-134-generic-lpae
Useful Commands
lsmod
- Show the status of modules in the Linux Kernel, more info here.

Tutorial 2: Building and booting a custom Linux kernel for ARM using kw
asd
A detailed guide can be found at: Building and booting a custom Linux kernel for ARM using kw
Parts
find ~/ -type f -name “postgis-2.0.0”
Troubleshooting
Results
Tutorial 3: Introduction to Linux kernel build configuration and modules
Parts
find ~/ -type f -name “postgis-2.0.0”
Troubleshooting
Results
Tutorial 4:
Parts
Troubleshooting
Results
Citation
If you found this useful, please cite this as:Novoa, Sebastián Aguilera (Mar 2025). Free Software Development. https://saguileran.github.io/blog/2025/free-software-development/.
or as a BibTeX entry:
@article{novoa2025free-software-development,
title = {Free Software Development},
author = {Novoa, Sebastián Aguilera},
year = {2025},
month = {Mar},
url = {https://saguileran.github.io/blog/2025/free-software-development/}
}