Zeppe-Lin Handbook

PREFACE

Zeppe-Lin Handbook. Because figuring it out yourself is overrated.

This guide gives you everything you need to install, configure, and maintain your Zeppe-Lin system.

Build the Zeppe-Lin system you need. This handbook shows you how (because nobody else will).

1. INTRODUCTION

1.1. About Zeppe-Lin

1.1.1. What is Zeppe-Lin?

Zeppe-Lin is a lightweight, source-based GNU/Linux distro for x86-64, forked from CRUX. Rooted in KISS and UNIX ideals, it gives you the simplicity and control to build a clear, transparent system.

1.1.2. Who is Zeppe-Lin For?

Zeppe-Lin is for the Linux user who understands the elegance of Plan 9's philosophy but still prefers a POSIX-compliant environment they can build themselves, brick by digital brick.

While we might not be serving up everything as a file quite yet, the spirit of Plan 9's elegance and simplicity resonates deeply within Zeppe-Lin.

1.2. Principles

Zeppe-Lin is guided by the following principles:

  1. Keep It Simple (KISS). Design and code with minimalism. No fluff -- just what's necessary.

  2. Respect the UNIX Spirit. Craft small, focused tools with text-based, modular configurations.

  3. Stay Transparent. Keep everything clear and understandable. Avoid needless abstraction.

  4. Enable User Control. Provide flexible tools for customization. Let users define their workflows.

  5. Build on a Minimal Base. Start lean. Add only what is explicitly required.

These principles ensure Zeppe-Lin remains simple, clear, and true to UNIX.

2. INSTALLATION

2.1. Supported Hardware

Zeppe-Lin Linux supports x86-64 processors (e.g., AMD Athlon 64, Intel Core, Intel Atom). Older architectures, such as i686, are incompatible.

2.2. Boot a Live System

Zeppe-Lin is distributed as a compiled tarball containing a root filesystem. To install it, first boot your computer using a Linux-based "Live CD/DVD/USB" system.

Launch your preferred Live system, open a terminal, and gain root privileges (e.g., sudo su).

2.3. Disk Partitions and Filesystems

This section describes "UEFI and LVM on LUKS" installation. Additional partitions/filesystems schemes may be added in the future.

2.3.1. UEFI and LVM on LUKS

Set up Zeppe-Lin on a fully encrypted disk (excluding the bootloader partition for UEFI compatibility) using dm-crypt with LUKS and an LVM container inside the encrypted partition.

2.3.1.1. Partition Scheme

Example device: /dev/sda. Replace with your actual device (e.g., /dev/sdb, /dev/nvme0n1).

Partition Filesystem Size Description
/dev/sda1 FAT32 512 MiB boot partition
/dev/sda2 none remaining space LUKS container
2.3.1.2. Create Partitions

Use parted(8) to partition the disk:

# parted /dev/sda
(parted) mklabel gpt
(parted) mkpart ESP fat32 1MiB 513MiB
(parted) set 1 boot on
(parted) name 1 efiboot
(parted) mkpart primary 513MiB 100%
(parted) name 2 luks
(parted) quit

Encrypt /dev/sda2 with LUKS and map it:

# cryptsetup luksFormat /dev/sda2
# cryptsetup luksOpen /dev/sda2 crypt
2.3.1.3. Create LVM Inside LUKS

Create a physical volume and volume group (e.g., zpln):

# pvcreate /dev/mapper/crypt
# vgcreate zpln /dev/mapper/crypt

Logical volume layout:

Volume name Filesystem Size Description
swap none 2 x RAM swap area
root ext4 remaining space root filesystem

Check RAM size with # free -m. For example, if you have 4 GiB RAM, a swap of 8 GiB (2 x RAM) is recommended.

Create the logical volumes:

# lvcreate -L 8G -n swap zpln
# lvcreate -l 100%FREE -n root zpln
2.3.1.4. Create Filesystems and Swap

Format partitions and volumes:

# mkfs.vfat -F32 /dev/sda1
# mkfs.ext4 /dev/zpln/root
# mkswap /dev/zpln/swap
# swapon /dev/zpln/swap
2.3.1.5. Mount Partitions

The /mnt directory is used as the default mount point in this handbook.

Mount the root volume:

# mount /dev/zpln/root /mnt

Mount the boot partition:

# mkdir /mnt/boot
# mount /dev/sda1 /mnt/boot

2.4. Install Base System

Download, verify, and extract the rootfs tarball to set up your base system.

2.4.1. Download Rootfs Tarball

Download the tarball directly into the /mnt directory (the default mount point) to avoid using live media RAM.

Get the latest release from the Zeppe-Lin pkgsrc-core releases page or run these commands (replace v1.0 with the version you need):

# cd /mnt
# VERSION=v1.0
# URL=https://github.com/zeppe-lin/pkgsrc-core/releases/download/
# wget -c ${URL}/${VERSION}/rootfs-${VERSION}-x86_64.tar.xz{,.sig}

2.4.2. Verify Downloaded Tarball

The tarball is signed with GPG. For security, verifying its integrity is highly recommended.

Import the public key:

# gpg --keyserver keyserver.ubuntu.com --recv-keys 59ec1986fbd902cf

Verify the signature:

# gpg --verify rootfs-${VERSION}-x86_64.tar.xz{.sig,}

2.4.3. Extract Rootfs Tarball

Once the rootfs tarball is downloaded and verified, extract its contents using the following command. Make sure to replace ${VERSION} with the version you downloaded:

# tar --numeric-owner --xattrs --xattrs-include='*' -xpf \
    rootfs-${VERSION}-x86_64.tar.xz

2.5. Chroot Into Base System

Mount necessary pseudo-filesystems:

# mount -B /dev /mnt/dev
# mount -B /tmp /mnt/tmp
# mount -B /run /mnt/run
# mount -t proc proc /mnt/proc
# mount -t sysfs none /mnt/sys
# mount -t devpts -o noexec,nosuid,gid=tty,mode=0620 devpts \
    /mnt/dev/pts

For UEFI systems, mount EFI variables:

# mount -B /sys/firmware/efi/efivars /mnt/sys/firmware/efi/efivars

Enter the chroot:

# chroot /mnt /bin/bash

Set the SHELL variable to /bin/bash to ensure tools like vim work properly:

(chrooted) # export SHELL=/bin/bash

2.6. Configure Base System

Set the root password:

(chrooted) # passwd root

Ensure proper ownership and permissions:

(chrooted) # chown root:root /
(chrooted) # chmod 755 /

Generate necessary locales (e.g., en_US):

(chrooted) # localedef -i en_US -f UTF-8 en_US.UTF-8

Set the system default locale by adding it to /etc/profile:

export LANG=en_US.UTF-8

See 6.1. Generating Locales for details.

Edit /etc/fstab to configure filesystems (see fstab(5)). Example for 2.3.1. UEFI and LVM on LUKS:

/dev/zpln/root  /      ext4  defaults,noatime,nodiratime  1 2
/dev/sda1       /boot  vfat  defaults,noatime,nodiratime  1 2
/dev/zpln/swap  swap   swap  defaults                     0 0

Edit /etc/rc.conf (see rc.conf(5)) to configure system settings like console font, keyboard, timezone, hostname, and services.

Configure network settings in /etc/rc.d/net, /etc/hosts, and /etc/resolv.conf (see 6.3. Networking).

2.6.1 Mount Additional Filesystems

Before building and installing additional software, mount additional pseudo-filesystems.

Shared memory (needed by some builds, e.g., python3):

(chrooted) # mount /dev/shm

Optional package build directory (RAM builds):

(chrooted) # mount /var/cache/pkgmk/work

2.6.2. Configure Users and Privileges

Add a regular user (set a specific UID now if needed):

