2021-01-29 12:31:58 +01:00
|
|
|
//
|
|
|
|
// FILE: M62429.cpp
|
|
|
|
// AUTHOR: Rob Tillaart
|
|
|
|
// PURPOSE: Arduino library for M62429 volume control IC
|
2022-11-15 10:47:38 +01:00
|
|
|
// VERSION: 0.3.6
|
2021-01-29 12:31:58 +01:00
|
|
|
// HISTORY: See M62429.cpp2
|
|
|
|
// URL: https://github.com/RobTillaart/M62429
|
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// HISTORY: see changelog.md
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
#include "M62429.h"
|
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
#define M62429_MAX_ATTN 87 // decibel
|
2022-02-12 09:23:23 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
void M62429::begin(uint8_t dataPin, uint8_t clockPin)
|
|
|
|
{
|
|
|
|
_data = dataPin;
|
|
|
|
_clock = clockPin;
|
|
|
|
pinMode(_data, OUTPUT);
|
|
|
|
pinMode(_clock, OUTPUT);
|
|
|
|
digitalWrite(_data, LOW);
|
|
|
|
digitalWrite(_clock, LOW);
|
|
|
|
|
|
|
|
_muted = false;
|
|
|
|
setVolume(2, 0);
|
|
|
|
}
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
int M62429::getVolume(uint8_t channel)
|
|
|
|
{
|
|
|
|
if (channel > 2) return M62429_CHANNEL_ERROR;
|
|
|
|
return _vol[channel & 1];
|
|
|
|
}
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
int M62429::setVolume(uint8_t channel, uint8_t volume)
|
|
|
|
{
|
|
|
|
if (channel > 2) return M62429_CHANNEL_ERROR;
|
|
|
|
if (_muted) return M62429_MUTED;
|
|
|
|
|
2022-02-12 09:23:23 +01:00
|
|
|
uint16_t attn = (M62429_MAX_ATTN * volume)/255;
|
|
|
|
_setAttn(channel, attn);
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// update cached values
|
2022-02-12 09:51:57 +01:00
|
|
|
if (channel == 0) _vol[0] = volume;
|
|
|
|
else if (channel == 1) _vol[1] = volume;
|
|
|
|
else _vol[0] = _vol[1] = volume;
|
2022-01-05 16:44:57 +01:00
|
|
|
return M62429_OK;
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2022-02-13 15:58:42 +01:00
|
|
|
int M62429::incr(uint8_t channel)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2022-02-13 15:58:42 +01:00
|
|
|
if (channel > 2) return M62429_CHANNEL_ERROR;
|
2022-02-12 14:33:18 +01:00
|
|
|
if (_muted) return M62429_MUTED;
|
2022-02-13 15:58:42 +01:00
|
|
|
|
|
|
|
if ( ((channel == 0) || (channel == 2)) && (_vol[0] < 255))
|
2022-02-12 09:51:57 +01:00
|
|
|
{
|
|
|
|
_vol[0]++;
|
|
|
|
setVolume(0, _vol[0]);
|
|
|
|
}
|
2022-02-13 15:58:42 +01:00
|
|
|
if ( ((channel == 1) || (channel == 2)) && (_vol[1] < 255))
|
2022-02-12 09:51:57 +01:00
|
|
|
{
|
|
|
|
_vol[1]++;
|
|
|
|
setVolume(1, _vol[1]);
|
|
|
|
}
|
2022-02-12 14:33:18 +01:00
|
|
|
return M62429_OK;
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2022-02-13 15:58:42 +01:00
|
|
|
int M62429::decr(uint8_t channel)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2022-02-13 15:58:42 +01:00
|
|
|
if (channel > 2) return M62429_CHANNEL_ERROR;
|
2022-02-12 14:33:18 +01:00
|
|
|
if (_muted) return M62429_MUTED;
|
2022-02-13 15:58:42 +01:00
|
|
|
|
|
|
|
if ( ((channel == 0) || (channel == 2)) && (_vol[0] > 0))
|
2022-02-12 09:51:57 +01:00
|
|
|
{
|
|
|
|
_vol[0]--;
|
|
|
|
setVolume(0, _vol[0]);
|
|
|
|
}
|
2022-02-13 15:58:42 +01:00
|
|
|
if ( ((channel == 1) || (channel == 2)) && (_vol[1] > 0))
|
2022-02-12 09:51:57 +01:00
|
|
|
{
|
|
|
|
_vol[1]--;
|
|
|
|
setVolume(1, _vol[1]);
|
|
|
|
}
|
2022-02-12 14:33:18 +01:00
|
|
|
return M62429_OK;
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2022-02-12 14:33:18 +01:00
|
|
|
int M62429::average()
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2022-02-12 14:33:18 +01:00
|
|
|
if (_muted) return M62429_MUTED;
|
2021-01-29 12:31:58 +01:00
|
|
|
uint8_t v = (((int)_vol[0]) + _vol[1]) / 2;
|
2022-02-12 09:51:57 +01:00
|
|
|
setVolume(2, v);
|
2022-02-12 14:33:18 +01:00
|
|
|
return M62429_OK;
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
void M62429::muteOn()
|
|
|
|
{
|
2022-02-12 14:33:18 +01:00
|
|
|
if (_muted) return;
|
2021-01-29 12:31:58 +01:00
|
|
|
_muted = true;
|
2022-11-15 10:47:38 +01:00
|
|
|
// if ((_vol[0] > 0) || (_vol[1] > 0)) _setAttn(2, 0);
|
|
|
|
_setAttn(2, 0); // mute must work unconditional.
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-05-28 13:36:28 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
void M62429::muteOff()
|
|
|
|
{
|
2022-02-12 14:33:18 +01:00
|
|
|
if (_muted == false) return;
|
2021-01-29 12:31:58 +01:00
|
|
|
_muted = false;
|
2022-02-13 15:58:42 +01:00
|
|
|
if (_vol[0] > 0) setVolume(0, _vol[0]);
|
|
|
|
if (_vol[1] > 0) setVolume(1, _vol[1]);
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2022-11-15 10:47:38 +01:00
|
|
|
// PRIVATE
|
2021-01-29 12:31:58 +01:00
|
|
|
//
|
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// attn = 0..M62429_MAX_ATTN
|
2022-02-12 09:23:23 +01:00
|
|
|
void M62429::_setAttn(uint8_t channel, uint8_t attn)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2022-02-12 09:23:23 +01:00
|
|
|
uint16_t databits = 0x0200; // D9 latch bit
|
2022-01-05 16:44:57 +01:00
|
|
|
databits |= ((attn & 0x03) << 7); // D8 - D7
|
2022-02-12 09:23:23 +01:00
|
|
|
databits |= (attn & 0x7C); // D6 - D2
|
2022-11-15 10:47:38 +01:00
|
|
|
// channel == 2 -> both 0x00 is default
|
2022-01-05 16:44:57 +01:00
|
|
|
if (channel == 0) databits |= 0x03; // D0 - D1
|
|
|
|
if (channel == 1) databits |= 0x02; // D0 - D1
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2022-01-05 16:44:57 +01:00
|
|
|
// write D0 - D9
|
2022-02-12 09:23:23 +01:00
|
|
|
for (uint8_t i = 0; i < 10; i++)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
2022-02-12 09:23:23 +01:00
|
|
|
digitalWrite(_data, databits & 0x01);
|
|
|
|
databits >>= 1;
|
2021-01-29 12:31:58 +01:00
|
|
|
digitalWrite(_clock, HIGH);
|
2022-11-15 10:47:38 +01:00
|
|
|
// Note if _clock pulses are long enough, _data pulses are too.
|
2022-02-15 17:59:16 +01:00
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
2021-12-21 12:50:11 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
digitalWrite(_data, LOW);
|
|
|
|
digitalWrite(_clock, LOW);
|
2022-02-15 17:59:16 +01:00
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
2022-01-05 16:44:57 +01:00
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// Send D10 HIGH bit (Latch signal)
|
2021-01-29 12:31:58 +01:00
|
|
|
digitalWrite(_data, HIGH);
|
|
|
|
digitalWrite(_clock, HIGH);
|
2022-02-15 17:59:16 +01:00
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
2021-12-21 12:50:11 +01:00
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// latch D10 signal requires _clock low before _data
|
|
|
|
// make _data dummy write to keep timing constant
|
2022-01-05 16:44:57 +01:00
|
|
|
digitalWrite(_data, HIGH);
|
2021-01-29 12:31:58 +01:00
|
|
|
digitalWrite(_clock, LOW);
|
2022-02-15 17:59:16 +01:00
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
2022-01-05 16:44:57 +01:00
|
|
|
|
|
|
|
digitalWrite(_data, LOW);
|
2022-02-15 17:59:16 +01:00
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2021-12-21 12:50:11 +01:00
|
|
|
|
2022-02-21 19:21:27 +01:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// M62429_RAW
|
|
|
|
//
|
|
|
|
void M62429_RAW::begin(uint8_t dataPin, uint8_t clockPin)
|
|
|
|
{
|
|
|
|
_data = dataPin;
|
|
|
|
_clock = clockPin;
|
|
|
|
pinMode(_data, OUTPUT);
|
|
|
|
pinMode(_clock, OUTPUT);
|
|
|
|
digitalWrite(_data, LOW);
|
|
|
|
digitalWrite(_clock, LOW);
|
|
|
|
|
|
|
|
setAttn(2, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int M62429_RAW::getAttn(uint8_t channel)
|
|
|
|
{
|
|
|
|
return _attn[channel & 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void M62429_RAW::setAttn(uint8_t channel, uint8_t attn)
|
|
|
|
{
|
|
|
|
uint16_t databits = 0x0200; // D9 latch bit
|
|
|
|
databits |= ((attn & 0x03) << 7); // D8 - D7
|
|
|
|
databits |= (attn & 0x7C); // D6 - D2
|
2022-11-15 10:47:38 +01:00
|
|
|
// channel == 2 -> both 0x00 is default
|
2022-02-21 19:21:27 +01:00
|
|
|
if (channel == 0) databits |= 0x03; // D0 - D1
|
|
|
|
if (channel == 1) databits |= 0x02; // D0 - D1
|
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// write D0 - D9
|
2022-02-21 19:21:27 +01:00
|
|
|
for (uint8_t i = 0; i < 10; i++)
|
|
|
|
{
|
|
|
|
digitalWrite(_data, databits & 0x01);
|
|
|
|
databits >>= 1;
|
|
|
|
digitalWrite(_clock, HIGH);
|
|
|
|
// Note if _clock pulses are long enough, _data pulses are too.
|
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
digitalWrite(_data, LOW);
|
|
|
|
digitalWrite(_clock, LOW);
|
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// Send D10 HIGH bit (Latch signal)
|
2022-02-21 19:21:27 +01:00
|
|
|
digitalWrite(_data, HIGH);
|
|
|
|
digitalWrite(_clock, HIGH);
|
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// latch D10 signal requires _clock low before _data
|
|
|
|
// make _data dummy write to keep timing constant
|
2022-02-21 19:21:27 +01:00
|
|
|
digitalWrite(_data, HIGH);
|
|
|
|
digitalWrite(_clock, LOW);
|
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
digitalWrite(_data, LOW);
|
|
|
|
#if M62429_CLOCK_DELAY > 0
|
|
|
|
delayMicroseconds(M62429_CLOCK_DELAY);
|
|
|
|
#endif
|
|
|
|
|
2022-11-15 10:47:38 +01:00
|
|
|
// update cached values
|
2022-02-21 19:21:27 +01:00
|
|
|
if (channel == 0) _attn[0] = attn;
|
|
|
|
else if (channel == 1) _attn[1] = attn;
|
|
|
|
else _attn[0] = _attn[1] = attn;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
// -- END OF FILE --
|
2021-12-21 12:50:11 +01:00
|
|
|
|