"текущий" в коде ядра Linux

Когда я проходил следующий код Linux char, я нашел указатель на структуру current в printk.

Я хочу знать, на какую структуру указывает current, и ее полные элементы.

Какую цель выполняет эта структура?

ssize_t sleepy_read (struct file *filp, char __user *buf, size_t count, loff_t *pos)
{
    printk(KERN_DEBUG "process %i (%s) going to sleep\n",
    current->pid, current->comm);
    wait_event_interruptible(wq, flag != 0);
    flag = 0;
    printk(KERN_DEBUG "awoken %i (%s)\n", current->pid, current->comm);
    return 0;
}

Ответ 1

Это указатель на текущий процесс, то есть процесс, который выдал системный вызов.

Из docs:

Текущий процесс

Хотя модули ядра не выполняются последовательно, как это делают приложения, большинство действий, выполняемых ядром, связаны с конкретным обработать. Код ядра может знать текущий процесс, доступ к глобальному элементу current, указателю на struct task_struct, который с версии 2.4 ядра объявлен в <asm/current.h>, включенный <linux/sched.h>. Текущий указатель относится к выполняемому в настоящее время пользовательскому процессу. Во время исполнения системного вызова, такого как открытый или прочитанный, текущий процесс является одним из который вызвал вызов. Код ядра может использовать специфичные для процесса информацию с использованием текущего, если это необходимо. Пример этого техника представлена ​​в разделе "Контроль доступа к файлу устройства", в Глава 5, "Расширенные операции Char" Операции с драйверами ".

На самом деле, ток уже не является глобальной переменной, как это был в первых ядрах Linux. Разработчики оптимизировали доступ к структура, описывающая текущий процесс, спрятав его в стеке стр. Вы можете посмотреть подробности текущего в <asm/current.h>. В то время как код, на который вы будете смотреть, может казаться волосатым, мы должны помнить, что Linux - это совместимая с SMP система, а глобальная переменная просто не будет работать, когда вы имеете дело с несколькими процессорами. Детали Однако реализация остается скрытой для других подсистем ядра, и драйвер устройства может просто включать и ссылаться на текущий процесс.

С точки зрения модуля ток аналогичен внешнему справка принтk. Модуль может ссылаться на текущий, где он сочтет нужным. Например, следующий оператор печатает идентификатор процесса и имя команды текущего процесса путем доступа к определенным полям в struct task_struct:

 printk("The process is \"%s\" (pid %i)\n",
  current->comm, current->pid);

Имя команды, хранящееся в current- > comm, является базовым именем файл программы, который выполняется текущим процессом.

Ответ 2

Вот полная структура, обозначающая "ток" на

task_struct
Each task_struct data structure describes a process or task in the system.

struct task_struct {
/* these are hardcoded - don't touch */
  volatile long        state;          /* -1 unrunnable, 0 runnable, >0 stopped */
  long                 counter;
  long                 priority;
  unsigned             long signal;
  unsigned             long blocked;   /* bitmap of masked signals */
  unsigned             long flags;     /* per process flags, defined below */
  int errno;
  long                 debugreg[8];    /* Hardware debugging registers */
  struct exec_domain   *exec_domain;
/* various fields */
  struct linux_binfmt  *binfmt;
  struct task_struct   *next_task, *prev_task;
  struct task_struct   *next_run,  *prev_run;
  unsigned long        saved_kernel_stack;
  unsigned long        kernel_stack_page;
  int                  exit_code, exit_signal;
  /* ??? */
  unsigned long        personality;
  int                  dumpable:1;
  int                  did_exec:1;
  int                  pid;
  int                  pgrp;
  int                  tty_old_pgrp;
  int                  session;
  /* boolean value for session group leader */
  int                  leader;
  int                  groups[NGROUPS];
  /* 
   * pointers to (original) parent process, youngest child, younger sibling,
   * older sibling, respectively.  (p->father can be replaced with 
   * p->p_pptr->pid)
   */
  struct task_struct   *p_opptr, *p_pptr, *p_cptr, 
                       *p_ysptr, *p_osptr;
  struct wait_queue    *wait_chldexit;  
  unsigned short       uid,euid,suid,fsuid;
  unsigned short       gid,egid,sgid,fsgid;
  unsigned long        timeout, policy, rt_priority;
  unsigned long        it_real_value, it_prof_value, it_virt_value;
  unsigned long        it_real_incr, it_prof_incr, it_virt_incr;
  struct timer_list    real_timer;
  long                 utime, stime, cutime, cstime, start_time;
/* mm fault and swap info: this can arguably be seen as either
   mm-specific or thread-specific */
  unsigned long        min_flt, maj_flt, nswap, cmin_flt, cmaj_flt, cnswap;
  int swappable:1;
  unsigned long        swap_address;
  unsigned long        old_maj_flt;    /* old value of maj_flt */
  unsigned long        dec_flt;        /* page fault count of the last time */
  unsigned long        swap_cnt;       /* number of pages to swap on next pass */
/* limits */
  struct rlimit        rlim[RLIM_NLIMITS];
  unsigned short       used_math;
  char                 comm[16];
/* file system info */
  int                  link_count;
  struct tty_struct    *tty;           /* NULL if no tty */
/* ipc stuff */
  struct sem_undo      *semundo;
  struct sem_queue     *semsleeping;
/* ldt for this task - used by Wine.  If NULL, default_ldt is used */
  struct desc_struct *ldt;
/* tss for this task */
  struct thread_struct tss;
/* filesystem information */
  struct fs_struct     *fs;
/* open file information */
  struct files_struct  *files;
/* memory management info */
  struct mm_struct     *mm;
/* signal handlers */
  struct signal_struct *sig;
#ifdef __SMP__
  int                  processor;
  int                  last_processor;
  int                  lock_depth;     /* Lock depth. 
                                          We can context switch in and out
                                          of holding a syscall kernel lock... */  
#endif   
};