Skip to content
/ gss Public

A simple package for Guided source separation (GSS)

License

Notifications You must be signed in to change notification settings

desh2608/gss

Repository files navigation

GPU-accelerated Guided Source Separation

Guided source separation is a type of blind source separation (blind = no training required) in which the mask estimation is guided by a diarizer output. The original method was proposed for the CHiME-5 challenge in this paper by Boeddeker et al.

It is a kind of target-speaker extraction method. The inputs to the model are:

  1. A multi-channel recording, e.g., from an array microphone, of a long, unsegmented, multi-talker session (possibly with overlapping speech)
  2. An RTTM file containing speaker segment boundaries

The system produces enhanced audio for each of the segments in the RTTM, removing the background speech and noise and "extracting" only the target speaker in the segment.

This repository contains a GPU implementation of this method in Python, along with CLI binaries to run the enhancement from shell. We also provide several example "recipes" for using the method.

Features

The core components of the tool are borrowed from pb_chime5 , but GPU support is added by porting most of the work to CuPy.

  • The main components of the pipeline --- WPE, mask estimation with CACGMM, and beamforming --- are now built into their own packages: WPE, CACGMM, and beamformer. The code is also directly included into this package for ease of installation.
  • The GSS implementation (see gss/core) has been stripped of CHiME-6 dataset-specific peculiarities (such as array naming conventions etc.)
  • We use Lhotse for simplified data loading, speaker activity generation, and RTTM representation. We provide examples in the recipes directory for how to use the gss module for several datasets. We are currently aiming to support LibriCSS, AMI, and AliMeeting.
  • The inference can be done on multi-node GPU environment. This makes it several times faster than the original CPU implementation.
  • We have implemented batch processing of segments (see this issue for details) to maximize GPU memory usage and provide additional speed-up.
  • We provide both Python modules and CLI for using the enhancement functions, which can be easily included in recipes from Kaldi, Icefall, ESPNet, etc.

Installation

Preparing to install

Create a new Conda environment:

conda create -n gss python=3.8

Install CuPy as follows (see https://docs.cupy.dev/en/stable/install.html for the appropriate version for your CUDA).

pip install cupy-cuda102

Install (basic)

pip install git+http://github.com/desh2608/gss

Install (advanced)

git clone https://github.com/desh2608/gss.git & cd gss
pip install -e '.[dev]'
pre-commit install # installs pre-commit hooks with style checks

Usage

Enhancing a single recording

For the simple case of target-speaker extraction given a multi-channel recording and an RTTM file denoting speaker segments, run the following:

export CUDA_VISIBLE_DEVICES=0
gss enhance recording \
  /path/to/sessionA.wav /path/to/rttm exp/enhanced_segs \
  --recording-id sessionA --min-segment-length 0.1 --max-segment-length 10.0 \
  --max-batch-duration 20.0 --num-buckets 2 -o exp/segments.jsonl.gz

Enhancing a corpus

See the recipes directory for usage examples. The main stages are as follows:

  1. Prepare Lhotse manifests. See this list of corpora currently supported in Lhotse. You can also apply GSS on your own dataset by preparing it as Lhotse manifests.

  2. If you are using an RTTM file to get segments (e.g. in CHiME-6 Track 2), convert the RTTMs to Lhotse-style supervision manifest.

  3. Create recording-level cut sets by combining the recording with its supervisions. These will be used to get speaker activities.

  4. Trim the recording-level cut set into segment-level cuts. These are the segments that will actually be enhanced.

  5. (Optional) Split the segments into as many parts as the number of GPU jobs you want to run. In the recipes, we submit the jobs through qsub , similar to Kaldi or ESPNet recipes. You can use the parallelization in those toolkits to additionally use a different scheduler such as SLURM.

  6. Run the enhancement on GPUs. The following options can be provided:

  • --num-channels: Number of channels to use for enhancement. By default, all channels are used.

  • --bss-iteration: Number of iterations of the CACGMM inference.

  • --context-duration: Context (in seconds) to include on both sides of the segment.

  • --min-segment-length: Any segment shorter than this value will be removed. This is particularly useful when using segments from a diarizer output since they often contain very small segments which are not relevant for ASR. A recommended setting is 0.1s.

  • --max-segment-length: Segments longer than this value will be chunked up. This is to prevent OOM errors since the segment STFTs are loaded onto the GPU. We use a setting of 15s in most cases.

  • --max-batch-duration: Segments from the same speaker will be batched together to increase GPU efficiency. We used 20s batches for enhancement on GPUs with 12G memory. For GPUs with larger memory, this value can be increased.

  • --max-batch-cuts: This sets an upper limit on the maximum number of cuts in a batch. To simulate segment-wise enhancement, set this to 1.

  • --num-workers: Number of workers to use for data-loading (default = 1). Use more if you increase the max-batch-duration .

  • --num-buckets: Number of buckets to use for sampling. Batches are drawn from the same bucket (see Lhotse's DynamicBucketingSampler for details).

  • --enhanced-manifest/-o: Path to manifest file to write the enhanced cut manifest. This is useful for cases when the supervisions need to be propagated to the enhanced segments, for downstream ASR tasks, for example.

Other details

Internally, we also have a fallback option to chunk up batches into increasingly smaller parts in case OOM error is encountered (see gss.core.enhancer.py ).

The enhanced wav files are named as recoid-spkid-start_end.wav, i.e., 1 wav file is generated for each segment in the RTTM. The "start" and "end" are padded to 6 digits, for example: 21.18 seconds is encoded as 002118 . This convention should be fine if your audio duration is under ~2.75 h (9999s), otherwise, you should change the padding in gss/core/enhancer.py .

For examples of how to generate RTTMs for guiding the separation, please refer to my diarizer toolkit.

Additional parameters: We have only made the most important parameters available in the top-level CLI. To play with other parameters, check out the gss.enhancer.get_enhancer() function.

Profiling: You can also profile your enhancement by providing the option --profiler-output test.pstats

to the gss enhance CLI. The generated pstats file can then be visualized using Snakeviz: snakeviz test.pstats . An example output can be seen here.

Contributing

Contributions for core improvements or new recipes are welcome. Please run the following before creating a pull request.

pre-commit install
pre-commit run # Running linter checks

Citations

Please refer to the original repository for papers related to GSS. If you used this code for your work, consider citing the repo by clicking on Cite this repository.

You can also cite our CHiME-6 paper which used diarization-based GSS for multi-talker speech reognition.

@article{Arora2020TheJM,
  title={The JHU Multi-Microphone Multi-Speaker ASR System for the CHiME-6 Challenge},
  author={Ashish Arora and Desh Raj and Aswin Shanmugam Subramanian and Ke Li and Bar Ben-Yair and Matthew Maciejewski and Piotr Żelasko and Paola Garc{\'i}a and Shinji Watanabe and Sanjeev Khudanpur},
  journal={ArXiv},
  year={2020},
  volume={abs/2006.07898}
}

About

A simple package for Guided source separation (GSS)

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published