- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试获取 flite speech synthesis library在我的 Mac 上工作,但 flite 库不支持我的声音架构。为了解决这个问题,我正在使用 PortAudio播放合成音频;所以我不得不在 audio.c
文件中进行一些修改,让 flite 能够使用该库。在 GNU AutoTools 中折腾了一段时间后,我设法让所有的东西都编译得很好,但随后我运行了程序并得到了这个输出:
$ ./flite -t "test"
frameIndex: 0
maxFrameIndex: 0
numChannels: 1
numSamples: 7225
sampleRate: 8000
=== Now playing back. ===
Waiting for playback to finish.
frameIndex in callback: -2008986336
maxFrameIndex in callback: 32655
numChannels in callback: 152579008
numSamples in callback: 0
sampleRate in callback: 0
Segmentation fault: 11
$ ./flite -t "test"
frameIndex: 0
maxFrameIndex: 0
numChannels: 1
numSamples: 7225
sampleRate: 8000
=== Now playing back. ===
Waiting for playback to finish.
frameIndex in callback: -71217888
maxFrameIndex in callback: 32712
numChannels in callback: 232979392
numSamples in callback: 0
sampleRate in callback: 0
Segmentation fault: 11
这是 audio.c
文件中的相关代码,当我提供命令行参数 -t
时会调用它。经过一些调试后,我在 playCallback()
函数中标记了出现段错误的感兴趣区域。
static int playCallback( const void *inputBuffer, void *outputBuffer,
unsigned long framesPerBuffer,
const PaStreamCallbackTimeInfo* timeInfo,
PaStreamCallbackFlags statusFlags,
void *userData )
{
cst_wave *data = (cst_wave*)userData;
short *rptr = &data->samples[data->frameIndex * data->num_channels];
short *wptr = (short*)outputBuffer;
unsigned int i;
int finished;
unsigned int framesLeft = cst_wave_maxFrameIndex(data) - cst_wave_frameIndex(data);
(void) inputBuffer; /* Prevent unused variable warnings. */
(void) timeInfo;
(void) statusFlags;
(void) userData;
printf("frameIndex in callback: %d\n", cst_wave_frameIndex(data));
printf("maxFrameIndex in callback: %d\n", cst_wave_maxFrameIndex(data));
printf("numChannels in callback: %d\n", cst_wave_num_channels(data));
printf("numSamples in callback: %d\n", cst_wave_num_samples(data));
printf("sampleRate in callback: %d\n\n", cst_wave_sample_rate(data));
if( framesLeft < framesPerBuffer )
{
/* final buffer... */
for( i=0; i<framesLeft; i++ )
{
*wptr++ = *rptr++; /* left */
if( cst_wave_num_channels(data) == 2 ) *wptr++ = *rptr++; /* right */
}
for( ; i<framesPerBuffer; i++ )
{
*wptr++ = 0; /* left */
if( cst_wave_num_channels(data) == 2) *wptr++ = 0; /* right */
}
data->frameIndex += framesLeft;
finished = paComplete;
}
else
{
for( i=0; i<framesPerBuffer; i++ )
{
*wptr++ = *rptr++; /* left */
if( cst_wave_num_channels(data) == 2 ) *wptr++ = *rptr++; /* right */
}
cst_wave_set_frameIndex(data, framesPerBuffer);
finished = paContinue;
}
return finished;
}
int play_wave(cst_wave *w)
{
PaStream* stream;
PaStreamParameters outputParameters;
cst_wave_set_frameIndex(w, 0);
cst_wave_set_maxFrameIndex(w, (cst_wave_num_samples(w) / cst_wave_sample_rate(w)) * cst_wave_num_channels(w) * sizeof(short));
int err = 0;
err = Pa_Initialize();
outputParameters.device = Pa_GetDefaultOutputDevice();
if (outputParameters.device == paNoDevice)
{
fprintf(stderr,"Error: No default output device.\n");
return -5;
}
printf("frameIndex: %d\n", cst_wave_frameIndex(w));
printf("maxFrameIndex: %d\n", cst_wave_maxFrameIndex(w));
printf("numChannels: %d\n", cst_wave_num_channels(w));
printf("numSamples: %d\n", cst_wave_num_samples(w));
printf("sampleRate: %d\n", cst_wave_sample_rate(w));
outputParameters.channelCount = cst_wave_num_channels(w);
outputParameters.sampleFormat = paInt16;
outputParameters.suggestedLatency = Pa_GetDeviceInfo( outputParameters.device )->defaultLowOutputLatency;
outputParameters.hostApiSpecificStreamInfo = NULL;
puts("=== Now playing back. ===");
err = Pa_OpenStream(&stream,
NULL, /* no input */
&outputParameters,
cst_wave_sample_rate(w),
512,
paClipOff,
playCallback,
&w);
if( stream )
{
err = Pa_StartStream( stream );
if( err != paNoError ) goto done;
puts("Waiting for playback to finish.");
while((err = Pa_IsStreamActive(stream)) == 1) Pa_Sleep(100);
if( err < 0 ) goto done;
err = Pa_CloseStream( stream );
if( err != paNoError ) goto done;
puts("Done.");
}
done:
Pa_Terminate();
free(cst_wave_samples(w));
}
因为它是相关的,我还稍微修改了 cst_wave.h
中的 cst_wave
结构,使其包含我必须存储的数据,并添加了一些#defines
到已经存在的那些:
typedef struct cst_wave_struct {
const char *type;
int frameIndex;
int maxFrameIndex;
int sample_rate;
int num_samples;
int num_channels;
short *samples;
} cst_wave;
#define cst_wave_num_samples(w) (w?w->num_samples:0)
#define cst_wave_num_channels(w) (w?w->num_channels:0)
#define cst_wave_sample_rate(w) (w?w->sample_rate:0)
#define cst_wave_samples(w) (w->samples)
#define cst_wave_frameIndex(w) (w->frameIndex)
#define cst_wave_maxFrameIndex(w) (w->maxFrameIndex)
#define cst_wave_set_num_samples(w,s) w->num_samples=s
#define cst_wave_set_num_channels(w,s) w->num_channels=s
#define cst_wave_set_sample_rate(w,s) w->sample_rate=s
#define cst_wave_set_frameIndex(w,s) w->frameIndex=s
#define cst_wave_set_maxFrameIndex(w,s) w->maxFrameIndex=s
按照@Rohan 的建议现在给我这个输出:
$ ./bin/flite -t "test"
frameIndex: 0
maxFrameIndex: 0
numChannels: 1
numSamples: 7225
sampleRate: 8000
=== Now playing back. ===
Waiting for playback to finish.
frameIndex in callback: 0
maxFrameIndex in callback: 0
numChannels in callback: 1
numSamples in callback: 7225
sampleRate in callback: 8000
Done.
flite(68929,0x7fff71c0d310) malloc: *** error for object 0x7fd6e2809800: pointer being freed was not allocated
*** set a breakpoint in malloc_error_break to debug
Abort trap: 6
为了解决这个问题,我删除了 free(cst_wave_samples(w));
。现在程序正常执行,没有明显的错误,但我的 Mac 上仍然没有音频输出。有什么建议吗?
最佳答案
在我看来问题可能出在其他地方。
总而言之,您添加评论的例程真的很微不足道。它基本上只是将一个充满数据的缓冲区从一个地方复制到另一个地方,如果数据没有填满输入缓冲区,则用零填充其余部分。如果我正在编写代码,我可能会按照这些一般思路做更多的事情:
const unsigned frame_size = sizeof(short) * data->num_channels;
char *source = &data->samples[data->frameIndex * data->num_channels];
char *dest = outputBuffer;
unsigned framesLeft = data->maxFrameIndex - data->frameIndex;
unsigned framesEmpty = framesPerBuffer - framesLeft;
memcpy(source, dest, framesLeft * frame_size);
memset(dest+framesLeft * frame_size, 0, framesEmpty * frame_size);
data->frameIndex += framesPerBuffer;
尽管写得相当笨拙,问题中的 if
/else
只是跳过了 memset
部分,如果需要的话被填充为零。
因此,这会将一个充满数据的缓冲区从一个地方复制到另一个地方,并用零填充任何剩余部分。如果您遇到段错误,则显然分配目标缓冲区的任何内容都没有在那里分配足够的空间。如果不做一些观察,就不可能猜测分配是否发生在 Pa_Initialize
、Pa_OpenStream
、Pa_StartStream
或其他地方——而且很可能您不太关心实际执行分配的代码,而是计算要分配多少空间的代码(可能在上述代码之一中,或者完全在其他地方)。
关于c - 使用 PortAudio 获取 flite 输出音频,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23820713/
我在安装 PyAudio 和 portaudio 时遇到问题。 当我执行 python -m pip install pyaudio 时出现此错误: src/_portaudiomodule.
我有一个主要使用 PyAudio 的实验室项目,为了进一步了解其工作方式,我做了一些测量,在本例中是回调之间的时间(使用回调模式)。 我计时了,得到了一个有趣的结果 (@256 chunk size,
我目前正在使用 live555、mpg123 和 portaudio 构建一个 mp3 流媒体/接收器,并且刚刚开始流式传输、解码和播放 mp3 的概念。 我的问题是当我需要用 portaudio 播
我正在使用Portaudio以便将声音录制到.raw文件中,但是我想仅在有声音时才开始录制,而在无声时才停止录制。 有没有办法用Portaudio做到这一点? 如果没有,您是否知道我该怎么做? 提前致
我再次需要你的帮助。 短版: 您可以将声音片段导入 Portaudio 吗?如果没有,我在哪里可以获得“真实”的声音示例,这些示例被转换为可以存储到我的声音缓冲区中的离散值? 加长版: 作为一个小型项
我正在尝试学习 DSP 编程以尝试创建自己的吉他效果,并且我正在使用 Portaudio 库来实现它。我遇到的问题是,无论我做什么,每当我在回调中读取输入缓冲区时,它都是空的。我已经尝试了几个不同的
我大约一周前开始使用 PortAudio 库。我已经检查了大部分教程/测试示例,但没有看到我需要的解决方案。我正在制作简单的音序器 - 我已经在钢琴卷轴上绘制了作为 block 的声音,但现在我需要以
由于我是 PortAudio 的新手,所以我尝试了一个来自互联网的示例程序。该程序能够通过回调函数记录麦克风的输入。 我想获取录制音频的每个样本,以数值(例如 float )表示。我不知道麦克风的记录
我正在尝试在 portaudio 中创建多个流。这是打开流所需要的 PaError Pa_OpenDefaultStream( PaStream** stream,
我有类似钢琴的应用程序,当我按下钢琴键时,它会播放声音。在我的应用程序中,我编写了自己的正弦波发生器。应用程序是用 Qt 编写的。我认为问题出在 portAudio 上,但我找不到任何解决方案。 我已
我正在使用 PortAudio 开发声音应用程序。我有一个结构数组,表示传递到主回调函数的各个振荡器的数据。 当我尝试在一条线上将两个波相加并播放时,它起作用了,但是当我尝试使用 for 循环来做到这
我在使用此 cmake 列表时遇到的一个问题是我收到错误消息 error: undefined reference to 'Pa_GetVersion'对于这段代码: 记录.cpp: #include
我在使用 PortAudio 时遇到问题,我不确定是我不太了解回调的工作原理,还是我做错了什么。我的假设是回调应该连续“on tick”触发,包含当前样本,但似乎我在打开和启动流时只收到一些回调,然后
我正在使用 PortAudio,这就是我现在打开流进行阅读的方式。 Pa_OpenDefaultStream(&stream, 1, 0, paFloat32, SAMPLE_RATE, SAMPLE
我是 PortAudio 的新手。我的目的是不断从我的电脑上的线路输入中捕获数据并实时处理数据。 我使用的是 44100 的采样率和 11025 的缓冲区大小 (frameCount)。我能够成功地做
我目前正在使用带有录音应用程序的 portaudio,我似乎在收集样本时遇到了一些问题。据我所知,只有一个样本被存储,回调只被调用一次,仅此而已,即使变量 NUM_OF_SECONDS 设置为 30
我想制作一个吉他效果器程序,所以我尝试从 PORTAUDIO API 运行示例代码 pa_fuzz.c。 它适用于 paWDMKS 主机 api。但是当我使用 Behringer ucg 102(无延
我正在使用 PortAudio 来实现实时音频处理。 我的首要任务是连续采集mic数据,提供100个样本进行处理(each FRAME = 100 samples at a time)到其他一些处理线
我在编写 C++ 程序时遇到了这个问题,所以我创建了一个最小的代码实例来更好地阐明问题: #include #include int main() { Pa_Initialize();
我想知道,你们知道我可以使用的用于 portaudio 的 cmake 文件吗? 提前致谢! 最佳答案 这是一个: # - Try to find Portaudio # Once done this
我是一名优秀的程序员,十分优秀!