原创 Contiki的NETWORK层间数据传输流程

2014-7-6 22:40 1168 16 16 分类: 消费电子
Contiki的NETWORK层之间的数据流路径:
Send:
Network->Mac->Rdc(->Frame802154)->Radio
Recv:
Radio->Rdc(->Frame802154)->Mac->Network
注:在此结构中,802154应该是属于RDC层的,
如果不将RDC算作一层,应该是介于Mac和Radio之间。是Mac层的出口。这也是Contiki的一项重要模块之一。在Contiki中ContikiMAC.c模块实现了RDC功能。论文《The ContikiMAC Radio Duty Cycling Protocol》详细描述了RDC的实现与理论基础。
 
在Contiki中接收采用事件驱动方式。接收通过一个PThread任务,等待数据到来事件,如果数据到来事件触发,则将数据则取出来。这个事件触发是在Radio的中断中设置的。
即:NETSTACK_RDC.input()的调用一般都是在Radio的一个PThread中,由于在Network.open的时候已经将每一层的回调函数设置完备,所以当接收到数据的时候,通过每一层的回调,将数据传递到Network_Network处,即协议栈的最顶部应用层处理。
 
以contiki\cpu\avr\radio\rf230bb\rf230bb.c文件为例
rf230_interrupt()函数中判断接收到数据, 调用process_poll(&rf230_process),发送PROCESS_EVENT_POLL事件。
 
在rf230_process PThread中收到PROCESS_EVENT_POLL,调用NETSTACK_RDC.input()进入数据接收处理流程。最后在Radio文件中,找到了调用process_poll()函数来,poll一个PROCESS_EVENT_POLL。

+aux_len;i++)>

附1:
以Contiki中example-mesh.c历程为例:
examples/example-mesh.c/example_mesh_process/mesh_open
 
send:
(在rime中:
mesh.c/mesh_send()
multihop_send()
unicast_send()
broadcast_send()
abc_send()(abc_send中调用rime_output)
)
rime.c/rime_output()/NETSTACK_MAC.send
nullmac.c/NETSTACK_RDC.send
(NETSTACK_FRAMER.create 生成framer)
nullrdc.c/NET_RADIO.send
nullrdc.c/mac_call_sent_callback 处理发送结束后的状态。
 
recv:
tf230bb/fr230bb.c/rf230_interrupt  调用process_poll产生PROCESS_EVENT_POLL。
pthread
rf230bb.c/rf230_process任务/NETSTACK_RDC.input
(NETSTACK_FRAMER.parse解析收到的数据帧,是否符合MAC层要求)
nullrdc.c/NETSTACK_MAC.input
nullmac.c/NETSTACK_NETWORK.input 调用rime.c/input()/abc_input()
通过rime协议栈的回调传递给
mesh.c/data_packet_received()此函数回调应用层的接收处理函数
example-mesh.c/example_recv()处理接收数据。
 
附2:
stunicast.c中
对于重发的处理,由在发送过程中设置timer如果在规定的时间内,如果没有数据返回则通过timer调用回调处理函数,如果有数据返回则在在recv_from_stunicast()调用stunicast_cancel()取消timer,避免调用回调。
在Contiki中,数据的发送与接收分别使用一个任务,这样可以避免之间的冲突问题。
NETWORK协议栈层之间的数据交互,通过buf来实现。这一点与Linux的sk_buff很像,所以导致了很多不必要的内存拷贝,这也是为什么Contiki内存占用很小的原因之一。

 

PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
我要评论
0
16
关闭 站长推荐上一条 /3 下一条