Commit 7833c3f8 authored by Takashi Iwai's avatar Takashi Iwai

ALSA: hda - Migrate hdac_stream into legacy driver

Embed hdac_stream object into azx_dev, and use a few basic helper
functions.  The most of helper codes for hdac_stream aren't still used
yet.

Also this commit disables the tracepoints temporarily due to build
problems.  It'll be enabled again later.
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent a43ff5ba
This diff is collapsed.
...@@ -59,36 +59,10 @@ enum { ...@@ -59,36 +59,10 @@ enum {
}; };
struct azx_dev { struct azx_dev {
struct snd_dma_buffer bdl; /* BDL buffer */ struct hdac_stream core;
u32 *posbuf; /* position buffer pointer */
unsigned int bufsize; /* size of the play buffer in bytes */
unsigned int period_bytes; /* size of the period in bytes */
unsigned int frags; /* number for period in the play buffer */
unsigned int fifo_size; /* FIFO size */
unsigned long start_wallclk; /* start + minimum wallclk */
unsigned long period_wallclk; /* wallclk for period */
void __iomem *sd_addr; /* stream descriptor pointer */
u32 sd_int_sta_mask; /* stream int status mask */
/* pcm support */
struct snd_pcm_substream *substream; /* assigned substream,
* set in PCM open
*/
unsigned int format_val; /* format value to be set in the
* controller and the codec
*/
unsigned char stream_tag; /* assigned stream */
unsigned char index; /* stream index */
int assigned_key; /* last device# key assigned to */
unsigned int opened:1;
unsigned int running:1;
unsigned int irq_pending:1; unsigned int irq_pending:1;
unsigned int prepared:1; unsigned int prepared:1;
unsigned int locked:1;
/* /*
* For VIA: * For VIA:
* A flag to ensure DMA position is 0 * A flag to ensure DMA position is 0
...@@ -96,19 +70,11 @@ struct azx_dev { ...@@ -96,19 +70,11 @@ struct azx_dev {
*/ */
unsigned int insufficient:1; unsigned int insufficient:1;
unsigned int wc_marked:1; unsigned int wc_marked:1;
unsigned int no_period_wakeup:1;
struct timecounter azx_tc;
struct cyclecounter azx_cc;
int delay_negative_threshold;
#ifdef CONFIG_SND_HDA_DSP_LOADER
/* Allows dsp load to have sole access to the playback stream. */
struct mutex dsp_mutex;
#endif
}; };
#define azx_stream(dev) (&(dev)->core)
#define stream_to_azx_dev(s) container_of(s, struct azx_dev, core)
/* CORB/RIRB */ /* CORB/RIRB */
struct azx_rb { struct azx_rb {
u32 *buf; /* CORB/RIRB buffer u32 *buf; /* CORB/RIRB buffer
...@@ -181,9 +147,6 @@ struct azx { ...@@ -181,9 +147,6 @@ struct azx {
spinlock_t reg_lock; spinlock_t reg_lock;
struct mutex open_mutex; /* Prevents concurrent open/close operations */ struct mutex open_mutex; /* Prevents concurrent open/close operations */
/* streams (x num_streams) */
struct azx_dev *azx_dev;
/* PCM */ /* PCM */
struct list_head pcm_list; /* azx_pcm list */ struct list_head pcm_list; /* azx_pcm list */
...@@ -253,17 +216,17 @@ struct azx { ...@@ -253,17 +216,17 @@ struct azx {
((chip)->io_ops->reg_readb((chip)->remap_addr + AZX_REG_##reg)) ((chip)->io_ops->reg_readb((chip)->remap_addr + AZX_REG_##reg))
#define azx_sd_writel(chip, dev, reg, value) \ #define azx_sd_writel(chip, dev, reg, value) \
((chip)->io_ops->reg_writel(value, (dev)->sd_addr + AZX_REG_##reg)) snd_hdac_stream_writel(&(dev)->core, reg, value)
#define azx_sd_readl(chip, dev, reg) \ #define azx_sd_readl(chip, dev, reg) \
((chip)->io_ops->reg_readl((dev)->sd_addr + AZX_REG_##reg)) snd_hdac_stream_readl(&(dev)->core, reg)
#define azx_sd_writew(chip, dev, reg, value) \ #define azx_sd_writew(chip, dev, reg, value) \
((chip)->io_ops->reg_writew(value, (dev)->sd_addr + AZX_REG_##reg)) snd_hdac_stream_writew(&(dev)->core, reg, value)
#define azx_sd_readw(chip, dev, reg) \ #define azx_sd_readw(chip, dev, reg) \
((chip)->io_ops->reg_readw((dev)->sd_addr + AZX_REG_##reg)) snd_hdac_stream_readw(&(dev)->core, reg)
#define azx_sd_writeb(chip, dev, reg, value) \ #define azx_sd_writeb(chip, dev, reg, value) \
((chip)->io_ops->reg_writeb(value, (dev)->sd_addr + AZX_REG_##reg)) snd_hdac_stream_writeb(&(dev)->core, reg, value)
#define azx_sd_readb(chip, dev, reg) \ #define azx_sd_readb(chip, dev, reg) \
((chip)->io_ops->reg_readb((dev)->sd_addr + AZX_REG_##reg)) snd_hdac_stream_readb(&(dev)->core, reg)
#define azx_has_pm_runtime(chip) \ #define azx_has_pm_runtime(chip) \
(!AZX_DCAPS_PM_RUNTIME || ((chip)->driver_caps & AZX_DCAPS_PM_RUNTIME)) (!AZX_DCAPS_PM_RUNTIME || ((chip)->driver_caps & AZX_DCAPS_PM_RUNTIME))
...@@ -278,7 +241,7 @@ unsigned int azx_get_pos_lpib(struct azx *chip, struct azx_dev *azx_dev); ...@@ -278,7 +241,7 @@ unsigned int azx_get_pos_lpib(struct azx *chip, struct azx_dev *azx_dev);
unsigned int azx_get_pos_posbuf(struct azx *chip, struct azx_dev *azx_dev); unsigned int azx_get_pos_posbuf(struct azx *chip, struct azx_dev *azx_dev);
/* Stream control. */ /* Stream control. */
void azx_stream_stop(struct azx *chip, struct azx_dev *azx_dev); void azx_stop_all_streams(struct azx *chip);
/* Allocation functions. */ /* Allocation functions. */
int azx_alloc_stream_pages(struct azx *chip); int azx_alloc_stream_pages(struct azx *chip);
......
...@@ -492,7 +492,7 @@ static void azx_init_pci(struct azx *chip) ...@@ -492,7 +492,7 @@ static void azx_init_pci(struct azx *chip)
static int azx_get_delay_from_lpib(struct azx *chip, struct azx_dev *azx_dev, static int azx_get_delay_from_lpib(struct azx *chip, struct azx_dev *azx_dev,
unsigned int pos) unsigned int pos)
{ {
struct snd_pcm_substream *substream = azx_dev->substream; struct snd_pcm_substream *substream = azx_dev->core.substream;
int stream = substream->stream; int stream = substream->stream;
unsigned int lpib_pos = azx_get_pos_lpib(chip, azx_dev); unsigned int lpib_pos = azx_get_pos_lpib(chip, azx_dev);
int delay; int delay;
...@@ -502,16 +502,16 @@ static int azx_get_delay_from_lpib(struct azx *chip, struct azx_dev *azx_dev, ...@@ -502,16 +502,16 @@ static int azx_get_delay_from_lpib(struct azx *chip, struct azx_dev *azx_dev,
else else
delay = lpib_pos - pos; delay = lpib_pos - pos;
if (delay < 0) { if (delay < 0) {
if (delay >= azx_dev->delay_negative_threshold) if (delay >= azx_dev->core.delay_negative_threshold)
delay = 0; delay = 0;
else else
delay += azx_dev->bufsize; delay += azx_dev->core.bufsize;
} }
if (delay >= azx_dev->period_bytes) { if (delay >= azx_dev->core.period_bytes) {
dev_info(chip->card->dev, dev_info(chip->card->dev,
"Unstable LPIB (%d >= %d); disabling LPIB delay counting\n", "Unstable LPIB (%d >= %d); disabling LPIB delay counting\n",
delay, azx_dev->period_bytes); delay, azx_dev->core.period_bytes);
delay = 0; delay = 0;
chip->driver_caps &= ~AZX_DCAPS_COUNT_LPIB_DELAY; chip->driver_caps &= ~AZX_DCAPS_COUNT_LPIB_DELAY;
chip->get_delay[stream] = NULL; chip->get_delay[stream] = NULL;
...@@ -551,13 +551,13 @@ static int azx_position_check(struct azx *chip, struct azx_dev *azx_dev) ...@@ -551,13 +551,13 @@ static int azx_position_check(struct azx *chip, struct azx_dev *azx_dev)
*/ */
static int azx_position_ok(struct azx *chip, struct azx_dev *azx_dev) static int azx_position_ok(struct azx *chip, struct azx_dev *azx_dev)
{ {
struct snd_pcm_substream *substream = azx_dev->substream; struct snd_pcm_substream *substream = azx_dev->core.substream;
int stream = substream->stream; int stream = substream->stream;
u32 wallclk; u32 wallclk;
unsigned int pos; unsigned int pos;
wallclk = azx_readl(chip, WALLCLK) - azx_dev->start_wallclk; wallclk = azx_readl(chip, WALLCLK) - azx_dev->core.start_wallclk;
if (wallclk < (azx_dev->period_wallclk * 2) / 3) if (wallclk < (azx_dev->core.period_wallclk * 2) / 3)
return -1; /* bogus (too early) interrupt */ return -1; /* bogus (too early) interrupt */
if (chip->get_position[stream]) if (chip->get_position[stream])
...@@ -577,17 +577,17 @@ static int azx_position_ok(struct azx *chip, struct azx_dev *azx_dev) ...@@ -577,17 +577,17 @@ static int azx_position_ok(struct azx *chip, struct azx_dev *azx_dev)
} }
} }
if (pos >= azx_dev->bufsize) if (pos >= azx_dev->core.bufsize)
pos = 0; pos = 0;
if (WARN_ONCE(!azx_dev->period_bytes, if (WARN_ONCE(!azx_dev->core.period_bytes,
"hda-intel: zero azx_dev->period_bytes")) "hda-intel: zero azx_dev->period_bytes"))
return -1; /* this shouldn't happen! */ return -1; /* this shouldn't happen! */
if (wallclk < (azx_dev->period_wallclk * 5) / 4 && if (wallclk < (azx_dev->core.period_wallclk * 5) / 4 &&
pos % azx_dev->period_bytes > azx_dev->period_bytes / 2) pos % azx_dev->core.period_bytes > azx_dev->core.period_bytes / 2)
/* NG - it's below the first next period boundary */ /* NG - it's below the first next period boundary */
return chip->bdl_pos_adj[chip->dev_index] ? 0 : -1; return chip->bdl_pos_adj[chip->dev_index] ? 0 : -1;
azx_dev->start_wallclk += wallclk; azx_dev->core.start_wallclk += wallclk;
return 1; /* OK, it's fine */ return 1; /* OK, it's fine */
} }
...@@ -598,7 +598,9 @@ static void azx_irq_pending_work(struct work_struct *work) ...@@ -598,7 +598,9 @@ static void azx_irq_pending_work(struct work_struct *work)
{ {
struct hda_intel *hda = container_of(work, struct hda_intel, irq_pending_work); struct hda_intel *hda = container_of(work, struct hda_intel, irq_pending_work);
struct azx *chip = &hda->chip; struct azx *chip = &hda->chip;
int i, pending, ok; struct hdac_bus *bus = azx_bus(chip);
struct hdac_stream *s;
int pending, ok;
if (!hda->irq_pending_warned) { if (!hda->irq_pending_warned) {
dev_info(chip->card->dev, dev_info(chip->card->dev,
...@@ -610,17 +612,17 @@ static void azx_irq_pending_work(struct work_struct *work) ...@@ -610,17 +612,17 @@ static void azx_irq_pending_work(struct work_struct *work)
for (;;) { for (;;) {
pending = 0; pending = 0;
spin_lock_irq(&chip->reg_lock); spin_lock_irq(&chip->reg_lock);
for (i = 0; i < chip->num_streams; i++) { list_for_each_entry(s, &bus->stream_list, list) {
struct azx_dev *azx_dev = &chip->azx_dev[i]; struct azx_dev *azx_dev = stream_to_azx_dev(s);
if (!azx_dev->irq_pending || if (!azx_dev->irq_pending ||
!azx_dev->substream || !s->substream ||
!azx_dev->running) !s->running)
continue; continue;
ok = azx_position_ok(chip, azx_dev); ok = azx_position_ok(chip, azx_dev);
if (ok > 0) { if (ok > 0) {
azx_dev->irq_pending = 0; azx_dev->irq_pending = 0;
spin_unlock(&chip->reg_lock); spin_unlock(&chip->reg_lock);
snd_pcm_period_elapsed(azx_dev->substream); snd_pcm_period_elapsed(s->substream);
spin_lock(&chip->reg_lock); spin_lock(&chip->reg_lock);
} else if (ok < 0) { } else if (ok < 0) {
pending = 0; /* too early */ pending = 0; /* too early */
...@@ -637,11 +639,14 @@ static void azx_irq_pending_work(struct work_struct *work) ...@@ -637,11 +639,14 @@ static void azx_irq_pending_work(struct work_struct *work)
/* clear irq_pending flags and assure no on-going workq */ /* clear irq_pending flags and assure no on-going workq */
static void azx_clear_irq_pending(struct azx *chip) static void azx_clear_irq_pending(struct azx *chip)
{ {
int i; struct hdac_bus *bus = azx_bus(chip);
struct hdac_stream *s;
spin_lock_irq(&chip->reg_lock); spin_lock_irq(&chip->reg_lock);
for (i = 0; i < chip->num_streams; i++) list_for_each_entry(s, &bus->stream_list, list) {
chip->azx_dev[i].irq_pending = 0; struct azx_dev *azx_dev = stream_to_azx_dev(s);
azx_dev->irq_pending = 0;
}
spin_unlock_irq(&chip->reg_lock); spin_unlock_irq(&chip->reg_lock);
} }
...@@ -671,7 +676,7 @@ static unsigned int azx_via_get_position(struct azx *chip, ...@@ -671,7 +676,7 @@ static unsigned int azx_via_get_position(struct azx *chip,
unsigned int fifo_size; unsigned int fifo_size;
link_pos = azx_sd_readl(chip, azx_dev, SD_LPIB); link_pos = azx_sd_readl(chip, azx_dev, SD_LPIB);
if (azx_dev->substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { if (azx_dev->core.substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
/* Playback, no problem using link position */ /* Playback, no problem using link position */
return link_pos; return link_pos;
} }
...@@ -680,8 +685,8 @@ static unsigned int azx_via_get_position(struct azx *chip, ...@@ -680,8 +685,8 @@ static unsigned int azx_via_get_position(struct azx *chip,
/* For new chipset, /* For new chipset,
* use mod to get the DMA position just like old chipset * use mod to get the DMA position just like old chipset
*/ */
mod_dma_pos = le32_to_cpu(*azx_dev->posbuf); mod_dma_pos = le32_to_cpu(*azx_dev->core.posbuf);
mod_dma_pos %= azx_dev->period_bytes; mod_dma_pos %= azx_dev->core.period_bytes;
/* azx_dev->fifo_size can't get FIFO size of in stream. /* azx_dev->fifo_size can't get FIFO size of in stream.
* Get from base address + offset. * Get from base address + offset.
...@@ -697,20 +702,20 @@ static unsigned int azx_via_get_position(struct azx *chip, ...@@ -697,20 +702,20 @@ static unsigned int azx_via_get_position(struct azx *chip,
} }
if (link_pos <= fifo_size) if (link_pos <= fifo_size)
mini_pos = azx_dev->bufsize + link_pos - fifo_size; mini_pos = azx_dev->core.bufsize + link_pos - fifo_size;
else else
mini_pos = link_pos - fifo_size; mini_pos = link_pos - fifo_size;
/* Find nearest previous boudary */ /* Find nearest previous boudary */
mod_mini_pos = mini_pos % azx_dev->period_bytes; mod_mini_pos = mini_pos % azx_dev->core.period_bytes;
mod_link_pos = link_pos % azx_dev->period_bytes; mod_link_pos = link_pos % azx_dev->core.period_bytes;
if (mod_link_pos >= fifo_size) if (mod_link_pos >= fifo_size)
bound_pos = link_pos - mod_link_pos; bound_pos = link_pos - mod_link_pos;
else if (mod_dma_pos >= mod_mini_pos) else if (mod_dma_pos >= mod_mini_pos)
bound_pos = mini_pos - mod_mini_pos; bound_pos = mini_pos - mod_mini_pos;
else { else {
bound_pos = mini_pos - mod_mini_pos + azx_dev->period_bytes; bound_pos = mini_pos - mod_mini_pos + azx_dev->core.period_bytes;
if (bound_pos >= azx_dev->bufsize) if (bound_pos >= azx_dev->core.bufsize)
bound_pos = 0; bound_pos = 0;
} }
...@@ -1063,7 +1068,6 @@ static int azx_free(struct azx *chip) ...@@ -1063,7 +1068,6 @@ static int azx_free(struct azx *chip)
{ {
struct pci_dev *pci = chip->pci; struct pci_dev *pci = chip->pci;
struct hda_intel *hda = container_of(chip, struct hda_intel, chip); struct hda_intel *hda = container_of(chip, struct hda_intel, chip);
int i;
if (azx_has_pm_runtime(chip) && chip->running) if (azx_has_pm_runtime(chip) && chip->running)
pm_runtime_get_noresume(&pci->dev); pm_runtime_get_noresume(&pci->dev);
...@@ -1082,8 +1086,7 @@ static int azx_free(struct azx *chip) ...@@ -1082,8 +1086,7 @@ static int azx_free(struct azx *chip)
if (chip->initialized) { if (chip->initialized) {
azx_clear_irq_pending(chip); azx_clear_irq_pending(chip);
for (i = 0; i < chip->num_streams; i++) azx_stop_all_streams(chip);
azx_stream_stop(chip, &chip->azx_dev[i]);
azx_stop_chip(chip); azx_stop_chip(chip);
} }
...@@ -1097,7 +1100,6 @@ static int azx_free(struct azx *chip) ...@@ -1097,7 +1100,6 @@ static int azx_free(struct azx *chip)
if (chip->region_requested) if (chip->region_requested)
pci_release_regions(chip->pci); pci_release_regions(chip->pci);
pci_disable_device(chip->pci); pci_disable_device(chip->pci);
kfree(chip->azx_dev);
#ifdef CONFIG_SND_HDA_PATCH_LOADER #ifdef CONFIG_SND_HDA_PATCH_LOADER
release_firmware(chip->fw); release_firmware(chip->fw);
#endif #endif
...@@ -1566,10 +1568,6 @@ static int azx_first_init(struct azx *chip) ...@@ -1566,10 +1568,6 @@ static int azx_first_init(struct azx *chip)
chip->capture_index_offset = 0; chip->capture_index_offset = 0;
chip->playback_index_offset = chip->capture_streams; chip->playback_index_offset = chip->capture_streams;
chip->num_streams = chip->playback_streams + chip->capture_streams; chip->num_streams = chip->playback_streams + chip->capture_streams;
chip->azx_dev = kcalloc(chip->num_streams, sizeof(*chip->azx_dev),
GFP_KERNEL);
if (!chip->azx_dev)
return -ENOMEM;
err = azx_alloc_stream_pages(chip); err = azx_alloc_stream_pages(chip);
if (err < 0) if (err < 0)
...@@ -1717,9 +1715,9 @@ static int substream_alloc_pages(struct azx *chip, ...@@ -1717,9 +1715,9 @@ static int substream_alloc_pages(struct azx *chip,
int ret; int ret;
mark_runtime_wc(chip, azx_dev, substream, false); mark_runtime_wc(chip, azx_dev, substream, false);
azx_dev->bufsize = 0; azx_dev->core.bufsize = 0;
azx_dev->period_bytes = 0; azx_dev->core.period_bytes = 0;
azx_dev->format_val = 0; azx_dev->core.format_val = 0;
ret = snd_pcm_lib_malloc_pages(substream, size); ret = snd_pcm_lib_malloc_pages(substream, size);
if (ret < 0) if (ret < 0)
return ret; return ret;
......
...@@ -104,9 +104,9 @@ static int substream_alloc_pages(struct azx *chip, ...@@ -104,9 +104,9 @@ static int substream_alloc_pages(struct azx *chip,
{ {
struct azx_dev *azx_dev = get_azx_dev(substream); struct azx_dev *azx_dev = get_azx_dev(substream);
azx_dev->bufsize = 0; azx_dev->core.bufsize = 0;
azx_dev->period_bytes = 0; azx_dev->core.period_bytes = 0;
azx_dev->format_val = 0; azx_dev->core.format_val = 0;
return snd_pcm_lib_malloc_pages(substream, size); return snd_pcm_lib_malloc_pages(substream, size);
} }
...@@ -290,12 +290,10 @@ static const struct dev_pm_ops hda_tegra_pm = { ...@@ -290,12 +290,10 @@ static const struct dev_pm_ops hda_tegra_pm = {
*/ */
static int hda_tegra_dev_free(struct snd_device *device) static int hda_tegra_dev_free(struct snd_device *device)
{ {
int i;
struct azx *chip = device->device_data; struct azx *chip = device->device_data;
if (chip->initialized) { if (chip->initialized) {
for (i = 0; i < chip->num_streams; i++) azx_stop_all_streams(chip);
azx_stream_stop(chip, &chip->azx_dev[i]);
azx_stop_chip(chip); azx_stop_chip(chip);
} }
...@@ -377,10 +375,6 @@ static int hda_tegra_first_init(struct azx *chip, struct platform_device *pdev) ...@@ -377,10 +375,6 @@ static int hda_tegra_first_init(struct azx *chip, struct platform_device *pdev)
chip->capture_index_offset = 0; chip->capture_index_offset = 0;
chip->playback_index_offset = chip->capture_streams; chip->playback_index_offset = chip->capture_streams;
chip->num_streams = chip->playback_streams + chip->capture_streams; chip->num_streams = chip->playback_streams + chip->capture_streams;
chip->azx_dev = devm_kcalloc(card->dev, chip->num_streams,
sizeof(*chip->azx_dev), GFP_KERNEL);
if (!chip->azx_dev)
return -ENOMEM;
err = azx_alloc_stream_pages(chip); err = azx_alloc_stream_pages(chip);
if (err < 0) if (err < 0)
......
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