// Copyright (c) 2023 Chen Qianhe Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #pragma once #include #include #include #include "./wav.h" #include "fastdeploy/fastdeploy_model.h" #include "fastdeploy/runtime.h" class Vad : public fastdeploy::FastDeployModel { public: enum class State { SIL = 0, START, SPEECH, END }; friend std::ostream& operator<<(std::ostream& os, const Vad::State& s); Vad(const std::string& model_file, const fastdeploy::RuntimeOption& custom_option = fastdeploy::RuntimeOption()); void Init(); void Reset(); void SetConfig(int sr, int frame_ms, float threshold, int min_silence_duration_ms, int speech_pad_left_ms, int speech_pad_right_ms); bool ForwardChunk(std::vector& chunk); const State& Postprocess(); const std::vector> GetResult( float removeThreshold = 0.0, float expandHeadThreshold = 0.0, float expandTailThreshold = 0, float mergeThreshold = 0.0) const; const std::vector GetStates() const { return states_; } int SampleRate() const { return sample_rate_; } int FrameMs() const { return frame_ms_; } int64_t WindowSizeSamples() const { return window_size_samples_; } float Threshold() const { return threshold_; } int MinSilenceDurationMs() const { return min_silence_samples_ / sample_rate_; } int SpeechPadLeftMs() const { return speech_pad_left_samples_ / sample_rate_; } int SpeechPadRightMs() const { return speech_pad_right_samples_ / sample_rate_; } int MinSilenceSamples() const { return min_silence_samples_; } int SpeechPadLeftSamples() const { return speech_pad_left_samples_; } int SpeechPadRightSamples() const { return speech_pad_right_samples_; } std::string ModelName() const override; private: bool Initialize(); private: std::once_flag init_; // input and output std::vector inputTensors_; std::vector outputTensors_; // model states bool triggerd_ = false; unsigned int speech_start_ = 0; unsigned int speech_end_ = 0; unsigned int temp_end_ = 0; unsigned int current_sample_ = 0; unsigned int current_chunk_size_ = 0; // MAX 4294967295 samples / 8sample per ms / 1000 / 60 = 8947 minutes float outputProb_; std::vector speakStart_; mutable std::vector speakEnd_; std::vector states_; /* ======================================================================== */ int sample_rate_ = 16000; int frame_ms_ = 32; // 32, 64, 96 for 16k float threshold_ = 0.5f; int64_t window_size_samples_; // support 256 512 768 for 8k; 512 1024 1536 // for 16k. int sr_per_ms_; // support 8 or 16 int min_silence_samples_; // sr_per_ms_ * frame_ms_ int speech_pad_left_samples_{0}; // usually 250ms int speech_pad_right_samples_{0}; // usually 0 /* ======================================================================== */ std::vector sr_; const size_t size_hc_ = 2 * 1 * 64; // It's FIXED. std::vector h_; std::vector c_; std::vector input_node_dims_; const std::vector sr_node_dims_ = {1}; const std::vector hc_node_dims_ = {2, 1, 64}; };