1888dcb7cSJohannes Berg /* 2888dcb7cSJohannes Berg * Apple Onboard Audio Alsa helpers 3888dcb7cSJohannes Berg * 4888dcb7cSJohannes Berg * Copyright 2006 Johannes Berg <johannes@sipsolutions.net> 5888dcb7cSJohannes Berg * 6888dcb7cSJohannes Berg * GPL v2, can be found in COPYING. 7888dcb7cSJohannes Berg */ 8888dcb7cSJohannes Berg #include <linux/module.h> 9888dcb7cSJohannes Berg #include "alsa.h" 10888dcb7cSJohannes Berg 11888dcb7cSJohannes Berg static int index = -1; 12888dcb7cSJohannes Berg module_param(index, int, 0444); 13888dcb7cSJohannes Berg MODULE_PARM_DESC(index, "index for AOA sound card."); 14888dcb7cSJohannes Berg 15888dcb7cSJohannes Berg static struct aoa_card *aoa_card; 16888dcb7cSJohannes Berg 17888dcb7cSJohannes Berg int aoa_alsa_init(char *name, struct module *mod, struct device *dev) 18888dcb7cSJohannes Berg { 19888dcb7cSJohannes Berg struct snd_card *alsa_card; 20888dcb7cSJohannes Berg int err; 21888dcb7cSJohannes Berg 22888dcb7cSJohannes Berg if (aoa_card) 23888dcb7cSJohannes Berg /* cannot be EEXIST due to usage in aoa_fabric_register */ 24888dcb7cSJohannes Berg return -EBUSY; 25888dcb7cSJohannes Berg 26*10768797STakashi Iwai err = snd_card_new(dev, index, name, mod, sizeof(struct aoa_card), 27bd7dd77cSTakashi Iwai &alsa_card); 28bd7dd77cSTakashi Iwai if (err < 0) 29bd7dd77cSTakashi Iwai return err; 30888dcb7cSJohannes Berg aoa_card = alsa_card->private_data; 31888dcb7cSJohannes Berg aoa_card->alsa_card = alsa_card; 32888dcb7cSJohannes Berg strlcpy(alsa_card->driver, "AppleOnbdAudio", sizeof(alsa_card->driver)); 33888dcb7cSJohannes Berg strlcpy(alsa_card->shortname, name, sizeof(alsa_card->shortname)); 34888dcb7cSJohannes Berg strlcpy(alsa_card->longname, name, sizeof(alsa_card->longname)); 35888dcb7cSJohannes Berg strlcpy(alsa_card->mixername, name, sizeof(alsa_card->mixername)); 36888dcb7cSJohannes Berg err = snd_card_register(aoa_card->alsa_card); 37888dcb7cSJohannes Berg if (err < 0) { 38888dcb7cSJohannes Berg printk(KERN_ERR "snd-aoa: couldn't register alsa card\n"); 39888dcb7cSJohannes Berg snd_card_free(aoa_card->alsa_card); 40888dcb7cSJohannes Berg aoa_card = NULL; 41888dcb7cSJohannes Berg return err; 42888dcb7cSJohannes Berg } 43888dcb7cSJohannes Berg return 0; 44888dcb7cSJohannes Berg } 45888dcb7cSJohannes Berg 46888dcb7cSJohannes Berg struct snd_card *aoa_get_card(void) 47888dcb7cSJohannes Berg { 48888dcb7cSJohannes Berg if (aoa_card) 49888dcb7cSJohannes Berg return aoa_card->alsa_card; 50888dcb7cSJohannes Berg return NULL; 51888dcb7cSJohannes Berg } 52888dcb7cSJohannes Berg EXPORT_SYMBOL_GPL(aoa_get_card); 53888dcb7cSJohannes Berg 54888dcb7cSJohannes Berg void aoa_alsa_cleanup(void) 55888dcb7cSJohannes Berg { 56888dcb7cSJohannes Berg if (aoa_card) { 57888dcb7cSJohannes Berg snd_card_free(aoa_card->alsa_card); 58888dcb7cSJohannes Berg aoa_card = NULL; 59888dcb7cSJohannes Berg } 60888dcb7cSJohannes Berg } 61888dcb7cSJohannes Berg 62888dcb7cSJohannes Berg int aoa_snd_device_new(snd_device_type_t type, 63888dcb7cSJohannes Berg void * device_data, struct snd_device_ops * ops) 64888dcb7cSJohannes Berg { 65888dcb7cSJohannes Berg struct snd_card *card = aoa_get_card(); 66888dcb7cSJohannes Berg int err; 67888dcb7cSJohannes Berg 68888dcb7cSJohannes Berg if (!card) return -ENOMEM; 69888dcb7cSJohannes Berg 70888dcb7cSJohannes Berg err = snd_device_new(card, type, device_data, ops); 71888dcb7cSJohannes Berg if (err) { 72888dcb7cSJohannes Berg printk(KERN_ERR "snd-aoa: failed to create snd device (%d)\n", err); 73888dcb7cSJohannes Berg return err; 74888dcb7cSJohannes Berg } 75888dcb7cSJohannes Berg err = snd_device_register(card, device_data); 76888dcb7cSJohannes Berg if (err) { 77888dcb7cSJohannes Berg printk(KERN_ERR "snd-aoa: failed to register " 78888dcb7cSJohannes Berg "snd device (%d)\n", err); 79888dcb7cSJohannes Berg printk(KERN_ERR "snd-aoa: have you forgotten the " 80888dcb7cSJohannes Berg "dev_register callback?\n"); 81888dcb7cSJohannes Berg snd_device_free(card, device_data); 82888dcb7cSJohannes Berg } 83888dcb7cSJohannes Berg return err; 84888dcb7cSJohannes Berg } 85888dcb7cSJohannes Berg EXPORT_SYMBOL_GPL(aoa_snd_device_new); 86888dcb7cSJohannes Berg 87888dcb7cSJohannes Berg int aoa_snd_ctl_add(struct snd_kcontrol* control) 88888dcb7cSJohannes Berg { 89888dcb7cSJohannes Berg int err; 90888dcb7cSJohannes Berg 91888dcb7cSJohannes Berg if (!aoa_card) return -ENODEV; 92888dcb7cSJohannes Berg 93888dcb7cSJohannes Berg err = snd_ctl_add(aoa_card->alsa_card, control); 94888dcb7cSJohannes Berg if (err) 95888dcb7cSJohannes Berg printk(KERN_ERR "snd-aoa: failed to add alsa control (%d)\n", 96888dcb7cSJohannes Berg err); 97888dcb7cSJohannes Berg return err; 98888dcb7cSJohannes Berg } 99888dcb7cSJohannes Berg EXPORT_SYMBOL_GPL(aoa_snd_ctl_add); 100