录音加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阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • 一.声音参数基本概念: 声音是连续模拟量,计算机将它离散化之后用数字表示,就有了以下几个名词术语。 样本长度(sa...
    cs1001阅读 2,686评论 0 2
  • CUDA从入门到精通(零):写在前面 本文原版链接: 在老板的要求下,本博主从2012年上高性能计算课程开始接触C...
    Pitfalls阅读 3,593评论 1 3
  • 一.声音参数基本概念: 声音是连续模拟量,计算机将它离散化之后用数字表示,就有了以下几个名词术语。 样本长度(sa...
    cs1001阅读 5,364评论 0 3
  • 教程一:视频截图(Tutorial 01: Making Screencaps) 首先我们需要了解视频文件的一些基...
    90后的思维阅读 4,648评论 0 3
  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,719评论 0 33