Commit aac8da65 authored by Alexander Shishkin's avatar Alexander Shishkin Committed by Greg Kroah-Hartman

intel_th: msu: Start handling IRQs

We intend to use the interrupt to detect Last Block condition in the MSU
driver, which we can use for double-buffering software-managed data
transfers.

Add an interrupt handler to the MSU driver.
Signed-off-by: default avatarAlexander Shishkin <alexander.shishkin@linux.intel.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 7b7036d4
...@@ -826,6 +826,28 @@ static const struct file_operations intel_th_output_fops = { ...@@ -826,6 +826,28 @@ static const struct file_operations intel_th_output_fops = {
.llseek = noop_llseek, .llseek = noop_llseek,
}; };
static irqreturn_t intel_th_irq(int irq, void *data)
{
struct intel_th *th = data;
irqreturn_t ret = IRQ_NONE;
struct intel_th_driver *d;
int i;
for (i = 0; i < th->num_thdevs; i++) {
if (th->thdev[i]->type != INTEL_TH_OUTPUT)
continue;
d = to_intel_th_driver(th->thdev[i]->dev.driver);
if (d && d->irq)
ret |= d->irq(th->thdev[i]);
}
if (ret == IRQ_NONE)
pr_warn_ratelimited("nobody cared for irq\n");
return ret;
}
/** /**
* intel_th_alloc() - allocate a new Intel TH device and its subdevices * intel_th_alloc() - allocate a new Intel TH device and its subdevices
* @dev: parent device * @dev: parent device
...@@ -865,6 +887,12 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata, ...@@ -865,6 +887,12 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata,
th->resource[nr_mmios++] = devres[r]; th->resource[nr_mmios++] = devres[r];
break; break;
case IORESOURCE_IRQ: case IORESOURCE_IRQ:
err = devm_request_irq(dev, devres[r].start,
intel_th_irq, IRQF_SHARED,
dev_name(dev), th);
if (err)
goto err_chrdev;
if (th->irq == -1) if (th->irq == -1)
th->irq = devres[r].start; th->irq = devres[r].start;
break; break;
...@@ -891,6 +919,10 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata, ...@@ -891,6 +919,10 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata,
return th; return th;
err_chrdev:
__unregister_chrdev(th->major, 0, TH_POSSIBLE_OUTPUTS,
"intel_th/output");
err_ida: err_ida:
ida_simple_remove(&intel_th_ida, th->id); ida_simple_remove(&intel_th_ida, th->id);
......
...@@ -8,6 +8,8 @@ ...@@ -8,6 +8,8 @@
#ifndef __INTEL_TH_H__ #ifndef __INTEL_TH_H__
#define __INTEL_TH_H__ #define __INTEL_TH_H__
#include <linux/irqreturn.h>
/* intel_th_device device types */ /* intel_th_device device types */
enum { enum {
/* Devices that generate trace data */ /* Devices that generate trace data */
...@@ -160,7 +162,7 @@ struct intel_th_driver { ...@@ -160,7 +162,7 @@ struct intel_th_driver {
void (*disable)(struct intel_th_device *thdev, void (*disable)(struct intel_th_device *thdev,
struct intel_th_output *output); struct intel_th_output *output);
/* output ops */ /* output ops */
void (*irq)(struct intel_th_device *thdev); irqreturn_t (*irq)(struct intel_th_device *thdev);
int (*activate)(struct intel_th_device *thdev); int (*activate)(struct intel_th_device *thdev);
void (*deactivate)(struct intel_th_device *thdev); void (*deactivate)(struct intel_th_device *thdev);
/* file_operations for those who want a device node */ /* file_operations for those who want a device node */
......
...@@ -102,6 +102,7 @@ struct msc_iter { ...@@ -102,6 +102,7 @@ struct msc_iter {
*/ */
struct msc { struct msc {
void __iomem *reg_base; void __iomem *reg_base;
void __iomem *msu_base;
struct intel_th_device *thdev; struct intel_th_device *thdev;
struct list_head win_list; struct list_head win_list;
...@@ -122,7 +123,8 @@ struct msc { ...@@ -122,7 +123,8 @@ struct msc {
/* config */ /* config */
unsigned int enabled : 1, unsigned int enabled : 1,
wrap : 1; wrap : 1,
do_irq : 1;
unsigned int mode; unsigned int mode;
unsigned int burst_len; unsigned int burst_len;
unsigned int index; unsigned int index;
...@@ -476,6 +478,40 @@ static void msc_buffer_clear_hw_header(struct msc *msc) ...@@ -476,6 +478,40 @@ static void msc_buffer_clear_hw_header(struct msc *msc)
} }
} }
static int intel_th_msu_init(struct msc *msc)
{
u32 mintctl, msusts;
if (!msc->do_irq)
return 0;
mintctl = ioread32(msc->msu_base + REG_MSU_MINTCTL);
mintctl |= msc->index ? M1BLIE : M0BLIE;
iowrite32(mintctl, msc->msu_base + REG_MSU_MINTCTL);
if (mintctl != ioread32(msc->msu_base + REG_MSU_MINTCTL)) {
dev_info(msc_dev(msc), "MINTCTL ignores writes: no usable interrupts\n");
msc->do_irq = 0;
return 0;
}
msusts = ioread32(msc->msu_base + REG_MSU_MSUSTS);
iowrite32(msusts, msc->msu_base + REG_MSU_MSUSTS);
return 0;
}
static void intel_th_msu_deinit(struct msc *msc)
{
u32 mintctl;
if (!msc->do_irq)
return;
mintctl = ioread32(msc->msu_base + REG_MSU_MINTCTL);
mintctl &= msc->index ? ~M1BLIE : ~M0BLIE;
iowrite32(mintctl, msc->msu_base + REG_MSU_MINTCTL);
}
/** /**
* msc_configure() - set up MSC hardware * msc_configure() - set up MSC hardware
* @msc: the MSC device to configure * @msc: the MSC device to configure
...@@ -1295,6 +1331,21 @@ static int intel_th_msc_init(struct msc *msc) ...@@ -1295,6 +1331,21 @@ static int intel_th_msc_init(struct msc *msc)
return 0; return 0;
} }
static irqreturn_t intel_th_msc_interrupt(struct intel_th_device *thdev)
{
struct msc *msc = dev_get_drvdata(&thdev->dev);
u32 msusts = ioread32(msc->msu_base + REG_MSU_MSUSTS);
u32 mask = msc->index ? MSUSTS_MSC1BLAST : MSUSTS_MSC0BLAST;
if (!(msusts & mask)) {
if (msc->enabled)
return IRQ_HANDLED;
return IRQ_NONE;
}
return IRQ_HANDLED;
}
static const char * const msc_mode[] = { static const char * const msc_mode[] = {
[MSC_MODE_SINGLE] = "single", [MSC_MODE_SINGLE] = "single",
[MSC_MODE_MULTI] = "multi", [MSC_MODE_MULTI] = "multi",
...@@ -1500,10 +1551,19 @@ static int intel_th_msc_probe(struct intel_th_device *thdev) ...@@ -1500,10 +1551,19 @@ static int intel_th_msc_probe(struct intel_th_device *thdev)
if (!msc) if (!msc)
return -ENOMEM; return -ENOMEM;
res = intel_th_device_get_resource(thdev, IORESOURCE_IRQ, 1);
if (!res)
msc->do_irq = 1;
msc->index = thdev->id; msc->index = thdev->id;
msc->thdev = thdev; msc->thdev = thdev;
msc->reg_base = base + msc->index * 0x100; msc->reg_base = base + msc->index * 0x100;
msc->msu_base = base;
err = intel_th_msu_init(msc);
if (err)
return err;
err = intel_th_msc_init(msc); err = intel_th_msc_init(msc);
if (err) if (err)
...@@ -1520,6 +1580,7 @@ static void intel_th_msc_remove(struct intel_th_device *thdev) ...@@ -1520,6 +1580,7 @@ static void intel_th_msc_remove(struct intel_th_device *thdev)
int ret; int ret;
intel_th_msc_deactivate(thdev); intel_th_msc_deactivate(thdev);
intel_th_msu_deinit(msc);
/* /*
* Buffers should not be used at this point except if the * Buffers should not be used at this point except if the
...@@ -1533,6 +1594,7 @@ static void intel_th_msc_remove(struct intel_th_device *thdev) ...@@ -1533,6 +1594,7 @@ static void intel_th_msc_remove(struct intel_th_device *thdev)
static struct intel_th_driver intel_th_msc_driver = { static struct intel_th_driver intel_th_msc_driver = {
.probe = intel_th_msc_probe, .probe = intel_th_msc_probe,
.remove = intel_th_msc_remove, .remove = intel_th_msc_remove,
.irq = intel_th_msc_interrupt,
.activate = intel_th_msc_activate, .activate = intel_th_msc_activate,
.deactivate = intel_th_msc_deactivate, .deactivate = intel_th_msc_deactivate,
.fops = &intel_th_msc_fops, .fops = &intel_th_msc_fops,
......
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
enum { enum {
REG_MSU_MSUPARAMS = 0x0000, REG_MSU_MSUPARAMS = 0x0000,
REG_MSU_MSUSTS = 0x0008, REG_MSU_MSUSTS = 0x0008,
REG_MSU_MINTCTL = 0x0004, /* MSU-global interrupt control */
REG_MSU_MSC0CTL = 0x0100, /* MSC0 control */ REG_MSU_MSC0CTL = 0x0100, /* MSC0 control */
REG_MSU_MSC0STS = 0x0104, /* MSC0 status */ REG_MSU_MSC0STS = 0x0104, /* MSC0 status */
REG_MSU_MSC0BAR = 0x0108, /* MSC0 output base address */ REG_MSU_MSC0BAR = 0x0108, /* MSC0 output base address */
...@@ -28,6 +29,8 @@ enum { ...@@ -28,6 +29,8 @@ enum {
/* MSUSTS bits */ /* MSUSTS bits */
#define MSUSTS_MSU_INT BIT(0) #define MSUSTS_MSU_INT BIT(0)
#define MSUSTS_MSC0BLAST BIT(16)
#define MSUSTS_MSC1BLAST BIT(24)
/* MSCnCTL bits */ /* MSCnCTL bits */
#define MSC_EN BIT(0) #define MSC_EN BIT(0)
...@@ -36,6 +39,11 @@ enum { ...@@ -36,6 +39,11 @@ enum {
#define MSC_MODE (BIT(4) | BIT(5)) #define MSC_MODE (BIT(4) | BIT(5))
#define MSC_LEN (BIT(8) | BIT(9) | BIT(10)) #define MSC_LEN (BIT(8) | BIT(9) | BIT(10))
/* MINTCTL bits */
#define MICDE BIT(0)
#define M0BLIE BIT(16)
#define M1BLIE BIT(24)
/* MSC operating modes (MSC_MODE) */ /* MSC operating modes (MSC_MODE) */
enum { enum {
MSC_MODE_SINGLE = 0, MSC_MODE_SINGLE = 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