mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02: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
					
				
					 4 changed files with 106 additions and 2 deletions
				
			
		| 
						 | 
					@ -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 a new issue