Commit 7835e090 authored by Takashi Iwai's avatar Takashi Iwai

ALSA: intel8x0: Allocate resources with device-managed APIs

This patch refactors the intel8x0 and intel8x0m driver codes using
devres and gets rid of the driver remove callback.

The conversion is fairly straightforward: each API call is replaced
with the device-managed API function, e.g. pci_enable_device() ->
pcim_enable_device(), and so on.  The buffer descriptor list is
allocated with a new API, snd_devm_alloc_pages().

A slight code structure change is that the intel8x0 object is
allocated as a card's private_data instead of the own lowlevel
snd_device object.  This simplifies the resource management.
And, the take-down procedure is triggered via card->private_free, and
it's registered at the end of the whole initialization, i.e. after the
all resources get properly managed.

The only not-devres-managed resource is the irq handler.  Since we
need to release at suspend and re-acquire at resume (otherwise
something weird happens on some machines), this is still managed
manually.  But the rest are all freed automatically.

The end result is a good amount of code reduction.

Link: https://lore.kernel.org/r/20210715075941.23332-6-tiwai@suse.deSigned-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent ac327f1b
...@@ -378,7 +378,7 @@ struct intel8x0 { ...@@ -378,7 +378,7 @@ struct intel8x0 {
spinlock_t reg_lock; spinlock_t reg_lock;
u32 bdbars_count; u32 bdbars_count;
struct snd_dma_buffer bdbars; struct snd_dma_buffer *bdbars;
u32 int_sta_reg; /* interrupt status register */ u32 int_sta_reg; /* interrupt status register */
u32 int_sta_mask; /* interrupt status mask */ u32 int_sta_mask; /* interrupt status mask */
}; };
...@@ -2528,8 +2528,9 @@ static int snd_intel8x0_chip_init(struct intel8x0 *chip, int probing) ...@@ -2528,8 +2528,9 @@ static int snd_intel8x0_chip_init(struct intel8x0 *chip, int probing)
return 0; return 0;
} }
static int snd_intel8x0_free(struct intel8x0 *chip) static void snd_intel8x0_free(struct snd_card *card)
{ {
struct intel8x0 *chip = card->private_data;
unsigned int i; unsigned int i;
if (chip->irq < 0) if (chip->irq < 0)
...@@ -2552,16 +2553,6 @@ static int snd_intel8x0_free(struct intel8x0 *chip) ...@@ -2552,16 +2553,6 @@ static int snd_intel8x0_free(struct intel8x0 *chip)
__hw_end: __hw_end:
if (chip->irq >= 0) if (chip->irq >= 0)
free_irq(chip->irq, chip); free_irq(chip->irq, chip);
if (chip->bdbars.area)
snd_dma_free_pages(&chip->bdbars);
if (chip->addr)
pci_iounmap(chip->pci, chip->addr);
if (chip->bmaddr)
pci_iounmap(chip->pci, chip->bmaddr);
pci_release_regions(chip->pci);
pci_disable_device(chip->pci);
kfree(chip);
return 0;
} }
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
...@@ -2831,12 +2822,6 @@ static void snd_intel8x0_proc_init(struct intel8x0 *chip) ...@@ -2831,12 +2822,6 @@ static void snd_intel8x0_proc_init(struct intel8x0 *chip)
snd_intel8x0_proc_read); snd_intel8x0_proc_read);
} }
static int snd_intel8x0_dev_free(struct snd_device *device)
{
struct intel8x0 *chip = device->device_data;
return snd_intel8x0_free(chip);
}
struct ich_reg_info { struct ich_reg_info {
unsigned int int_sta_mask; unsigned int int_sta_mask;
unsigned int offset; unsigned int offset;
...@@ -2880,19 +2865,15 @@ static int snd_intel8x0_inside_vm(struct pci_dev *pci) ...@@ -2880,19 +2865,15 @@ static int snd_intel8x0_inside_vm(struct pci_dev *pci)
return result; return result;
} }
static int snd_intel8x0_create(struct snd_card *card, static int snd_intel8x0_init(struct snd_card *card,
struct pci_dev *pci, struct pci_dev *pci,
unsigned long device_type, unsigned long device_type)
struct intel8x0 **r_intel8x0)
{ {
struct intel8x0 *chip; struct intel8x0 *chip = card->private_data;
int err; int err;
unsigned int i; unsigned int i;
unsigned int int_sta_masks; unsigned int int_sta_masks;
struct ichdev *ichdev; struct ichdev *ichdev;
static const struct snd_device_ops ops = {
.dev_free = snd_intel8x0_dev_free,
};
static const unsigned int bdbars[] = { static const unsigned int bdbars[] = {
3, /* DEVICE_INTEL */ 3, /* DEVICE_INTEL */
...@@ -2925,17 +2906,10 @@ static int snd_intel8x0_create(struct snd_card *card, ...@@ -2925,17 +2906,10 @@ static int snd_intel8x0_create(struct snd_card *card,
}; };
const struct ich_reg_info *tbl; const struct ich_reg_info *tbl;
*r_intel8x0 = NULL; err = pcim_enable_device(pci);
err = pci_enable_device(pci);
if (err < 0) if (err < 0)
return err; return err;
chip = kzalloc(sizeof(*chip), GFP_KERNEL);
if (chip == NULL) {
pci_disable_device(pci);
return -ENOMEM;
}
spin_lock_init(&chip->reg_lock); spin_lock_init(&chip->reg_lock);
chip->device_type = device_type; chip->device_type = device_type;
chip->card = card; chip->card = card;
...@@ -2961,38 +2935,23 @@ static int snd_intel8x0_create(struct snd_card *card, ...@@ -2961,38 +2935,23 @@ static int snd_intel8x0_create(struct snd_card *card,
chip->fix_nocache = 1; /* enable workaround */ chip->fix_nocache = 1; /* enable workaround */
err = pci_request_regions(pci, card->shortname); err = pci_request_regions(pci, card->shortname);
if (err < 0) { if (err < 0)
kfree(chip);
pci_disable_device(pci);
return err; return err;
}
if (device_type == DEVICE_ALI) { if (device_type == DEVICE_ALI) {
/* ALI5455 has no ac97 region */ /* ALI5455 has no ac97 region */
chip->bmaddr = pci_iomap(pci, 0, 0); chip->bmaddr = pcim_iomap(pci, 0, 0);
goto port_inited; } else {
} if (pci_resource_flags(pci, 2) & IORESOURCE_MEM) /* ICH4 and Nforce */
chip->addr = pcim_iomap(pci, 2, 0);
if (pci_resource_flags(pci, 2) & IORESOURCE_MEM) /* ICH4 and Nforce */ else
chip->addr = pci_iomap(pci, 2, 0); chip->addr = pcim_iomap(pci, 0, 0);
else if (pci_resource_flags(pci, 3) & IORESOURCE_MEM) /* ICH4 */
chip->addr = pci_iomap(pci, 0, 0); chip->bmaddr = pcim_iomap(pci, 3, 0);
if (!chip->addr) { else
dev_err(card->dev, "AC'97 space ioremap problem\n"); chip->bmaddr = pcim_iomap(pci, 1, 0);
snd_intel8x0_free(chip);
return -EIO;
} }
if (pci_resource_flags(pci, 3) & IORESOURCE_MEM) /* ICH4 */
chip->bmaddr = pci_iomap(pci, 3, 0);
else
chip->bmaddr = pci_iomap(pci, 1, 0);
port_inited:
if (!chip->bmaddr) {
dev_err(card->dev, "Controller space ioremap problem\n");
snd_intel8x0_free(chip);
return -EIO;
}
chip->bdbars_count = bdbars[device_type]; chip->bdbars_count = bdbars[device_type];
/* initialize offsets */ /* initialize offsets */
...@@ -3028,21 +2987,20 @@ static int snd_intel8x0_create(struct snd_card *card, ...@@ -3028,21 +2987,20 @@ static int snd_intel8x0_create(struct snd_card *card,
/* allocate buffer descriptor lists */ /* allocate buffer descriptor lists */
/* the start of each lists must be aligned to 8 bytes */ /* the start of each lists must be aligned to 8 bytes */
if (snd_dma_alloc_pages(intel8x0_dma_type(chip), &pci->dev, chip->bdbars = snd_devm_alloc_pages(&pci->dev, intel8x0_dma_type(chip),
chip->bdbars_count * sizeof(u32) * ICH_MAX_FRAGS * 2, chip->bdbars_count * sizeof(u32) *
&chip->bdbars) < 0) { ICH_MAX_FRAGS * 2);
snd_intel8x0_free(chip); if (!chip->bdbars)
dev_err(card->dev, "cannot allocate buffer descriptors\n");
return -ENOMEM; return -ENOMEM;
}
/* tables must be aligned to 8 bytes here, but the kernel pages /* tables must be aligned to 8 bytes here, but the kernel pages
are much bigger, so we don't care (on i386) */ are much bigger, so we don't care (on i386) */
int_sta_masks = 0; int_sta_masks = 0;
for (i = 0; i < chip->bdbars_count; i++) { for (i = 0; i < chip->bdbars_count; i++) {
ichdev = &chip->ichd[i]; ichdev = &chip->ichd[i];
ichdev->bdbar = ((__le32 *)chip->bdbars.area) + ichdev->bdbar = ((__le32 *)chip->bdbars->area) +
(i * ICH_MAX_FRAGS * 2); (i * ICH_MAX_FRAGS * 2);
ichdev->bdbar_addr = chip->bdbars.addr + ichdev->bdbar_addr = chip->bdbars->addr +
(i * sizeof(u32) * ICH_MAX_FRAGS * 2); (i * sizeof(u32) * ICH_MAX_FRAGS * 2);
int_sta_masks |= ichdev->int_sta_mask; int_sta_masks |= ichdev->int_sta_mask;
} }
...@@ -3076,28 +3034,24 @@ static int snd_intel8x0_create(struct snd_card *card, ...@@ -3076,28 +3034,24 @@ static int snd_intel8x0_create(struct snd_card *card,
chip->codec_isr_bits |= chip->codec_bit[i]; chip->codec_isr_bits |= chip->codec_bit[i];
err = snd_intel8x0_chip_init(chip, 1); err = snd_intel8x0_chip_init(chip, 1);
if (err < 0) { if (err < 0)
snd_intel8x0_free(chip);
return err; return err;
}
/* request irq after initializaing int_sta_mask, etc */ /* request irq after initializaing int_sta_mask, etc */
/* NOTE: we don't use devm version here since it's released /
* re-acquired in PM callbacks.
* It's released explicitly in snd_intel8x0_free(), too.
*/
if (request_irq(pci->irq, snd_intel8x0_interrupt, if (request_irq(pci->irq, snd_intel8x0_interrupt,
IRQF_SHARED, KBUILD_MODNAME, chip)) { IRQF_SHARED, KBUILD_MODNAME, chip)) {
dev_err(card->dev, "unable to grab IRQ %d\n", pci->irq); dev_err(card->dev, "unable to grab IRQ %d\n", pci->irq);
snd_intel8x0_free(chip);
return -EBUSY; return -EBUSY;
} }
chip->irq = pci->irq; chip->irq = pci->irq;
card->sync_irq = chip->irq; card->sync_irq = chip->irq;
err = snd_device_new(card, SNDRV_DEV_LOWLEVEL, chip, &ops); card->private_free = snd_intel8x0_free;
if (err < 0) {
snd_intel8x0_free(chip);
return err;
}
*r_intel8x0 = chip;
return 0; return 0;
} }
...@@ -3163,9 +3117,11 @@ static int snd_intel8x0_probe(struct pci_dev *pci, ...@@ -3163,9 +3117,11 @@ static int snd_intel8x0_probe(struct pci_dev *pci,
int err; int err;
struct shortname_table *name; struct shortname_table *name;
err = snd_card_new(&pci->dev, index, id, THIS_MODULE, 0, &card); err = snd_devm_card_new(&pci->dev, index, id, THIS_MODULE,
sizeof(*chip), &card);
if (err < 0) if (err < 0)
return err; return err;
chip = card->private_data;
if (spdif_aclink < 0) if (spdif_aclink < 0)
spdif_aclink = check_default_spdif_aclink(pci); spdif_aclink = check_default_spdif_aclink(pci);
...@@ -3199,23 +3155,16 @@ static int snd_intel8x0_probe(struct pci_dev *pci, ...@@ -3199,23 +3155,16 @@ static int snd_intel8x0_probe(struct pci_dev *pci,
buggy_irq = 0; buggy_irq = 0;
} }
err = snd_intel8x0_create(card, pci, pci_id->driver_data, &chip); err = snd_intel8x0_init(card, pci, pci_id->driver_data);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
card->private_data = chip;
err = snd_intel8x0_mixer(chip, ac97_clock, ac97_quirk); err = snd_intel8x0_mixer(chip, ac97_clock, ac97_quirk);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
err = snd_intel8x0_pcm(chip); err = snd_intel8x0_pcm(chip);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
snd_intel8x0_proc_init(chip); snd_intel8x0_proc_init(chip);
...@@ -3233,24 +3182,17 @@ static int snd_intel8x0_probe(struct pci_dev *pci, ...@@ -3233,24 +3182,17 @@ static int snd_intel8x0_probe(struct pci_dev *pci,
} }
err = snd_card_register(card); err = snd_card_register(card);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
pci_set_drvdata(pci, card); pci_set_drvdata(pci, card);
return 0; return 0;
} }
static void snd_intel8x0_remove(struct pci_dev *pci)
{
snd_card_free(pci_get_drvdata(pci));
}
static struct pci_driver intel8x0_driver = { static struct pci_driver intel8x0_driver = {
.name = KBUILD_MODNAME, .name = KBUILD_MODNAME,
.id_table = snd_intel8x0_ids, .id_table = snd_intel8x0_ids,
.probe = snd_intel8x0_probe, .probe = snd_intel8x0_probe,
.remove = snd_intel8x0_remove,
.driver = { .driver = {
.pm = INTEL8X0_PM_OPS, .pm = INTEL8X0_PM_OPS,
}, },
......
...@@ -182,7 +182,7 @@ struct intel8x0m { ...@@ -182,7 +182,7 @@ struct intel8x0m {
spinlock_t reg_lock; spinlock_t reg_lock;
struct snd_dma_buffer bdbars; struct snd_dma_buffer *bdbars;
u32 bdbars_count; u32 bdbars_count;
u32 int_sta_reg; /* interrupt status register */ u32 int_sta_reg; /* interrupt status register */
u32 int_sta_mask; /* interrupt status mask */ u32 int_sta_mask; /* interrupt status mask */
...@@ -947,8 +947,9 @@ static int snd_intel8x0m_chip_init(struct intel8x0m *chip, int probing) ...@@ -947,8 +947,9 @@ static int snd_intel8x0m_chip_init(struct intel8x0m *chip, int probing)
return 0; return 0;
} }
static int snd_intel8x0m_free(struct intel8x0m *chip) static void snd_intel8x0m_free(struct snd_card *card)
{ {
struct intel8x0m *chip = card->private_data;
unsigned int i; unsigned int i;
if (chip->irq < 0) if (chip->irq < 0)
...@@ -962,16 +963,6 @@ static int snd_intel8x0m_free(struct intel8x0m *chip) ...@@ -962,16 +963,6 @@ static int snd_intel8x0m_free(struct intel8x0m *chip)
__hw_end: __hw_end:
if (chip->irq >= 0) if (chip->irq >= 0)
free_irq(chip->irq, chip); free_irq(chip->irq, chip);
if (chip->bdbars.area)
snd_dma_free_pages(&chip->bdbars);
if (chip->addr)
pci_iounmap(chip->pci, chip->addr);
if (chip->bmaddr)
pci_iounmap(chip->pci, chip->bmaddr);
pci_release_regions(chip->pci);
pci_disable_device(chip->pci);
kfree(chip);
return 0;
} }
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
...@@ -1047,47 +1038,30 @@ static void snd_intel8x0m_proc_init(struct intel8x0m *chip) ...@@ -1047,47 +1038,30 @@ static void snd_intel8x0m_proc_init(struct intel8x0m *chip)
snd_intel8x0m_proc_read); snd_intel8x0m_proc_read);
} }
static int snd_intel8x0m_dev_free(struct snd_device *device)
{
struct intel8x0m *chip = device->device_data;
return snd_intel8x0m_free(chip);
}
struct ich_reg_info { struct ich_reg_info {
unsigned int int_sta_mask; unsigned int int_sta_mask;
unsigned int offset; unsigned int offset;
}; };
static int snd_intel8x0m_create(struct snd_card *card, static int snd_intel8x0m_init(struct snd_card *card,
struct pci_dev *pci, struct pci_dev *pci,
unsigned long device_type, unsigned long device_type)
struct intel8x0m **r_intel8x0m)
{ {
struct intel8x0m *chip; struct intel8x0m *chip = card->private_data;
int err; int err;
unsigned int i; unsigned int i;
unsigned int int_sta_masks; unsigned int int_sta_masks;
struct ichdev *ichdev; struct ichdev *ichdev;
static const struct snd_device_ops ops = {
.dev_free = snd_intel8x0m_dev_free,
};
static const struct ich_reg_info intel_regs[2] = { static const struct ich_reg_info intel_regs[2] = {
{ ICH_MIINT, 0 }, { ICH_MIINT, 0 },
{ ICH_MOINT, 0x10 }, { ICH_MOINT, 0x10 },
}; };
const struct ich_reg_info *tbl; const struct ich_reg_info *tbl;
*r_intel8x0m = NULL; err = pcim_enable_device(pci);
err = pci_enable_device(pci);
if (err < 0) if (err < 0)
return err; return err;
chip = kzalloc(sizeof(*chip), GFP_KERNEL);
if (chip == NULL) {
pci_disable_device(pci);
return -ENOMEM;
}
spin_lock_init(&chip->reg_lock); spin_lock_init(&chip->reg_lock);
chip->device_type = device_type; chip->device_type = device_type;
chip->card = card; chip->card = card;
...@@ -1095,37 +1069,21 @@ static int snd_intel8x0m_create(struct snd_card *card, ...@@ -1095,37 +1069,21 @@ static int snd_intel8x0m_create(struct snd_card *card,
chip->irq = -1; chip->irq = -1;
err = pci_request_regions(pci, card->shortname); err = pci_request_regions(pci, card->shortname);
if (err < 0) { if (err < 0)
kfree(chip);
pci_disable_device(pci);
return err; return err;
}
if (device_type == DEVICE_ALI) { if (device_type == DEVICE_ALI) {
/* ALI5455 has no ac97 region */ /* ALI5455 has no ac97 region */
chip->bmaddr = pci_iomap(pci, 0, 0); chip->bmaddr = pcim_iomap(pci, 0, 0);
goto port_inited; } else {
} if (pci_resource_flags(pci, 2) & IORESOURCE_MEM) /* ICH4 and Nforce */
chip->addr = pcim_iomap(pci, 2, 0);
if (pci_resource_flags(pci, 2) & IORESOURCE_MEM) /* ICH4 and Nforce */ else
chip->addr = pci_iomap(pci, 2, 0); chip->addr = pcim_iomap(pci, 0, 0);
else if (pci_resource_flags(pci, 3) & IORESOURCE_MEM) /* ICH4 */
chip->addr = pci_iomap(pci, 0, 0); chip->bmaddr = pcim_iomap(pci, 3, 0);
if (!chip->addr) { else
dev_err(card->dev, "AC'97 space ioremap problem\n"); chip->bmaddr = pcim_iomap(pci, 1, 0);
snd_intel8x0m_free(chip);
return -EIO;
}
if (pci_resource_flags(pci, 3) & IORESOURCE_MEM) /* ICH4 */
chip->bmaddr = pci_iomap(pci, 3, 0);
else
chip->bmaddr = pci_iomap(pci, 1, 0);
port_inited:
if (!chip->bmaddr) {
dev_err(card->dev, "Controller space ioremap problem\n");
snd_intel8x0m_free(chip);
return -EIO;
} }
/* initialize offsets */ /* initialize offsets */
...@@ -1153,19 +1111,19 @@ static int snd_intel8x0m_create(struct snd_card *card, ...@@ -1153,19 +1111,19 @@ static int snd_intel8x0m_create(struct snd_card *card,
/* allocate buffer descriptor lists */ /* allocate buffer descriptor lists */
/* the start of each lists must be aligned to 8 bytes */ /* the start of each lists must be aligned to 8 bytes */
if (snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, &pci->dev, chip->bdbars = snd_devm_alloc_pages(&pci->dev, SNDRV_DMA_TYPE_DEV,
chip->bdbars_count * sizeof(u32) * ICH_MAX_FRAGS * 2, chip->bdbars_count * sizeof(u32) *
&chip->bdbars) < 0) { ICH_MAX_FRAGS * 2);
snd_intel8x0m_free(chip); if (!chip->bdbars)
return -ENOMEM; return -ENOMEM;
}
/* tables must be aligned to 8 bytes here, but the kernel pages /* tables must be aligned to 8 bytes here, but the kernel pages
are much bigger, so we don't care (on i386) */ are much bigger, so we don't care (on i386) */
int_sta_masks = 0; int_sta_masks = 0;
for (i = 0; i < chip->bdbars_count; i++) { for (i = 0; i < chip->bdbars_count; i++) {
ichdev = &chip->ichd[i]; ichdev = &chip->ichd[i];
ichdev->bdbar = ((__le32 *)chip->bdbars.area) + (i * ICH_MAX_FRAGS * 2); ichdev->bdbar = ((__le32 *)chip->bdbars->area) + (i * ICH_MAX_FRAGS * 2);
ichdev->bdbar_addr = chip->bdbars.addr + (i * sizeof(u32) * ICH_MAX_FRAGS * 2); ichdev->bdbar_addr = chip->bdbars->addr + (i * sizeof(u32) * ICH_MAX_FRAGS * 2);
int_sta_masks |= ichdev->int_sta_mask; int_sta_masks |= ichdev->int_sta_mask;
} }
chip->int_sta_reg = ICH_REG_GLOB_STA; chip->int_sta_reg = ICH_REG_GLOB_STA;
...@@ -1174,27 +1132,23 @@ static int snd_intel8x0m_create(struct snd_card *card, ...@@ -1174,27 +1132,23 @@ static int snd_intel8x0m_create(struct snd_card *card,
pci_set_master(pci); pci_set_master(pci);
err = snd_intel8x0m_chip_init(chip, 1); err = snd_intel8x0m_chip_init(chip, 1);
if (err < 0) { if (err < 0)
snd_intel8x0m_free(chip);
return err; return err;
}
/* NOTE: we don't use devm version here since it's released /
* re-acquired in PM callbacks.
* It's released explicitly in snd_intel8x0m_free(), too.
*/
if (request_irq(pci->irq, snd_intel8x0m_interrupt, IRQF_SHARED, if (request_irq(pci->irq, snd_intel8x0m_interrupt, IRQF_SHARED,
KBUILD_MODNAME, chip)) { KBUILD_MODNAME, chip)) {
dev_err(card->dev, "unable to grab IRQ %d\n", pci->irq); dev_err(card->dev, "unable to grab IRQ %d\n", pci->irq);
snd_intel8x0m_free(chip);
return -EBUSY; return -EBUSY;
} }
chip->irq = pci->irq; chip->irq = pci->irq;
card->sync_irq = chip->irq; card->sync_irq = chip->irq;
err = snd_device_new(card, SNDRV_DEV_LOWLEVEL, chip, &ops); card->private_free = snd_intel8x0m_free;
if (err < 0) {
snd_intel8x0m_free(chip);
return err;
}
*r_intel8x0m = chip;
return 0; return 0;
} }
...@@ -1232,9 +1186,11 @@ static int snd_intel8x0m_probe(struct pci_dev *pci, ...@@ -1232,9 +1186,11 @@ static int snd_intel8x0m_probe(struct pci_dev *pci,
int err; int err;
struct shortname_table *name; struct shortname_table *name;
err = snd_card_new(&pci->dev, index, id, THIS_MODULE, 0, &card); err = snd_devm_card_new(&pci->dev, index, id, THIS_MODULE,
sizeof(*chip), &card);
if (err < 0) if (err < 0)
return err; return err;
chip = card->private_data;
strcpy(card->driver, "ICH-MODEM"); strcpy(card->driver, "ICH-MODEM");
strcpy(card->shortname, "Intel ICH"); strcpy(card->shortname, "Intel ICH");
...@@ -1246,23 +1202,16 @@ static int snd_intel8x0m_probe(struct pci_dev *pci, ...@@ -1246,23 +1202,16 @@ static int snd_intel8x0m_probe(struct pci_dev *pci,
} }
strcat(card->shortname," Modem"); strcat(card->shortname," Modem");
err = snd_intel8x0m_create(card, pci, pci_id->driver_data, &chip); err = snd_intel8x0m_init(card, pci, pci_id->driver_data);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
card->private_data = chip;
err = snd_intel8x0m_mixer(chip, ac97_clock); err = snd_intel8x0m_mixer(chip, ac97_clock);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
err = snd_intel8x0m_pcm(chip); err = snd_intel8x0m_pcm(chip);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
snd_intel8x0m_proc_init(chip); snd_intel8x0m_proc_init(chip);
...@@ -1270,24 +1219,16 @@ static int snd_intel8x0m_probe(struct pci_dev *pci, ...@@ -1270,24 +1219,16 @@ static int snd_intel8x0m_probe(struct pci_dev *pci,
card->shortname, chip->irq); card->shortname, chip->irq);
err = snd_card_register(card); err = snd_card_register(card);
if (err < 0) { if (err < 0)
snd_card_free(card);
return err; return err;
}
pci_set_drvdata(pci, card); pci_set_drvdata(pci, card);
return 0; return 0;
} }
static void snd_intel8x0m_remove(struct pci_dev *pci)
{
snd_card_free(pci_get_drvdata(pci));
}
static struct pci_driver intel8x0m_driver = { static struct pci_driver intel8x0m_driver = {
.name = KBUILD_MODNAME, .name = KBUILD_MODNAME,
.id_table = snd_intel8x0m_ids, .id_table = snd_intel8x0m_ids,
.probe = snd_intel8x0m_probe, .probe = snd_intel8x0m_probe,
.remove = snd_intel8x0m_remove,
.driver = { .driver = {
.pm = INTEL8X0M_PM_OPS, .pm = INTEL8X0M_PM_OPS,
}, },
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment