ALSA / ASoC: Como carregar dispositivos / drivers corretamente?

10

Estou usando o Buildroot para criar um sistema Linux embarcado (2.6.39.2) para o microcontrolador NXP LPC3250.

Neste momento, estou tentando colocar o ALSA / ASoC em funcionamento, mas estou tendo alguns problemas para fazer com que os módulos funcionem juntos. (Eu acho!)

Algumas informações importantes:

A placa com a qual estou testando é a placa de desenvolvimento Embedded Artists 3250 V2 . V2 é diferente de V1 porque não tem uma tela LCD, mas inclui um codec de áudio I2S: O NXP UDA1380. O suporte à placa para o EA3250 V1 está incluído na versão LPCLinux do kernel. Há também uma placa de desenvolvimento diferente, chamada Phytec 3250 , que contém o mesmo chip de codec UDA1380. A distribuição LPCLinux também tem suporte para a placa Phytec, junto com o chip codec de áudio. Pelo que eu percebi, a placa Phytec 3250 tem o codec UDA1380 no endereço I2C 0x18 . Na minha placa EA3250 V2, o codec de áudio está localizado no endereço I2C 0x1a . (verifiquei que o chip está ligado e posso me comunicar com ele usando o pacote I2C-tools. Ele responde ao i2cdetect e posso ler registros do chip corretamente usando i2cget.)

Modificando a fonte:

Eu precisava editar os arquivos do driver Phytec 3250 para alterar o endereço do chip do codec. Eu editei esta seção de lpc3xxx-uda1380.c :

static struct snd_soc_dai_link phy3250_uda1380_dai[] = {
       {
                  .name           = "uda1380",
                  .stream_name    = "UDA1380 Duplex",
  #if defined(CONFIG_SND_LPC32XX_USEI2S1)
                  .cpu_dai_name   = "lpc3xxx-i2s1",
  #else
                  .cpu_dai_name   = "lpc3xxx-i2s0",
  #endif
                  .codec_dai_name = "uda1380-hifi",
                  .init           = phy3250_uda1380_init,
                  .platform_name  = "lpc3xxx-audio.0",
         //EDIT// .codec_name     = "uda1380-codec.0-0018",  //EDIT//
                  .codec_name     = "uda1380-codec.0-001a",
                  .ops            = &phy3250_uda1380_ops,
          },
  };

Depois de fazer essa alteração, fui em frente e construí o sistema novamente e tudo compilou OK. Depois de inicializar no sistema, tenho os seguintes módulos (além dos módulos principais padrão) em /lib/modules/2.6.39.2/kernel/sound :

 ./soc/codecs: snd-soc-uda1380.ko          <-- ASoC codec driver
./soc/lpc3xxx: snd-soc-lpc3xxx-i2s.ko      <-- ASoC DAI
               snd-soc-lpc3xxx-uda1380.ko  <-- ASoC machine driver
               snd-soc-lpc3xxx.ko          <-- ASoC platform driver

Agora, como eu realmente vinculo todas essas coisas?

Apenas inserir os módulos com modprobe não fornece o dispositivo para o ALSA / ASoC. Eu não consigo detectar a placa de som. Isso significa que agora preciso criar um novo dispositivo chamado uda1380-codec no endereço 0x1a e ligá-lo a um driver? Eu tentei fazer o seguinte:
echo uda1380-codec 0x01a > /sys/bus/i2c/devices/i2c-0/new_device
e recebi:
i2c i2c-0: new_device: Instantiated device uda1380-codec at 0x1a
Em seguida, tento ligar um driver ao dispositivo:
echo 0x1a > /sys/bus/i2c/drivers/uda1380-codec/bind
e recebi:
sh: write error: No such device

Eu recebo este erro por tudo que eu tento! Tenho a sensação de que não estou criando o dispositivo corretamente e, em seguida, não sei como vinculá-lo ao driver correto.

Nota Bene:

Eu estava brincando com isso ontem à noite e de alguma forma consegui que o ASoC acordasse e pelo menos sondasse o cartão. Eu estava jogando com diferentes ligações, eu acho. Era tarde e difícil lembrar meus passos, mas consegui pelo menos o seguinte erro:

uda1380-codec 0-001a: asoc: failed to probe CODEC uda1380-codec.0-001a: -22
asoc: failed to instantiate card LPC32XX: -22

Não consegui recriar este erro!

Editar:

Confirmei que meu código modificado está sendo compilado, portanto, o driver deve estar falando com o endereço correto agora. Depois de carregar manualmente os módulos, a saída de lsmod é:

