Skip to content

Memory Panama Design

github-actions[bot] edited this page Jun 16, 2026 · 5 revisions

πŸ’Ύ Off-Heap Panama Design

Spector Memory achieves zero garbage collection pressure by storing all vector data and cognitive headers off-heap using Java Project Panama's Foreign Function & Memory API. No memory record ever touches the JVM heap.


Why Off-Heap?

In a standard JVM application, objects live on the heap and are managed by the garbage collector. For AI memory workloads, this creates problems:

On-Heap (Traditional) Off-Heap (Panama)
GC pauses (10-100ms for large heaps) Zero GC pauses β€” data is invisible to GC
Object overhead (16-24 bytes per object header) Zero overhead β€” raw bytes, no object headers
Memory fragmentation over time Compact β€” contiguous byte arrays
Heap size limits JVM config System memory β€” limited only by OS
Serialization required for persistence Direct mmap β€” bytes are already on disk

Panama Architecture

MemorySegment β€” The Core Abstraction

Every memory record is stored in a MemorySegment β€” a contiguous off-heap byte buffer managed by an Arena. Fields are read and written directly at byte offsets β€” no Java objects are created, no deserialization occurs.

Key properties:

  • Shared Arena β€” thread-safe for concurrent reads across Virtual Threads
  • 64-byte alignment β€” ensures SIMD-friendly access patterns and cache-line-aligned header reads
  • GC-invisible β€” the garbage collector never sees this memory

Arena Lifecycle

graph LR
    A["Create shared Arena"] --> B["Allocate off-heap<br/>(bytes + alignment)"]
    B --> C["MemorySegment<br/>(off-heap buffer)"]
    C -->|"read/write"| D["SIMD Scorer<br/>Virtual Threads"]
    C -->|"close Arena"| E["Memory released<br/>back to OS"]

    style A fill:#3498db,color:white
    style C fill:#2ecc71,color:white
    style E fill:#e74c3c,color:white
Loading

warning: Lifetime Management Unlike heap objects, off-heap memory is not garbage collected. You must explicitly close the Arena when done. SpectorMemory implements AutoCloseable and closes all arenas in its close() method. Always use try-with-resources.


Three Storage Modes

flowchart TD
    subgraph "Volatile (In-Memory)"
        WM["Working Memory<br/><i>circular buffer</i>"]
        PM["Procedural Memory<br/><i>learned procedures</i>"]
    end

    subgraph "Persistent (mmap)"
        EM["Episodic Memory<br/><i>time-partitioned files</i>"]
    end

    subgraph "Compact (Header-Only)"
        SM["Semantic Memory<br/><i>metadata slab, no vectors</i>"]
    end

    style WM fill:#e74c3c,color:white
    style PM fill:#e74c3c,color:white
    style EM fill:#0984e3,color:white
    style SM fill:#00b894,color:white
Loading

1. Arena-Allocated (Working, Procedural)

Volatile, in-memory segments for transient data.

Characteristics:

  • Fast allocation (~1Β΅s)
  • Lost on JVM shutdown
  • No file I/O overhead
  • Fixed capacity

2. mmap-Backed (Episodic)

Persistent, memory-mapped files for durable storage. The OS handles paging data between RAM and disk.

Characteristics:

  • Persists across JVM restarts
  • OS handles paging to/from disk
  • Lazy loading β€” only mapped pages are in physical RAM
  • Atomic flush for durability

3. Header-Only Slab (Semantic)

Compact metadata-only storage (no vectors).

Characteristics:

  • Minimal memory footprint (64B per record vs. ~832B for full records)
  • Fast metadata scans (tag match, importance, valence, arousal)
  • No vector data β€” re-embed at query time if needed

Binary Record Format

Versioned Header Layout

The cognitive record format uses a 64-byte cache-line-aligned header via the HeaderLayout sealed interface. The header occupies exactly one CPU cache line for optimal sequential scan performance. See Synapse β€” Tags & Scoring for the full byte-level specification.

graph LR
    subgraph "64B Header + Vector"
        H["Header (64B)"] --> V["INT8 Vector (NB)"]
    end

    style H fill:#27ae60,color:white
    style V fill:#2ecc71,color:white
Loading

