Skip to main content

eXPeditious Data Transfer

Project description

xpdt: eXPeditious Data Transfer

PyPI version

About

xpdt is (yet another) language for defining data-types and generating code for serializing and deserializing them. It aims to produce code with little or no overhead, especially in the case where some fields aren't required, and is based on fixed-length representations allowing for branch-free zero-copy deserialization and (at-most-)one-copy writes (source to buffer).

The generated C code, in particular, is highly optimized and often permits the elimination of data-copying for writes and enables optimizations such as loop-unrolling when deserializing fixed-length objects. This can lead to read speeds in excess of 500 million objects per second (~1.8 nsec per object).

Examples

The xpdt source language looks similar to C struct definitions:

struct timestamp {
	u32	tv_sec;
	u32	tv_nsec;
};

struct point {
	i32	x;
	i32	y;
	i32	z;
};

struct line {
	timestamp	time;
	point		line_start;
	point		line_end;
	bytes		comment;
};

Fixed width integer types from 8 to 128 bit are supported, along with the bytes type, which is a variable-length sequence of bytes.

Target Languages

The following target languages are currently supported:

  • C
  • Python

The C code is very highly optimized.

The Python code is about as well optimized for CPython as I can make it. It uses typed NamedTuple for objects, which has some small overhead over regular tuples, and it uses struct.Struct to do the packing/unpacking. I have also code-golfed the generated bytecodes down to what I think is minimal given the design constraints. As a result, performance of the pure Python code is comparable to a JSON library implemented in C or Rust.

For better performance in Python, it may be desirable to develop a Cython target. In some instances CFFI structs may be more performant since they can avoid the creation/destruction of an object for each record.

Target languages are implemented purely as jinja2 templates.

Serialization format

The serialization format for fixed-length objects is simply a packed C struct, with little-endian fields.

For any object which contains variable length fields (eg. bytes or utf8):

  • a 32bit unsigned record length is prepended to the struct, this allows efficient skipping of the whole record
  • all variable-length fields are converted to u32 and contain the length, in bytes, of the data
  • all variable-length contents are appended after the struct in the order in which they appear

For example, following the example above, the serialization would be:

u32 tot_len # = 41
u32 time.tv_sec
u32 time.tv_usec
i32 line_start.x
i32 line_start.y
i32 line_start.z
i32 line_end.x
i32 line_end.y
i32 line_end.z
u32 comment # = 5
u8 'H'
u8 'e'
u8 'l'
u8 'l'
u8 'o'

Why Variable-Length Data at the End?

Placing variable-length fields at the end of the struct provides significant performance benefits:

  1. xpdt is optimized for fast zero-copy deserialization, it especially tries to avoid adding overhead in the case where only a subset of fields are required. If variable length fields could come in the middle of a struct, reading the length and skipping the field would incur a cost even when the field is being skipped.
  2. Potentially large strings and payloads isolated: Only when the variable-length field is needed do you pay the cost of calculating their offsets. This is ideal for descriptive strings or optional metadata that is rarely accessed.

Features

The feature-set is, as of now, pretty slim.

There are no array / sequence / map types, and no keyed unions.

Support for such things may be added in future provided that suitable implementations exist. An implementation is suitable if:

  • It admits a zero (or close to zero) overhead implementation
  • it causes no overhead when the feature isn't being used

License

The compiler is released under the GPLv3.

The C support code/headers are released under the MIT license.

The generated code is yours.

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

xpdt-0.4.1.tar.gz (38.4 kB view details)

Uploaded Source

Built Distribution

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

xpdt-0.4.1-py3-none-any.whl (42.5 kB view details)

Uploaded Python 3

File details

Details for the file xpdt-0.4.1.tar.gz.

File metadata

  • Download URL: xpdt-0.4.1.tar.gz
  • Upload date:
  • Size: 38.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.3

File hashes

Hashes for xpdt-0.4.1.tar.gz
Algorithm Hash digest
SHA256 e737a9b6d64ae2cb630ee14048c5f55f60bb3c92f5211fde9e45c20a65b68b4f
MD5 e9f0a6078ed4bc1438386272523320b0
BLAKE2b-256 20fa1986953af52cd808afb05b4f12422852332ef31aefe7313d8d26e326e71c

See more details on using hashes here.

File details

Details for the file xpdt-0.4.1-py3-none-any.whl.

File metadata

  • Download URL: xpdt-0.4.1-py3-none-any.whl
  • Upload date:
  • Size: 42.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.3

File hashes

Hashes for xpdt-0.4.1-py3-none-any.whl
Algorithm Hash digest
SHA256 32f4080dcc79250f718a28caaeb117a23a558ac1f8d5bf1f2290cf6d7cb0a67c
MD5 0dba602b49cb99c557b32550b61a40e4
BLAKE2b-256 0002e264f9f9cc548d2db9d73c87eaa8ef70e4d8645b7fbeec34e97bec6c85b2

See more details on using hashes here.

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