錄音加vad

#include#include#include#include#include//#include "stdio.h"#include "wb_vad.h"#include#includeusing namespace std;// 用于存儲語言塊的信息class BlockInfo{? ? public:? ? ? ? bool valid;// 是否有效語音? ? ? ? char* p;// 語音塊的首地址? ? ? ? int len;// 語言塊的大小? ? public:? ? ? ? BlockInfo()? ? ? ? {? ? ? ? ? ? valid = false;? ? ? ? ? ? p = NULL;? ? ? ? ? ? len = 0;? ? ? ? }? ? ? ? ~BlockInfo()? ? ? ? {? ? ? ? ? ? if(NULL == p) return;? ? ? ? ? ? delete p;? ? ? ? }? ? ? ? void set(char* ptr, int length)? ? ? ? {? ? ? ? ? ? p = ptr;? ? ? ? ? ? valid = true;? ? ? ? ? ? len = length;? ? ? ? }};queueglobal_full;queueglobal_valid;int flag = 0; // 當前錄音是停止的int irstart = 0;// 判斷是否重啟char filename[1024] = "0";void? vad(char* ptr, int size){? ? int valitSize = 0;// 有效數據長度? ? int totalCount = size/512 + (size % 512) > 0 ? 1 : 0;// 總數據塊 數量? ? int useTotalCount = 1;// 使用過的數據塊 數量? ? int i, temp;? ? float indata[FRAME_LEN];? ? VadVars *vadstate;? ? wb_vad_init(&(vadstate));//vad初始化? ? while (useTotalCount <= totalCount)? ? {? ? ? ? int pos = 0;// 被處理的數據塊的使用偏移量? ? ? ? char *p = ptr + (useTotalCount -1) * 512;// 被處理數據塊在ptr 中的首地址? ? ? ? int cursize = (useTotalCount != totalCount) ? 512 : (size - (useTotalCount-1)*512);// 被處理的數據塊的大小? ? ? ? ++useTotalCount;? ? ? ? // 獲取并轉換成vad 待處理數據? ? ? ? for (i = 0; i65535 / 2)? ? ? ? ? ? ? ? indata[i] = indata[i] - 65536;? ? ? ? }// 存儲語音塊的信息到 global_full? ? ? ? BlockInfo* bi = new BlockInfo();? ? ? ? if (1 == wb_vad(vadstate, indata))//進行vad檢測? ? ? ? {? ? ? ? ? ? char* buf = new char[pos];? ? ? ? ? ? memcpy(buf, p, pos);? ? ? ? ? ? bi->set(buf, pos);? ? ? ? ? ? valitSize += cursize;? ? ? ? }? ? ? ? global_full.push(bi);? ? }? ? wb_vad_exit(&(vadstate));? ? fcloseall();}// 刪除有效語音塊中的數據(刪除所有、刪除3s)void clean_queue(queue* q, bool cleanall){? ? if(NULL == q) return ;? ? BlockInfo* bi = NULL;? ? if(cleanall)? ? {? ? ? ? while(q->size() > 0)? ? ? ? {? ? ? ? ? ? bi = q->front();? ? ? ? ? ? q->pop();? ? ? ? ? ? delete bi;? ? ? ? }? ? }? ? else? ? {? ? ? ? int count = 62*3;? ? ? ? while(q->size() > 0 && count > 0)? ? ? ? {? ? ? ? ? ? bi = q->front();? ? ? ? ? ? q->pop();? ? ? ? ? ? delete bi;? ? ? ? ? ? --count;? ? ? ? }? ? }}// 檢查有效語音塊集合的狀態// 0:save, 1: clean all, 2:clean by 3s, 3:ingoreint check_save(bool flag){? ? int size = global_valid.size();? ? if(flag)? ? {? ? ? ? return(size > 62*3*2) ? 2 : 3;? ? }? ? else? ? {? ? ? ? return (size < 62 || size > 62*3) ? 1 : 0;? ? }}char* get_time(){? ? struct tm *newtime;? ? char* outfile = new char[128]{0};? ? time_t t1 = time(NULL);? ? newtime = localtime(&t1);? ? strftime(outfile,128,"data_%Y%m%d_%H%M%S.wav",newtime);? ? return outfile;}// 存儲語音塊集合中的數據到文件void save_queue(queue* q)

