Menu

Cross-Platform Abstraction Patterns

Relevant source files

This document explains the systematic approaches used throughout fastfetch to support 10+ operating systems (Linux, Windows, macOS, FreeBSD, OpenBSD, NetBSD, DragonFly BSD, Android, Haiku, SunOS) while maintaining code clarity and minimizing duplication. It covers file organization, conditional compilation strategies, common interface patterns, and platform-specific API abstractions.

For information about the overall module system architecture, see Module System Architecture. For build system configuration, see Cross-Platform Build Configuration.

File Organization Strategy

Fastfetch uses a strict naming convention to organize platform-specific implementations, making it immediately clear which code runs on which platform.

Naming Convention Pattern

Platform-specific implementation files follow the pattern: <module>_<platform>.c

Platform Suffixes:

  • _linux.c - GNU/Linux
  • _windows.c - Windows
  • _apple.c / _apple.m - macOS (.m for Objective-C code)
  • _bsd.c - FreeBSD and DragonFly BSD shared implementation
  • _nbsd.c - NetBSD-specific
  • _obsd.c - OpenBSD-specific
  • _android.c - Android-specific (when different from Linux)
  • _sunos.c - SunOS/Solaris/OpenIndiana
  • _nosupport.c - Stub implementation for unsupported platforms

Sources: src/detection/cpu/cpu_linux.c src/detection/cpu/cpu_windows.c src/detection/cpu/cpu_apple.c src/detection/cpu/cpu_bsd.c src/detection/cpu/cpu_nbsd.c src/detection/cpu/cpu.h src/detection/cpu/cpu.c

Directory Structure Example

src/detection/gpu/ ├── gpu.h # Common interface, types, constants ├── gpu.c # Shared logic, name cleanup ├── gpu_linux.c # Linux PCI, DRM, sysfs detection ├── gpu_windows.c # Windows CM API, DirectX detection ├── gpu_apple.c # macOS IOKit, Metal detection ├── gpu_bsd.c # FreeBSD PCI, DRM detection ├── gpu_sunos.c # SunOS libdevinfo detection ├── gpu_pci.c # Shared PCI ID database lookup ├── gpu_drm.c # Shared DRM ioctl wrappers └── gpu_driver_specific.h # Driver-specific detection 

Sources: src/detection/gpu/gpu.h src/detection/gpu/gpu.c src/detection/gpu/gpu_linux.c src/detection/gpu/gpu_windows.c src/detection/gpu/gpu_apple.c src/detection/gpu/gpu_bsd.c src/detection/gpu/gpu_sunos.c src/detection/gpu/gpu_pci.c

Build-Time Platform Abstraction

CMake performs platform detection at configuration time and conditionally includes the appropriate source files in the build.

Platform Detection in CMake

CMakeLists.txt23-50 defines platform detection logic:

Sources: CMakeLists.txt23-50

Conditional Source Inclusion

CMakeLists.txt502-742 demonstrates conditional source file inclusion for Linux:

PlatformSource Files AddedKey APIs Used
Linuxcpu_linux.c, gpu_linux.c, disk_linux.c, etc./proc, /sys, ioctl(), DRM, Wayland
Windowscpu_windows.c, gpu_windows.c, etc.Registry, WMI, CM API, DirectX
macOScpu_apple.c, gpu_apple.c, etc.sysctl(), IOKit, CoreFoundation
FreeBSDcpu_bsd.c, gpu_bsd.c, etc.sysctl(), kvm, DRM
NetBSDcpu_nbsd.c, disk_bsd.c, etc.sysctl(), proplib
OpenBSDcpu_obsd.c, disk_bsd.c, etc.sysctl(), kvm

Sources: CMakeLists.txt502-742

Feature Detection with CMake Options

CMakeLists.txt72-96 defines optional feature flags that control conditional compilation:

These create FF_HAVE_* preprocessor macros used throughout the codebase:

  • FF_HAVE_VULKAN - Vulkan GPU detection support
  • FF_HAVE_WAYLAND - Wayland display server support
  • FF_HAVE_DRM - Direct Rendering Manager support
  • FF_HAVE_DBUS - D-Bus IPC support

