Contiki协议栈Rime:包属性packetbuf_attr

2024-04-08 04:32

本文主要是介绍Contiki协议栈Rime:包属性packetbuf_attr,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

更多的Contiki协议栈知识,请参考索引目录:
《Contiki协议栈:索引目录》

1 概述

  包属性其实属于下一篇博客《Contiki协议栈Rime:缓冲区管理packetbuf management》的一部分,但是它比较难以理解,所以单独抽出一篇博客对它做介绍。
  为了兼容其他协议,Rime不定义任何头部格式,而用包属性代替。一种属性是一种头部字段的抽象。当Rime协议栈中的子协议要发送一个包时,会传递一个属性链表下来。属性链表里包含了若干个属性,这些属性最终会被转换成头部的字段。
  该部分代码位于:contiki/core/net/packetbuf.[ch]

2 相关定义

packetbuf_attrs packetbuf_addrs

  Rime中定义了关于包属性的数组:

struct packetbuf_attr packetbuf_attrs[PACKETBUF_NUM_ATTRS];
struct packetbuf_addr packetbuf_addrs[PACKETBUF_NUM_ADDRS];

  其中,struct packetbuf_attr和struct packetbuf_addr分别定义为

typedef uint16_t packetbuf_attr_t;struct packetbuf_attr {packetbuf_attr_t val; // 一个16位无符号整型
};
struct packetbuf_addr {linkaddr_t addr; // 上一篇博客介绍的节点地址,16位或者64位
};

  数组的大小分别定义为:

#if NETSTACK_CONF_WITH_RIME
#define PACKETBUF_NUM_ADDRS 4
#else /* NETSTACK_CONF_WITH_RIME */
#define PACKETBUF_NUM_ADDRS 2
#endif /* NETSTACK_CONF_WITH_RIME */
#define PACKETBUF_NUM_ATTRS (PACKETBUF_ATTR_MAX - PACKETBUF_NUM_ADDRS)

  为什么PACKETBUF_NUM_ATTRS定义为PACKETBUF_ATTR_MAX - PACKETBUF_NUM_ADDRS?
  这是因为在定义属性的枚举变量时,将ADDR也当做了一种特殊属性,这样做的好处是可以将ADDR和ATTR统一处理(统一放在属性链表中,当Rime协议栈中的子协议将属性链表传递下来时,统一写到缓冲区packetbuf的头部)。请看下面

enum {PACKETBUF_ATTR_NONE,.../* Scope 2 attributes: used between end-to-end nodes. */
#if NETSTACK_CONF_WITH_RIMEPACKETBUF_ATTR_HOPS,PACKETBUF_ATTR_TTL,PACKETBUF_ATTR_EPACKET_ID,PACKETBUF_ATTR_EPACKET_TYPE,PACKETBUF_ATTR_ERELIABLE,          // 其实ATTR在这里就定义完了
#endif /* NETSTACK_CONF_WITH_RIME */    /* These must be last */PACKETBUF_ADDR_SENDER,             // 这里是ADDR的定义PACKETBUF_ADDR_RECEIVER,
#if NETSTACK_CONF_WITH_RIMEPACKETBUF_ADDR_ESENDER,PACKETBUF_ADDR_ERECEIVER,
#endif /* NETSTACK_CONF_WITH_RIME */PACKETBUF_ATTR_MAX                // 这里才是PACKETBUF_ATTR_MAX,包含ATTR+ADDR,所以ATTR=MAX-ADDR
};

struct packetbuf_attrlist

  前面说了,Rime协议栈的子协议在发送包时会传递一个属性链表下来,它的定义为:

struct packetbuf_attrlist {uint8_t type;uint8_t len;  // 单位是bit
};

  type:属性的类型。它直接对应于属性数组packetbuf_attrs或者packetbuf_addrs的下标
  len:该属性的长度。需要注意的是,此处长度的单位是位,而不是字节。属性的类型是packetbuf_attr_t,即uint16_t,因此可以推断,len的取值范围应该是0~16。(本宝宝好厉害呀,居然想到这儿了~~~)

3 相关函数

packetbuf_attr_clear

void packetbuf_attr_clear(void)
{int i;memset(packetbuf_attrs, 0, sizeof(packetbuf_attrs));//其实也可以用memset(packetbuf_addrs, 0, sizeof(packetbuf_addrs))吧?for(i = 0; i < PACKETBUF_NUM_ADDRS; ++i) {linkaddr_copy(&packetbuf_addrs[i].addr, &linkaddr_null);}
}

  很简单,清空两个属性数组packetbuf_attrs和packetbuf_addrs。

packetbuf_attr_copyto

void packetbuf_attr_copyto(struct packetbuf_attr *attrs, struct packetbuf_addr *addrs)
{memcpy(attrs, packetbuf_attrs, sizeof(packetbuf_attrs));memcpy(addrs, packetbuf_addrs, sizeof(packetbuf_addrs));
}

  很简单,将两个属性数组的内容拷贝给attrs和addrs

packetbuf_attr_copyfrom

