Hannes Reinecke <hare@xxxxxxx> wrote:Indeed. That will work nicely.
>> Any comments/suggestions welcome; otherwise please apply.
> > > I suggest you just use a semaphore, initialised to a suitable value:
> > > static struct semaphore foo = __SEMAPHORE_INITIALIZER(foo, 50);
> > > {
> ...
> down(&foo);
> ...
> up(&foo);
> ...
> }
> Hmm; looks good, but: It's not possible to reliably change the maximum number of processes on the fly.
The trivial way of course it when the waitqueue is empty and no processes are holding the semaphore. But it's quite non-obvious how this should work if processes are already holding the semaphore.
We would need to wait for those processes to finish, setting the length of the queue to 0 (to disallow any other process from grabbing the semaphore), and atomically set the queue length to the new value.
Apart from the fact that we would need a worker thread for that (otherwise the calling process might block indefinitely), there is no guarantee that the queue ever will become empty, as hotplug processes might be generated at any time.
Or is there an easier way?
Well if you want to increase the maximum number by ten you do:
for (i = 0; i < 10; i++)
up(&foo);
and similarly for decreasing the limit. That will involve doing down()s,Hmm. Really? I mean, is there really an effect when the semaphore waitqueue is active?
which will automatically wait for the current number of threads to fall to
the desired level.
But I don't really see a need to tune this on the fly - probably the worseOch aye, there is. If the calling parameters to call_usermodehelper are stored temporarily, you can allow for the call to return immediately when not enough resources are available. This way, it is even possible to stop khelper processes altogether (and effectively queueing all events) and re-enable the khelper processes at a later stage.
problem occurs during bootup when the operator cannot perform tuning.
So a __setup parameter seems to be the best way of providing tunability. Initialise the semaphore in usermodehelper_init().Which is what I've done.