mirror of
https://github.com/edk2-porting/linux-next.git
synced 2024-12-27 14:43:58 +08:00
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: Alexander Shishkin <alexander.shishkin@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
7b7036d47c
commit
aac8da6517
@ -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,
|
|||||||
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,
|
|||||||
|
|
||||||
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 @@
|
|||||||
#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 {
|
|||||||
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 {
|
|||||||
*/
|
*/
|
||||||
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 {
|
|||||||
|
|
||||||
/* 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)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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 @@
|
|||||||
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 {
|
|||||||
|
|
||||||
/* 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 {
|
|||||||
#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,
|
||||||
|
Loading…
Reference in New Issue
Block a user