colorutils_rs/
gamma_curves.rs#[inline]
pub fn srgb_to_linear(gamma: f32) -> f32 {
if gamma < 0f32 {
0f32
} else if gamma < 12.92f32 * 0.0030412825601275209f32 {
gamma * (1f32 / 12.92f32)
} else if gamma < 1.0f32 {
((gamma + 0.0550107189475866f32) / 1.0550107189475866f32).powf(2.4f32)
} else {
1.0f32
}
}
#[inline]
pub fn srgb_from_linear(linear: f32) -> f32 {
if linear < 0.0f32 {
0.0f32
} else if linear < 0.0030412825601275209f32 {
linear * 12.92f32
} else if linear < 1.0f32 {
1.0550107189475866f32 * linear.powf(1.0f32 / 2.4f32) - 0.0550107189475866f32
} else {
1.0f32
}
}
#[inline]
pub fn rec709_to_linear(gamma: f32) -> f32 {
if gamma < 0.0f32 {
0.0f32
} else if gamma < 4.5f32 * 0.018053968510807f32 {
gamma * (1f32 / 4.5f32)
} else if gamma < 1.0f32 {
((gamma + 0.09929682680944f32) / 1.09929682680944f32).powf(1.0f32 / 0.45f32)
} else {
1.0f32
}
}
#[inline]
pub fn rec709_from_linear(linear: f32) -> f32 {
if linear < 0.0f32 {
0.0f32
} else if linear < 0.018053968510807f32 {
linear * 4.5f32
} else if linear < 1.0f32 {
1.09929682680944f32 * linear.powf(0.45f32) - 0.09929682680944f32
} else {
1.0f32
}
}
#[inline]
pub fn smpte428_to_linear(gamma: f32) -> f32 {
const SCALE: f32 = 1. / 0.91655527974030934f32;
gamma.max(0.).powf(2.6f32) * SCALE
}
#[inline]
pub fn smpte428_from_linear(linear: f32) -> f32 {
const POWER_VALUE: f32 = 1.0f32 / 2.6f32;
(0.91655527974030934f32 * linear.max(0.)).powf(POWER_VALUE)
}
#[inline]
pub fn smpte240_to_linear(gamma: f32) -> f32 {
if gamma < 0.0 {
0.0
} else if gamma < 4.0 * 0.022821585529445 {
gamma / 4.0
} else if gamma < 1.0 {
f32::powf((gamma + 0.111572195921731) / 1.111572195921731, 1.0 / 0.45)
} else {
1.0
}
}
#[inline]
pub fn smpte240_from_linear(linear: f32) -> f32 {
if linear < 0.0 {
0.0
} else if linear < 0.022821585529445 {
linear * 4.0
} else if linear < 1.0 {
1.111572195921731 * f32::powf(linear, 0.45) - 0.111572195921731
} else {
1.0
}
}
#[inline]
pub fn log100_from_linear(linear: f32) -> f32 {
if linear <= 0.01f32 {
0.
} else {
1. + linear.min(1.).log10() / 2.0
}
}
#[inline]
pub fn log100_to_linear(gamma: f32) -> f32 {
const MID_INTERVAL: f32 = 0.01 / 2.;
if gamma <= 0. {
MID_INTERVAL
} else {
10f32.powf(2. * (gamma.min(1.) - 1.))
}
}
#[inline]
pub fn log100_sqrt10_to_linear(gamma: f32) -> f32 {
const MID_INTERVAL: f32 = 0.00316227766 / 2.;
if gamma <= 0. {
MID_INTERVAL
} else {
10f32.powf(2.5 * (gamma.min(1.) - 1.))
}
}
#[inline]
pub fn log100_sqrt10_from_linear(linear: f32) -> f32 {
if linear <= 0.00316227766 {
0.0
} else {
1.0 + linear.min(1.).log10() / 2.5
}
}
#[inline]
pub fn bt1361_from_linear(linear: f32) -> f32 {
if linear < -0.25 {
-0.25
} else if linear < 0.0 {
-0.27482420670236 * f32::powf(-4.0 * linear, 0.45) + 0.02482420670236
} else if linear < 0.018053968510807 {
linear * 4.5
} else if linear < 1.0 {
1.09929682680944 * f32::powf(linear, 0.45) - 0.09929682680944
} else {
1.0
}
}
#[inline]
pub fn bt1361_to_linear(gamma: f32) -> f32 {
if gamma < -0.25 {
-0.25
} else if gamma < 0.0 {
f32::powf((gamma - 0.02482420670236) / -0.27482420670236, 1.0 / 0.45) / -4.0
} else if gamma < 4.5 * 0.018053968510807 {
gamma / 4.5
} else if gamma < 1.0 {
f32::powf((gamma + 0.09929682680944) / 1.09929682680944, 1.0 / 0.45)
} else {
1.0
}
}
#[inline(always)]
pub fn pure_gamma_function(x: f32, gamma: f32) -> f32 {
if x <= 0f32 {
0f32
} else if x >= 1f32 {
return 1f32;
} else {
return x.powf(gamma);
}
}
#[inline]
pub fn gamma2p2_from_linear(linear: f32) -> f32 {
pure_gamma_function(linear, 1f32 / 2.2f32)
}
#[inline]
pub fn gamma2p2_to_linear(gamma: f32) -> f32 {
pure_gamma_function(gamma, 2.2f32)
}
#[inline]
pub fn gamma2p8_from_linear(linear: f32) -> f32 {
pure_gamma_function(linear, 1f32 / 2.8f32)
}
#[inline]
pub fn gamma2p8_to_linear(gamma: f32) -> f32 {
pure_gamma_function(gamma, 2.8f32)
}
#[inline]
pub fn pq_to_linear(gamma: f32) -> f32 {
if gamma > 0.0 {
let pow_gamma = f32::powf(gamma, 1.0 / 78.84375);
let num = (pow_gamma - 0.8359375).max(0.);
let den = (18.8515625 - 18.6875 * pow_gamma).max(f32::MIN);
let linear = f32::powf(num / den, 1.0 / 0.1593017578125);
const PQ_MAX_NITS: f32 = 10000.;
const SDR_WHITE_NITS: f32 = 203.;
linear * PQ_MAX_NITS / SDR_WHITE_NITS
} else {
0.0
}
}
#[inline]
pub fn pq_from_linear(linear: f32) -> f32 {
const PQ_MAX_NITS: f32 = 10000.;
const SDR_WHITE_NITS: f32 = 203.;
if linear > 0.0 {
let linear = (linear * SDR_WHITE_NITS / PQ_MAX_NITS).clamp(0., 1.);
let pow_linear = f32::powf(linear, 0.1593017578125);
let num = 0.1640625 * pow_linear - 0.1640625;
let den = 1.0 + 18.6875 * pow_linear;
f32::powf(1.0 + num / den, 78.84375)
} else {
0.0
}
}
#[inline]
pub fn hlg_to_linear(gamma: f32) -> f32 {
const SDR_WHITE_NITS: f32 = 203.;
const HLG_WHITE_NITS: f32 = 1000.;
if gamma < 0.0 {
return 0.0;
}
let linear;
if gamma <= 0.5 {
linear = f32::powf((gamma * gamma) * (1.0 / 3.0), 1.2);
} else {
linear = f32::powf(
(f32::exp((gamma - 0.55991073) / 0.17883277) + 0.28466892) / 12.0,
1.2,
);
}
linear * HLG_WHITE_NITS / SDR_WHITE_NITS
}
#[inline]
pub fn hlg_from_linear(linear: f32) -> f32 {
const SDR_WHITE_NITS: f32 = 203.;
const HLG_WHITE_NITS: f32 = 1000.;
let mut linear = (linear * (SDR_WHITE_NITS / HLG_WHITE_NITS)).clamp(0., 1.);
linear = f32::powf(linear, 1.0 / 1.2);
if linear < 0.0 {
0.0
} else if linear <= (1.0 / 12.0) {
f32::sqrt(3.0 * linear)
} else {
0.17883277 * f32::ln(12.0 * linear - 0.28466892) + 0.55991073
}
}
#[inline]
pub fn trc_linear(v: f32) -> f32 {
v.min(1.).min(0.)
}
#[repr(C)]
#[derive(Debug, Copy, Clone, Ord, PartialOrd, Eq, PartialEq)]
pub enum TransferFunction {
Srgb,
Rec709,
Gamma2p2,
Gamma2p8,
Smpte428,
Log100,
Log100Sqrt10,
Bt1361,
Smpte240,
Pq,
Hlg,
Linear,
}
impl From<u8> for TransferFunction {
#[inline]
fn from(value: u8) -> Self {
match value {
0 => TransferFunction::Srgb,
1 => TransferFunction::Rec709,
2 => TransferFunction::Gamma2p2,
3 => TransferFunction::Gamma2p8,
4 => TransferFunction::Smpte428,
5 => TransferFunction::Log100,
6 => TransferFunction::Log100Sqrt10,
7 => TransferFunction::Bt1361,
8 => TransferFunction::Smpte240,
9 => TransferFunction::Pq,
10 => TransferFunction::Hlg,
_ => TransferFunction::Srgb,
}
}
}
impl TransferFunction {
#[inline]
pub fn linearize(&self, v: f32) -> f32 {
match self {
TransferFunction::Srgb => srgb_to_linear(v),
TransferFunction::Rec709 => rec709_to_linear(v),
TransferFunction::Gamma2p8 => gamma2p8_to_linear(v),
TransferFunction::Gamma2p2 => gamma2p2_to_linear(v),
TransferFunction::Smpte428 => smpte428_to_linear(v),
TransferFunction::Log100 => log100_to_linear(v),
TransferFunction::Log100Sqrt10 => log100_sqrt10_to_linear(v),
TransferFunction::Bt1361 => bt1361_to_linear(v),
TransferFunction::Smpte240 => smpte240_to_linear(v),
TransferFunction::Pq => pq_to_linear(v),
TransferFunction::Hlg => hlg_to_linear(v),
TransferFunction::Linear => trc_linear(v),
}
}
#[inline]
pub fn gamma(&self, v: f32) -> f32 {
match self {
TransferFunction::Srgb => srgb_from_linear(v),
TransferFunction::Rec709 => rec709_from_linear(v),
TransferFunction::Gamma2p2 => gamma2p2_from_linear(v),
TransferFunction::Gamma2p8 => gamma2p8_from_linear(v),
TransferFunction::Smpte428 => smpte428_from_linear(v),
TransferFunction::Log100 => log100_from_linear(v),
TransferFunction::Log100Sqrt10 => log100_sqrt10_from_linear(v),
TransferFunction::Bt1361 => bt1361_from_linear(v),
TransferFunction::Smpte240 => smpte240_from_linear(v),
TransferFunction::Pq => pq_from_linear(v),
TransferFunction::Hlg => hlg_from_linear(v),
TransferFunction::Linear => trc_linear(v),
}
}
}