(chrooted) # useradd --shell /bin/bash --create-home  \
    --groups audio,video,scanner,cdrom,input,users    \
    --uid 1000 --user-group <USERNAME>

Set the password:

(chrooted) # passwd <USERNAME>

To grant administrative privileges (e.g., as Ubuntu does), first add the user to the wheel group:

(chrooted) # usermod -aG wheel <USERNAME>

And second, grant the users in the wheel group to be root via sudo(8) by creating the file /etc/sudoers.d/00_wheel with the following content:

%wheel ALL=(ALL:ALL) ALL

2.7. Update the Base System

Before building and installing additional packages, it is highly recommended to update the freshly installed Zeppe-Lin system.

2.7.1. Prepare Pkgsrc Collections

Packages' sources are organized into collections (see 5.2.2. Collections for details).

Clone the necessary collections, keeping in mind that each depends on the previous one (replace 1.x with the current release branch):

(chrooted) # cd /usr/src/
(chrooted) # URL=https://github.com/zeppe-lin
(chrooted) # git clone ${URL}/pkgsrc-core    --branch 1.x
(chrooted) # git clone ${URL}/pkgsrc-system  --branch 1.x
(chrooted) # git clone ${URL}/pkgsrc-xorg    --branch 1.x
(chrooted) # git clone ${URL}/pkgsrc-desktop --branch 1.x

Enable the cloned collections in /etc/pkgman.conf. By default, only pkgsrc-core collection is enabled.

2.7.2. Perform the System Update

Perform a full system update with dependency handling and sorting, and stop if the installation of at least one package fails:

(chrooted) # pkgman sysup --deps --depsort --group

Merge any rejected files during update:

(chrooted) # rejmerge

Check for broken packages and rebuild as needed:

(chrooted) # pkgman update -fr --depsort $(revdep)

Here, revdep(1) identifies packages with broken dependencies, and if any are reported, pkgman update -fr rebuilds the affected packages.

2.8. Install Essential Packages

Install packages required for a minimal, functional workstation:

(chrooted) # pkgman install --deps --group \
    cryptsetup e2fsprogs dosfstools grub2 grub2-efi iw gnupg \
    lvm2 pinentry wireless-tools wpa-supplicant dhcpcd iputils

This is a generic setup; users can add any additional packages they require based on their specific needs.

2.9. Kernel Setup

Two methods are available: build and install a package or compile manually.

2.9.1. Kernel Package

Build and install the packaged Linux kernel:

(chrooted) # pkgman install --deps --group \
    --config-append="runscripts no" linux

The linux package includes a post-install script that runs mkinitramfs(8) and updates the GRUB config. This script is disabled here through --config-append="runscripts no" to run manually later.

If you installed the kernel using this method, proceed to 2.10. Kernel Firmware if your system requires additional firmware, or 2.11. Initramfs otherwise.

2.9.2. Manual Kernel Compilation

This method suits those wanting a minimal kernel or needing sources for driver building (e.g., Nvidia, VirtualBox).

Download sources:

(chrooted) # pkgman install -do linux
(chrooted) # KV=$(pkgman printf %v --filter=linux)
(chrooted) # tar -xvf \
    /var/cache/pkgmk/sources/linux-${KV}.tar.?z -C /usr/src/

Apply Zeppe-Lin patches:

(chrooted) # cd /usr/src/linux-${KV}
(chrooted) # for p in $(pkgman path linux)/*.patch; do \
    [ -f "$p" ] && patch -Np1 -i $p; done

Configure kernel (choose one of the following):

Build and install:

(chrooted) # make -j$(nproc) all
(chrooted) # cp arch/x86/boot/bzImage /boot/vmlinuz-${KV}
(chrooted) # cp .config /boot/config-${KV}
(chrooted) # make modules_install

Next steps:

2.10. Kernel Firmware

Some hardware (like certain Wi-Fi or graphics cards) needs extra firmware.

Install non-free firmware if required:

(chrooted) # pkgman install --deps --group linux-firmware

Skip this if you prefer a completely free software system.

2.11. Initramfs

First, install mkinitramfs:

(chrooted) # pkgman install --deps --group mkinitramfs

The /etc/mkinitramfs/config file controls how mkinitramfs(8) builds the initial ramdisk image used at boot. It controls which modules, hooks, and settings are included in the initramfs image.

Here's an example for 2.3.1. UEFI and LVM on LUKS:

# /etc/mkinitramfs/config
hostonly=1 #(optional, creates smaller initramfs)
compress="gzip --fast"
hooks="eudev luks lvm resume"
root=/dev/zpln/root
root_type=ext4
resume=/dev/zpln/swap
luks_name=crypt
luks_root=/dev/sda2
# End of file.

See mkinitramfs.config(5) for more information.

Now, prepare an initramfs. If you installed the Linux kernel manually, you have already set KV variable to kernel version. Otherwise, fetch the kernel version from the package source:

(chrooted) # KV=$(pkgman printf %v --filter=linux)

Finally, generate the initramfs image:

(chrooted) # mkinitramfs -o /boot/initramfs-${KV}.img -k ${KV}

2.12. Bootloader

This section covers installing GRUB2 as your bootloader. Support for other bootloaders may be added later.

2.12.1. GRUB2

Create /etc/default/grub with:

# Set the delay before booting:
GRUB_TIMEOUT=3
# Show ZPLN in the GRUB menu:
GRUB_DISTRIBUTOR=ZPLN
# Set kernel parameters (quiet boot, swap for hibernation):
GRUB_CMDLINE_LINUX_DEFAULT="quiet resume=/dev/zpln/swap"

Next, install GRUB2 on your target drive. Example device: /dev/sda. Replace with your actual device (e.g., /dev/sdb, /dev/nvme0n1).

For UEFI, ensure the EFI System Partition (ESP) is mounted at /boot/efi inside the chroot. Then install GRUB with EFI support:

(chrooted) # grub-install --target=x86_64-efi \
    --efi-directory=/boot /dev/sda

For legacy BIOS booting, install GRUB directly to the disk's MBR:

(chrooted) # grub-install --target=i386-pc /dev/sda

Finally, generate the GRUB2 configuration file based on /etc/default/grub and detected kernels:

(chrooted) # grub-mkconfig -o /boot/grub/grub.cfg

2.13. Post-installation Tasks

2.13.1. Install Xorg and Drivers

To find video and input drivers, run:

(chrooted) # pkgman search -vv xf86-

Common video drivers include xorg-xf86-video-intel (Intel), xorg-xf86-video-amdgpu/xorg-xf86-video-ati (AMD), and xorg-xf86-video-nouveau (NVIDIA). For NVIDIA, the nouveau-firmware is also recommended.

For input, modern drivers like xorg-xf86-input-libinput and xorg-xf86-input-evdev are recommended. Legacy options like xorg-xf86-input-keyboard and xorg-xf86-input-mouse are also available.

After selecting drivers for your hardware, install xorg meta-package and additional drivers with:

(chrooted) # pkgman install --deps --group xorg [DRIVERS]

Replace [DRIVERS] with the ones matching your system.

2.13.2. Install a Window Manager

Zeppe-Lin focuses on lightweight and efficient window managers rather than complex desktop environments like GNOME or KDE. This approach avoids unnecessary dependencies and simplifies configuration.

Zeppe-Lin provides package sources for several window managers, but users are welcome to choose and contribute package sources for any window manager they prefer.

2.13.2.1. Ratpoison

Currently, the pkgsrc-desktop repository provides the ratpoison window manager. You can install it with:

(chrooted) # pkgman install --deps --group ratpoison
2.13.2.2. Unofficial: Window Maker

For users interested in a retro, intuitive window manager, you can install Window Maker from the unofficial pkgsrc-wmaker repository.

