contiki-进程

进程的结构

Contiki的进程由两部分组成:进程控制块和进程线程。进程控制块存储在内存中,它包含进程运行时的信息,比如:进程名进程状态指向进程线程的指针

进程线程是存储在ROM中的一个代码块。

进程控制块PCB(process control block)

 

 1 struct process {
 2   struct process *next;
 3 #if PROCESS_CONF_NO_PROCESS_NAMES
 4 #define PROCESS_NAME_STRING(process) ""
 5 #else
 6   const char *name;
 7 #define PROCESS_NAME_STRING(process) (process)->name
 8 #endif
 9   PT_THREAD((* thread)(struct pt *, process_event_t, process_data_t));
10   struct pt pt;
11   unsigned char state, needspoll;
12 };
struct pt {
lc_t lc;
};
/** \hideinitializer */
typedef unsigned short lc_t;

 

  进程控制块包含每个进程的信息,比如进程状态、指向进程的线程的指针、进程的文本名称。进程控制块只在内核内部使用,不能被进程直接访问。

用户代码不能直接访问进程控制块的任何成员。

  进程控制块是轻量级的,只需要几个字节的内存。进程控制块的结构体如上面所示。该结构体中任何成员都不能被直接访问,只有进程管理函数能够访问这些成员。

  进程控制块的第一个成员 struct process *next,指向进程链表中的下一个进程控制块。

  成员const char *name,指向进程的文本类型的名字。

  成员PT_THREAD((* thread)(struct pt *, process_event_t, process_data_t)),一个函数指针,指向了进程的线程。

  成员unsigned char state, needspoll,是内部标志,当进程被轮询时,通过函数process_poll()修改该标志。

进程的创建过程

  进程控制块不是直接定义和声明的,而是通过宏PROCESS()。

PROCESS(hello_world_process, "Hello world");

如上例所示,该宏有两个参数,用于访问该进程的进程控制块变量名hello_world_process、用于调试和打印进程的进程文本名字"Hello world"。

进程线程

进程线程是一个单一的protothread,由进程调度器调度。例子如下:

1 PROCESS_THREAD(hello_world_process, ev, data)  
2 {  
3   PROCESS_BEGIN();  
4   
5   printf("Hello, world\n");  
6   
7   PROCESS_END();  
8 }  

 Protothreads

  当在等待某个事件发生时,protothread允许系统运行其它活动。protothread的概念是在开发Contiki的过程中提出来的,但是这个概念不是与Contiki绑定在一起的。protothread也可以很好地运行在许多其它的系统中。

  Contiki运行在内存受限的系统之上,减小内存负载显得尤为重要。protothread提供了一种很好的方法,可以让C函数在没有传统线程内存负载的情况下,以类似线程的方式运行。

  protothread可以看作是一个常规的C函数。该函数使用两个特殊的宏作为开始和结束:PROCESS_BEGIN()和PROCESS_END()。

  C预处理器实现了protothread的主要操作:

 1 struct pt {
 2   lc_t lc;
 3 };
 4 
 5 #define PT_WAITING 0
 6 #define PT_YIELDED 1
 7 #define PT_EXITED  2
 8 #define PT_ENDED   3
 9 
10 #define PT_INIT(pt)   LC_INIT((pt)->lc)
11 
12 #define PT_BEGIN(pt) { char PT_YIELD_FLAG = 1;\
13                  if (PT_YIELD_FLAG) {;} LC_RESUME((pt)->lc)
14 
15 #define PT_END(pt) LC_END((pt)->lc); PT_YIELD_FLAG = 0; \
16                    PT_INIT(pt); return PT_ENDED; }   
17              
18 #define PT_WAIT_UNTIL(pt, condition)            \
19   do {                        \
20     LC_SET((pt)->lc);                \
21     if(!(condition)) {                \
22       return PT_WAITING;            \
23     }                        \
24   } while(0)
25 
26 #define PT_EXIT(pt)                \
27   do {                        \
28     PT_INIT(pt);                \
29     return PT_EXITED;            \
30   } while(0)

进程中的Protothreads

  Contiki进程自身实现了一套protothread,它允许进程等待即将到来的事件。因此,Contiki进程中使用的protothread语句与上面介绍的纯protothread语句有微小的差异。

  Contiki进程中使用的进程相关的protothread宏:

1 PROCESS_BEGIN(); // Declares the beginning of a process' protothread.   
2 PROCESS_END(); // Declares the end of a process' protothread.   
3 PROCESS_EXIT(); // Exit the process. 
4 PROCESS_WAIT_EVENT(); // Wait for any event.   
5 PROCESS_WAIT_EVENT_UNTIL(); // Wait for an event, but with a condition.  
6 PROCESS_YIELD(); // Wait for any event, equivalent to PROCESS_WAIT_EVENT().  
7 PROCESS_WAIT_UNTIL(); // Wait for a given condition; may not yield the process.  
8 PROCESS_PAUSE(); // Temporarily yield the process. 

事件

  Contiki中,进程接收到一个事件后就会运行。Contiki中有两种事件:异步事件和同步事件。

  当一个异步事件被发出时,该事件被放到内核中的事件队列中,并在一段时间后被传递到接收进程中。

  

 

  当一个同步事件被发出时,该事件被立即传递到接收进程中。

  

异步事件

  异步事件在被发出一段时间后才能被传递到接收进程。在事件被发出后和被传递前的这段时间,它被保存在Contiki内核的事件队列中。

