irq: irqdomain: Change irq_domain_add to detect errors

It should not be valid to add an irq_domain with a logical irq
range that overlaps with another already registered to the
system. Return an error on such an occurrence.

This change also inherently sorts the irq_domains by logical
irq_base as they are added.

Change-Id: Idef697dbe4584d783703d053fbf09f1b17e62ab0
Signed-off-by: Michael Bohan <mbohan@codeaurora.org>
This commit is contained in:
Michael Bohan
2012-01-11 11:45:16 -08:00
parent b8635c3f53
commit 6f4df41a29
2 changed files with 32 additions and 5 deletions

View File

@@ -89,7 +89,7 @@ static inline unsigned int irq_domain_to_irq(struct irq_domain *d,
hw < d->hwirq_base + d->nr_irq; \
hw++, irq = irq_domain_to_irq(d, hw))
extern void irq_domain_add(struct irq_domain *domain);
extern int irq_domain_add(struct irq_domain *domain);
extern void irq_domain_del(struct irq_domain *domain);
extern void irq_domain_register(struct irq_domain *domain);
extern void irq_domain_register_irq(struct irq_domain *domain, int hwirq);

View File

@@ -15,13 +15,34 @@ static DEFINE_MUTEX(irq_domain_mutex);
*
* Adds a irq_domain structure. The irq_domain must at a minimum be
* initialized with an ops structure pointer, and either a ->to_irq hook or
* a valid irq_base value. Everything else is optional.
* a valid irq_base value. The irq range must be mutually exclusive with
* domains already registered. Everything else is optional.
*/
void irq_domain_add(struct irq_domain *domain)
int irq_domain_add(struct irq_domain *domain)
{
struct irq_domain *curr;
uint32_t d_highirq = domain->irq_base + domain->nr_irq - 1;
if (!domain->nr_irq)
return -EINVAL;
mutex_lock(&irq_domain_mutex);
list_add(&domain->list, &irq_domain_list);
/* insert in ascending order of domain->irq_base */
list_for_each_entry(curr, &irq_domain_list, list) {
uint32_t c_highirq = curr->irq_base + curr->nr_irq - 1;
if (domain->irq_base < curr->irq_base &&
d_highirq < curr->irq_base) {
break;
}
if (d_highirq <= c_highirq) {
mutex_unlock(&irq_domain_mutex);
return -EINVAL;
}
}
list_add_tail(&domain->list, &curr->list);
mutex_unlock(&irq_domain_mutex);
return 0;
}
/**
@@ -146,6 +167,7 @@ unsigned int irq_create_of_mapping(struct device_node *controller,
list_for_each_entry(domain, &irq_domain_list, list) {
if (!domain->ops->dt_translate)
continue;
rc = domain->ops->dt_translate(domain, controller,
intspec, intsize, &hwirq, &type);
if (rc == 0)
@@ -209,6 +231,7 @@ EXPORT_SYMBOL_GPL(irq_domain_simple_ops);
void irq_domain_add_simple(struct device_node *controller, int irq_base)
{
struct irq_domain *domain;
int rc;
domain = kzalloc(sizeof(*domain), GFP_KERNEL);
if (!domain) {
@@ -219,7 +242,11 @@ void irq_domain_add_simple(struct device_node *controller, int irq_base)
domain->irq_base = irq_base;
domain->of_node = of_node_get(controller);
domain->ops = &irq_domain_simple_ops;
irq_domain_add(domain);
rc = irq_domain_add(domain);
if (rc) {
WARN(1, "Unable to create irq domain\n");
return;
}
irq_domain_register(domain);
}
EXPORT_SYMBOL_GPL(irq_domain_add_simple);