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

intel_th: Rework resource passing between glue layers and core

Currently, MMIO resource numbers in the TH driver core correspond to
PCI BAR numbers, because in the beginning there was only the PCI glue
layer. This created some confusion when the ACPI glue layer was added.

To avoid confusion and remove glue-specific code from the driver core,
split the resource indices between core and glue layers and change the
API so that the driver core receives the MMIO resources in the same
fixed order. At the same time, make the IRQ always be a parameter to
intel_th_alloc() instead of sometimes passing it as a resource.
Signed-off-by: default avatarAlexander Shishkin <alexander.shishkin@linux.intel.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 85d49eb1
...@@ -37,15 +37,23 @@ MODULE_DEVICE_TABLE(acpi, intel_th_acpi_ids); ...@@ -37,15 +37,23 @@ MODULE_DEVICE_TABLE(acpi, intel_th_acpi_ids);
static int intel_th_acpi_probe(struct platform_device *pdev) static int intel_th_acpi_probe(struct platform_device *pdev)
{ {
struct acpi_device *adev = ACPI_COMPANION(&pdev->dev); struct acpi_device *adev = ACPI_COMPANION(&pdev->dev);
struct resource resource[TH_MMIO_END];
const struct acpi_device_id *id; const struct acpi_device_id *id;
struct intel_th *th; struct intel_th *th;
int i, r, irq = -1;
id = acpi_match_device(intel_th_acpi_ids, &pdev->dev); id = acpi_match_device(intel_th_acpi_ids, &pdev->dev);
if (!id) if (!id)
return -ENODEV; return -ENODEV;
th = intel_th_alloc(&pdev->dev, (void *)id->driver_data, for (i = 0, r = 0; i < pdev->num_resources && r < TH_MMIO_END; i++)
pdev->resource, pdev->num_resources, -1); if (pdev->resource[i].flags & IORESOURCE_IRQ)
irq = pdev->resource[i].start;
else if (pdev->resource[i].flags & IORESOURCE_MEM)
resource[r++] = pdev->resource[i];
th = intel_th_alloc(&pdev->dev, (void *)id->driver_data, resource, r,
irq);
if (IS_ERR(th)) if (IS_ERR(th))
return PTR_ERR(th); return PTR_ERR(th);
......
...@@ -491,7 +491,7 @@ static const struct intel_th_subdevice { ...@@ -491,7 +491,7 @@ static const struct intel_th_subdevice {
.flags = IORESOURCE_MEM, .flags = IORESOURCE_MEM,
}, },
{ {
.start = 1, /* use resource[1] */ .start = TH_MMIO_SW,
.end = 0, .end = 0,
.flags = IORESOURCE_MEM, .flags = IORESOURCE_MEM,
}, },
...@@ -584,7 +584,6 @@ intel_th_subdevice_alloc(struct intel_th *th, ...@@ -584,7 +584,6 @@ intel_th_subdevice_alloc(struct intel_th *th,
struct intel_th_device *thdev; struct intel_th_device *thdev;
struct resource res[3]; struct resource res[3];
unsigned int req = 0; unsigned int req = 0;
bool is64bit = false;
int r, err; int r, err;
thdev = intel_th_device_alloc(th, subdev->type, subdev->name, thdev = intel_th_device_alloc(th, subdev->type, subdev->name,
...@@ -594,18 +593,12 @@ intel_th_subdevice_alloc(struct intel_th *th, ...@@ -594,18 +593,12 @@ intel_th_subdevice_alloc(struct intel_th *th,
thdev->drvdata = th->drvdata; thdev->drvdata = th->drvdata;
for (r = 0; r < th->num_resources; r++)
if (th->resource[r].flags & IORESOURCE_MEM_64) {
is64bit = true;
break;
}
memcpy(res, subdev->res, memcpy(res, subdev->res,
sizeof(struct resource) * subdev->nres); sizeof(struct resource) * subdev->nres);
for (r = 0; r < subdev->nres; r++) { for (r = 0; r < subdev->nres; r++) {
struct resource *devres = th->resource; struct resource *devres = th->resource;
int bar = 0; /* cut subdevices' MMIO from resource[0] */ int bar = TH_MMIO_CONFIG;
/* /*
* Take .end == 0 to mean 'take the whole bar', * Take .end == 0 to mean 'take the whole bar',
...@@ -614,8 +607,6 @@ intel_th_subdevice_alloc(struct intel_th *th, ...@@ -614,8 +607,6 @@ intel_th_subdevice_alloc(struct intel_th *th,
*/ */
if (!res[r].end && res[r].flags == IORESOURCE_MEM) { if (!res[r].end && res[r].flags == IORESOURCE_MEM) {
bar = res[r].start; bar = res[r].start;
if (is64bit)
bar *= 2;
res[r].start = 0; res[r].start = 0;
res[r].end = resource_size(&devres[bar]) - 1; res[r].end = resource_size(&devres[bar]) - 1;
} }
...@@ -812,8 +803,7 @@ static const struct file_operations intel_th_output_fops = { ...@@ -812,8 +803,7 @@ static const struct file_operations intel_th_output_fops = {
/** /**
* 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
* @devres: parent's resources * @devres: resources indexed by th_mmio_idx
* @ndevres: number of resources
* @irq: irq number * @irq: irq number
*/ */
struct intel_th * struct intel_th *
...@@ -823,12 +813,8 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata, ...@@ -823,12 +813,8 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata,
struct intel_th *th; struct intel_th *th;
int err, r; int err, r;
if (irq == -1) if (ndevres < TH_MMIO_END)
for (r = 0; r < ndevres; r++) return ERR_PTR(-EINVAL);
if (devres[r].flags & IORESOURCE_IRQ) {
irq = devres[r].start;
break;
}
th = kzalloc(sizeof(*th), GFP_KERNEL); th = kzalloc(sizeof(*th), GFP_KERNEL);
if (!th) if (!th)
...@@ -849,7 +835,8 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata, ...@@ -849,7 +835,8 @@ intel_th_alloc(struct device *dev, struct intel_th_drvdata *drvdata,
th->dev = dev; th->dev = dev;
th->drvdata = drvdata; th->drvdata = drvdata;
th->resource = devres; for (r = 0; r < ndevres; r++)
th->resource[r] = devres[r];
th->num_resources = ndevres; th->num_resources = ndevres;
th->irq = irq; th->irq = irq;
......
...@@ -225,9 +225,9 @@ int intel_th_set_output(struct intel_th_device *thdev, ...@@ -225,9 +225,9 @@ int intel_th_set_output(struct intel_th_device *thdev,
unsigned int master); unsigned int master);
int intel_th_output_enable(struct intel_th *th, unsigned int otype); int intel_th_output_enable(struct intel_th *th, unsigned int otype);
enum { enum th_mmio_idx {
TH_MMIO_CONFIG = 0, TH_MMIO_CONFIG = 0,
TH_MMIO_SW = 2, TH_MMIO_SW = 1,
TH_MMIO_END, TH_MMIO_END,
}; };
...@@ -244,7 +244,7 @@ enum { ...@@ -244,7 +244,7 @@ enum {
* @hub: "switch" subdevice (GTH) * @hub: "switch" subdevice (GTH)
* @resource: resources of the entire controller * @resource: resources of the entire controller
* @num_thdevs: number of devices in the @thdev array * @num_thdevs: number of devices in the @thdev array
* @num_resources: number or resources in the @resource array * @num_resources: number of resources in the @resource array
* @irq: irq number * @irq: irq number
* @id: this Intel TH controller's device ID in the system * @id: this Intel TH controller's device ID in the system
* @major: device node major for output devices * @major: device node major for output devices
...@@ -256,7 +256,7 @@ struct intel_th { ...@@ -256,7 +256,7 @@ struct intel_th {
struct intel_th_device *hub; struct intel_th_device *hub;
struct intel_th_drvdata *drvdata; struct intel_th_drvdata *drvdata;
struct resource *resource; struct resource resource[TH_MMIO_END];
int (*activate)(struct intel_th *); int (*activate)(struct intel_th *);
void (*deactivate)(struct intel_th *); void (*deactivate)(struct intel_th *);
unsigned int num_thdevs; unsigned int num_thdevs;
......
...@@ -17,7 +17,12 @@ ...@@ -17,7 +17,12 @@
#define DRIVER_NAME "intel_th_pci" #define DRIVER_NAME "intel_th_pci"
#define BAR_MASK (BIT(TH_MMIO_CONFIG) | BIT(TH_MMIO_SW)) enum {
TH_PCI_CONFIG_BAR = 0,
TH_PCI_STH_SW_BAR = 2,
};
#define BAR_MASK (BIT(TH_PCI_CONFIG_BAR) | BIT(TH_PCI_STH_SW_BAR))
#define PCI_REG_NPKDSC 0x80 #define PCI_REG_NPKDSC 0x80
#define NPKDSC_TSACT BIT(5) #define NPKDSC_TSACT BIT(5)
...@@ -66,6 +71,10 @@ static int intel_th_pci_probe(struct pci_dev *pdev, ...@@ -66,6 +71,10 @@ static int intel_th_pci_probe(struct pci_dev *pdev,
const struct pci_device_id *id) const struct pci_device_id *id)
{ {
struct intel_th_drvdata *drvdata = (void *)id->driver_data; struct intel_th_drvdata *drvdata = (void *)id->driver_data;
struct resource resource[TH_MMIO_END] = {
[TH_MMIO_CONFIG] = pdev->resource[TH_PCI_CONFIG_BAR],
[TH_MMIO_SW] = pdev->resource[TH_PCI_STH_SW_BAR],
};
struct intel_th *th; struct intel_th *th;
int err; int err;
...@@ -77,8 +86,8 @@ static int intel_th_pci_probe(struct pci_dev *pdev, ...@@ -77,8 +86,8 @@ static int intel_th_pci_probe(struct pci_dev *pdev,
if (err) if (err)
return err; return err;
th = intel_th_alloc(&pdev->dev, drvdata, pdev->resource, th = intel_th_alloc(&pdev->dev, drvdata, resource, TH_MMIO_END,
DEVICE_COUNT_RESOURCE, pdev->irq); pdev->irq);
if (IS_ERR(th)) if (IS_ERR(th))
return PTR_ERR(th); return PTR_ERR(th);
......
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