diff --git a/utilities/Cargo.toml b/utilities/Cargo.toml
index 59772f20b631ccb8dad528f713861b5badf3334d..72ab82977dd4553305fc187894c496b7886080ec 100644
--- a/utilities/Cargo.toml
+++ b/utilities/Cargo.toml
@@ -7,6 +7,7 @@ edition = "2018"
 [dependencies]
 serde = { version = "1.0", optional = true }
 codec = { package = "parity-scale-codec", version = "1.0.0", default-features = false }
+primitives = { package = "substrate-primitives",  git = "https://github.com/paritytech/substrate.git", default-features = false }
 sr-primitives = { git = "https://github.com/paritytech/substrate.git", default-features = false }
 runtime-io = { package = "sr-io", git = "https://github.com/paritytech/substrate.git", default-features = false }
 rstd = { package = "sr-std", git = "https://github.com/paritytech/substrate.git", default-features = false }
@@ -15,7 +16,6 @@ frame-system = { git = "https://github.com/paritytech/substrate.git", default-fe
 frame-support = { git = "https://github.com/paritytech/substrate.git", default-features = false }
 
 [dev-dependencies]
-primitives = { package = "substrate-primitives",  git = "https://github.com/paritytech/substrate.git", default-features = false }
 
 clear_on_drop = { version = "0.2.3", features = ["no_cc"] }	# https://github.com/paritytech/substrate/issues/4179
 
diff --git a/utilities/src/fixed128.rs b/utilities/src/fixed128.rs
new file mode 100644
index 0000000000000000000000000000000000000000..0085c769cb36b310bc9f83b5f037785cd606edb4
--- /dev/null
+++ b/utilities/src/fixed128.rs
@@ -0,0 +1,256 @@
+use codec::{Decode, Encode};
+use primitives::U256;
+use rstd::convert::{Into, TryFrom, TryInto};
+use sr_primitives::traits::{Bounded, Saturating};
+
+/// An unsigned fixed point number. Can hold any value in the range [0, 340_282_366_920_938_463_464]
+/// with fixed point accuracy of 10 ** 18
+#[derive(Encode, Decode, Default, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
+pub struct FixedU128(u128);
+
+const DIV: u128 = 1_000_000_000_000_000_000;
+
+impl FixedU128 {
+	/// create self from a natural number
+	///
+	/// Note that this might be lossy
+	pub fn from_natural(int: u128) -> Self {
+		Self(int.saturating_mul(DIV))
+	}
+
+	pub fn accuracy() -> u128 {
+		DIV
+	}
+
+	/// raw constructor. Equal to `parts / DIV`.
+	pub fn from_parts(parts: u128) -> Self {
+		Self(parts)
+	}
+
+	/// creates self from a rational number. Equal to `n/d`
+	///
+	/// Note that this might be lossy
+	pub fn from_rational(n: u128, d: u128) -> Self {
+		Self(
+			(U256::from(n).saturating_mul(U256::from(DIV)) / U256::from(d).max(U256::one()))
+				.try_into()
+				.unwrap_or_else(|_| Bounded::max_value()),
+		)
+	}
+
+	/// consume self and return the inner value.
+	///
+	/// This should only be used for testing.
+	#[cfg(any(feature = "std", test))]
+	pub fn into_inner(self) -> u128 {
+		self.0
+	}
+
+	// checked add for FixedU128
+	pub fn checked_add(&self, rhs: &Self) -> Option<Self> {
+		self.0.checked_add(rhs.0).map(Self)
+	}
+
+	// checked sub for FixedU128
+	pub fn checked_sub(&self, rhs: &Self) -> Option<Self> {
+		self.0.checked_sub(rhs.0).map(Self)
+	}
+
+	// checked mul for FixedU128
+	pub fn checked_mul(&self, rhs: &Self) -> Option<Self> {
+		if let Some(r) = U256::from(self.0)
+			.checked_mul(U256::from(rhs.0))
+			.and_then(|n| n.checked_div(U256::from(DIV)))
+		{
+			if let Ok(r) = TryInto::<u128>::try_into(r) {
+				return Some(Self(r));
+			}
+		}
+
+		None
+	}
+
+	// checked div for FixedU128
+	pub fn checked_div(&self, rhs: &Self) -> Option<Self> {
+		if let Some(r) = U256::from(self.0)
+			.checked_mul(U256::from(DIV))
+			.and_then(|n| n.checked_div(U256::from(rhs.0)))
+		{
+			if let Ok(r) = TryInto::<u128>::try_into(r) {
+				return Some(Self(r));
+			}
+		}
+
+		None
+	}
+
+	/// checked mul for type N
+	pub fn checked_mul_int<N>(&self, other: &N) -> Option<N>
+	where
+		N: Copy + TryFrom<u128> + TryInto<u128>,
+	{
+		if let Ok(n) = N::try_into(*other) {
+			if let Some(n) = U256::from(self.0)
+				.checked_mul(U256::from(n))
+				.and_then(|n| n.checked_div(U256::from(DIV)))
+			{
+				if let Ok(r) = TryInto::<u128>::try_into(n) {
+					if let Ok(r) = TryInto::<N>::try_into(r) {
+						return Some(r);
+					}
+				}
+			}
+		}
+
+		None
+	}
+
+	/// checked div for type N
+	pub fn checked_div_int<N>(&self, other: &N) -> Option<N>
+	where
+		N: Copy + TryFrom<u128> + TryInto<u128>,
+	{
+		if let Ok(n) = N::try_into(*other) {
+			if let Some(n) = self.0.checked_div(n).and_then(|n| n.checked_div(DIV)) {
+				if let Ok(r) = TryInto::<N>::try_into(n) {
+					return Some(r);
+				}
+			}
+		}
+
+		None
+	}
+}
+
+impl Saturating for FixedU128 {
+	fn saturating_add(self, rhs: Self) -> Self {
+		Self(self.0.saturating_add(rhs.0))
+	}
+
+	fn saturating_mul(self, rhs: Self) -> Self {
+		Self(
+			(U256::from(self.0).saturating_mul(U256::from(rhs.0)) / U256::from(DIV))
+				.try_into()
+				.unwrap_or_else(|_| Bounded::max_value()),
+		)
+	}
+
+	fn saturating_sub(self, rhs: Self) -> Self {
+		Self(self.0.saturating_sub(rhs.0))
+	}
+}
+
+impl Bounded for FixedU128 {
+	fn max_value() -> Self {
+		Self(u128::max_value())
+	}
+
+	fn min_value() -> Self {
+		Self(0u128)
+	}
+}
+
+impl rstd::fmt::Debug for FixedU128 {
+	#[cfg(feature = "std")]
+	fn fmt(&self, f: &mut rstd::fmt::Formatter) -> rstd::fmt::Result {
+		write!(f, "FixedU128({},{})", self.0 / DIV, (self.0 % DIV) / 1000)
+	}
+
+	#[cfg(not(feature = "std"))]
+	fn fmt(&self, _: &mut rstd::fmt::Formatter) -> rstd::fmt::Result {
+		Ok(())
+	}
+}
+
+#[cfg(test)]
+mod tests {
+	use super::*;
+
+	fn max() -> FixedU128 {
+		FixedU128::from_parts(u128::max_value())
+	}
+
+	#[test]
+	fn fixed128_semantics() {
+		assert_eq!(FixedU128::from_rational(5, 2).0, 5 * 1_000_000_000_000_000_000 / 2);
+		assert_eq!(FixedU128::from_rational(5, 2), FixedU128::from_rational(10, 4));
+		assert_eq!(FixedU128::from_rational(5, 0), FixedU128::from_rational(5, 1));
+
+		// biggest value that can be created.
+		assert_ne!(max(), FixedU128::from_natural(340_282_366_920_938_463_463));
+		assert_eq!(max(), FixedU128::from_natural(340_282_366_920_938_463_464));
+	}
+
+	#[test]
+	fn fixed128_operation() {
+		let a = FixedU128::from_natural(2);
+		let b = FixedU128::from_natural(1);
+		assert_eq!(a.checked_add(&b), Some(FixedU128::from_natural(1 + 2)));
+		assert_eq!(a.checked_sub(&b), Some(FixedU128::from_natural(2 - 1)));
+		assert_eq!(a.checked_mul(&b), Some(FixedU128::from_natural(1 * 2)));
+		assert_eq!(a.checked_div(&b), Some(FixedU128::from_rational(2, 1)));
+
+		let a = FixedU128::from_rational(5, 2);
+		let b = FixedU128::from_rational(3, 2);
+		assert_eq!(a.checked_add(&b), Some(FixedU128::from_rational(8, 2)));
+		assert_eq!(a.checked_sub(&b), Some(FixedU128::from_rational(2, 2)));
+		assert_eq!(a.checked_mul(&b), Some(FixedU128::from_rational(15, 4)));
+		assert_eq!(a.checked_div(&b), Some(FixedU128::from_rational(10, 6)));
+
+		let a = FixedU128::from_natural(120);
+		let b = 2i32;
+		assert_eq!(a.checked_div_int::<i32>(&b), Some(60));
+
+		let a = FixedU128::from_rational(20, 1);
+		let b = 2i32;
+		assert_eq!(a.checked_div_int::<i32>(&b), Some(10));
+
+		let a = FixedU128::from_natural(120);
+		let b = 2i32;
+		assert_eq!(a.checked_mul_int::<i32>(&b), Some(240));
+
+		let a = FixedU128::from_rational(1, 2);
+		let b = 20i32;
+		assert_eq!(a.checked_mul_int::<i32>(&b), Some(10));
+	}
+
+	#[test]
+	fn checked_div_with_zero_should_be_none() {
+		let a = FixedU128::from_natural(1);
+		let b = FixedU128::from_natural(0);
+		assert_eq!(a.checked_div(&b), None);
+	}
+
+	#[test]
+	fn checked_div_int_with_zero_should_be_none() {
+		let a = FixedU128::from_natural(1);
+		let b = 0i32;
+		assert_eq!(a.checked_div_int(&b), None);
+	}
+
+	#[test]
+	fn under_flow_should_be_none() {
+		let a = FixedU128::from_natural(2);
+		let b = FixedU128::from_natural(3);
+		assert_eq!(a.checked_sub(&b), None);
+	}
+
+	#[test]
+	fn over_flow_should_be_none() {
+		let a = FixedU128::from_parts(u128::max_value() - 1);
+		let b = FixedU128::from_parts(2);
+		assert_eq!(a.checked_add(&b), None);
+
+		let a = FixedU128::max_value();
+		let b = FixedU128::from_rational(2, 1);
+		assert_eq!(a.checked_mul(&b), None);
+
+		let a = FixedU128::from_natural(255);
+		let b = 2u8;
+		assert_eq!(a.checked_mul_int(&b), None);
+
+		let a = FixedU128::from_natural(256);
+		let b = 1u8;
+		assert_eq!(a.checked_div_int(&b), None);
+	}
+}
diff --git a/utilities/src/lib.rs b/utilities/src/lib.rs
index 6b864cc0748b4ff660f614de5dfc7b822e8af9c4..d158e8b59ef60f2561488a81355ecc8f48917fbf 100644
--- a/utilities/src/lib.rs
+++ b/utilities/src/lib.rs
@@ -1,5 +1,7 @@
 #![cfg_attr(not(feature = "std"), no_std)]
 
+pub mod fixed128;
 pub mod linked_item;
 
+pub use fixed128::FixedU128;
 pub use linked_item::{LinkedItem, LinkedList};