2021-01-29 12:31:58 +01:00
|
|
|
#pragma once
|
|
|
|
//
|
|
|
|
// FILE: gamma.h
|
|
|
|
// AUTHOR: Rob Tillaart
|
2021-11-02 15:15:11 +01:00
|
|
|
// VERSION: 0.2.0
|
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
|
|
|
|
// 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-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
|
|
|
|
#include "Arduino.h"
|
|
|
|
|
2021-11-02 15:15:11 +01:00
|
|
|
#define GAMMA_LIB_VERSION (F("0.2.0"))
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
|
|
|
|
class GAMMA
|
|
|
|
{
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
GAMMA(uint16_t size = 32)
|
|
|
|
{
|
|
|
|
_shift = 7;
|
|
|
|
// force power of 2; get shift & mask right
|
|
|
|
for (uint16_t s = 2; s < 512; s <<= 1)
|
|
|
|
{
|
|
|
|
if (size <= s)
|
|
|
|
{
|
|
|
|
_size = s;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
_shift--;
|
|
|
|
}
|
|
|
|
_mask = (1 << _shift) - 1;
|
|
|
|
_interval = 256 / _size;
|
2021-11-02 15:15:11 +01:00
|
|
|
// removed malloc from constructor for ESP32
|
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
|
|
|
|