Decimal library

This is a Rust fixed-point numeric library targeting blockchain. It was created purely for practical reasons as a fast and simple way to use checked math with given decimal precision.

It has achieved present form over several iterations, first being implemented inside Synthetify protocol. The current version leverages macros, traits and generics to exchange dozens of lines of error prone code with a single line and generating the rest. In this form it is used inside of Invariant protocol and was audited as a part of it.

It allows a definition of multiple types with different precisions and primitive types and calculations in between them, see below for a quick example.

Quickstart

The library is used by adding a macro #[decimal(k)], where k is a desired decimal precision (number decimal places after the dot).

This macro generates an implementation of several generic traits for each struct it is called on allowing basic operations in between them.

Basic example

Having imported the library you can declare a type like so:

#[decimal(2)]
#[derive(Default, PartialEq, Debug, Clone, Copy)]
struct Percentage(u32);

Deserialization

Named structs can be deserialized without a problem like so:

#[decimal(6)]
#[zero_copy]
#[derive(AnchorSerialize, AnchorDeserialize, ...)]
pub struct Price {
    pub v: u128,
}

Basic operations

All methods generated by the macro use checked math and panic on overflow. Operators are overloaded where possible for ease of use.

Basic example using types defined above would look like this:

let price = Price::from_integer(10); // this corresponds with 10 * 10^k so 10^7 in this case

let discount = Percentage::new(10); // using new doesn't account for decimal places so 0.10 here

// addition expects being called for left and right values being of the same type
// multiplication doesn't so you can be used like this:
let price = price * (Percentage::from_integer(1) - discount); // the resulting type is always the type of the left value

For more examples continue to the walkthrough.rs

Parameters for the macro

As mentioned the first argument of macro controls the amount of places after the dot. It can range between 0 and 38. It can be read by Price::scale(). The second one is optional and can be a bit harder to grasp

The Big Type

The second argument taken has a weird name of a big type. It sets the type that is to be used when calling the methods with _big_ in the name. Its purpose is to avoid temporary overflows so an overflow that occurs while calculating theS return value despite that value fitting in the given type. Consider the example below

#[decimal(2)]
#[derive(Default, std::fmt::Debug, Clone, Copy, PartialEq)]
struct Percentage(u8);

let p = Percentage(110); // 110% or 1.1

// assert_eq!(p * p, Percentage(121));      <- this would panic
assert_eq!(p.big_mul(p), Percentage(121));  <- this will work fine

To understand why it works like that look at the multiplication of decimal does under the hood:

What happens inside (on the math side)

Most of this library uses really basic math, a few things that might not be obvious are listed below

Keeping the scale

An multiplication of two percentages (scale of 2) using just the values would look like this :

(x / 10^2) / (y / 10^2) = x/y

(God i hate gh for not allowing LaTeX)

Using numbers it would look like this:

10% / 10% = 10 / 10 = 1

Which is obviously wrong. What we need is multiplying everything by 10^scale at every division. So it should look like this

(x / 10^scale) / (y / 10^scale) × 10^scale = x / y × 10^scale

Which checks out with the example above

In general at every multiplication of values there needs to be a division, and vice versa. This was the first purpose of this library - to abstract it away to make for less code, bugs and wasted time.

The important thing here is that multiplication has to occur before division to keep the precision, but this is also abstracted away.

Rounding errors

By default every method rounds down but has a counterpart ending with up rounding the opposite way.

Rounding works by addition of denominator - 1 to the numerator, so the mulup_ would look like so:

(x × y + 10^scale - 1) / 10^scale

For example for 10% × 1%

(10 × 1 + (10^2 - 1)) / (10^2) = 109 / 100 = 1%

What happens inside (on a code level)

As you do know by this point the whole library is in a form of macro. Inside of it is an implementation of several traits in a generic form to allow calling methods between any two of the implementations.