To install Window Maker, first clone the repository:

(chrooted) # cd /usr/src
(chrooted) # URL=https://github.com/zeppe-lin
(chrooted) # git clone ${URL}/pkgsrc-wmaker --branch 1.x

Next, you need to tell pkgman(1) about this new repository by adding the following line to /etc/pkgman.conf:

pkgsrcdir /usr/src/pkgsrc-wmaker

Then, install it with:

(chrooted) # pkgman install --deps --group wmaker

Optional meta-packages for Window Maker include wmaker-dockapps for added functionality and wmaker-themes for customization.

2.14. Reboot

Exit the chroot, unmount all filesystems, and reboot:

(chrooted) # exit
# cd /
# umount -R /mnt
# shutdown -r now

The GRUB menu should appear, allowing you to boot into Zeppe-Lin.

3. UPGRADING

WARNING! THIS SECTION IS UNDER DEVELOPMENT AND CANNOT BE USED AS A MANUAL!

Upgrading Zeppe-Lin requires careful planning. This section outlines methods for updating your system. Follow the steps as presented to ensure a reliable upgrade.

Upgrade methods:

  1. Quick binary upgrade (core components).
  2. Full update from sources.

Before starting, read the release notes for the target version and back up important data.

3.1. Release Notes

Read the release notes for the new Zeppe-Lin version. They contain essential details about new features, removed packages, and compatibility issues. The latest release notes you can obtain here.

3.2. Set Release And Sync

Synchronize your local pkgsrc collections to match the new release branch. Replace 1.x with the specific release you are upgrading to:

git -C /usr/src/pkgsrc-core    switch 1.x
git -C /usr/src/pkgsrc-system  switch 1.x
git -C /usr/src/pkgsrc-xorg    switch 1.x
git -C /usr/src/pkgsrc-desktop switch 1.x

3.3. Binary Upgrade

Use precompiled binary packages for faster core system upgrades, especially to avoid build issues during initial updates.

3.3.1. Download and Extract binpkgs

Download the tarball and its signature for your upgrade, replacing v1.0 with the target version:

$ VERSION=v1.0
$ URL=https://github.com/zeppe-lin/pkgsrc-core/releases/download
$ wget -c ${URL}/${VERSION}/binpkgs-${VERSION}-x86_64.tar.xz{,.sig}

Verify the tarball with gpg(1):

$ gpg --keyserver keyserver.ubuntu.com --recv-keys 59ec1986fbd902cf
$ gpg --verify binpkgs-${VERSION}-x86_64.tar.xz{.sig,}

Extract the tarball after successful verification:

$ mkdir -p ~/tmp/binpkgs
$ tar -xf binpkgs-${VERSION}-x86_64.tar.xz -C ~/tmp/binpkgs

3.3.2. Upgrade Core Packages

Navigate to the directory and upgrade core packages:

$ cd ~/tmp/binpkgs
$ for pkg in *.pkg.tar.gz; do sudo pkgadd -u $pkg; done

pkgadd -u upgrades each package. New, non-upgradable packages will return an error, which the loop skips, ensuring all upgradeable packages are updated.

3.3.3. Install New Core Packages

Install new core packages introduced in the release:

$ cd ~/tmp/binpkgs
$ for pkg in *.pkg.tar.gz; do sudo pkgadd $pkg; done

pkgadd installs each package. Already installed packages return an error, which the loop skips to ensure all new packages are installed.

4. PACKAGE MANAGEMENT

4.1. Introduction

Zeppe-Lin's package management system revolves around packages and package sources. Low-level tools handle atomic operations like building and installation, working with individual package sources or packages as instructed. The high-level tool pkgman manages the package source collections and orchestrates these low-level tasks for system-wide software management.

To work with packages and package sources, Zeppe-Lin provides:

4.2. What is a Package?

A Zeppe-Lin package is a compressed archive (e.g., tar.gz) that contains only the software files. It avoids any kind of embedded metadata, ensuring simplicity and transparency in its structure.

4.2.1. Package Naming

Zeppe-Lin packages follow a clear naming convention:

name#version-release.pkg.tar.gz

Components:

Example:

ed#1.21-1.pkg.tar.gz

The extension .pkg.tar.gz denotes a Zeppe-Lin package. Other common compression formats like tar.bz2, tar.xz, and tar.zst are also supported.

4.2.2. Package Database

The package database, located at /var/lib/pkg/db, tracks installed packages and their contents. Each package entry follows this format:

<name>                 # Package name
<version>-<release>    # Version and release
<dir>/                 # Top-level directory
<dir>/<subdir>         # Subdirectory
<dir>/<subdir>/<file>  # File within subdirectory
(Blank line)           # Separates package entries

Example entry for ed:

ed
1.21-1
bin/
bin/ed
usr/
usr/share/
usr/share/man/
usr/share/man/man1/
usr/share/man/man1/ed.1.gz
(Blank line)

4.3. Low-level Tools: pkgutils

pkgutils provides the core utilities to handle packages:

4.3.1. Using pkgutils

4.3.1.1. Installing a Package

Install a package file with pkgadd(8):

# pkgadd bash#5.0.18-1.pkg.tar.gz

By default, pkgadd(8) prevents overwriting existing files. Use -f or --force to override this behavior (use with caution!):

# pkgadd -f bash#5.0.18-1.pkg.tar.gz
4.3.1.2. Upgrading a Package

Upgrade a package with the -u or --upgrade option:

# pkgadd -u bash#5.0.18-1.pkg.tar.gz

This replaces the installed package. If the package is not installed, it will return an error. You can even "upgrade" to an older version.

Files that should not be upgraded (based on /etc/pkgadd.conf) are placed in /var/lib/pkg/rejected/.

4.3.1.3. Removing a Package

To remove a package, use pkgrm(8) followed by the package name:

# pkgrm bash
4.3.1.4. Inspecting Packages

Use pkginfo(1) to get information about packages.

List installed packages and versions:

$ pkginfo -i

List files in a package:

$ pkginfo -l bash

List files in a package archive (before installing):

$ pkginfo -l grep#2.5-1.pkg.tar.gz

Find which package owns a file:

$ pkginfo -o /usr/bin/ls

See pkgadd(8), pkgrm(8), and pkginfo(1) for details.

4.3.2. Configuring pkgadd

You can customize pkgadd(8) behavior during package upgrades using /etc/pkgadd.conf. Rules define whether files should be upgraded or preserved.

4.3.2.1. Rule Format

Each rule in /etc/pkgadd.conf follows this format:

EVENT   PATTERN   ACTION

Rules are processed sequentially, with the last matching rule taking priority.

4.3.2.2. Example

Example /etc/pkgadd.conf entries:

UPGRADE   ^etc/.*$              NO
UPGRADE   ^var/log/.*$          NO
UPGRADE   ^etc/X11/xorg.conf$   YES

These rules will:

See pkgadd.conf(5) for details.

4.4. Low-level Tools: rejmerge

rejmerge(8) resolves files rejected during upgrades using pkgadd -u. Rejected files are moved to /var/lib/pkg/rejected/ for manual handling, while rejmerge(8) provides a way to manage these files semi-automatically.

4.4.1. Using rejmerge

If files are present in /var/lib/pkg/rejected, rejmerge(8) prompts you to decide their fate. For each file, you can choose:

=======> file
(diff output between installed file and rejected file)
[K]eep [U]pgrade [M]erge [D]iff [S]kip

Legend:

See rejmerge(8) for details.

4.4.2. Configuring rejmerge

Configurations for rejmerge(8) can be adjusted in /etc/rejmerge.conf. Key settings:

4.4.2.1. Example

Use unified, colorized output for differences:

