micd: scale sound volume with ambient noise level (#26399)
* test changing sound volume * test changing sound volume * create system/hardware/pc/hardware.h * implement Hardware::set_volume using pactl * soundd: use Hardware::set_volume * add sounddevice dependency * sounddevice example * simple micd * cleanup * remove this * fix process config * add to release files * hardware: get sound input device * no more offroad * debug * calculate volume from all measurements since last update * use microphone noise level to update sound volume * fix scale * mute microphone during alerts * log raw noise level * hardware: reduce tici min volume * improve scale * add package * clear measurements on muted * change default to min volume and respond quicker * fixes Co-authored-by: Shane Smiskol <shane@smiskol.com> * logarithmic scaling * fix * respond quicker * fixes * tweak scaling * specify default device * Revert "hardware: get sound input device" This reverts commit 50f594f7a3bab005023482bc793147a8c8dae5d7. * tuning * forgot to update submaster * tuning * don't mute microphone, and clip measurement * remove submaster * fixes * tuning * implement Hardware::set_volume using pactl * Revert "test changing sound volume" This reverts committaco4bbd870746
. * draft * draft * calculate sound pressure level in dB * fix setting * faster filter * start at initial value * don't run command in background * pactl: use default sink * use sound pressure db * tuning * bump up max volume threshold * update filter slower * fix divide by zero * bump cereal Co-authored-by: Shane Smiskol <shane@smiskol.com> old-commit-hash:108ff15f5d
parent
8a4a390d6f
commit
9bcf83f493
9 changed files with 78 additions and 7 deletions
@ -1 +1 @@ |
|||||||
Subproject commit 19a0c46b71150a8dabc5644eb24f261feee45b9c |
Subproject commit dbc9846ac9c9e735ee2f4a281ce079cfff7ea285 |
@ -1,3 +1,3 @@ |
|||||||
version https://git-lfs.github.com/spec/v1 |
version https://git-lfs.github.com/spec/v1 |
||||||
oid sha256:616bfb7e1761adc9f7a27571d1bde023727147971f5426207ee09fab592f97bc |
oid sha256:88495068650500a25401ed83cf3f2c4eb2216a1c3f8c2dcda0fa33bfd801b85f |
||||||
size 561251 |
size 560742 |
||||||
|
@ -0,0 +1,67 @@ |
|||||||
|
#!/usr/bin/env python3 |
||||||
|
import sounddevice as sd |
||||||
|
import numpy as np |
||||||
|
|
||||||
|
from cereal import messaging |
||||||
|
from common.filter_simple import FirstOrderFilter |
||||||
|
from common.realtime import Ratekeeper |
||||||
|
from system.swaglog import cloudlog |
||||||
|
|
||||||
|
RATE = 10 |
||||||
|
DT_MIC = 1. / RATE |
||||||
|
REFERENCE_SPL = 2 * 10 ** -5 # newtons/m^2 |
||||||
|
|
||||||
|
|
||||||
|
class Mic: |
||||||
|
def __init__(self, pm): |
||||||
|
self.pm = pm |
||||||
|
self.rk = Ratekeeper(RATE) |
||||||
|
|
||||||
|
self.measurements = np.empty(0) |
||||||
|
self.spl_filter = FirstOrderFilter(0, 4, DT_MIC, initialized=False) |
||||||
|
|
||||||
|
def update(self): |
||||||
|
# self.measurements contains amplitudes from -1 to 1 which we use to |
||||||
|
# calculate an uncalibrated sound pressure level |
||||||
|
if len(self.measurements) > 0: |
||||||
|
# https://www.engineeringtoolbox.com/sound-pressure-d_711.html |
||||||
|
sound_pressure = np.sqrt(np.mean(self.measurements ** 2)) # RMS of amplitudes |
||||||
|
sound_pressure_level = 20 * np.log10(sound_pressure / REFERENCE_SPL) if sound_pressure > 0 else 0 # dB |
||||||
|
self.spl_filter.update(sound_pressure_level) |
||||||
|
else: |
||||||
|
sound_pressure = 0 |
||||||
|
sound_pressure_level = 0 |
||||||
|
|
||||||
|
self.measurements = np.empty(0) |
||||||
|
|
||||||
|
msg = messaging.new_message('microphone') |
||||||
|
msg.microphone.soundPressure = float(sound_pressure) |
||||||
|
msg.microphone.soundPressureDb = float(sound_pressure_level) |
||||||
|
msg.microphone.filteredSoundPressureDb = float(self.spl_filter.x) |
||||||
|
|
||||||
|
self.pm.send('microphone', msg) |
||||||
|
self.rk.keep_time() |
||||||
|
|
||||||
|
def callback(self, indata, frames, time, status): |
||||||
|
self.measurements = np.concatenate((self.measurements, indata[:, 0])) |
||||||
|
|
||||||
|
def micd_thread(self, device=None): |
||||||
|
if device is None: |
||||||
|
device = "sysdefault" |
||||||
|
|
||||||
|
with sd.InputStream(device=device, channels=1, samplerate=44100, callback=self.callback) as stream: |
||||||
|
cloudlog.info(f"micd stream started: {stream.samplerate=} {stream.channels=} {stream.dtype=} {stream.device=}") |
||||||
|
while True: |
||||||
|
self.update() |
||||||
|
|
||||||
|
|
||||||
|
def main(pm=None, sm=None): |
||||||
|
if pm is None: |
||||||
|
pm = messaging.PubMaster(['microphone']) |
||||||
|
|
||||||
|
mic = Mic(pm) |
||||||
|
mic.micd_thread() |
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__": |
||||||
|
main() |
Loading…
Reference in new issue