这些文章已经写了好几年了,可能已经过时了。在MSN space和QQzone几经辗转之后,我想也许这些技术文章还是放在搞技术的博客中更能帮助人。于是做了一个艰难的决定,把这些文章一篇篇搬过来!绝对是原创的。
实际上MD中线程的调度都是植根于linux中的调度,我这里为什么要单独拿出来讨论讨论?因为我觉得MD的线程写法挺有意思,让我受益匪浅。
在MD中使用的守护线程都是系统线程,它的优先级是非常高的。先来看MD线程的数据结构mdk_thread_t。内容非常简单,最关键的就是run函数指针,才是线程中真正的工作函数;线程通过mddev指针来访问MD的数据;线程中会有个等待队列,以便于线程的睡眠与唤醒;flags字段在MD线程中有点控制标志的味道;毫无疑问tsk就是指向该线程在linux中的结构;timeout字段用于定时。
MD线程是如何创建的呢?代码在md_register_thread中可以找到。这个函数分配并初始化了了一个mdk_thread_t结构。然后调用kthread_run函数创建了一个内核线程,这个内核线程执行的函数是md_thread,这个函数我们接下来会重点讨论。与此相对,MD线程的结束是通过md_unregister_thread函数,停止线程并释放分配的mdk_thread_t结构。
虽然md_register_thread启动的线程并没有直接用mdk_thread_t的run函数,但是这个run函数才真正决定了这个线程的工作内容。那就跳转到md_thread里面去看看。这个函数太简单了,核心就是一个while循环。而且循环体中的内容也非常简单,无非就是如果这个MD线程检测到THREAD_WAKEUP标志,就起来执行run函数,执行结束后就会被加入到等待队列中等待下一次的唤醒。所以说md_thread函数实际上完成一种线程调度的封装。
需要唤醒MD线程的时候就要调用md_wakeup_thread函数,它的工作就是打上THREAD_WAKEUP标志,然后调用内核wake_up函数唤醒等待对列中的线程。
好了,似乎短短几段就把MD线程的相关内容说完了,如果结合我们在RAID-5中介绍的raid5d,就更清楚他们是怎么工作的了。虽然MD线程的内容并不深,但我觉得这种方式非常漂亮,值得我学习。