linux内核函数kernel

    技术2022-05-19  24

    linux内核函数kernel_thread

    本信息来源于www.poluoluo.com,原文地址:http://www.poluoluo.com/server/201004/82854.html

     

    设备驱动程序中,如果需要几个并发执行的人物,可以启动内核线程,启动内和县城的函数为:

      int kernel_thread (int ( * fn )( void * ), void * arg, unsigned long flags);

      kernel_thread函数的作用是产生一个新的线程

      内核线程实际上就是一个共享父进程地址空间的进程,它有自己的系统堆栈.

      内核线程和进程都是通过do_fork()函数来产生的,系统中规定的最大进程数与

      线程数由fork_init来决定:

      [/arch/kernel/process.c/fork_init()]

      void __init fork_init(unsigned long mempages)

      {

      #ifndef __HAVE_ARCH_TASK_STRUCT_ALLOCATOR

      #ifndef ARCH_MIN_TASKALIGN

      #define ARCH_MIN_TASKALIGN   L1_CACHE_BYTES

      #endif

      /* 在slab高速缓存中建立task_struct结构专用的缓冲区队列 */

      task_struct_cachep =

      kmem_cache_create("task_struct", sizeof(struct task_struct),

      ARCH_MIN_TASKALIGN, SLAB_PANIC, NULL, NULL);

      #endif

      /*

      把默认线程数设置到一个安全值,因为内核中总的线程占用的空间

      可能要内存一半还要多.

      参数mempages系统中总的物理内存结构大小,它等于mempages/PAGESIZE.

      比如我机器的内存是512m,那么在我的系统最多能同时产生线程数为

      (512*2^20/2^12) / 2^3 = 512*2^5 = 16384

      */

      max_threads = mempages / (8 * THREAD_SIZE / PAGE_SIZE);

      /*

      * 启动系统的时候至少需要20个线程

      */

      if(max_threads < 20)

      max_threads = 20;

      /*

      * 每个进程最多产生max_threads/2,也就是线程总数的一半,在我的机器上为8192.

      */

      init_task.signal->rlim[RLIMIT_NPROC].rlim_cur = max_threads/2;

      init_task.signal->rlim[RLIMIT_NPROC].rlim_max = max_threads/2;

      }

      kernel_thread原形在/arch/kernel/process.c中.

      (*fn)(void *)为要执行的函数的指针,arg为函数参数,flags为do_fork产生线程时的标志.

      int kernel_thread(int (*fn)(void *), void * arg, unsigned long flags)

      {

      struct pt_regs regs;

      memset(®s, 0, sizeof(regs));

      regs.ebx = (unsigned long) fn;   /* ebx指向函数地址 */

      regs.edx = (unsigned long) arg;   /* edx指向参数 */

      regs.xds = __USER_DS;

      regs.xes = __USER_DS;

      regs.orig_eax = -1;

      regs.eip = (unsigned long) kernel_thread_helper;

      regs.xcs = __KERNEL_CS;

      regs.eflags = X86_EFLAGS_IF | X86_EFLAGS_SF | X86_EFLAGS_PF | 0x2;

      /* 利用do_fork来产生一个新的线程,共享父进程地址空间,并且不允许调试子进程 */

      return do_fork(flags | CLONE_VM | CLONE_UNTRACED, 0, ®s, 0, NULL, NULL);

      }

      [/arch/i386/kernel/process.c/kernel_thread_helper]

      extern void kernel_thread_helper(void); /* 定义成全局变量 */

      __asm__(".section .text/n"

      ".align 4/n"

      "kernel_thread_helper:/n/t"

      "movl

    转载请注明原文地址: https://ibbs.8miu.com/read-2216900.html

    最新回复(0)