Sources: CMakeLists.txt72-96 src/common/init.c166-261

Common Interface Pattern

All platform-specific implementations conform to a common interface defined in the module's header file. This allows platform-agnostic code to call detection functions without knowing which platform implementation will execute.

Detection Function Signature Pattern

Example from GPU detection:

src/detection/gpu/gpu.h48-49 defines the common interface:

Each platform implements this signature:

Sources: src/detection/gpu/gpu.h48-49 src/detection/gpu/gpu_linux.c309-551 src/detection/gpu/gpu_windows.c22-421 src/detection/gpu/gpu_apple.c83-155 src/detection/gpu/gpu_bsd.c155-309

Result Structure Uniformity

All platforms populate the same result structure, ensuring consistent output regardless of platform:

src/detection/gpu/gpu.h30-46 defines FFGPUResult:

FieldTypePurpose
indexuint32_tGPU index for multi-GPU systems
typeFFGPUTypeIntegrated vs Discrete classification
vendorFFstrbufVendor name (AMD, NVIDIA, Intel, etc.)
nameFFstrbufGPU model name
driverFFstrbufDriver name and version
platformApiFFstrbufDetection method (DRM, IOKit, CM API, etc.)
temperaturedoubleGPU temperature in °C
coreCountint32_tNumber of GPU cores/compute units
frequencyuint32_tMaximum frequency in MHz
dedicated.total/useduint64_tDedicated VRAM in bytes
shared.total/useduint64_tShared memory in bytes

Sources: src/detection/gpu/gpu.h30-46

Error Handling Convention

Detection functions return const char* error messages or NULL on success:

Sources: src/detection/gpu/gpu.c1-134 src/modules/gpu/gpu.c1-239

Runtime Conditional Compilation

In addition to build-time source selection, the codebase uses preprocessor directives for runtime conditional compilation within individual source files.

Platform Detection Macros

Common preprocessor macros used for platform detection:

MacroPlatformDefined By
__linux__LinuxGCC/Clang
_WIN32WindowsMSVC/MinGW
__APPLE__macOSGCC/Clang
__FreeBSD__FreeBSDGCC/Clang
__OpenBSD__OpenBSDGCC/Clang
__NetBSD__NetBSDGCC/Clang
__DragonFly__DragonFly BSDGCC/Clang
__sunSunOSGCC/Clang
__ANDROID__AndroidNDK
__HAIKU__HaikuGCC

Feature Availability Macros

src/fastfetch_config.h.in (generated by CMake) defines feature availability:

Conditional Code Blocks

src/detection/gpu/gpu_linux.c12-33 demonstrates feature-gated code:

Example from GPU detection: src/detection/gpu/gpu_linux.c106-126

Sources: src/detection/gpu/gpu_linux.c12-33 src/detection/gpu/gpu_linux.c106-126

Architecture-Specific Code

src/detection/gpu/gpu_linux.c24-33 shows architecture-specific conditional compilation:

This pattern enables Apple Silicon GPU support on ARM64 Linux systems.

Sources: src/detection/gpu/gpu_linux.c24-33

Platform-Specific API Usage

Each platform uses its native APIs for system information detection. The abstraction layer hides these differences behind common interfaces.

API Mapping by Platform

Sources: src/detection/cpu/cpu_linux.c1-600 src/detection/cpu/cpu_windows.c1-300 src/detection/cpu/cpu_apple.c1-200 src/detection/cpu/cpu_bsd.c1-150

Linux: /proc and /sys Filesystems

src/detection/cpu/cpu_linux.c14-139 uses /sys/class/hwmon and /sys/class/thermal for temperature:

src/detection/gpu/gpu_linux.c128-193 uses /sys/class/drm/card*/device/ for AMD GPU info:

Sources: src/detection/cpu/cpu_linux.c14-139 src/detection/gpu/gpu_linux.c128-193

Windows: Registry and Performance Counters

src/detection/gpu/gpu_windows.c113-180 uses Configuration Manager API and Registry:

src/detection/cpu/cpu_windows.c20-139 uses Performance Library for CPU temperature:

Sources: src/detection/gpu/gpu_windows.c113-180 src/detection/cpu/cpu_windows.c20-139

macOS: IOKit and sysctl

src/detection/gpu/gpu_apple.c83-155 uses IOKit for GPU detection:

src/detection/cpu/cpu_apple.c47-135 uses IOKit for frequency detection on Apple Silicon:

Sources: src/detection/gpu/gpu_apple.c83-155 src/detection/cpu/cpu_apple.c47-135

BSD: sysctl and kvm

src/detection/cpu/cpu_bsd.c33-80 uses sysctl() for CPU information:

Sources: src/detection/cpu/cpu_bsd.c33-80

Code Sharing Strategies

Fastfetch employs several strategies to share code across platforms, reducing duplication while maintaining platform-specific optimizations.

Shared Helper Functions

src/detection/gpu/gpu_pci.c1-234 provides shared PCI ID database lookup used by Linux, BSD, and SunOS:

Example usage: src/detection/gpu/gpu_linux.c345-350

Sources: src/detection/gpu/gpu_pci.c1-234 src/detection/gpu/gpu_linux.c345-350

Platform Groups

Some platforms share significant code. FreeBSD and DragonFly BSD use cpu_bsd.c, while more specialized implementations exist for NetBSD and OpenBSD:

CMakeLists.txt661-753 shows BSD platform grouping:

Sources: CMakeLists.txt661-753

Conditional Platform-Specific Code within Shared Files

src/detection/gpu/gpu_bsd.c155-309 contains FreeBSD-specific DRM detection alongside general BSD PCI code:

Sources: src/detection/gpu/gpu_bsd.c155-309

Abstract Helper Layers

src/common/sysctl.c (referenced in BSD/macOS code) provides platform-agnostic sysctl() wrappers:

These wrappers hide platform differences in sysctl() usage between BSD variants and macOS.

Sources: src/detection/cpu/cpu_bsd.c33-80 src/detection/cpu/cpu_apple.c47-135

Fallback and Detection Method Selection

Many modules support multiple detection methods with runtime fallback mechanisms.

Multi-Method Detection Pattern

src/detection/gpu/gpu_linux.c309-551 implements a detection method hierarchy:

Implementation: src/detection/gpu/gpu_linux.c309-551

Sources: src/detection/gpu/gpu_linux.c309-551

Driver-Specific Enhancement

src/detection/gpu/gpu_linux.c106-307 shows conditional driver-specific detection:

This pattern allows basic detection to work everywhere, with optional enhanced information when specific drivers are available.

Sources: src/detection/gpu/gpu_linux.c106-307

Best Practices and Guidelines

Based on the patterns observed throughout the codebase:

1. File Organization

  • Always use platform suffixes: *_linux.c, *_windows.c, *_apple.c, *_bsd.c
  • Keep common interface in *.h and shared logic in base *.c file
  • Place platform-specific headers (like drm.h) within #ifdef guards

2. Interface Design

  • Define function signatures in common header: const char* ffDetect*Impl(const FF*Options*, FF*Result*)
  • Return NULL on success, error message string on failure
  • Use consistent result structure types across platforms

3. Conditional Compilation

  • Use CMake for build-time source file selection whenever possible
  • Use #ifdef FF_HAVE_* for optional features that require external libraries
  • Use #ifdef __linux__ / _WIN32 / etc. only when necessary within shared files

4. Code Reuse

  • Extract truly platform-agnostic logic to base *.c files (e.g., gpu_pci.c)
  • Create shared helper libraries for common tasks (e.g., sysctl.c)
  • Group similar platforms (FreeBSD+DragonFly) when behavior is identical

5. Fallback Mechanisms

  • Implement detection method hierarchies (try native API → fallback to generic API)
  • Allow user override via options when multiple methods exist
  • Always provide graceful degradation

6. Error Handling

  • Return descriptive error messages for debugging
  • Use FF_*_UNSET constants for undetected values rather than magic numbers
  • Allow partial detection success (e.g., detect GPU name but not temperature)

Sources: All patterns derived from comprehensive analysis of src/detection/ subdirectories and CMakeLists.txt