Skip to main content

Reduced Python frontend for eBPF

Project description

Dark‐mode image

PyPI version Downloads Build Status License

Python-BPF is an LLVM IR generator for eBPF programs written in Python. It uses llvmlite to generate LLVM IR and then compiles to LLVM object files. These object files can be loaded into the kernel for execution. Python-BPF performs compilation without relying on BCC.

Note: This project is under active development and not ready for production use.


Overview

  • Generate eBPF programs directly from Python.
  • Compile to LLVM object files for kernel execution.
  • Built with llvmlite for IR generation.
  • Supports maps, helpers, and global definitions for BPF.
  • Companion project: pylibbpf, which provides the bindings required for object loading and execution.

Try It Out!

Run

curl -s https://raw.githubusercontent.com/pythonbpf/Python-BPF/refs/heads/master/tools/setup.sh | sudo bash

Installation

Dependencies:

  • clang
  • Python ≥ 3.8

Install via pip:

pip install pythonbpf pylibbpf

Example Usage

import time
from pythonbpf import bpf, map, section, bpfglobal, BPF
from pythonbpf.helper import pid
from pythonbpf.maps import HashMap
from pylibbpf import *
from ctypes import c_void_p, c_int64, c_uint64, c_int32
import matplotlib.pyplot as plt


# This program attaches an eBPF tracepoint to sys_enter_clone,
# counts per-PID clone syscalls, stores them in a hash map,
# and then plots the distribution as a histogram using matplotlib.
# It provides a quick view of process creation activity over 10 seconds.

@bpf
@map
def hist() -> HashMap:
    return HashMap(key=c_int32, value=c_uint64, max_entries=4096)


@bpf
@section("tracepoint/syscalls/sys_enter_clone")
def hello(ctx: c_void_p) -> c_int64:
    process_id = pid()
    one = 1
    prev = hist.lookup(process_id)
    if prev:
        previous_value = prev + 1
        print(f"count: {previous_value} with {process_id}")
        hist.update(process_id, previous_value)
        return c_int64(0)
    else:
        hist.update(process_id, one)
    return c_int64(0)


@bpf
@bpfglobal
def LICENSE() -> str:
    return "GPL"


b = BPF()
b.load_and_attach()
hist = BpfMap(b, hist)
print("Recording")
time.sleep(10)

counts = list(hist.values())

plt.hist(counts, bins=20)
plt.xlabel("Clone calls per PID")
plt.ylabel("Frequency")
plt.title("Syscall clone counts")
plt.show()

Architecture

Python-BPF provides a complete pipeline to write, compile, and load eBPF programs in Python:

  1. Python Source Code

    • Users write BPF programs in Python using decorators like @bpf, @map, @section, and @bpfglobal.
    • Maps (hash maps), helpers (e.g., ktime, deref), and tracepoints are defined using Python constructs, preserving a syntax close to standard Python.
  2. AST Generation

    • The Python ast module parses the source code into an Abstract Syntax Tree (AST).
    • Decorators and type annotations are captured to determine BPF maps, tracepoints, and global variables.
  3. LLVM IR Emission

    • The AST is transformed into LLVM Intermediate Representation (IR) using llvmlite.
    • IR captures BPF maps, control flow, assignments, and calls to helper functions.
    • Debug information is emitted for easier inspection.
  4. LLVM Object File Compilation

    • The LLVM IR (.ll) is compiled into a BPF target object file (.o) using llc -march=bpf -O2.
    • This produces a kernel-loadable ELF object file containing the BPF bytecode.
  5. libbpf Integration (via pylibbpf)

    • The compiled object file can be loaded into the kernel using pylibbpf.
    • Maps, tracepoints, and program sections are initialized, and helper functions are resolved.
    • Programs are attached to kernel hooks (e.g., syscalls) for execution.
  6. Execution in Kernel

    • The kernel executes the loaded eBPF program.
    • Hash maps, helpers, and global variables behave as defined in the Python source.
    • Output can be read via BPF maps, helper functions, or trace printing.

This architecture eliminates the need for embedding C code in Python, allowing full Python tooling support while generating true BPF object files ready for kernel execution.


Development

  1. Create a virtual environment and activate it:

    python3 -m venv .venv
    source .venv/bin/activate
    
  2. Install dependencies:

    make install
    

    Then, run any example in examples

  3. Verify an object file with the kernel verifier:

    ./tools/check.sh check execve2.o
    
  4. Run an object file using bpftool:

    ./tools/check.sh run execve2.o
    
  5. Explore LLVM IR output from clang in examples/c-form by running make.


Resources


Authors


Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pythonbpf-0.1.6.tar.gz (60.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pythonbpf-0.1.6-py3-none-any.whl (73.5 kB view details)

Uploaded Python 3

File details

Details for the file pythonbpf-0.1.6.tar.gz.

File metadata

  • Download URL: pythonbpf-0.1.6.tar.gz
  • Upload date:
  • Size: 60.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pythonbpf-0.1.6.tar.gz
Algorithm Hash digest
SHA256 362a55658ce91a0fb4b1388dc893172e2a720b1475894b214bbb5c0c240a010b
MD5 2624b4225e6bb43e4b4aeaf29d74a72a
BLAKE2b-256 7367188554bd14cbcf288be0d19aa9b3940dcc91d4f17d9abc26a5f8128e5356

See more details on using hashes here.

Provenance

The following attestation bundles were made for pythonbpf-0.1.6.tar.gz:

Publisher: python-publish.yml on pythonbpf/Python-BPF

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pythonbpf-0.1.6-py3-none-any.whl.

File metadata

  • Download URL: pythonbpf-0.1.6-py3-none-any.whl
  • Upload date:
  • Size: 73.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pythonbpf-0.1.6-py3-none-any.whl
Algorithm Hash digest
SHA256 2ced61fefb13e81d852c4e72a5fee784ac8a3c99653cd4f6d10cb276e1fdbb20
MD5 092234cf87870603080f2f369e9ba2e4
BLAKE2b-256 b37610cb09cea7820663a8ebb0044697ed22f31ad76cfbf9ac51d3c4d7a8c487

See more details on using hashes here.

Provenance

The following attestation bundles were made for pythonbpf-0.1.6-py3-none-any.whl:

Publisher: python-publish.yml on pythonbpf/Python-BPF

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page