rejmerge_diff() { diff -u --color=always $1 $2 > $3 ; }

Use sdiff(1) for merging files:

rejmerge_merge() { sdiff -o $3 $1 $2 ; }

See rejmerge.conf(5) for more details.

4.5. Low-level Tools: pkgmk

pkgmk(8) builds Zeppe-Lin packages (compressed archives) from source files using a Pkgfile. These files, along with the supporting directory, form a Package Source -- a self-contained unit that includes all files needed to build and install software.

A Package Source typically contains:

A Package Source is the basic unit of software organization in Zeppe-Lin. While this section introduces Package Sources in the context of building packages, 5. PACKAGE SOURCES AND COLLECTIONS explores their organization into Collections and Repositories for broader software management workflows.

4.5.1. Pkgfile Format

The Pkgfile defines:

4.5.1.1. Example
name=hello
version=2.12.1
release=1
source=https://ftp.gnu.org/gnu/hello/$name-$version.tar.gz
    
build() {
    cd $name-$version
    ./configure --prefix=/usr --disable-nls
    make
    make DESTDIR=$PKG install
    rm -rf $PKG/usr/info
}

See pkgmk.Pkgfile(5) for details.

4.5.2. Using pkgmk

4.5.2.1. Building a Package

To build a package, create a directory named after the package, cd into this directory, place the Pkgfile inside, and run:

$ fakeroot pkgmk -d -cf /dev/null

On the first build, pkgmk(8) generates:

The final package (e.g., hello#2.12.1-1.pkg.tar.gz) will be in the current directory.

4.5.3. Configuring pkgmk

Customize pkgmk(8) with /etc/pkgmk.conf. Key settings:

4.5.3.1. Example

Define backup URLs:

PKGMK_SOURCE_MIRRORS="http://fileserver.intranet/sources/
                      http://mirror2/sources/"

Set a unique work directory:

PKGMK_WORK_DIR="/var/cache/pkgmk/work/$name-$$"

Using $name (package name) and $$ (process ID) avoids conflicts during simultaneous builds.

See pkgmk.conf(5) for full options.

4.6. High-level Tool: pkgman

pkgman(1) serves as a versatile interface for managing Zeppe-Lin's package ecosystem. Building on tools like pkgutils and pkgmk, it unifies and simplifies workflows while supporting a wide range of package management tasks.

pkgman(1) empowers users to:

With its adaptable design, pkgman(1) integrates seamlessly into diverse system management scenarios, making it an indispensable tool for maintaining and customizing Zeppe-Lin.

4.6.1. Using pkgman

4.6.1.1. List Installed Packages

List installed packages:

$ pkgman list

Include versions:

$ pkgman list -v

Include versions and descriptions:

$ pkgman list -vv
4.6.1.2. Query Package Information

Display detailed package information:

$ pkgman info acl

Example output:

Name:         acl
Path:         /usr/src/pkgsrc-core
Version:      2.3.1
Release:      1
Description:  Access Control List filesystem support
URL:          http://savannah.nongnu.org/projects/acl
Dependencies: attr
4.6.1.3. Search Package Sources

Search by name:

$ pkgman search -vv glibc

Search using regular expressions:

$ pkgman search -vv --regex '^(glib)c?$'

Search by description:

$ pkgman dsearch -vv archive
4.6.1.4. Query Dependencies

Show direct dependencies:

$ pkgman dep vim

Show recursive dependencies:

$ pkgman dep vim --recursive

Show detailed dependency tree:

$ pkgman dep vim --recursive --tree
4.6.1.5. Build and Install Packages

Build and install a package with dependencies (recommended):

# pkgman install --deps --group xterm

Use --group to stop installation if a dependency fails. Retry failed installations:

# pkgman install --deps --group --force xterm

--force skips already installed packages during the retry.

4.6.1.6. Update Installed Packages

Identify outdated packages:

$ pkgman diff --deps --full

Update a single package:

# pkgman update --deps --group bind

Upgrade all packages in the system:

# pkgman sysup --deps --depsort --group

4.6.2. Configuring pkgman

pkgman(1) is configured via /etc/pkgman.conf. Key settings:

See pkgman.conf(5) for more details.

4.7. The Essentials (outro)

This chapter introduced Zeppe-Lin's tools for managing individual packages and basic system updates.

In the next chapter, we introduce Collections of Package Sources -- a modular approach to organizing and managing software -- with pkgman serving as the central orchestration tool.

5. PACKAGE SOURCES AND COLLECTIONS

5.1. Introduction

Zeppe-Lin organizes software sources into three key components:

These components provide a structured approach to building, managing, and distributing software on Zeppe-Lin. Tools like pkgmk(8) and pkgman(1) are used to interact with these components, supporting workflows such as dependency management, package building, and system updates.

5.2. Organizing Package Sources

5.2.1. Package Source: The Core Unit

In Zeppe-Lin, a Package Source is the fundamental unit for building software. Each one represents a single application, library, or utility, organized as a directory with all the necessary files to build and install the software.

Essential files include:

Some Package Sources may include additional files to support custom builds or configurations:

Together, these files create a self-contained environment for building and managing software.

5.2.2. Collections

Collections are logical groupings of Package Sources. A Collection is simply a directory containing multiple Package Sources with a shared purpose, like system utilities or desktop applications.

Example Collection structure:

Collection (e.g., pkgsrc-core)
|
|-- acl/ (Package Source)
|   |-- Pkgfile
|   |-- .md5sum
|   `-- .footprint
|
|-- bash/ (Package Source)
|   |-- Pkgfile
|   |-- .md5sum
|   `-- .footprint
|
`-- ... (more Package Sources)

This logical organization enables modular software management. Instead of dealing with individual packages in isolation, Collections provide a structured approach to grouping and managing related sets of software.

Common Collections in Zeppe-Lin:

These Collections form the foundation of Zeppe-Lin's software ecosystem, allowing users to selectively enable the software they need.

5.2.3. Repositories: Distributing and Versioning Collections

Collections are organized into Repositories, hosted on systems like Git. It's important to note that while a repository can contain multiple collections, Zeppe-Lin's official repositories adopt a strategy where each repository primarily focuses on a single collection. This design choice is driven by the benefits of allowing users to selectively clone only the collections they need. Repositories provide:

The official Zeppe-Lin repositories, following this one-collection-per-repository approach, include:

Repositories ensure that collections are both modular and manageable, allowing users to selectively clone only the collections they need.

5.3. Managing Package Sources

5.3.1. Setting Up Collections

To access software for installation in Zeppe-Lin, download the official package source collections and enable them for pkgman(1).

5.3.1.1. Syncing Official Repositories

Clone the repositories from the Zeppe-Lin GitHub account:

# cd /usr/src/
# URL=https://github.com/zeppe-lin
# git clone ${URL}/pkgsrc-core    --branch 1.x
# git clone ${URL}/pkgsrc-system  --branch 1.x
# git clone ${URL}/pkgsrc-xorg    --branch 1.x
# git clone ${URL}/pkgsrc-desktop --branch 1.x

Replace 1.x with the branch matching your Zeppe-Lin release (e.g., 1.x for version 1.0 or 1.1). Check the Zeppe-Lin website or release notes for the correct branch.

5.3.1.2. Enabling Collections in pkgman.conf

Edit /etc/pkgman.conf to enable package sources collections. For example:

pkgsrcdir /usr/src/pkgsrc-core
pkgsrcdir /usr/src/pkgsrc-system
pkgsrcdir /usr/src/pkgsrc-xorg
pkgsrcdir /usr/src/pkgsrc-desktop

For detailed configuration options, see pkgman.conf(5).

5.3.1.3. Verify Your Setup

Confirm that pkgman(1) recognizes the collections:

$ pkgman list --all

This lists all package sources from enabled collections.

5.3.2. Exploring and Installing Packages

With collections enabled, you can explore and install packages using pkgman(1).

5.3.2.1. Listing Available Packages

List all package sources:

$ pkgman list --all

Example output:

-- list ([i] = installed)
[i] acl
[i] bash
[ ] vim
[ ] zathura
...

Legend:

To display the full path of each package source (helpful for identifying which collection a package belongs to), use the --path option:

$ pkgman list --all --path

Example output:

[i] /usr/src/pkgsrc-core/acl
[ ] /usr/src/pkgsrc-system/vim
5.3.2.2. Searching for Packages

If you're looking for specific software, use the search command. For example, to search for packages related to alsa:

$ pkgman search alsa

Example output:

-- search ([i] = installed)
[ ] alsa-lib
[ ] alsa-ucm-conf
[i] alsa-utils

The output indicates which packages are installed and available.

For a detailed search that includes collection paths, add the --path option:

$ pkgman search --path alsa
5.3.2.3. Building and Installing Packages

Build and install a package with dependencies:

# pkgman install --deps --group <package_name>

Example:

# pkgman install --deps --group vim

Build and install multiple packages:

# pkgman install --deps --group vim zathura

For verbose output (show building process on stdout):

# pkgman install -vv --deps --group vim zathura

Dry run:

# pkgman install -vv --test --deps --group vim zathura

Run pkgman install --help or see pkgman-install(8) for information about all installation options.

5.3.3. Keeping Your System Up-to-Date

Maintaining your Zeppe-Lin system involves two primary tasks:

  1. Sync your Repositories regularly to stay up-to-date.
  2. Updating installed packages to their latest versions.
5.3.3.1. Synchronizing Local Collections

Regularly sync local collections with remote repositories using git pull:

# git -C /usr/src/pkgsrc-core     pull
# git -C /usr/src/pkgsrc-system   pull
# git -C /usr/src/pkgsrc-xorg     pull
# git -C /usr/src/pkgsrc-desktop  pull

Include unofficial collections if applicable.

Automate Syncing: Set up a cron job with crond(8):

# Weekly sync at 3:00 AM on Sundays
0 3 * * 0 /root/pkgsrc-sync.sh

Example script (/root/pkgsrc-sync.sh):

#!/bin/sh
LOCKFILE="/var/lock/pkgsrc-sync.lock"
exec 9>"$LOCKFILE"
if ! flock -n 9; then
    echo "Another sync in progress."
    exit 1
fi
git -C /usr/src/pkgsrc-core     pull
git -C /usr/src/pkgsrc-system   pull
git -C /usr/src/pkgsrc-xorg     pull
git -C /usr/src/pkgsrc-desktop  pull
echo "Repositories synced."

For logging, replace echo with logger.

5.3.3.2. Checking for Package Updates

Identify outdated packages with pkgman diff:

$ pkgman diff --deps --full

Example output:

-- Differences between installed packages and packages sources tree
Package         Installed      Available
feh             3.7.2-1        3.7.2-2
screen          4.9.0-1        5.0.0-1
...

-- Packages not found in the packages sources tree
Package         Installed      Required by
libncurses      6.2-1          vim

Review this output to determine which packages need updates. Packages in the "not found" section may have been removed from the collection or installed manually from other sources.

5.3.3.3. Updating Packages

Use pkgman sysup for a system upgrade.

Step 1: Download all required source files first:

# pkgman sysup -do

This ensures that all necessary files are retrieved before the upgrade begins, avoiding interruptions during the build process.

Step 2: Upgrade Packages:

# pkgman sysup --deps --depsort --group

These options ensure proper dependency handling and upgrade order.

5.3.3.4. Updating Specific Packages

Update individual packages with pkgman update. For example:

# pkgman update --deps --depsort --group screen feh

This command updates only the specified packages while resolving their dependencies.

5.3.4. Advanced Package Management

5.3.4.1. Detecting Broken Dependencies with revdep(1)

revdep(1) checks installed packages for missing or incorrect shared libraries. Use it to verify system stability and identify packages needing a rebuild after updates.

Examples:

  1. Check system-wide for broken libraries:

    $ revdep
    
  2. Rebuild affected packages:

    # pkgman update -fr $(revdep)
    
  3. Inspect a specific package:

    $ revdep -p bash
    

Run revdep regularly after updates or repository syncs to ensure all packages remain functional.

See revdep(1) and revdep.d(5) for more information.

5.3.4.2. Ignoring Packages During Updates

Prevent updates to specific packages during system upgrades with --ignore. Example:

# pkgman sysup --deps --depsort --group --ignore=vim,screen

Regularly revisit ignored packages to avoid security risks and dependency conflicts.

5.3.4.3. Locking Packages

Lock a package to prevent upgrades indefinitely:

# pkgman lock vim

Unlock a package to resume updates:

# pkgman unlock vim

Locked packages are useful for maintaining specific software versions critical to workflows.

5.3.4.4. Handling Lower Available Versions

Occasionally, a package source may revert to a lower version than the one installed on your system. This often happens when maintainers roll back to a stable release after issues with a newer version. By default, pkgman does not prioritize installed versions over available ones.

To override this behavior, enable the preferhigher option:

# pkgman --config-set="preferhigher yes" sysup --deps --depsort

This ensures pkgman retains the higher installed version during upgrades. Use this feature cautiously, as higher versions might be less stable or unsupported.

5.3.4.5. Practical Considerations

5.4. Extending Software Sources

5.4.1. Adding a Local Collection

This section explains how to create custom collections. A local collection allows you to add new packages that are not available in the official repositories.

5.4.1.1. Creating the Collection Directory

Create your local collection directory in /usr/src, alongside official Zeppe-Lin collections. Using directories with broader read permissions ensures that unprivileged users like pkgmk can access the collection.

Example:

# cd /usr/src
# mkdir mynewcollection
5.4.1.2. Enabling the Collection in pkgman.conf

Add your collection to /etc/pkgman.conf. List it above official collections to give it search priority:

pkgsrcdir /usr/src/mynewcollection
pkgsrcdir /usr/src/pkgsrc-core
pkgsrcdir /usr/src/pkgsrc-system
pkgsrcdir /usr/src/pkgsrc-xorg

Save the file and proceed to verify the setup.

5.4.1.3. Verifying the Collection Setup

Verify that pkgman(1) recognizes your collection:

$ pkgman dumpconfig

Check the "Packages sources directories" section for your collection:

Packages sources directories: <n>
  /usr/src/mynewcollection <-- here
  /usr/src/pkgsrc-core
  /usr/src/pkgsrc-system
  /usr/src/pkgsrc-xorg
  ...

If it appears, the setup is complete, and you can start adding package sources.

5.4.2. Creating Custom Packages

After setting up your local collection, the next step is to add a package source. This involves creating a directory for the package and writing a Pkgfile with build instructions and metadata.

5.4.2.1. Creating the Package Directory

Create a subdirectory for the new package in your local collection. Ensure the directory name matches the package name in the Pkgfile:

# mkdir /usr/src/mynewcollection/hello
# cd /usr/src/mynewcollection/hello

Important: Naming consistency is critical. A mismatch between the directory name and name in the Pkgfile cause build error.

5.4.2.2. Writing the Pkgfile

The Pkgfile serves two purposes:

  1. Provides build instructions for pkgmk(8).
  2. Supplies metadata for pkgman(1) to enable dependency resolution, search, and queries.

Example Pkgfile:

# Description: Example package based on GNU hello
# URL:         https://www.gnu.org/software/hello/
# Depends on:  glibc
  
name=hello
version=2.12.1
release=1
source=https://ftp.gnu.org/gnu/hello/$name-$version.tar.gz
  
build() {
    cd $name-$version
  
    ./configure --prefix=/usr --disable-nls
  
    make V=1
    make DESTDIR=$PKG install
  
    rm -rf $PKG/usr/share/info
}
5.4.2.3. Key Metadata Headers for pkgman

Include these headers in the Pkgfile for optimal integration with pkgman(1):

Without these headers, pkgman(1) cannot display metadata or resolve dependencies.

5.4.2.4. Verifying the Package Source

Verify your package metadata with:

$ pkgman info hello

Expected output:

Name:         hello
Path:         /usr/src/mynewcollection/hello
Version:      2.12.1
Release:      1
Description:  Example package based on GNU hello
URL:          https://www.gnu.org/software/hello/
Depends on:   glibc

If metadata is missing, review the Pkgfile for errors.

5.4.3. Building and Installing Custom Packages

After creating a custom package source, the next step is to build and install the package using pkgman(1) and pkgmk(8), addressing any potential permission issues.

5.4.3.1. Building the Package

Build your custom package with:

# pkgman install -vv --deps hello

This command:

  1. Downloads source files if not cached.
  2. Builds the package using pkgmk(8).
  3. Installs the package with pkgadd(8).
5.4.3.2. Resolving Permissions for Build Files

There are at least two options to resolve permissions case.

  1. Manually Updating Files.

    Navigate to the package directory and generate the files:

    # cd /usr/src/mynewcollection/hello
    # pkgmk -um   # Update .md5sum
    # pkgmk -uf   # Update .footprint
    
  2. Adjusting Permissions.

    Grant write access to pkgmk using setfacl(1):

    # setfacl -m u:pkgmk:rwx /usr/src/mynewcollection/hello
    

    Rebuild the package afterward:

    # pkgman update -fr hello
    
5.4.3.3. Verifying Generated Files

After addressing warnings, confirm the presence of required files:

$ ls -l /usr/src/mynewcollection/hello
-rw-r--r--  1 pkgmk pkgmk  251 .footprint
-rw-r--r--  1 pkgmk pkgmk   54 .md5sum
-rw-r--r--  1 root  root   359 Pkgfile
5.4.3.4. Verifying the Installation

Ensure the package is installed on your system:

$ pkgman isinst hello

Expected output:

package 'hello' is installed

Run:

$ hello

Expected output:

Hello, world!

5.4.4. Building Packages from Bleeding-Edge Sources

Some packages need to be built directly from version control systems (VCS) like Git or Subversion, instead of fixed release archives. This section outlines two approaches: a stable approach and a bleeding-edge approach.

5.4.4.1. Stable Commit Approach

For production, pinning to a specific commit ensures consistent builds. Example:

# Description: Ratpoison window manager
# URL:         https://www.nongnu.org/ratpoison/
# Depends on:  xorg-server
  
name=ratpoison
version=1.4.9-db94d49  # Release + commit hash
release=1
  
# Custom function to fetch or clone git repository
download_git() {
    if [ -d $PKGMK_SOURCE_DIR/$name ]; then
        git -C $PKGMK_SOURCE_DIR/$name fetch --depth=1
        git -C $PKGMK_SOURCE_DIR/$name clean -f
        git -C $PKGMK_SOURCE_DIR/$name reset --hard origin/$2
    else
        git -C $PKGMK_SOURCE_DIR clone --depth=1 $1 -b $2 $name
    fi
    cp -r $PKGMK_SOURCE_DIR/$name $PKGMK_WORK_DIR/src/$name
}
  
build() {
    download_git https://git.savannah.nongnu.org/git/ratpoison.git master
    cd ratpoison
    git checkout ${version#*-}  # Pin commit 'db94d49'
      
    ./configure --prefix=/usr
    make V=1
    make DESTDIR=$PKG install
}

Pinning builds guarantees stability and shields against upstream changes.

5.4.4.2. Bleeding-Edge Approach

For testing, dynamically set the version to the current date (YYYYMMDD) to track upstream changes. Example:

# Description: Ruby interpreter (bleeding-edge)
# URL:         http://www.ruby-lang.org
# Depends on:  gdbm libgmp openssl zlib libffi libyaml
  
name=ruby-scm
version=$(date +%Y%m%d)  # Version updates daily to track build date
release=1
  
download_git() {
    if [ -d $PKGMK_SOURCE_DIR/$name ]; then
        git -C $PKGMK_SOURCE_DIR/$name fetch --depth=1
        git -C $PKGMK_SOURCE_DIR/$name clean -f
        git -C $PKGMK_SOURCE_DIR/$name reset --hard origin/$2
    else
        git -C $PKGMK_SOURCE_DIR clone --depth=1 $1 -b $2 $name
    fi
    cp -r $PKGMK_SOURCE_DIR/$name $PKGMK_WORK_DIR/src/$name
}
  
build() {
    download_git https://github.com/ruby/ruby master
    cd ruby-scm
      
    ./configure --prefix=/usr
    make V=1
    make DESTDIR=$PKG install
}

"Daily Updates": If the installed version is outdated, pkgman(1) flags it during pkgman diff, prompting updates:

$ pkgman diff
  
-- Differences between installed packages and packages sources tree
ruby-scm                        20250216-1          20250401-1

This approach is ideal for development but unsuitable for production environments.

5.4.4.3. Subversion Repository Download Function

For Subversion-based builds, use a download function like this:

download_svn() {
    if [ -d $PKGMK_SOURCE_DIR/$name ]; then
        svn cleanup $PKGMK_SOURCE_DIR/$name
        svn up $PKGMK_SOURCE_DIR/$name
    else
        svn co $1 $PKGMK_SOURCE_DIR/$name
    fi
    cp -r $PKGMK_SOURCE_DIR/$name $PKGMK_WORK_DIR/src/$name
}

Example usage in a Pkgfile:

download_svn https://svn.project.org/svn/program/trunk

Add subversion to dependencies for compatibility. Unlike git, which resides in pkgsrc-core, subversion requires explicit declaration.

5.4.4.4. Best Practices

5.5. Tips and Best Practices

5.5.1. Renicing Builds

Package builds can reduce system responsiveness due to high CPU and I/O usage. Mitigate this by adjusting process priorities using nice(1) and ionice(1):

Update the makecommand in /etc/pkgman.conf:

makecommand sudo -H -u pkgmk nice -n10 ionice -c2 -n6 fakeroot pkgmk

Explanation:

These changes help the build process interfere less with interactive tasks, improving system responsiveness.

5.5.2. RAM Builds for Faster Package Compilation

Using RAM for temporary build files (tmpfs) can significantly speed up package compilation in Zeppe-Lin.

5.5.2.1. Enable RAM Builds in Zeppe-Lin
  1. Uncomment this line in /etc/fstab:

    pkgmk /var/cache/pkgmk/work tmpfs size=<SIZE>,uid=<UID>,defaults 0 0
    

    Replace SIZE (e.g., 16G). The UID for the pkgmk user is preconfigured.

  2. Mount tmpfs:

    # mount pkgmk
    
  3. Remove -pipe from CFLAGS and CXXFLAGS in /etc/pkgmk.conf.

5.5.2.2. RAM Build Configuration (Educational)

For users interested in customizing the setup:

  1. Creating a dedicated build user:

    # useradd -b /var/cache/ -m -s /bin/false -U pkgmk
    
  2. Setting up build directories and assigning ownership:

    # mkdir -p /var/cache/pkgmk/{sources,packages,work}
    # chown -R pkgmk:pkgmk /var/cache/pkgmk/{sources,packages,work}
    
  3. Configuring tmpfs in /etc/fstab:

    pkgmk /var/cache/pkgmk/work tmpfs size=<SIZE>,uid=<UID>,defaults 0 0
    
  4. Setting the work directory in /etc/pkgmk.conf:

    PKGMK_WORK_DIR="/var/cache/pkgmk/work/$name"
    
  5. Removing -pipe from CFLAGS and CXXFLAGS in /etc/pkgmk.conf.

5.5.3. Unprivileged Builds in Zeppe-Lin

Zeppe-Lin builds packages as an unprivileged user (typically pkgmk) by default. This secure and modular setup isolates the build process to mitigate risks. If you are using Zeppe-Lin, these steps are already configured out of the box. There is no need to repeat them.

However, this guide is provided for educational purposes, to show how it was achieved, or for users wanting to adapt this approach for other distributions or customizations.

  1. Create a Dedicated Build User. A restricted user is used to perform package builds securely:

    # useradd -b /var/cache/ -m -s /bin/false -U pkgmk
    

    This creates a user and group named pkgmk with limited shell access.

  2. Set Up Build Directories. Directories for sources, packages, and temporary files are created and assigned ownership to the pkgmk user:

    # mkdir -p /var/cache/pkgmk/{sources,packages,work}
    # chown -R pkgmk:pkgmk /var/cache/pkgmk/{sources,packages,work}
    
  3. Configure pkgmk. In /etc/pkgmk.conf, the paths for these directories are defined:

    PKGMK_SOURCE_DIR="/var/cache/pkgmk/sources"
    PKGMK_PACKAGE_DIR="/var/cache/pkgmk/packages"
    PKGMK_WORK_DIR="/var/cache/pkgmk/work/$name-$$"
    
  4. Integrate pkgman. In /etc/pkgman.conf, the makecommand variable is updated to run pkgmk(8) as the unprivileged pkgmk user via sudo(8) and fakeroot(1):

    makecommand sudo -H -u pkgmk fakeroot pkgmk
    

This structure ensures package builds are securely isolated while remaining flexible for adaptations.

6. CONFIGURATION

6.1. Generating Locales

In Zeppe-Lin, glibc package includes a minimal set of locales by default, with C.UTF-8 generated for compatibility with pkgmk(8). To create additional locales, replace en_US with your desired locale and run:

# localedef -i en_US -f UTF-8 en_US.UTF-8

Set the system default locale by adding it to /etc/profile:

export LANG=en_US.UTF-8

This ensures consistent locale settings across your system.

6.2. System Initialization and Configuration

Zeppe-Lin uses a lightweight BSD-style init system to manage startup, services, and shutdown. This system relies on shell scripts located in /etc and /etc/rc.d, offering an efficient and flexible way to control processes.

6.2.1. System States (Runlevels)

Runlevels define the operating state of the system, such as halting, rebooting, or normal operation. Zeppe-Lin's runlevels are configured in /etc/inittab:

Runlevel State Purpose
0 Halt Shut down the system completely
1 (S) Single-User Mode Maintenance mode with minimal setup
2 Multi-User Mode Full system operation with multiple users
3-5 Unused Reserved for custom use
6 Reboot Restart the system

By default, Zeppe-Lin boots into multi-user mode (runlevel 2), which enables all services and user access. Single-user mode (runlevel 1 or S) is ideal for system maintenance, providing a minimal environment.

6.2.2. Core Init Scripts

Zeppe-Lin's init system uses the following scripts to manage system processes:

File/Directory Purpose
/etc/rc Main system boot script, invoked by init(8)
/etc/rc.modules Kernel module initialization, invoked by /etc/rc
/etc/rc.single Single-user mode script for maintenance tasks
/etc/rc.multi Multi-user mode script for normal operation
/etc/rc.local Custom commands executed after /etc/rc.multi during normal boot
/etc/rc.d Directory containing scripts to control individual services
/etc/rc.shutdown Script invoked by shutdown(8) to gracefully stop services

Customize system behavior by modifying /etc/rc.local for personal commands or /etc/rc.modules to load specific kernel modules.

See rc(8) for details.

6.2.3. Configuring Runlevels and Init Behavior

The /etc/inittab file determines actions for each runlevel. Each entry follows this format: id:runlevel:action:command

Key Components:

Example /etc/inittab Configuration:

# Set default runlevel to multi-user mode
id:2:initdefault:

# Core startup scripts
rc::sysinit:/etc/rc         # Main boot script
rs:S1:wait:/etc/rc.single   # Single-user mode startup
rm:2:wait:/etc/rc.multi     # Multi-user mode startup
rd:06:wait:/etc/rc.shutdown # Graceful shutdown

# Virtual console login prompts
c1:2:respawn:/sbin/agetty --noclear 38400 tty1 linux
c2:2:respawn:/sbin/agetty 38400 tty2 linux

This configuration ensures a smooth startup, runtime, and shutdown process. See inittab(5) for configuration details.

6.2.4. Configuring System Behavior (/etc/rc.conf)

The /etc/rc.conf file centralizes system configuration settings, such as hostname, timezone, and services. Example:

HOSTNAME="myhostname"
TIMEZONE="Europe/Berlin"
SERVICES="lo crond sshd"

Services listed in SERVICES correspond to scripts in /etc/rc.d/, which are started by /etc/rc.multi and stopped by /etc/rc.shutdown (in reverse order).

See rc.conf(5) for additional configuration settings.

6.2.5. Service Control Scripts (/etc/rc.d/)

The /etc/rc.d/ directory contains individual scripts to manage services. Each script supports commands like:

Example usage:

/etc/rc.d/sshd start
/etc/rc.d/sshd status

See rc.d(7) for more information.

6.2.6. Kernel Module Management (/etc/rc.modules)

The /etc/rc.modules file is used to load specific kernel modules at boot. It begins by running /sbin/depmod -a to map module dependencies, ensuring proper functionality. After this, include commands like:

/sbin/modprobe virtio_net

For modules requiring parameters, it's better to configure settings in /etc/modprobe.d/ for persistent management. For example, to set options for the snd_hda_intel module, create a file like /etc/modprobe.d/snd_hda_intel.conf with the following content:

options snd_hda_intel probe_mask=8

This approach ensures parameters are automatically applied when the module is reloaded, making manual specification unnecessary and adhering to best practices.

The /etc/rc.modules script is invoked by /etc/rc during the boot process.

6.2.7. Custom Startup Commands (/etc/rc.local)

User-defined actions are added to /etc/rc.local, executed as last step of a normal boot (invoked by /etc/rc.multi). Example:

# Start a monitoring app
/usr/bin/my_monitoring_app &

6.2.8. From Boot to Login: System Startup Sequence

Here's the sequence from power-up to login. It differs slightly with or without initramfs.

6.2.8.1. Without Initramfs
  1. Power On: System starts.
  2. Bootloader: Loads the kernel into memory.
  3. Kernel starts: Initializes hardware and memory.
  4. /sbin/init starts: The first user-space process (PID 1).
  5. /etc/inittab is read: init gets its instructions.
  6. /etc/rc runs: The main boot script executes.
  7. Runlevel is set: The default runlevel (usually 2) is chosen.
  8. /etc/rc.multi starts services: Services from /etc/rc.conf are launched.
  9. /etc/rc.local runs: Your custom commands are executed.
  10. Login prompts: The system is ready for you.
6.2.8.2. With Initramfs
  1. Power On: System starts.
  2. Bootloader: Loads kernel and initramfs into memory.
  3. Kernel starts: Initializes basic hardware.
  4. initramfs mounts: A temporary root filesystem in RAM.
  5. /init from initramfs runs: Performs early setup (e.g., storage drivers).
  6. Real root mounts: Your main filesystem is mounted.
  7. Root is switched: System moves from initramfs to your real root.
  8. /sbin/init on real root starts: The init process from your main system.
  9. (Steps 5-10 from 6.2.8.1. Without Initramfs follow).

6.2.9. Minimal Mode (Single-User Mode)

Runlevel 1 or S boots the system with a minimal setup for maintenance. It runs /etc/rc.single, providing a root shell for troubleshooting.

6.2.10. Graceful Shutdown Procedure

When shutting down or rebooting, /etc/rc.shutdown is invoked to stop services, unmount filesystems, and safely power off the system.

6.3. Networking

6.3.1. Hostname Configuration

The hostname identifies your system on the network.

To set it temporarily, use:

# hostname <new_hostname>

For a permanent setting, edit /etc/rc.conf and set:

HOSTNAME='<new_hostname>'

For proper network function, edit /etc/hosts and add your hostname to the loopback lines:

127.0.0.1 localhost <new_hostname>
::1 localhost <new_hostname>

This ensures your system recognizes its own name.

6.3.2. Static Address

To configure a static IP, edit /etc/rc.d/net. Set the network interface, IP, subnet mask, and gateway. For example:

DEV=enp11s0
ADDR=192.168.1.100
MASK=24
GW=192.168.1.1

Set DNS in /etc/resolv.conf:

search <your internal domain>
nameserver <your DNS server>

Start, stop or restart the service via:

# /etc/rc.d/net [start|stop|restart]

6.3.3. Dynamic Address

To get an IP address automatically via DHCP, install the dhcpcd package and edit /etc/rc.d/dhcpcd to specify the network interface (e.g., enp5s0):

DEV=enp5s0

Start, stop, or restart the service via:

# /etc/rc.d/dhcpcd [start|stop|restart]

Check dhcpcd's README for potential issues:

$ pkgman readme dhcpcd

6.3.4. Wireless Network

Ensure the wireless interface isn't blocked. Use rfkill(8):

$ rfkill list
# rfkill unblock <ID|TYPE>

Install the wpa-supplicant package and edit /etc/rc.d/wpa_supplicant to set the wireless interface:

DEV=wlp1s0

Update /etc/wpa_supplicant.conf based on your network:

6.3.4.1. WPA-PSK (Pre-Shared Key)

Generate a key with wpa_passphrase(8):

# wpa_passphrase <MYSSID> <PASSPHRASE>

Example /etc/wpa_supplicant.conf:

network={
    ssid="YOUR_NETWORK_NAME"
    psk="THE_GENERATED_KEY"
}
6.3.4.2. WPA-EAP (Enterprise)

Generate a password hash:

# echo -n <PASSPHRASE> | iconv -t utf16le | openssl md4

Example /etc/wpa_supplicant.conf:

network={
    ssid="YOUR_NETWORK_NAME"
    key_mgmt=WPA-EAP
    identity="YOUR_USERNAME"
    password=hash:<YOUR_PASSWORD_HASH>
}
6.3.4.3. WEP

Example /etc/wpa_supplicant.conf for WEP:

network={
    ssid="YOUR_NETWORK_NAME"
    key_mgmt=NONE
    wep_key0="YOUR_WEP_KEY"
    wep_tx_keyidx=0
    auth_alg=SHARED
}

For automatic IP, ensure the wireless interface is set in /etc/rc.d/dhcpcd. Start the services:

# /etc/rc.d/wpa_supplicant start
# /etc/rc.d/dhcpcd start

6.4. Time and Date Configuration

6.4.1. Timezone Setup

Identify your timezone from /usr/share/zoneinfo (e.g., America/New_York).

Set the timezone immediately:

# ln -sfn /usr/share/zoneinfo/<Your_Timezone> /etc/localtime

Verify the symbolic link:

# ls -l /etc/localtime

To make it persistent, add the timezone to /etc/rc.conf:

TIMEZONE='<Your_Timezone>'

6.4.2. System Clock Setup

Manually set the system clock:

# date MMDDhhmmYYYY.ss

Example:

# date 032705352025.00

For accuracy, use NTP (described next).

6.4.3. Network Time Protocol (NTP) Configuration

Use an NTP client, such as chrony, to synchronize the system clock.

After installing chrony, start, stop, or restart the service via:

# /etc/rc.d/chrony [start|stop|restart]

6.5. User Authentication and Environment

This section explains Zeppe-Lin's approach to user authentication, environment setup, and PAM (Pluggable Authentication Modules).

6.5.1. Password Management

Zeppe-Lin uses strong password encryption with SHA512 for secure user accounts.

Applications requiring user authentication via crypt(3) must link against libcrypt (-lcrypt). Zeppe-Lin's libcrypt supports SHA512 and older encryption methods like DES.

6.5.2. User Environment Configuration

User account creation via useradd(8) is governed by /etc/login.defs. Key settings include:

Zeppe-Lin follows a minimalist approach -- home directories created with useradd -m are empty by default. Customize user environments by:

Default shell PATH (via /etc/profile) is set to /sbin:/usr/sbin:/bin:/usr/bin. This allows access to administrative commands (e.g., sudo(8)) without specifying full paths.

6.5.3. Pluggable Authentication Modules (PAM)

Zeppe-Lin uses PAM (linux-pam package) for flexible user authentication. PAM's configuration files are in /etc/pam.d/, where each file corresponds to a specific login service.

Common PAM modules include:

For advanced configurations, consult module-specific man pages (e.g., man pam_env).

7. FAQ

  1. Why is the distribution named Zeppe-Lin?

    It's our 'Whole Lotta Love' for Linux and GNU, just renamed!

  2. I've looked everywhere, but I can't find the Zeppe-Lin installation CD/DVD. Did I miss something?

    Ah, the elusive Zeppe-Lin installation disc! Fear not, intrepid user, your eyesight is likely just fine. You see, Zeppe-Lin is so cutting-edge, so minimal, that it decided to ditch physical media altogether!

    Instead of hunting for a shiny disc, you embark on a digital treasure hunt to download the root filesystem. Think of it as our way of saving trees... or maybe we just figured you already have a perfectly good Live CD lying around.

    So, no, you didn't miss it. There's no disc. Now, about that rootfs download... (See 2.4.1. Download Rootfs Tarball)

  3. Where's the package/feature I need? Who can add it for me?

    The short answer is: there's no nanny here. Zeppe-Lin is built by its users, for its users. If you need a package, or any other feature, you are the person to make it happen. Our community's strength comes from people like you taking the initiative.

    If you want it, make it. Just be sure your work aligns with the Zeppe-Lin Principles and packaging(7) guidelines before submitting your pull request to the appropriate repository. We appreciate your contribution, and we're looking forward to seeing what you'll add.

8. REPORTING BUGS

For bug reports regarding this site and handbook, please use the issue tracker: https://github.com/zeppe-lin/zeppe-lin.github.io/issues.

This section outlines the legal and licensing information for Zeppe-Lin, including copyrights, licenses, and disclaimers.

9.1. Software Licenses

Zeppe-Lin integrates open-source software projects, each with its own license. For licensing details, consult the source code of the respective packages included in your system.

9.2. Build Scripts Licensing

Zeppe-Lin build scripts retain their original licensing structure.

9.2.1. Attribution

9.2.2. License

These scripts are released under the GNU General Public License version 3 or later (GPLv3+): https://gnu.org/licenses/gpl.html.

9.3. Handbook Licensing

9.3.1. Attribution

9.3.2. License

This handbook is licensed under the Creative Commons Attribution-ShareAlike 3.0 Unported License (CC-BY-SA 3.0): https://creativecommons.org/licenses/by-sa/3.0/.

9.4. Disclaimer

Zeppe-Lin and its documentation are distributed "AS IS" without warranty of any kind. This includes, but is not limited to, implied warranties of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. Use it at YOUR OWN RISK.