141 lines
2.5 KiB
C
Raw Normal View History

2021-01-29 12:31:58 +01:00
#pragma once
//
// FILE: gamma.h
// AUTHOR: Rob Tillaart
2021-12-18 17:16:37 +01:00
// VERSION: 0.2.1
2021-01-29 12:31:58 +01:00
// DATE: 2020-08-08
// PURPOSE: Arduino Library to efficiently hold a gamma lookup table
// 0.1.0 2020-08-08 initial release
2021-12-18 17:16:37 +01:00
// 0.1.1 2020-12-24 arduino-ci + unit test
2021-11-02 15:15:11 +01:00
// 0.2.0 2021-11-02 update build-CI, badges
// add begin() - fixes ESP32 crash.
2021-12-18 17:16:37 +01:00
// 0.2.1 2021-12-18 update library.json, license,
// add constants, minor edits.
2021-01-29 12:31:58 +01:00
#include "Arduino.h"
2021-12-18 17:16:37 +01:00
#define GAMMA_LIB_VERSION (F("0.2.1"))
#define GAMMA_DEFAULT_SIZE 32
#define GAMMA_MAX_SIZE 256
2021-01-29 12:31:58 +01:00
class GAMMA
{
public:
2021-12-18 17:16:37 +01:00
GAMMA(uint16_t size = GAMMA_DEFAULT_SIZE)
2021-01-29 12:31:58 +01:00
{
_shift = 7;
// force power of 2; get shift & mask right
2021-12-18 17:16:37 +01:00
for (uint16_t s = 2; s <= GAMMA_MAX_SIZE; s <<= 1)
2021-01-29 12:31:58 +01:00
{
if (size <= s)
{
_size = s;
break;
}
_shift--;
}
_mask = (1 << _shift) - 1;
2021-12-18 17:16:37 +01:00
_interval = GAMMA_MAX_SIZE / _size;
2021-01-29 12:31:58 +01:00
}
2021-11-02 15:15:11 +01:00
2021-01-29 12:31:58 +01:00
~GAMMA()
{
if (_table) free(_table);
}
2021-11-02 15:15:11 +01:00
void begin()
{
if (_table == NULL)
{
_table = (uint8_t *)malloc(_size + 1);
}
setGamma(2.8);
}
2021-01-29 12:31:58 +01:00
void setGamma(float gamma)
{
if (_gamma != gamma)
{
yield(); // keep ESP happy
_gamma = gamma;
for (uint16_t i = 0; i < _size; i++)
{
_table[i] = pow(i * _interval * (1.0 / 255), _gamma) * 255 + 0.5;
}
_table[_size] = 255; // anchor for interpolation..
}
}
2021-11-02 15:15:11 +01:00
2021-01-29 12:31:58 +01:00
float getGamma()
{
return _gamma;
};
2021-11-02 15:15:11 +01:00
2021-01-29 12:31:58 +01:00
uint8_t operator[] (uint8_t idx)
{
if (_interval == 1) return _table[idx];
// else interpolate
uint8_t i = idx >> _shift;
uint8_t m = idx & _mask;
// exact element shortcut
if ( m == 0 ) return _table[i];
// interpolation
uint16_t delta = _table[i+1] - _table[i];
delta = (delta * m + _interval/2) >> _shift; // == /_interval;
return _table[i] + delta;
}
2021-11-02 15:15:11 +01:00
2021-01-29 12:31:58 +01:00
uint16_t size()
{
return _size + 1;
};
2021-11-02 15:15:11 +01:00
2021-01-29 12:31:58 +01:00
int distinct()
{
int last = _table[0];
int cnt = 1;
for (uint16_t i = 1; i < _size; i++)
{
if (_table[i] == last) continue;
last = _table[i];
cnt++;
}
return cnt;
}
2021-11-02 15:15:11 +01:00
2021-01-29 12:31:58 +01:00
void dump()
{
for (uint16_t i = 0; i < _size; i++)
{
Serial.println(_table[i]);
}
}
private:
uint8_t _shift = 0;
uint8_t _mask = 0;
uint16_t _size = 0;
uint8_t _interval = 0;
float _gamma = 0;
2021-11-02 15:15:11 +01:00
uint8_t * _table = NULL;
2021-01-29 12:31:58 +01:00
};
2021-11-02 15:15:11 +01:00
2021-01-29 12:31:58 +01:00
// -- END OF FILE --
2021-11-02 15:15:11 +01:00