160 lines
5.9 KiB
Markdown
Raw Normal View History

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)
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)
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
takes time enough.
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.
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
The interface is straightforward
2022-02-15 17:59:16 +01:00
Use **\#include "M62429.h"**
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
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
2022-02-15 17:59:16 +01:00
- separate releaseNotes.md?
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
2022-11-15 10:47:38 +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.
would add a lot of extra testing. the user can implement a **setVOlume(chan, 0)**
- **mute50()** reduces levels with 50% (rounded down?).
user can implement this **setVolume(chan, getVolume(chan)/2);**
2021-01-29 12:31:58 +01:00