ALSA在Linux3.1上的一些改进


Android迈进了4.0,相应的Linux内核也进入了3.x时代。之后的一个工作估计要将2.6.32的驱动移植到3.x上面来。因此趁现在有空,看看alsa在这方面有什么改动。

总的来说,架构大的改动是不大可能的。codec中几个关键结构体没有大的变化,如snd_soc_dai_ops、snd_soc_dai_driver(相当于2.6.32中的snd_soc_dai),倒是以前的snd_soc_codec_device重定义为snd_soc_codec_driver,这个算是最明显的。


注册用结构体-snd_soc_codec_driver


2.6.32:

[cpp]
  1. /* codec device */  
  2. struct snd_soc_codec_device {  
  3.     int (*probe)(struct platform_device *pdev);  
  4.     int (*remove)(struct platform_device *pdev);  
  5.     int (*suspend)(struct platform_device *pdev, pm_message_t state);  
  6.     int (*resume)(struct platform_device *pdev);  
  7. };  
3.1.1: [cpp]
  1. /* codec driver */  
  2. struct snd_soc_codec_driver {  
  3.   
  4.     /* driver ops */  
  5.     int (*probe)(struct snd_soc_codec *);  
  6.     int (*remove)(struct snd_soc_codec *);  
  7.     int (*suspend)(struct snd_soc_codec *,  
  8.             pm_message_t state);  
  9.     int (*resume)(struct snd_soc_codec *);  
  10.   
  11.     /* Default control and setup, added after probe() is run */  
  12.     const struct snd_kcontrol_new *controls;  
  13.     int num_controls;  
  14.     const struct snd_soc_dapm_widget *dapm_widgets;  
  15.     int num_dapm_widgets;  
  16.     const struct snd_soc_dapm_route *dapm_routes;  
  17.     int num_dapm_routes;  
  18.   
  19.     /* codec wide operations */  
  20.     int (*set_sysclk)(struct snd_soc_codec *codec,  
  21.               int clk_id, unsigned int freq, int dir);  
  22.     int (*set_pll)(struct snd_soc_codec *codec, int pll_id, int source,  
  23.         unsigned int freq_in, unsigned int freq_out);  
  24.   
  25.     /* codec IO */  
  26.     unsigned int (*read)(struct snd_soc_codec *, unsigned int);  
  27.     int (*write)(struct snd_soc_codec *, unsigned int, unsigned int);  
  28.     int (*display_register)(struct snd_soc_codec *, char *,  
  29.                 size_t, unsigned int);  
  30.     int (*volatile_register)(struct snd_soc_codec *, unsigned int);  
  31.     int (*readable_register)(struct snd_soc_codec *, unsigned int);  
  32.     int (*writable_register)(struct snd_soc_codec *, unsigned int);  
  33.     short reg_cache_size;  
  34.     short reg_cache_step;  
  35.     short reg_word_size;  
  36.     const void *reg_cache_default;  
  37.     short reg_access_size;  
  38.     const struct snd_soc_reg_access *reg_access_default;  
  39.     enum snd_soc_compress_type compress_type;  
  40.   
  41.     /* codec bias level */  
  42.     int (*set_bias_level)(struct snd_soc_codec *,  
  43.                   enum snd_soc_bias_level level);  
  44.   
  45.     void (*seq_notifier)(struct snd_soc_dapm_context *,  
  46.                  enum snd_soc_dapm_type, int);  
  47.   
  48.     /* probe ordering - for components with runtime dependencies */  
  49.     int probe_order;  
  50.     int remove_order;  
  51. };  

位于snd_soc_codec_driver中的一些codec IO成员函数和set_bias_level回调函数原来都放在另外一个结构体snd_soc_codec中,现在放置在这里了,这是根据源码结构调整的结果。事实真正需要设置的成员也不是很多,如下:

[cpp]
  1. static struct snd_soc_codec_driver soc_codec_dev_wm9713 = {  
  2.     .probe =    wm9713_soc_probe,  
  3.     .remove =   wm9713_soc_remove,  
  4.     .suspend =  wm9713_soc_suspend,  
  5.     .resume =   wm9713_soc_resume,  
  6.     .read = ac97_read,  
  7.     .write = ac97_write,  
  8.     .set_bias_level = wm9713_set_bias_level,  
  9.     .reg_cache_size = ARRAY_SIZE(wm9713_reg),  
  10.     .reg_word_size = sizeof(u16),  
  11.     .reg_cache_step = 2,  
  12.     .reg_cache_default = wm9713_reg,  
  13.     .dapm_widgets = wm9713_dapm_widgets,  
  14.     .num_dapm_widgets = ARRAY_SIZE(wm9713_dapm_widgets),  
  15.     .dapm_routes = wm9713_audio_map,  
  16.     .num_dapm_routes = ARRAY_SIZE(wm9713_audio_map),  
  17. };  

probe、remove、suspend、resume相信不用累述了,volatile_register函数判断指定的寄存器是否volatile,reg_cache_size一般为寄存器数目,reg_word_size为寄存器的字长,reg_cache_default为寄存器默认值配置表。

而dapm_widgets、dapm_routes就比较“犀利”了。之前的dapm widgets和routes分别通过函数snd_soc_dapm_new_controls和snd_soc_dapm_add_routes来注册的(当然现在还保留这些接口),现在则可以填入到这个结构体,在soc-core里注册,省了不少功夫:

[cpp]
  1. static int soc_probe_codec(struct snd_soc_card *card,  
  2.                struct snd_soc_codec *codec)  
  3. {  
  4.     //...   
  5.   
  6.     if (driver->dapm_widgets)  
  7.         snd_soc_dapm_new_controls(&codec->dapm, driver->dapm_widgets,  
  8.                       driver->num_dapm_widgets);  
  9.   
  10.     //...   
  11.   
  12.     if (driver->controls)  
  13.         snd_soc_add_controls(codec, driver->controls,  
  14.                      driver->num_controls);  
  15.     if (driver->dapm_routes)  
  16.         snd_soc_dapm_add_routes(&codec->dapm, driver->dapm_routes,  
  17.                     driver->num_dapm_routes);  
  18.     //...   
  19. }  


同时可看到dirver ops的函数参数有所不同了,以前是struct platform_device *pdev,现在改为struct snd_soc_codec *codec,这个与codec的注册函数snd_soc_register_codec和设备的drvdata有关,之后会逐一分析。

  • 1
  • 2
  • 3
  • 下一页

相关内容