How to join a thread in Linux kernel?

前端 未结 2 1336
长情又很酷
长情又很酷 2021-01-06 19:07

The main question is: How we can wait for a thread in Linux kernel to complete? I have seen a few post concerned about proper way of handling threads in Linux kernel but i\'

相关标签:
2条回答
  • 2021-01-06 19:39

    kthread_stop() is a kernel's way for wait thread to end.

    Aside from waiting, kthread_stop() also sets should_stop flag for waited thread and wake up it, if needed. It is usefull for threads which repeat some actions infinitely.

    As for single-shot tasks, it is usually simpler to use works for them, instead of kthreads.

    EDIT: Note: kthread_stop() can be called only when kthread(task_struct) structure is not freed.

    Either thread function should return only after it found kthread_should_stop() return true, or get_task_struct() should be called before start thread (and put_task_struct() should be called after kthread_stop()).

    0 讨论(0)
  • 2021-01-06 19:45

    AFAIK there is no equivalent of pthread_join() in kernel. Also, I feel like your pattern (of starting bunch of threads and waiting only for one of them) is not really common in kernel. That being said, there kernel does have few synchronization mechanism that may be used to accomplish your goal.

    Note that those mechanisms will not guarantee that the thread finished, they will only let main thread know that they finished doing the work they were supposed to do. It may still take some time to really stop this tread and free all resources.

    Semaphores

    You can create a locked semaphore, then call down in your main thread. This will put it to sleep. Then you will up this semaphore inside of your thread just before exiting. Something like:

    struct semaphore sem;
    
    int func(void *arg) {
        struct semaphore *sem = (struct semaphore*)arg; // you could use global instead
    
        // do something
    
        up(sem);
        return 0;
    }
    
    int init_module(void) {
        // some initialization
        init_MUTEX_LOCKED(&sem);
        kthread_run(&func, (void*) &sem, "Creating thread");
        down(&sem); // this will block until thread runs up()
    }
    

    This should work but is not the most optimal solution. I mention this as it's a known pattern that is also used in userspace. Semaphores in kernel are designed for cases where it's mostly available and this case has high contention. So a similar mechanism optimized for this case was created.

    Completions

    You can declare completions using:

    struct completion comp;
    init_completion(&comp);
    

    or:

    DECLARE_COMPLETION(comp);
    

    Then you can use wait_for_completion(&comp); instead of down() to wait in main thread and complete(&comp); instead of up() in your thread.

    Here's the full example:

    DECLARE_COMPLETION(comp);
    struct my_data {
        int id;
        struct completion *comp;
    };
    
    int func(void *arg) {
        struct my_data *data = (struct my_data*)arg;
        // doing something
    
        if (data->id == 3)
            complete(data->comp);
    
        return 0;
    }
    
    int init_module(void) {
        struct my_data *data[] = kmalloc(sizeof(struct my_data)*N, GFP_KERNEL);
    
        // some initialization
        for (int i=0; i<N; i++) {
            data[i]->comp = &comp;
            data[i]->id = i;
            kthread_run(func, (void*) data[i], "my_thread%d", i);
        }
        wait_for_completion(&comp); // this will block until some thread runs complete()
    }
    

    Multiple threads

    I don't really see why you would start 5 identical threads and only want to wait for 3rd one but of course you could send different data to each thread, with a field describing it's id, and then call up or complete only if this id equals 3. That's shown in the completion example. There are other ways to do this, this is just one of them.

    Word of caution

    Go read some more about those mechanisms before using any of them. There are some important details I did not write about here. Also those examples are simplified and not tested, they are here just to show the overall idea.

    0 讨论(0)
提交回复
热议问题