That code is not quite correct, you need something like this.
spin_lock(&driver_lock);
drv = scan_driver_list();
if (drv && drv->owner)
if (!try_inc_mod_count(drv->owner))
drv = NULL;
spin_unlock(&driver_lock); /* either failed or protected by use count */
if (drv && drv->open)
drv->open();
>...but, what if someone has unloaded the module before you get to the if
>statement? The memory for the module has been freed, including drv itself.
It is assumed that the module unload routine will call
driver_unregister() which will also take the driver_lock. The
interaction between driver_lock in your code and the unregister routine
via module unload, together with the interaction between unload_lock in
sys_delete_module and try_inc_mod_count will prevent races, provided
you code it right. And provided that your code that does
try_inc_mod_count is in built in code, not in the module itself.
An alternative to driver_lock is BKL, provided you do not have high
activity on your open routine.
open:
lock_kernel();
drv = scan_driver_list();
if (drv && drv->owner)
if (!try_inc_mod_count(drv->owner))
drv = NULL;
unlock_kernel();
if (drv && drv->open)
drv->open();
That works because sys_delete_module(), including all the module clean
up code, runs under BKL. The module_cleanup routine will unregister
from driver_list under BKL so it cannot race with the open code.
use:
drv->func(); /* protected by non-zero mod use count */
You only need to lock drv and do try_inc_mod_count() if the use count
might be 0 at the start of the call. IOW, you only need that code on a
drv->open() or equivalent function. Once the use count is non-zero,
the module is locked down; it is assumed that drv belongs to the module
and is also protected.
close:
if (drv->owner) /* protected by non-zero mod use count */
__MOD_INC_USE_COUNT(drv->owner);
drv->close();
if (drv->owner)
__MOD_DEC_USE_COUNT(drv->owner);
There is a very small race on close where the module does
MOD_DEC_USE_COUNT() which can take the count to 0 but the close routine
has not returned from the module yet. Bumping the use count around the
close call closes that race.
Preventing module unload races with the current infrastructure is
complex and fragile. But I've said that before :).
ps. I am going to be off the net for a few days.
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/