void
packetbuf_attr_copyfrom(struct packetbuf_attr *attrs, struct packetbuf_addr *addrs)
{memcpy(packetbuf_attrs, attrs, sizeof(packetbuf_attrs));memcpy(packetbuf_addrs, addrs, sizeof(packetbuf_addrs));
}

  很简单,从attrs和addrs拷贝到两个属性数组中。

packetbuf_set_attr

int packetbuf_set_attr(uint8_t type, const packetbuf_attr_t val)
{
packetbuf_attrs[type].val = val;
return 1;
}
  很简单,设置属性数组中对应属性的值。

packetbuf_attr

packetbuf_attr_t
packetbuf_attr(uint8_t type)
{return packetbuf_attrs[type].val;
}

  很简单,返回属性数组中对于属性的值。

packetbuf_set_addr

int packetbuf_set_addr(uint8_t type, const linkaddr_t *addr)
{linkaddr_copy(&packetbuf_addrs[type - PACKETBUF_ADDR_FIRST].addr, addr);return 1;
}

  很简单,设置属性数组中对应属性的值。

packetbuf_addr

const linkaddr_t *packetbuf_addr(uint8_t type)
{return &packetbuf_addrs[type - PACKETBUF_ADDR_FIRST].addr;
}

  很简单,返回属性数组中对于属性的值。

4 小结

  这么抽象的东西,居然把它给阐释清楚了,我得意的笑~~
  但是说实话,初次接触时,难免会有很多地方难以理解,没关系,在继续看完我的后续几篇博客后,整个路就完全通了,然后大家都可以嘚瑟了O(∩_∩)O哈哈~

这篇关于Contiki协议栈Rime:包属性packetbuf_attr的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/884571

相关文章

HTTP 与 SpringBoot 参数提交与接收协议方式

《HTTP与SpringBoot参数提交与接收协议方式》HTTP参数提交方式包括URL查询、表单、JSON/XML、路径变量、头部、Cookie、GraphQL、WebSocket和SSE,依据... 目录HTTP 协议支持多种参数提交方式,主要取决于请求方法(Method)和内容类型(Content-Ty

Python如何调用另一个类的方法和属性

《Python如何调用另一个类的方法和属性》在Python面向对象编程中,类与类之间的交互是非常常见的场景,本文将详细介绍在Python中一个类如何调用另一个类的方法和属性,大家可以根据需要进行选择... 目录一、前言二、基本调用方式通过实例化调用通过类继承调用三、高级调用方式通过组合方式调用通过类方法/静

Java对接MQTT协议的完整实现示例代码

《Java对接MQTT协议的完整实现示例代码》MQTT是一个基于客户端-服务器的消息发布/订阅传输协议,MQTT协议是轻量、简单、开放和易于实现的,这些特点使它适用范围非常广泛,:本文主要介绍Ja... 目录前言前置依赖1. MQTT配置类代码解析1.1 MQTT客户端工厂1.2 MQTT消息订阅适配器1.

Linux中的自定义协议+序列反序列化用法

《Linux中的自定义协议+序列反序列化用法》文章探讨网络程序在应用层的实现,涉及TCP协议的数据传输机制、结构化数据的序列化与反序列化方法,以及通过JSON和自定义协议构建网络计算器的思路,强调分层... 目录一,再次理解协议二,序列化和反序列化三,实现网络计算器3.1 日志文件3.2Socket.hpp

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

spring中的@MapperScan注解属性解析

《spring中的@MapperScan注解属性解析》@MapperScan是Spring集成MyBatis时自动扫描Mapper接口的注解,简化配置并支持多数据源,通过属性控制扫描路径和过滤条件,利... 目录一、核心功能与作用二、注解属性解析三、底层实现原理四、使用场景与最佳实践五、注意事项与常见问题六

Linux线程之线程的创建、属性、回收、退出、取消方式

《Linux线程之线程的创建、属性、回收、退出、取消方式》文章总结了线程管理核心知识:线程号唯一、创建方式、属性设置(如分离状态与栈大小)、回收机制(join/detach)、退出方法(返回/pthr... 目录1. 线程号2. 线程的创建3. 线程属性4. 线程的回收5. 线程的退出6. 线程的取消7.

python删除xml中的w:ascii属性的步骤

《python删除xml中的w:ascii属性的步骤》使用xml.etree.ElementTree删除WordXML中w:ascii属性,需注册命名空间并定位rFonts元素,通过del操作删除属... 可以使用python的XML.etree.ElementTree模块通过以下步骤删除XML中的w:as

如何在Spring Boot项目中集成MQTT协议

《如何在SpringBoot项目中集成MQTT协议》本文介绍在SpringBoot中集成MQTT的步骤,包括安装Broker、添加EclipsePaho依赖、配置连接参数、实现消息发布订阅、测试接口... 目录1. 准备工作2. 引入依赖3. 配置MQTT连接4. 创建MQTT配置类5. 实现消息发布与订阅

使用Python进行GRPC和Dubbo协议的高级测试

《使用Python进行GRPC和Dubbo协议的高级测试》GRPC(GoogleRemoteProcedureCall)是一种高性能、开源的远程过程调用(RPC)框架,Dubbo是一种高性能的分布式服... 目录01 GRPC测试安装gRPC编写.proto文件实现服务02 Dubbo测试1. 安装Dubb