内核负责将事件队列中的事件传递到接收进程。内核循环遍历事件队列,通过调用进程将队列中的事件传递到进程中。

  异步事件的接收者可以是一个特殊进程(何为特殊进程?),也可以是所有正在运行的通用进程。当接收者是一个特殊进程,内核就调用该进程并传递事件到该进程中。当事件接收者是系统中的通用进程,内核将一个接一个地顺序传递事件到所有的进程中。

  异步事件通过函数process_post()发出。

  

 1 int
 2 process_post(struct process *p, process_event_t ev, process_data_t data)
 3 {
 4   static process_num_events_t snum;
 5 
 6   if(PROCESS_CURRENT() == NULL) {
 7     PRINTF("process_post: NULL process posts event %d to process '%s', nevents %d\n",
 8        ev,PROCESS_NAME_STRING(p), nevents);
 9   } else {
10     PRINTF("process_post: Process '%s' posts event %d to process '%s', nevents %d\n",
11        PROCESS_NAME_STRING(PROCESS_CURRENT()), ev,
12        p == PROCESS_BROADCAST? "<broadcast>": PROCESS_NAME_STRING(p), nevents);
13   }
14   
15   if(nevents == PROCESS_CONF_NUMEVENTS) {
16 #if DEBUG
17     if(p == PROCESS_BROADCAST) {
18       printf("soft panic: event queue is full when broadcast event %d was posted from %s\n", ev, PROCESS_NAME_STRING(process_current));
19     } else {
20       printf("soft panic: event queue is full when event %d was posted to %s frpm %s\n", ev, PROCESS_NAME_STRING(p), PROCESS_NAME_STRING(process_current));
21     }
22 #endif /* DEBUG */
23     return PROCESS_ERR_FULL;
24   }
25   
26   snum = (process_num_events_t)(fevent + nevents) % PROCESS_CONF_NUMEVENTS;
27   events[snum].ev = ev;
28   events[snum].data = data;
29   events[snum].p = p;
30   ++nevents;
31 
32 #if PROCESS_CONF_STATS
33   if(nevents > process_maxevents) {
34     process_maxevents = nevents;
35   }
36 #endif /* PROCESS_CONF_STATS */
37   
38   return PROCESS_ERR_OK;
39 }

  process_post()的内部实现很简单(还是理清头绪才简单)。先检查当前事件队列的大小,检查是否还有可以存放事件的空间,然后再做决定,如果没有足够的空间,该函数返回一个错误,如果有足够的空间,该函数将事件插入到事件队列的末尾,然后返回。

同步事件

  与异步事件不同的是,同步事件被发出后不经过事件队列,会被直接传递。同步事件只能被发出给一个特定进程。由于同步事件直接被传递,因此传递一个同步事件在功能上等同于一个函数调用:接收进程被直接调用,发送进程在接收进程完成处理事件前一直处于阻塞状态。不过,接收进程不会被告诉所发出的事件是同步事件还是异步事件。

  同步事件是通过函数process_post_synch()发出。

1 void
2 process_post_synch(struct process *p, process_event_t ev, process_data_t data)
3 {
4   struct process *caller = process_current;
5 
6   call_process(p, ev, data);
7   process_current = caller;
8 }

轮询(不太理解)

  轮询请求是一个特殊的事件。进程可以通过调用函数process_poll()请求被轮询。进程请求轮询后,会尽可能快地被调用。当轮询到该进程时,会传递一个特殊的事件到进程中。

事件标识(zhi)符

  事件被事件标识符所标识。事件标识符是一个8比特,一个字节的数组,会被传递到接收进程中。接收进程可以根据接收到的不同的事件标识符来做相应不同的处理。

  事件标识符的范围是0~255,在127一下的事件标识符可以在一个用户进程中自由使用,在128以上的事件标识符只能在不同的进程间使用。在128以上的事件标识符被内核所管理。

  从128开始的数字被内核静态分配,用于实现不同的目的。

  Contiki内核保留的事件标识符:

 1 #define PROCESS_EVENT_NONE            0x80  //该事件标识符 没有被使用
 2 #define PROCESS_EVENT_INIT            0x81 //该事件被发送到一个正在初始化的新进程中
 3 #define PROCESS_EVENT_POLL            0x82     //该事件被发送到一个轮询进程中
 4 #define PROCESS_EVENT_EXIT            0x83 //该事件被发送到一个正在被内核杀死的进程中。
                           //进程接收到该事件后,因为可能不会被再次调用,因此它可以选择清空自己分派到的资源
5 #define PROCESS_EVENT_SERVICE_REMOVED 0x84 6 #define PROCESS_EVENT_CONTINUE 0x85 //该事件被内核发送到一个执行了PROCESS_YIELD()而正在等待的进程 7 #define PROCESS_EVENT_MSG 0x86 //该事件被发送到一个已经接收到通信消息的进程。
                           //它一般被用于IP栈去通知进程有消息到来了,也可以用与两个进程间表示一个通用消息到来了。
8 #define PROCESS_EVENT_EXITED 0x87 /*当一个进程将要退出时,该事件被发送到所有进程。
                           发送事件的同时,还会发送一个指向正在退出的进程的进程控制块的指针。
                           当接收到该事件时,接收进程将清除将要退出进程所分配的状态*/
9 #define PROCESS_EVENT_TIMER 0x88 //该事件被发送给一个事件定时器etimer到期的进程 10 #define PROCESS_EVENT_COM 0x89 11 #define PROCESS_EVENT_MAX 0x8a

  除静态分配事件号之外,进程可以分配用于进程间的大于128的事件标识符。被分配的事件标识符被存储在一个变量中,接收进程可以使用该变量来匹配事件标识符。

摘录自:http://blog.csdn.net/tidyjiang/article/details/51378589

  

 

posted @ 2016-11-23 12:02  一只奋斗的考拉  阅读(1090)  评论(0编辑  收藏  举报