A solution for the Neuralink Compression Challenge.
Go to file
2024-05-27 10:29:25 +02:00
.gitignore Changed everything 2024-05-25 17:31:08 +02:00
analysis.ipynb Cleared verbose outputs in notebook 2024-05-25 17:33:29 +02:00
bitstream.py BinomialHuffman Bitstream Encoder, lets gooo 2024-05-26 23:54:31 +02:00
config.yaml More tuning lel 2024-05-27 10:29:15 +02:00
data_processing.py Cover egde case: Negative correlation in lead activations 2024-05-26 23:55:31 +02:00
main.py Fix eval code 2024-05-27 10:28:51 +02:00
models.py Better ResNets 2024-05-26 23:56:02 +02:00
README.md README: Outline arch change; seperate feature extraction 2024-05-27 10:29:25 +02:00
requirements.txt 3 2024-05-25 00:53:30 +02:00
spikey.svg Pretty README with logo 2024-05-26 14:42:31 +02:00
utils.py Fix eval code 2024-05-27 10:28:51 +02:00

Spikey

This repository contains a solution for the Neuralink Compression Challenge. The challenge involves compressing raw electrode recordings from a Neuralink implant. These recordings are taken from the motor cortex of a non-human primate while playing a video game.

Challenge Overview

The Neuralink N1 implant generates approximately 200 Mbps of electrode data (1024 electrodes @ 20 kHz, 10-bit resolution) and can transmit data wirelessly at about 1 Mbps. This means a compression ratio of over 200x is required. The compression must run in real-time (< 1 ms) and consume low power (< 10 mW, including radio).

Data Analysis

The analysis.ipynb notebook contains a detailed analysis of the data. We found that there is sometimes significant cross-correlation between the different leads, so we find it vital to use this information for better compression. This cross-correlation allows us to improve the accuracy of our predictions and reduce the overall amount of data that needs to be transmitted. As part of the analysis, we also note that achieving a 200x compression ratio is highly unlikely to be possible and is also nonsensical; a very close reproduction is sufficient.

Algorithm Overview

1 - Thread Topology Reconstruction

As the first step, we analyze readings from the leads to construct an approximate topology of the threads in the brain. The distance metric we generate only approximately represents true Euclidean distances, but rather the 'distance' in common activity. This topology must only be computed once for a given implant and may be updated for thread movements but is not part of the regular compression/decompression process.

2 - Predictive Architecture

The main workhorse of our compression approach is a predictive model running both in the compressor and decompressor. With good predictions of the data, only the error between the prediction and actual data must be transmitted. We make use of the previously constructed topology to allow the predictive model's latent to represent the activity of brain regions based on the reading of the threads instead of just for threads themselves.

We separate the predictive model into four parts:

  1. Feature Extraction: This module processes a given history of readings for a single thread and extracts relevant features (using mostly wavelet and Fourier transforms). Highly configurable, this module performs the heavy lifting of signal analysis, allowing shallow neural networks to handle the rest. (Full disclosure: I have no idea what half of the implemented wavelet transforms actually do. We just throw anything sensible at the problem and will narrow down later; making effective use of the fact that 'fuck around' and 'find out' are positively correlated.)

  2. Latent Projector: This takes the feature vectors and projects them into a latent space. The latent projector can be configured as a fully connected network or an RNN (LSTM) with an arbitrary shape.

  3. MiddleOut (Message Passer): For each lead, this module performs message passing according to the thread topology. Their latent representations along with their distance metrics are used to generate region latent representations. This is done by training a fully connected layer to map from (our_latent, their_latent, metric) -> region_latent and then averaging over all region_latent values to get the final representation.

  4. Predictor: This module takes the new latent representation from the MiddleOut module and predicts the next timestep. The goal is to minimize the prediction error during training. It can be configured to be an FCNN of arbitrary shape.

The neural networks used are rather small, making it possible to meet the latency and power requirements if implemented more efficiently.

If we were to give up on lossless compression, one could expand MiddleOut to form a joint latent over all threads and transmit that.

3 - Efficient Bitstream Encoding

Based on an expected distribution of deltas that have to be transmitted, an efficient Huffman-like binary format is used for encoding the data.

TODO

  • Our flagship bitstream encoder builds an optimal Huffman tree assuming the deltas are binomially distributed. This should be updated when we know a more precise approximation of the delta distribution.
  • All trained models still mostly suck. I'm not beating a compression ratio of ~2x (not counting the bitstream encoder). Probably a bug somewhere in our code.

Installation

To install the necessary dependencies, create a virtual environment and install the requirements:

python3 -m venv env
source env/bin/activate
pip install -r requirements.txt

Usage

Training

Requires Slate, which is not currently publicly available. Install via (requires repo access):

pip install -e git+ssh://git@dominik-roth.eu/dodox/Slate.git#egg=slate

To train the model, run:

python main.py <config_file.yaml> <exp_name>