1

我目前正在学习使用 alsa API (libasound) 进行开发。我想将 PCM 声音发送到我的 USB 声卡。

我运行这段代码:

#include <stdio.h>
#include <stdlib.h>
#include <alsa/asoundlib.h>

#include "helloPi.h"

int main(int argc, char *argv[]) {
    int i;
    int err;

    snd_pcm_t *playback_handle;
    snd_pcm_hw_params_t *hw_params;

    if ((err = snd_pcm_open(&playback_handle, "hw:1,0", SND_PCM_STREAM_PLAYBACK,
        0)) < 0) {
        fprintf(stderr, "cannot open audio device %s (%s)\n", argv[1],
            snd_strerror(err));
        exit(1);
    }

    if ((err = snd_pcm_hw_params_malloc(&hw_params)) < 0) {
        fprintf(stderr, "cannot allocate hardware parameter structure (%s)\n",
            snd_strerror(err));
        exit(1);
    }

    if ((err = snd_pcm_hw_params_any(playback_handle, hw_params)) < 0){
        fprintf(stderr, "cannot initialize hardware parameter structure (%s)\n",
            snd_strerror(err));
        exit(1);
    }

    if ((err = snd_pcm_hw_params_set_access(playback_handle, hw_params,
        SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) {
        fprintf(stderr, "cannot set access type (%s)\n", snd_strerror(err));
        exit(1);
    }

    if ((err = snd_pcm_hw_params_set_format(playback_handle, hw_params,
        SND_PCM_FORMAT_S16_LE)) < 0) {
        fprintf(stderr, "cannot set sample format (%s)\n", snd_strerror(err));
        exit(1);
    }

    unsigned int freq = 44100;

    if ((err = snd_pcm_hw_params_set_rate_near(playback_handle, hw_params,
        &freq, 0)) < 0) {
        fprintf(stderr, "cannot set sample rate (%s)\n", snd_strerror(err));
        exit(1);
    }

    if ((err = snd_pcm_hw_params_set_channels(playback_handle, hw_params, 2))
        < 0) {
        fprintf(stderr, "cannot set channel count (%s)\n", snd_strerror(err));
        exit(1);
    }

    if ((err = snd_pcm_hw_params(playback_handle, hw_params)) < 0) {
        fprintf(stderr, "cannot set parameters (%s)\n", snd_strerror(err));
        exit(1);
    }

    snd_pcm_hw_params_free(hw_params);

    if ((err = snd_pcm_prepare(playback_handle)) < 0) {
        fprintf(stderr, "cannot prepare audio interface for use (%s)\n",
            snd_strerror(err));
        exit(1);
    }

    for (i = 0; i < 100; ++i) {
        if ((err = snd_pcm_writei(playback_handle, buf, 2048)) < 0) {
            fprintf(stderr, "write to audio interface failed (%s)\n",
                snd_strerror(err));
            exit(1);
        }
    }

    snd_pcm_close(playback_handle);
    exit(0);
  }

注意:buf是在 HelloPi.h 中声明的包含声波的数组。

当我将snd_pcm_open(&playback_handle, "hw:1,0", SND_PCM_STREAM_PLAYBACK,0)hw:0,0一起使用时,我的笔记本电脑的内部声卡可以正常工作。但是,如果我使用hw:1,0来使用 USB 声卡,则什么也没有发生(甚至没有错误!)。

如果我在终端中运行:

aplay -l

我有以下结果:

carte 0: PCH [HDA Intel PCH], périphérique 0: ALC283 Analog [ALC283 Analog]
  Sous-périphériques: 1/1
  Sous-périphérique #0: subdevice #0
carte 0: PCH [HDA Intel PCH], périphérique 3: HDMI 0 [HDMI 0]
  Sous-périphériques: 1/1
  Sous-périphérique #0: subdevice #0
carte 1: Set [C-Media USB Headphone Set], périphérique 0: USB Audio [USB Audio]
  Sous-périphériques: 1/1
  Sous-périphérique #0: subdevice #0

你好

我知道 USB 卡没问题,因为我有声音:

speaker-test -Dhw:1,0 -c2 -twav

我不知道为什么我的代码用 hw:1,0 发出声音...

我希望你们中的一些人会帮助我!谢谢,

马克西姆。

4

1 回答 1

0

我找到了一个解决方案,我将示例中提供的代码与 asound 库一起使用。

/*
 *  This extra small demo sends a random samples to your speakers.
 */

#include <alsa/asoundlib.h>
static char *device = "hw:1,0";         /* playback device */

snd_output_t *output = NULL;
unsigned char buffer[16*1024];              /* some random data */

int main(void)
{
        int err;
        unsigned int i;
        snd_pcm_t *handle;
        snd_pcm_sframes_t frames;

        for (i = 0; i < sizeof(buffer); i++)
                buffer[i] = random() & 0xff;

    if ((err = snd_pcm_open(&handle, device, SND_PCM_STREAM_PLAYBACK, 0)) < 0) {
        printf("Playback1 open error: %s\n", snd_strerror(err));
        exit(EXIT_FAILURE);
    }
    if ((err = snd_pcm_set_params(handle,
                                  SND_PCM_FORMAT_S16,
                                  SND_PCM_ACCESS_RW_INTERLEAVED,
                                  2,
                                  44100,
                                  0,
                                  50000)) < 0) {    /* 0.5sec */
        printf("Playback2 open error: %s\n", snd_strerror(err));
        exit(EXIT_FAILURE);
    }

        for (i = 0; i < 16; i++) {
                frames = snd_pcm_writei(handle, buffer, sizeof(buffer));
                if (frames < 0)
                        frames = snd_pcm_recover(handle, frames, 0);
                if (frames < 0) {
                        printf("snd_pcm_writei failed: %s\n", snd_strerror(frames));
                        break;
                }
                if (frames > 0 && frames < (long)sizeof(buffer))
                        printf("Short write (expected %li, wrote %li)\n", (long)sizeof(buffer), frames);
        }

    snd_pcm_close(handle);
    return 0;
}

它可以工作,但我仍然必须找出为什么第一个代码不好。

再见,马克西姆。

于 2016-07-30T17:56:34.363 回答