Layout (64 bytes) β€” Cache-Line Aligned

 0                   1                   2                   3
 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| ver(1B)|flg(1B)| val(1B)| aro(1B)| importance (4B)            |  ← Offset 0
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                                                               |
+                      timestamp (8B)                           +  ← Offset 8
|                                                               |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|              agent_recall_count (4B)                          |  ← Offset 16
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|              exact_norm (4B)                                  |  ← Offset 20
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                                                               |
+                    synapticTags (8B)                          +  ← Offset 24
|                                                               |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|  centroid_id (2B) |  _pad0 (2B) | storage_strength (4B)       |  ← Offset 32
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|      spector_recall_cnt (4B)    |   _reserved_f1 (4B)         |  ← Offset 40
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                                                               |
+                    last_auto_ltp (8B)                         +  ← Offset 48
|                                                               |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                                                               |
+                    _reserved_l1 (8B)                          +  ← Offset 56
|                                                               |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                                                               |
+              Quantized Vector β€” INT8[N]                       +  ← Offset 64
|              (dequantize: float = byte Γ— scale + min)         |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
                  stride = 64 + N bytes per record

Memory Cost

Header Stride (768-dim) 1M Records Alignment
64B 832B ~793 MB 1Γ— cache line (64B)

Field Access Patterns

The header layout is designed for sequential access in the scoring hot-loop. Fields are ordered by access frequency:

Phase 1: flags        (offset 1,  1B)  β€” First check, highest skip rate
Phase 2: synapticTags (offset 24, 8B)  β€” Second check, eliminates 99%
Phase 3: valence      (offset 2,  1B)  β€” Third check (profile-dependent)
Phase 4: importance   (offset 4,  4B)  β€” Fourth check
Phase 4: timestamp    (offset 8,  8B)  β€” Read with importance
Phase 4: recallCount  (offset 16, 4B)  β€” Reconsolidation adjustment
Phase 4: arousal      (offset 3,  1B)  β€” Arousal-modulated decay
Phase 4: storageStr   (offset 36, 4B)  β€” Two-Factor S(t)
Phase 5: vector       (offset 64, NB)  β€” Only if all filters pass

tip: Cache Line Optimization The 64-byte header occupies exactly one CPU cache line. During sequential scans, each header read hits exactly one cache line β€” no split-line loads, no false sharing. The CPU prefetcher can pre-fetch the next record's header while the current one is being scored.


Episodic Partition File Format

Each episodic partition file has a 64-byte metadata header:

Offset   Size   Field            Description
──────   ────   ─────            ───────────
  0       4B    magic            0x45504943 ("EPIC" in ASCII)
  4       4B    version          Format version (1)
  8       4B    count            Number of live records
 12       4B    tombstoneCount   Number of tombstoned records
 16       4B    capacity         Maximum records in partition
 20       4B    state            PartitionState ordinal
 24       4B    stride           Record stride in bytes
 28      36B    reserved         Future use (alignment padding)

File naming: episodic-{yyyyMMdd}.mem (e.g., episodic-20260527.mem)

Partition capacity: Default 10,000 records per partition. At 832 bytes/record (768-dim INT8), each partition file is ~8 MB.


Thread Safety Model

Component Thread Safety Mechanism
Shared Arena βœ… Concurrent reads Built-in Panama support
Segment reads βœ… Lock-free Direct memory access
Segment writes ⚠️ Single writer Synchronized on partition append
Reverse index βœ… Lock-free reads CAS-based updates
Partition metadata ⚠️ Single writer Metadata header writes serialized

Recall: Multiple Virtual Threads read different partitions concurrently β€” zero contention because each partition's MemorySegment is disjoint.

Ingestion: Writes are serialized per partition (one writer at a time) but different partitions can accept writes concurrently.


Zero-Copy Data Path

graph LR
    A["πŸ’Ύ Disk"] -->|mmap| B["MemorySegment"]
    B -->|"direct read"| C["SIMD Registers"]
    C --> D["βœ… Score"]

    style A fill:#3498db,color:white
    style B fill:#2ecc71,color:white
    style D fill:#00b894,color:white
Loading

No Java objects created. No serialization. No deserialization. No GC pressure.

The entire data path from persistent storage to CPU computation operates on raw bytes. The JVM heap is used only for the top-K result set β€” typically 5-20 small result records.


Next Steps

  • :material-speedometer: [[Performance|Memory--Performance]] β€” benchmark results
  • :material-brain: [[Architecture|Memory--Architecture]] β€” system design
  • :material-lightning-bolt: [[6-Phase Scoring Pipeline|Memory--Scoring-Pipeline]] β€” the SIMD hot-loop
  • :material-tag: [[Synapse β€” Tags & Scoring|Memory--Synapse]] β€” versioned header byte maps, arousal decay, Bloom filter
  • :material-flask: [[Labs β€” Research Roadmap|Labs--Roadmap]] β€” Dynamic Quantization (SQ4), Two-Factor Memory

🏠 Home


Clone this wiki locally