Module                      Size  Used by    Not tainted
snd_soc_lpc3xxx_uda1380     2087  0 
snd_soc_lpc3xxx             3089  0 
snd_soc_lpc3xxx_i2s         4089  1 
snd_soc_uda1380            10865  0 
snd_soc_core               51549  4 snd_soc_lpc3xxx_uda1380,snd_soc_lpc3xxx,snd_soc_lpc3xxx_i2s,snd_soc_uda1380
snd_pcm                    52098  2 snd_soc_lpc3xxx,snd_soc_core
snd_timer                  15590  1 snd_pcm
snd_page_alloc              3021  1 snd_pcm
snd                        37286  3 snd_soc_core,snd_pcm,snd_timer

Isso parece correto?

E minha tabela de dispositivos:

# Audio stuff
/dev/audio      c       666     0       29      14      4       -       -       -
#/dev/audio1    c       666     0       29      14      20      -       -       -
/dev/dsp        c       666     0       29      14      3       -       -       -
#/dev/dsp1      c       666     0       29      14      19      -       -       -
#/dev/sndstat   c       666     0       29      14      6       -       -       -
/dev/mixer      c       666     0       29      14      0       -       -       -
/dev/snd        d       755     0       29      -       -       -       -       -
/dev/snd/controlC0      c       666     0       29      116     0       -       -       -
/dev/snd/pcmC0D0c       c       666     0       29      116     24      -       -       -
/dev/snd/pcmC0D0p       c       666     0       29      116     16      -       -       -
/dev/snd/seq    c       666     0       29      116     1       -       -       -
/dev/snd/timer  c       666     0       29      116     33      -       -       -
    
por dext0rb 26.12.2012 / 21:40

1 resposta

3

É necessário editar o arquivo da placa que define os dispositivos da plataforma. Eu precisava modificar arch/arm/mach-lpc32xx/ea3250.c :

Adicione isto:

/*
 * Platform Data for UDA1380 Audiocodec.
 * As there are no GPIOs for codec power & reset pins,
 * dummy GPIO numbers are used.
 */
static struct uda1380_platform_data uda1380_info = {
    .gpio_power = LPC32XX_GPIO(LPC32XX_GPO_P3_GRP,10),
    .gpio_reset = LPC32XX_GPIO(LPC32XX_GPO_P3_GRP,2),
    .dac_clk    = UDA1380_DAC_CLK_WSPLL,
};

Edite isso para incluir o codec:

static struct i2c_board_info __initdata ea3250_i2c_board_info [] = {
        {   I2C_BOARD_INFO("uda1380", 0x1a),
            .platform_data = &uda1380_info,
        }, 
#if defined (CONFIG_LEDS_PCA9532)
        {
            I2C_BOARD_INFO("pca9532", I2C_PCA9532_ADDR),
            .platform_data = &ea3250_leds,
        },
#endif
#if defined (CONFIG_FB_ARMCLCD)
        {
            /* 8Kb Configuration EEPROM on display board */
            I2C_BOARD_INFO("ea_i2c_disp_cfg", LCDB_CONFIG_EEPROM_I2C_ADDR),
        },
        {
            I2C_BOARD_INFO("ea_i2c_video", LCDB_PCA9532_I2C_ADDR),
        },
#endif
#if defined (CONFIG_EEPROM_AT24)
        {
            I2C_BOARD_INFO("24c256", I2C_24LC256_ADDR),
        },
#endif
    };
#endif

Agora tenho todos os dispositivos:

# cat cards
 0 [LPC32XX        ]:  - LPC32XX
                      LPC32XX
# cat devices
  2: [ 0- 0]: digital audio playback
  3: [ 0- 0]: digital audio capture
  4: [ 0]   : control
 33:        : timer

# cat pcm
00-00: UDA1380 Duplex uda1380-hifi-0 :  : playback 1 : capture 1

Não consigo encontrar aplay para detectar nada, mas talvez seja uma questão diferente.

EDIT: Sim, esse foi um problema diferente. O número na frente dos dispositivos mostrados por cat devices deve corresponder ao menor número de dispositivo nas entradas do seu dispositivo / dev / snd. Tudo parece bem no ALSA terminar agora, mas eu não tenho dados I2S vindo do LPC3250 ...

EDIT2: RESOLVIDO RESOLVIDO RESOLVIDO. Se não houver dados / relógio I2S, verifique se o registro mux de saída está configurado corretamente para conectar os pinos de saída ao periférico I2S.

    
por 23.01.2013 / 00:28