{

if(NULL == q) return;

char* savename = get_time();

int fd = open(savename, O_CREAT|O_WRONLY|O_TRUNC, 0777);

if(fd < 0)

{

printf("Fail to open or create file!\n");

exit(1);

}

BlockInfo* bi = NULL;

while(global_valid.size() > 0)

{

bi = global_valid.front();

global_valid.pop();

write(fd, bi->p, bi->len);

delete bi;

}

delete savename;

close(fd);

}

// 檢查語音塊集合

void check_queue()

{

if(0 ==global_full.size()) return;

BlockInfo* bi = NULL;

int check_save_ret = 0;

// 對每一塊語音進行檢查

while(global_full.size() > 0)

{

bi =? global_full.front();

global_full.pop();

printf("vaild count %d\n", global_valid.size());

// 判斷有效語音集合的狀態

if(false == bi->valid)

{

delete bi;

check_save_ret = check_save(false);

// 0:save, 1: clean all, 2:clean by 3s, 3:ingore

}

else

{

global_valid.push(bi);

// 0:save, 1: clean all, 2:clean by 3s, 3:ingore

check_save_ret = check_save(true);

}

// 針對有效語音塊集合的狀態 做對應動作( 0:save, 1: clean all, 2:clean by 3s, 3:ingore)

switch(check_save_ret)

{

// 0:save, 1: clean all, 2:clean by 3s, 3:ingore

case 0:

save_queue(&global_valid);

break;

case 1:

clean_queue(&global_valid, true);

break;

case 2:

clean_queue(&global_valid, false);

break;

case 3:

break;

default:

printf("logic err\n");

break;

}

}

}

void start()

{

printf("準備錄音...\n");

flag = 1;

long loops;

int rc;

int size;

snd_pcm_t *handle;

snd_pcm_hw_params_t *params;

unsigned int val;

int dir;

snd_pcm_uframes_t frames;

char *buffer;

rc = snd_pcm_open(&handle, "default", SND_PCM_STREAM_CAPTURE, 0);

if (rc < 0)

{

fprintf(stderr, "unable to open pcm device: %s\n", snd_strerror(rc));

exit(1);

}

/* Allocate a hardware parameters object. */

snd_pcm_hw_params_alloca(?ms);

/* Fill it in with default values. */

if((rc=snd_pcm_hw_params_any(handle, params))<0)

{

fprintf(stderr,"cannot initialize hd params (%s)\n",snd_strerror(rc));

exit(1);

}

/* Set the desired hardware parameters. */

/* Interleaved mode */

if((rc=snd_pcm_hw_params_set_access(handle, params, SND_PCM_ACCESS_RW_INTERLEAVED))<0)

{

fprintf(stderr,"cannot set access type (%s)\n",snd_strerror(rc));

exit(1);

}

/* Signed 16-bit little-endian format */

if((rc=snd_pcm_hw_params_set_format(handle, params, SND_PCM_FORMAT_S16_LE))<0)

{

fprintf(stderr,"cannot set sample format (%s)\n",snd_strerror(rc));

exit(1);

}

/* One channels (stereo) */

if((rc=snd_pcm_hw_params_set_channels(handle, params, 1))<0)

{

fprintf(stderr,"cannot set channels (%s)\n",snd_strerror(rc));

exit(1);

}

/* 16000 bits/second sampling rate (CD quality) */

val = 16000;

if((rc=snd_pcm_hw_params_set_rate_near(handle, params, &val, &dir))<0)

{

fprintf(stderr,"cannot set sample rate_near (%s)\n",snd_strerror(rc));

exit(1);

}

/* Set period size to 32 frames. */

frames = 32;

if((rc=snd_pcm_hw_params_set_period_size_near(handle, params, &frames, &dir))<0)

{

fprintf(stderr,"cannot set perios_size_near (%s)\n",snd_strerror(rc));

exit(1);

}

/* Write the parameters to the driver */

rc = snd_pcm_hw_params(handle, params);

if (rc < 0)

{

fprintf(stderr,

"unable to set hw parameters: %s\n",

snd_strerror(rc));

exit(1);

}

/* Use a buffer large enough to hold one period */

if((rc=snd_pcm_hw_params_get_period_size(params, &frames, &dir))<0)

{

fprintf(stderr,"cannot get period_size (%s)\n",snd_strerror(rc));

exit(1);

}

size = frames*8; /* 2 bytes/sample, 1 channels */

buffer = (char *) malloc(size);

printf("%d\n",frames);

printf("%d\n",size);

while (flag)

{

rc = snd_pcm_readi(handle, buffer, frames*4);

if (rc == -EPIPE)

{

/* EPIPE means overrun */

fprintf(stderr, "overrun occurred\n");

snd_pcm_prepare(handle);

}

else if (rc < 0)

{

fprintf(stderr, "error from read: %s\n", snd_strerror(rc));

}

else if (rc != (int)frames*4)

{

fprintf(stderr, "short read, read %d frames\n", rc);

}

vad(buffer,size);

// 檢查語音塊集合

check_queue();

}

if((rc=snd_pcm_drain(handle))<0)

{

fprintf(stderr,"cannot drain (%s)\n",snd_strerror(rc));

exit(1);

}

if((snd_pcm_close(handle))<0)

{

fprintf(stderr,"cannot close (%s)\n",snd_strerror(rc));

exit(1);

}

free(buffer);

// 錄音結束

printf("錄音完成,準備寫入文件\n");

if(irstart)

{

start();

}

}

