ALSA: echoaudio: Allocate resources with device-managed APIs

This patch converts the resource management in PCI echoaudio drivers
with devres as a clean up.  Each manual resource management is
converted with the corresponding devres helper, the page allocations
are done with the devres helper, and the card object release is
managed now via card->private_free instead of a lowlevel snd_device.
The irq handler is still managed manually because it's re-acquired at
PM suspend/resume.

This should give no user-visible functional changes.

Link: https://lore.kernel.org/r/20210715075941.23332-33-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
Takashi Iwai 2021-07-15 09:58:54 +02:00
parent 5eba4c646d
commit 9c211bf392
2 changed files with 48 additions and 122 deletions
sound/pci/echoaudio

@ -1882,105 +1882,63 @@ static irqreturn_t snd_echo_interrupt(int irq, void *dev_id)
Module construction / destruction Module construction / destruction
******************************************************************************/ ******************************************************************************/
static int snd_echo_free(struct echoaudio *chip) static void snd_echo_free(struct snd_card *card)
{ {
struct echoaudio *chip = card->private_data;
if (chip->comm_page) if (chip->comm_page)
rest_in_peace(chip); rest_in_peace(chip);
if (chip->irq >= 0) if (chip->irq >= 0)
free_irq(chip->irq, chip); free_irq(chip->irq, chip);
if (chip->comm_page)
snd_dma_free_pages(&chip->commpage_dma_buf);
iounmap(chip->dsp_registers);
release_and_free_resource(chip->iores);
pci_disable_device(chip->pci);
/* release chip data */ /* release chip data */
free_firmware_cache(chip); free_firmware_cache(chip);
kfree(chip);
return 0;
} }
static int snd_echo_dev_free(struct snd_device *device)
{
struct echoaudio *chip = device->device_data;
return snd_echo_free(chip);
}
/* <--snd_echo_probe() */ /* <--snd_echo_probe() */
static int snd_echo_create(struct snd_card *card, static int snd_echo_create(struct snd_card *card,
struct pci_dev *pci, struct pci_dev *pci)
struct echoaudio **rchip)
{ {
struct echoaudio *chip; struct echoaudio *chip = card->private_data;
int err; int err;
size_t sz; size_t sz;
static const struct snd_device_ops ops = {
.dev_free = snd_echo_dev_free,
};
*rchip = NULL;
pci_write_config_byte(pci, PCI_LATENCY_TIMER, 0xC0); pci_write_config_byte(pci, PCI_LATENCY_TIMER, 0xC0);
err = pci_enable_device(pci); err = pcim_enable_device(pci);
if (err < 0) if (err < 0)
return err; return err;
pci_set_master(pci); pci_set_master(pci);
/* Allocate chip if needed */ /* Allocate chip if needed */
if (!*rchip) { spin_lock_init(&chip->lock);
chip = kzalloc(sizeof(*chip), GFP_KERNEL); chip->card = card;
if (!chip) { chip->pci = pci;
pci_disable_device(pci); chip->irq = -1;
return -ENOMEM; chip->opencount = 0;
} mutex_init(&chip->mode_mutex);
dev_dbg(card->dev, "chip=%p\n", chip); chip->can_set_rate = 1;
spin_lock_init(&chip->lock);
chip->card = card;
chip->pci = pci;
chip->irq = -1;
chip->opencount = 0;
mutex_init(&chip->mode_mutex);
chip->can_set_rate = 1;
} else {
/* If this was called from the resume function, chip is
* already allocated and it contains current card settings.
*/
chip = *rchip;
}
/* PCI resource allocation */ /* PCI resource allocation */
err = pci_request_regions(pci, ECHOCARD_NAME);
if (err < 0)
return err;
chip->dsp_registers_phys = pci_resource_start(pci, 0); chip->dsp_registers_phys = pci_resource_start(pci, 0);
sz = pci_resource_len(pci, 0); sz = pci_resource_len(pci, 0);
if (sz > PAGE_SIZE) if (sz > PAGE_SIZE)
sz = PAGE_SIZE; /* We map only the required part */ sz = PAGE_SIZE; /* We map only the required part */
chip->iores = request_mem_region(chip->dsp_registers_phys, sz, chip->dsp_registers = devm_ioremap(&pci->dev, chip->dsp_registers_phys, sz);
ECHOCARD_NAME);
if (!chip->iores) {
dev_err(chip->card->dev, "cannot get memory region\n");
snd_echo_free(chip);
return -EBUSY;
}
chip->dsp_registers = ioremap(chip->dsp_registers_phys, sz);
if (!chip->dsp_registers) { if (!chip->dsp_registers) {
dev_err(chip->card->dev, "ioremap failed\n"); dev_err(chip->card->dev, "ioremap failed\n");
snd_echo_free(chip);
return -ENOMEM; return -ENOMEM;
} }
if (request_irq(pci->irq, snd_echo_interrupt, IRQF_SHARED, if (request_irq(pci->irq, snd_echo_interrupt, IRQF_SHARED,
KBUILD_MODNAME, chip)) { KBUILD_MODNAME, chip)) {
dev_err(chip->card->dev, "cannot grab irq\n"); dev_err(chip->card->dev, "cannot grab irq\n");
snd_echo_free(chip);
return -EBUSY; return -EBUSY;
} }
chip->irq = pci->irq; chip->irq = pci->irq;
@ -1988,39 +1946,29 @@ static int snd_echo_create(struct snd_card *card,
dev_dbg(card->dev, "pci=%p irq=%d subdev=%04x Init hardware...\n", dev_dbg(card->dev, "pci=%p irq=%d subdev=%04x Init hardware...\n",
chip->pci, chip->irq, chip->pci->subsystem_device); chip->pci, chip->irq, chip->pci->subsystem_device);
card->private_free = snd_echo_free;
/* Create the DSP comm page - this is the area of memory used for most /* Create the DSP comm page - this is the area of memory used for most
of the communication with the DSP, which accesses it via bus mastering */ of the communication with the DSP, which accesses it via bus mastering */
if (snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, &chip->pci->dev, chip->commpage_dma_buf =
sizeof(struct comm_page), snd_devm_alloc_pages(&pci->dev, SNDRV_DMA_TYPE_DEV,
&chip->commpage_dma_buf) < 0) { sizeof(struct comm_page));
dev_err(chip->card->dev, "cannot allocate the comm page\n"); if (!chip->commpage_dma_buf)
snd_echo_free(chip);
return -ENOMEM; return -ENOMEM;
} chip->comm_page_phys = chip->commpage_dma_buf->addr;
chip->comm_page_phys = chip->commpage_dma_buf.addr; chip->comm_page = (struct comm_page *)chip->commpage_dma_buf->area;
chip->comm_page = (struct comm_page *)chip->commpage_dma_buf.area;
err = init_hw(chip, chip->pci->device, chip->pci->subsystem_device); err = init_hw(chip, chip->pci->device, chip->pci->subsystem_device);
if (err >= 0) if (err >= 0)
err = set_mixer_defaults(chip); err = set_mixer_defaults(chip);
if (err < 0) { if (err < 0) {
dev_err(card->dev, "init_hw err=%d\n", err); dev_err(card->dev, "init_hw err=%d\n", err);
snd_echo_free(chip);
return err; return err;
} }
err = snd_device_new(card, SNDRV_DEV_LOWLEVEL, chip, &ops);
if (err < 0) {
snd_echo_free(chip);
return err;
}
*rchip = chip;
/* Init done ! */
return 0; return 0;
} }
/* constructor */ /* constructor */
static int snd_echo_probe(struct pci_dev *pci, static int snd_echo_probe(struct pci_dev *pci,
const struct pci_device_id *pci_id) const struct pci_device_id *pci_id)
@ -2040,17 +1988,15 @@ static int snd_echo_probe(struct pci_dev *pci,
} }
i = 0; i = 0;
err = snd_card_new(&pci->dev, index[dev], id[dev], THIS_MODULE, err = snd_devm_card_new(&pci->dev, index[dev], id[dev], THIS_MODULE,
0, &card); sizeof(*chip), &card);
if (err < 0) if (err < 0)
return err; return err;
chip = card->private_data;
chip = NULL; /* Tells snd_echo_create to allocate chip */ err = snd_echo_create(card, pci);
err = snd_echo_create(card, pci, &chip); if (err < 0)
if (err < 0) {
snd_card_free(card);
return err; return err;
}
strcpy(card->driver, "Echo_" ECHOCARD_NAME); strcpy(card->driver, "Echo_" ECHOCARD_NAME);
strcpy(card->shortname, chip->card_name); strcpy(card->shortname, chip->card_name);
@ -2066,7 +2012,6 @@ static int snd_echo_probe(struct pci_dev *pci,
err = snd_echo_new_pcm(chip); err = snd_echo_new_pcm(chip);
if (err < 0) { if (err < 0) {
dev_err(chip->card->dev, "new pcm error %d\n", err); dev_err(chip->card->dev, "new pcm error %d\n", err);
snd_card_free(card);
return err; return err;
} }
@ -2075,7 +2020,6 @@ static int snd_echo_probe(struct pci_dev *pci,
err = snd_echo_midi_create(card, chip); err = snd_echo_midi_create(card, chip);
if (err < 0) { if (err < 0) {
dev_err(chip->card->dev, "new midi error %d\n", err); dev_err(chip->card->dev, "new midi error %d\n", err);
snd_card_free(card);
return err; return err;
} }
} }
@ -2085,64 +2029,64 @@ static int snd_echo_probe(struct pci_dev *pci,
snd_echo_vmixer.count = num_pipes_out(chip) * num_busses_out(chip); snd_echo_vmixer.count = num_pipes_out(chip) * num_busses_out(chip);
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_vmixer, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_vmixer, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#ifdef ECHOCARD_HAS_LINE_OUT_GAIN #ifdef ECHOCARD_HAS_LINE_OUT_GAIN
err = snd_ctl_add(chip->card, err = snd_ctl_add(chip->card,
snd_ctl_new1(&snd_echo_line_output_gain, chip)); snd_ctl_new1(&snd_echo_line_output_gain, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif #endif
#else /* ECHOCARD_HAS_VMIXER */ #else /* ECHOCARD_HAS_VMIXER */
err = snd_ctl_add(chip->card, err = snd_ctl_add(chip->card,
snd_ctl_new1(&snd_echo_pcm_output_gain, chip)); snd_ctl_new1(&snd_echo_pcm_output_gain, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif /* ECHOCARD_HAS_VMIXER */ #endif /* ECHOCARD_HAS_VMIXER */
#ifdef ECHOCARD_HAS_INPUT_GAIN #ifdef ECHOCARD_HAS_INPUT_GAIN
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_line_input_gain, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_line_input_gain, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif #endif
#ifdef ECHOCARD_HAS_INPUT_NOMINAL_LEVEL #ifdef ECHOCARD_HAS_INPUT_NOMINAL_LEVEL
if (!chip->hasnt_input_nominal_level) { if (!chip->hasnt_input_nominal_level) {
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_intput_nominal_level, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_intput_nominal_level, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
} }
#endif #endif
#ifdef ECHOCARD_HAS_OUTPUT_NOMINAL_LEVEL #ifdef ECHOCARD_HAS_OUTPUT_NOMINAL_LEVEL
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_output_nominal_level, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_output_nominal_level, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif #endif
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_vumeters_switch, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_vumeters_switch, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_vumeters, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_vumeters, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#ifdef ECHOCARD_HAS_MONITOR #ifdef ECHOCARD_HAS_MONITOR
snd_echo_monitor_mixer.count = num_busses_in(chip) * num_busses_out(chip); snd_echo_monitor_mixer.count = num_busses_in(chip) * num_busses_out(chip);
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_monitor_mixer, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_monitor_mixer, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif #endif
#ifdef ECHOCARD_HAS_DIGITAL_IN_AUTOMUTE #ifdef ECHOCARD_HAS_DIGITAL_IN_AUTOMUTE
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_automute_switch, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_automute_switch, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif #endif
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_channels_info, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_channels_info, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#ifdef ECHOCARD_HAS_DIGITAL_MODE_SWITCH #ifdef ECHOCARD_HAS_DIGITAL_MODE_SWITCH
/* Creates a list of available digital modes */ /* Creates a list of available digital modes */
@ -2153,7 +2097,7 @@ static int snd_echo_probe(struct pci_dev *pci,
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_digital_mode_switch, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_digital_mode_switch, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif /* ECHOCARD_HAS_DIGITAL_MODE_SWITCH */ #endif /* ECHOCARD_HAS_DIGITAL_MODE_SWITCH */
#ifdef ECHOCARD_HAS_EXTERNAL_CLOCK #ifdef ECHOCARD_HAS_EXTERNAL_CLOCK
@ -2167,37 +2111,32 @@ static int snd_echo_probe(struct pci_dev *pci,
chip->clock_src_ctl = snd_ctl_new1(&snd_echo_clock_source_switch, chip); chip->clock_src_ctl = snd_ctl_new1(&snd_echo_clock_source_switch, chip);
err = snd_ctl_add(chip->card, chip->clock_src_ctl); err = snd_ctl_add(chip->card, chip->clock_src_ctl);
if (err < 0) if (err < 0)
goto ctl_error; return err;
} }
#endif /* ECHOCARD_HAS_EXTERNAL_CLOCK */ #endif /* ECHOCARD_HAS_EXTERNAL_CLOCK */
#ifdef ECHOCARD_HAS_DIGITAL_IO #ifdef ECHOCARD_HAS_DIGITAL_IO
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_spdif_mode_switch, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_spdif_mode_switch, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
#endif #endif
#ifdef ECHOCARD_HAS_PHANTOM_POWER #ifdef ECHOCARD_HAS_PHANTOM_POWER
if (chip->has_phantom_power) { if (chip->has_phantom_power) {
err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_phantom_power_switch, chip)); err = snd_ctl_add(chip->card, snd_ctl_new1(&snd_echo_phantom_power_switch, chip));
if (err < 0) if (err < 0)
goto ctl_error; return err;
} }
#endif #endif
err = snd_card_register(card); err = snd_card_register(card);
if (err < 0) if (err < 0)
goto ctl_error; return err;
dev_info(card->dev, "Card registered: %s\n", card->longname); dev_info(card->dev, "Card registered: %s\n", card->longname);
pci_set_drvdata(pci, chip); pci_set_drvdata(pci, chip);
dev++; dev++;
return 0; return 0;
ctl_error:
dev_err(card->dev, "new control error %d\n", err);
snd_card_free(card);
return err;
} }
@ -2299,18 +2238,6 @@ static SIMPLE_DEV_PM_OPS(snd_echo_pm, snd_echo_suspend, snd_echo_resume);
#define SND_ECHO_PM_OPS NULL #define SND_ECHO_PM_OPS NULL
#endif /* CONFIG_PM_SLEEP */ #endif /* CONFIG_PM_SLEEP */
static void snd_echo_remove(struct pci_dev *pci)
{
struct echoaudio *chip;
chip = pci_get_drvdata(pci);
if (chip)
snd_card_free(chip->card);
}
/****************************************************************************** /******************************************************************************
Everything starts and ends here Everything starts and ends here
******************************************************************************/ ******************************************************************************/
@ -2320,7 +2247,6 @@ static struct pci_driver echo_driver = {
.name = KBUILD_MODNAME, .name = KBUILD_MODNAME,
.id_table = snd_echo_ids, .id_table = snd_echo_ids,
.probe = snd_echo_probe, .probe = snd_echo_probe,
.remove = snd_echo_remove,
.driver = { .driver = {
.pm = SND_ECHO_PM_OPS, .pm = SND_ECHO_PM_OPS,
}, },

@ -348,7 +348,7 @@ struct echoaudio {
struct pci_dev *pci; struct pci_dev *pci;
unsigned long dsp_registers_phys; unsigned long dsp_registers_phys;
struct resource *iores; struct resource *iores;
struct snd_dma_buffer commpage_dma_buf; struct snd_dma_buffer *commpage_dma_buf;
int irq; int irq;
#ifdef ECHOCARD_HAS_MIDI #ifdef ECHOCARD_HAS_MIDI
struct snd_rawmidi *rmidi; struct snd_rawmidi *rmidi;