Skip to content

Commit

Permalink
PNP: Avoid leaving unregistered device objects in lists
Browse files Browse the repository at this point in the history
pnp_register_protocol() and __pnp_add_device() both have a problem
that if device_register() fails, the objects they create will be left
in the lists they have been put one beforehand.  Unfortunately, that
is not handled by the callers of those routines either, so in case
of a device registration errors the PNP bus type's data structures
will end up in an inconsistent state.

Make pnp_register_protocol() and __pnp_add_device() remove the
objects from the lists if device registration fails.

Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
  • Loading branch information
Rafael J. Wysocki committed Mar 18, 2015
1 parent 38f6b38 commit 71150d2
Showing 1 changed file with 40 additions and 13 deletions.
53 changes: 40 additions & 13 deletions drivers/pnp/core.c
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,13 @@ void *pnp_alloc(long size)
return result;
}

static void pnp_remove_protocol(struct pnp_protocol *protocol)
{
mutex_lock(&pnp_lock);
list_del(&protocol->protocol_list);
mutex_unlock(&pnp_lock);
}

/**
* pnp_protocol_register - adds a pnp protocol to the pnp layer
* @protocol: pointer to the corresponding pnp_protocol structure
Expand All @@ -50,12 +57,13 @@ void *pnp_alloc(long size)
*/
int pnp_register_protocol(struct pnp_protocol *protocol)
{
int nodenum;
struct list_head *pos;
int nodenum, ret;

INIT_LIST_HEAD(&protocol->devices);
INIT_LIST_HEAD(&protocol->cards);
nodenum = 0;

mutex_lock(&pnp_lock);

/* assign the lowest unused number */
Expand All @@ -67,12 +75,18 @@ int pnp_register_protocol(struct pnp_protocol *protocol)
}
}

protocol->number = nodenum;
dev_set_name(&protocol->dev, "pnp%d", nodenum);

list_add_tail(&protocol->protocol_list, &pnp_protocols);

mutex_unlock(&pnp_lock);

protocol->number = nodenum;
dev_set_name(&protocol->dev, "pnp%d", nodenum);
return device_register(&protocol->dev);
ret = device_register(&protocol->dev);
if (ret)
pnp_remove_protocol(protocol);

return ret;
}

/**
Expand All @@ -81,9 +95,7 @@ int pnp_register_protocol(struct pnp_protocol *protocol)
*/
void pnp_unregister_protocol(struct pnp_protocol *protocol)
{
mutex_lock(&pnp_lock);
list_del(&protocol->protocol_list);
mutex_unlock(&pnp_lock);
pnp_remove_protocol(protocol);
device_unregister(&protocol->dev);
}

Expand Down Expand Up @@ -158,18 +170,36 @@ struct pnp_dev *pnp_alloc_dev(struct pnp_protocol *protocol, int id,
return dev;
}

static void pnp_delist_device(struct pnp_dev *dev)
{
mutex_lock(&pnp_lock);
list_del(&dev->global_list);
list_del(&dev->protocol_list);
mutex_unlock(&pnp_lock);
}

int __pnp_add_device(struct pnp_dev *dev)
{
int ret;

pnp_fixup_device(dev);
dev->status = PNP_READY;

mutex_lock(&pnp_lock);

list_add_tail(&dev->global_list, &pnp_global);
list_add_tail(&dev->protocol_list, &dev->protocol->devices);

mutex_unlock(&pnp_lock);
if (dev->protocol->can_wakeup)

ret = device_register(&dev->dev);
if (ret)
pnp_delist_device(dev);
else if (dev->protocol->can_wakeup)
device_set_wakeup_capable(&dev->dev,
dev->protocol->can_wakeup(dev));
return device_register(&dev->dev);

return ret;
}

/*
Expand Down Expand Up @@ -204,10 +234,7 @@ int pnp_add_device(struct pnp_dev *dev)

void __pnp_remove_device(struct pnp_dev *dev)
{
mutex_lock(&pnp_lock);
list_del(&dev->global_list);
list_del(&dev->protocol_list);
mutex_unlock(&pnp_lock);
pnp_delist_device(dev);
device_unregister(&dev->dev);
}

Expand Down

0 comments on commit 71150d2

Please sign in to comment.