2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/M62429/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
2021-12-21 12:50:11 +01:00
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/M62429/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/M62429/actions/workflows/arduino-lint.yml)
|
|
|
|
[![JSON check](https://github.com/RobTillaart/M62429/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/M62429/actions/workflows/jsoncheck.yml)
|
2023-11-08 19:04:11 +01:00
|
|
|
[![GitHub issues](https://img.shields.io/github/issues/RobTillaart/M62429.svg)](https://github.com/RobTillaart/M62429/issues)
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/M62429/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/M62429.svg?maxAge=3600)](https://github.com/RobTillaart/M62429/releases)
|
2023-11-08 19:04:11 +01:00
|
|
|
[![PlatformIO Registry](https://badges.registry.platformio.org/packages/robtillaart/library/M62429.svg)](https://registry.platformio.org/libraries/robtillaart/M62429)
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-12-21 12:50:11 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
# M62429
|
|
|
|
|
2021-12-21 12:50:11 +01:00
|
|
|
Arduino library for M62429 volume control IC.
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
## Description
|
|
|
|
|
2022-02-21 19:21:27 +01:00
|
|
|
#### M62429
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
This library is used to set the attenuation (volume) of the
|
|
|
|
M62429 IC a.k.a. FM62429.
|
|
|
|
|
|
|
|
The communication needs a minimum delay of 1.6 microseconds.
|
|
|
|
This is defined in the library in **M62429_CLOCK_DELAY** == 2
|
|
|
|
For AVR (UNO, slow device) it is defined as 0, as the digitalWrite
|
2023-11-08 19:04:11 +01:00
|
|
|
takes time enough.
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
For faster processors this define can be overruled runtime by setting it
|
2021-12-21 12:50:11 +01:00
|
|
|
before including "M62429.h" or by defining it as command line parameter.
|
|
|
|
|
2023-11-08 19:04:11 +01:00
|
|
|
|
2022-02-21 19:21:27 +01:00
|
|
|
#### M62429_RAW
|
|
|
|
|
|
|
|
The library also implements a M62429_RAW class which has a minimalistic interface.
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2022-02-21 19:21:27 +01:00
|
|
|
## Interface M62429
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2023-11-08 19:04:11 +01:00
|
|
|
```cpp
|
|
|
|
#include "M62429.h"
|
|
|
|
```
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2023-11-08 19:04:11 +01:00
|
|
|
The interface is straightforward
|
2022-02-15 17:59:16 +01:00
|
|
|
|
2023-11-08 19:04:11 +01:00
|
|
|
- **M62429()** Constructor
|
2021-12-21 12:50:11 +01:00
|
|
|
- **void begin(uint8_t dataPin, uint8_t clockPin)** defines the clock and data pin.
|
2021-01-29 12:31:58 +01:00
|
|
|
One has to create one object per IC.
|
2022-02-13 15:58:42 +01:00
|
|
|
- **int getVolume(uint8_t channel)** channel is 0, 1 or 2 (both).
|
2022-02-12 14:33:18 +01:00
|
|
|
In the latter case the volume of channel 0 is used as volume of both channels.
|
2022-02-13 15:58:42 +01:00
|
|
|
See remark future section.
|
2021-12-21 12:50:11 +01:00
|
|
|
- **int setVolume(uint8_t channel, uint8_t volume)**
|
2022-02-13 15:58:42 +01:00
|
|
|
- channel = 0, 1, 2 = both
|
|
|
|
- volume = 0 .. 255
|
2022-02-12 14:33:18 +01:00
|
|
|
- Note: if system is muted, no changes are made.
|
2022-02-13 15:58:42 +01:00
|
|
|
- **int incr(uint8_t channel = 2)** increment volume (both channels is default) until max (255) is reached.
|
2021-01-29 12:31:58 +01:00
|
|
|
This is another way to set volume that is better suited for a rotary
|
2022-02-13 15:58:42 +01:00
|
|
|
encoder or a \[+\] button.
|
2022-02-12 14:33:18 +01:00
|
|
|
- Note: if system is muted, no changes are made.
|
2022-02-13 15:58:42 +01:00
|
|
|
- **int decr(uint8_t channel = 2)** decrement volume (both channels is default) until 0 is reached.
|
2022-02-12 14:33:18 +01:00
|
|
|
- Note: if system is muted, no changes are made.
|
|
|
|
- **int average()** averages the 2 channels to same = average level.
|
2021-01-29 12:31:58 +01:00
|
|
|
Sort of set balance in the middle functionality.
|
2022-02-12 14:33:18 +01:00
|
|
|
- Note: if system is muted, no changes are made.
|
2022-02-13 15:58:42 +01:00
|
|
|
- **void muteOn()** silences both channels but remembers the volume.
|
2021-01-29 12:31:58 +01:00
|
|
|
GetVolume() will return the 'saved' volume value.
|
2021-12-21 12:50:11 +01:00
|
|
|
- **void muteOff()** resets the volume per channel again.
|
|
|
|
- **bool isMuted()** returns the muted state.
|
|
|
|
|
2022-02-12 09:23:23 +01:00
|
|
|
Note: the volume goes from 0..255 while the actual steps go from 0..87.
|
2022-02-12 14:33:18 +01:00
|
|
|
Therefore not every step in volume will make a "real" step (roughly 1 in 3).
|
|
|
|
This choice is made as the range 0..255 is more often used than the 0..87 range
|
|
|
|
and therefore better fits other sensors and devices.
|
|
|
|
|
|
|
|
|
|
|
|
#### Error codes
|
|
|
|
|
|
|
|
The functions **getVolume(), setVolume(), incr(), decr()** and **average()**
|
|
|
|
can return one of the error codes.
|
|
|
|
|
|
|
|
|
|
|
|
| value | name | notes |
|
|
|
|
|:-----:|:---------------------|:----------|
|
|
|
|
| 0 | M62429_OK | no error |
|
|
|
|
| -1 | M62429_MUTED | system is muted, use **muteOff()** |
|
|
|
|
| -10 | M62429_CHANNEL_ERROR | channel must be 0, 1 or 2 |
|
2022-02-12 09:23:23 +01:00
|
|
|
|
2021-12-21 12:50:11 +01:00
|
|
|
|
|
|
|
## Operation
|
|
|
|
|
|
|
|
See examples
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
|
2022-02-21 19:21:27 +01:00
|
|
|
|
|
|
|
## Interface M62429_RAW
|
|
|
|
|
|
|
|
The interface is minimalistic.
|
|
|
|
No parameter or other checks are made.
|
|
|
|
|
|
|
|
Use **\#include "M62429.h"**
|
|
|
|
|
|
|
|
- **void begin(uint8_t dataPin, uint8_t clockPin)** defines the clock and data pin.
|
|
|
|
One has to create one object per IC.
|
|
|
|
- **int getAttn(uint8_t channel)** channel is 0, 1 or 2 (both).
|
|
|
|
In the latter case the attenuation of channel 0 is used as attenuation of both channels.
|
|
|
|
- **void setAttn(uint8_t channel, uint8_t attn)**
|
|
|
|
- channel = 0, 1, 2 = both
|
|
|
|
- attenuation = 0 .. 87
|
|
|
|
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
## Future
|
|
|
|
|
2023-11-08 19:04:11 +01:00
|
|
|
#### Must
|
|
|
|
|
|
|
|
- improve documentation
|
|
|
|
|
|
|
|
#### Should
|
|
|
|
|
|
|
|
#### Could
|
|
|
|
|
2022-01-05 16:44:57 +01:00
|
|
|
#### Mixer
|
|
|
|
|
|
|
|
- Control multiple M62429 IC's
|
|
|
|
- One shared dataPin and clockPin per IC.
|
|
|
|
- use a PCF8574 / PCF8575 as selector
|
|
|
|
- would allow for 16 stereo or 32 mono channels.
|
|
|
|
Runtime configuration mono / stereo would be cool.
|
|
|
|
|
|
|
|
```
|
|
|
|
PCF8574 or PCF8575 is used as device selector
|
|
|
|
AND port takes DATA and CLOCK to right M62429
|
|
|
|
identical schema for all 8/16 ports
|
|
|
|
|
|
|
|
PCF8575 AND M62429
|
|
|
|
+---------+ +---+ +---------+
|
|
|
|
| |=======+======| & |==========| data |
|
|
|
|
| | +===|======| | | |
|
|
|
|
- I2C - | | | | +---+ | |
|
|
|
|
| | | | | |
|
|
|
|
| | | | +---+ | |
|
|
|
|
| | | +======| & |==========| clock |
|
|
|
|
| | | +====| | | |
|
|
|
|
| | | | +---+ | |
|
|
|
|
+---------+ | | +---------+
|
|
|
|
| |
|
|
|
|
DATA ==============+ |
|
|
|
|
CLOCK ====================+
|
|
|
|
```
|
|
|
|
|
|
|
|
|
2022-02-12 14:33:18 +01:00
|
|
|
#### Volume pan model
|
2022-01-05 16:44:57 +01:00
|
|
|
|
|
|
|
- model with one **volume(0..100%)** and one **balance(-100..100)** == **pan()**.
|
2022-02-12 14:33:18 +01:00
|
|
|
- Also a **left()** and **right()** incremental balance might be added.
|
|
|
|
|
|
|
|
Does this model work better than 2 separate volume channels?
|
|
|
|
|
|
|
|
|
|
|
|
#### Other
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
- change **getVolume(both)** to return max of the two channels?
|
2022-02-12 14:33:18 +01:00
|
|
|
would be better.
|
|
|
|
- **min()** to reduce volume to minimum of both channels.
|
|
|
|
- create **setAttn(channel, attn)** and **getAttn()** for low level control.
|
|
|
|
- needs cached attn values.
|
2022-01-05 16:44:57 +01:00
|
|
|
|
2023-11-08 19:04:11 +01:00
|
|
|
|
|
|
|
#### Wont
|
|
|
|
|
2022-01-05 16:44:57 +01:00
|
|
|
- **muteOff()** should increase gradually. takes too much blocking time.
|
2022-02-12 14:33:18 +01:00
|
|
|
- **Mute()** could be per channel, default = both / all.
|
2023-11-08 19:04:11 +01:00
|
|
|
would add a lot of extra testing. the user can implement a **setVolume(chan, 0)**
|
2022-02-12 14:33:18 +01:00
|
|
|
- **mute50()** reduces levels with 50% (rounded down?).
|
|
|
|
user can implement this **setVolume(chan, getVolume(chan)/2);**
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2023-11-08 19:04:11 +01:00
|
|
|
|
|
|
|
## Support
|
|
|
|
|
|
|
|
If you appreciate my libraries, you can support the development and maintenance.
|
|
|
|
Improve the quality of the libraries by providing issues and Pull Requests, or
|
|
|
|
donate through PayPal or GitHub sponsors.
|
|
|
|
|
|
|
|
Thank you,
|
|
|
|
|