__ _ _ __
/ _|_ __ _ _ __ _ __ _| (_)/ _|_ _
| |_| '__| | | |/ _` |/ _` | | | |_| | | |
| _| | | |_| | (_| | (_| | | | _| |_| |
|_| |_| \__,_|\__, |\__,_|_|_|_| \__, |
|___/ |___/
Traditionally, Puppy Linux has been based on two kinds of installation:
- "Full installation", where the operating system is installed on a bootable partition, like most other distros.
- "Frugal installation", where operating system consists of a squashfs image placed on a bootable partition, a writable file system image placed alongside it (the savefile) and an initramfs that mounts an aufs union file system where the squashfs image is a read-only layer, and changes to the file system are saved in the savefile.
In Puppy jargon, this is translated into a variable called PUPMODE. PUPMODE=2 means full installation, and PUPMODE=13 means frugal installation.
This is an over-simplified explanation because PUPMODE is already explained in detail by Barry Kauler, the original creator of Puppy and the architect behind its boot process.
Generally, the full installation type is reserved to cases where frugal installation is impossible, because the frugal installation type has many advantages:
- It takes less space: the squashfs image containing the operating system is compressed.
- It's easier to install, update, inspect and repair: the operating system itself is just the kernel, the initramfs and a squashfs image; the savefile can be deleted to repair or reset the operating system, and backup is a matter of copying the savefile.
- It makes it possible to install multiple operating systems (say, different variants of Puppy) on the same partition.
- Applications start faster and the operating system feels more responsive, because the smaller, compressed form of applications is faster to read from the disk, or because it's much faster to read from RAM and
pfix=copy
is specified.
However, inability to perform a frugal installation is not a purely theoretical problem:
- Some non-x86 devices have boot loaders that don't support initramfs boot.
- Sometimes, one might wish to take the highly modified kernel from a different distro (say, from Chrome OS) in its binary form and use it to boot Puppy, but that kernel is built without initramfs support.
- The Puppy initramfs is sensitive to boot device types (e.g. it must contain the required drivers), file system types (e.g. guess_fstype needs to be able to detect the file system on the boot partition) and so on; something along the boot process might not work in some configuration where non-initramfs boot would work just fine.
- Sometimes, the kernel command-line of kernels borrowed from other operating systems is hardcoded in the kernel image.
- The use of a savefile is inconvenient: it's slower than saving directly on a partition, it's hard to decide how much space to allocate for the savefile, and limiting the save file size to allow future operating system upgrades is a big compromise on computers with a small 16 GB disk.
frugalify is a small, static executable that can be placed on a bootable partition and configured to act as PID 1 via the init= kernel parameter.
frugalify simulates what the Puppy initramfs does:
- It re-runs itself from RAM, so the frugalify executable on disk can be replaced (for example, with a later version).
- It looks for squashfs images on the partition mounted by the kernel.
- It locks the image contents into RAM, unless
pfix=nocopy
is specified. - It creates the /upper directory on the partition, or mounts a
tmpfs
when / is read-only orpfix=ram
is specified. - It mounts a union file system, with /upper/save as the upper layer.
- It re-mounts the boot partition under /mnt/%s, to replace /dev/root in /proc/mounts with the actual block device name. Some old Puppy scripts try to detect the mount point of a partition using the block device name and /proc/mounts, instead of using its unique major:minor combination and /proc/self/mountinfo.
- It adds a bind mount at /initrd and creates the /mnt/home symlink, to simuate what the Puppy initramfs does.
- It runs the Puppy init script and a login shell under the union file system. Until commit 42350b2, frugalify used to pass control to /sbin/init, but now it runs the init script and starts a login shell without passing through busybox init, getty, login, etc', in order to speed up the boot process.
The result is an initramfs-less Puppy installation that combines the advantages of both installation methods:
- The operating system is small, because it's compressed.
- Applications start quickly, because they are already in RAM.
- Updates, repair, etc' are easy, because the operating system is the kernel, the frugalify executable and a squashfs image.
- Persistent storage is implemented using a disk partition, and the user does not have to reserve space for it, or think about available free space (i.e. empty the browser cache) all the time.
- It's portable: if the kernel can mount the partition, a semi-frugal installation using frugalify is possible.
frugalify supports:
The aufs variant of frugalify supports encryption of the /upper directory using file system level encryption.
In every boot, the user is required to specify a passphrase. frugalify computes its SHA512 using mbedtls to generate a 64-byte encryption key.
frugalify spawns a child process, fstrimd, which simulates fstrim(8) once a week.
frugalify spawns two child processes responsible for logging: syslogd and klogd.
frugalify supports the following Puppy boot options:
pfix=nocopy
: disables the locking of squashfs image contents to RAMpfix=ram
: disables persistency
These options should be specified in the kernel command-line.
Pre-built, statically-linked and portable binaries linked against musl are available in the releases page.
frugalify is free and unencumbered software released under the terms of the MIT license; see COPYING for the license text.
The ASCII art logo at the top was made using FIGlet.