void _savename(const char* savename)

{

strncpy(filename, savename, strlen(savename));

}

void stop()

{

// 讓錄音停止

flag = 0;

irstart = 0;

}

void rstart(const char* savename)

{

_savename(savename);

stop();

irstart = 1;

}

void* thread_cb(void* savename)

{

_savename((char*)savename);

start();

return NULL;

}

void start_with_thread(const char* savename)

{

pthread_t pid;

pthread_create(&pid, NULL, thread_cb, (void*)savename);

pthread_detach(pid);

}

int main(void)

{

int test = 0;

// 1 錄音

// 2 停止錄音

// 3 重啟

while(1)

{

scanf("%d", &test);

if(1 == test)

{

start_with_thread("start.wav");

}

else if(2 == test)

{

stop();

//savefile();

}

else if(3 == test)

{

rstart("restartname.wav");

}

else

{

break;

}

}

sleep(1);

printf("程序退出\n");

}

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,505評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,556評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,463評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,009評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,778評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,218評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,281評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,436評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,969評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,795評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,993評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,537評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,229評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,659評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,917評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,687評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,990評論 2 374

推薦閱讀更多精彩內容

  • 一.聲音參數基本概念: 聲音是連續模擬量,計算機將它離散化之后用數字表示,就有了以下幾個名詞術語。 樣本長度(sa...
    cs1001閱讀 2,773評論 0 2
  • CUDA從入門到精通(零):寫在前面 本文原版鏈接: 在老板的要求下,本博主從2012年上高性能計算課程開始接觸C...
    Pitfalls閱讀 3,636評論 1 3
  • 一.聲音參數基本概念: 聲音是連續模擬量,計算機將它離散化之后用數字表示,就有了以下幾個名詞術語。 樣本長度(sa...
    cs1001閱讀 5,441評論 0 3
  • 教程一:視頻截圖(Tutorial 01: Making Screencaps) 首先我們需要了解視頻文件的一些基...
    90后的思維閱讀 4,733評論 0 3
  • 背景 一年多以前我在知乎上答了有關LeetCode的問題, 分享了一些自己做題目的經驗。 張土汪:刷leetcod...
    土汪閱讀 12,762評論 0 33