通过这几天对openvswitch代码的分析,以及项目的须要,须要对openflow 1.0进行一定的扩展,发现网上没有这方面的教程,尽管在搞懂ovs代码架构,floodlight controller中利用的事件驱动模型之后,会认为并非难事,可是对于刚入门SDN的同学来说,须要一番折腾,这里简单记录一下,希望帮助到其它人。

环境配置:2host + 1 OVS + floodlight

软件版本号: openvswitch 1.9.0 , floodlight0.85

在尝试对ovs中的openflow协议进行扩展之前至少应该理清的是,ovs怎样与SDN controller进行通信,收发of msg的过程,处理of action的流程,特别是怎样解析来自controller的消息;此外由于Floodlight利用的是netty框架,所以还要理解netty事件驱动的原理。接下来添加一个简单的openflow message以及相应的action ,然后看ovs能否成功接收来自controller的消息。

1.在  include/openflow/openflow-1.0.h中扩展of协议,添加我们的消息类型。(里面的字段依据自己的业务逻辑须要而设置)

struct ofp10_action_fp_update {

ovs_be16 type;

ovs_be16 len;

ovs_be32 vector;

};

OFP_ASSERT(sizeof(struct ofp10_action_fp_update) == 8);

struct ofp_fp_update{

ovs_be32 buffer_id;           /* ID assigned by datapath or UINT32_MAX. */

ovs_be16 in_port;             /* Packet's input port (OFPP_NONE if none). */

ovs_be16 other;         /* other fields reserved. */

struct ofp_action_header actions[0]; /* The action length is inferred

from the length field in the

header. */

};

OFP_ASSERT(sizeof(struct ofp_fp_update) == 8);

2.在 lib/ofp-msg.h 更新ofpraw这个结构体(非常重要,由于当ovs收到来自controller的消息之后就会利用ofpraw里面格式化的定义来验证消息的有效性,从而进行兴许处理)和ofptype枚举体。须要注意的是对ofpraw的更改一定要遵循固定的格式,凝视中讲的非常清楚。

enum ofpraw {

/* Immutable standard messages.

*

* The OpenFlow standard promises to preserve these messages and their numbers

* in future versions, so we mark them as <all>, which covers every OpenFlow

* version numbered 0x01...0xff, rather than as OF1.0+, which covers only

* OpenFlow versions that we otherwise implement.

* Without <all> here, then we would fail to decode "hello" messages that

* announce a version higher than we understand, even though there still could

* be a version in common with the peer that we do understand.  The <all>

* keyword is less useful for the other messages, because our OpenFlow channels

* accept only OpenFlow messages with a previously negotiated version.

*/

....................................................

/* OFPT 1.0 (13): struct ofp_packet_out, uint8_t[]. */

OFPRAW_OFPT10_PACKET_OUT,

/* OFPT 1.1+ (13): struct ofp11_packet_out, uint8_t[]. */

OFPRAW_OFPT11_PACKET_OUT,

/* OFPT 1.0 (20): struct ofp_fp_update, struct ofp_action_header[]. */

OFPRAW_OFPT10_FP_UPDATE,

};

enum ofptype {

/* Immutable messages. */

OFPTYPE_HELLO,               /* OFPRAW_OFPT_HELLO. */

OFPTYPE_ERROR,               /* OFPRAW_OFPT_ERROR. */

......................................

/* Controller command messages. */

OFPTYPE_PACKET_OUT,          /* OFPRAW_OFPT10_PACKET_OUT.

* OFPRAW_OFPT11_PACKET_OUT. */

OFPTYPE_FLOW_MOD,            /* OFPRAW_OFPT10_FLOW_MOD.

* OFPRAW_OFPT11_FLOW_MOD.

* OFPRAW_NXT_FLOW_MOD. */

OFPTYPE_PORT_MOD,            /* OFPRAW_OFPT10_PORT_MOD.

* OFPRAW_OFPT11_PORT_MOD. */

OFPTYPE_FP_UPDATE,          /* OFPRAW_OFPT10_FP_UPDATE. */

};

3. 在lib/ofp-util.h 中加入对应的数据结构代表对应的通用类型结构,由于ovs用户空间在解码对应的消息的时候须要针对详细的类型运行对应的动作,所以也要更新 ofp-actions.h

// ofp-util.h

struct ofputil_fp_update {

struct ofpact *ofpacts;     /* Actions. */

size_t ofpacts_len;         /* Size of ofpacts in bytes. */

};

// 对这个消息编解码辅助函数

enum ofperr ofputil_decode_fp_update(struct ofputil_fp_update *,

const struct ofp_header *,

struct ofpbuf *ofpacts);

struct ofpbuf *ofputil_encode_fp_update(const struct ofputil_fp_update *,

enum ofputil_protocol protocol);

// ofp-actions.h

struct ofpact_fp_update {

struct ofpact ofpact;

uint32_t vector;    /* 自己定义*/

};

4. 在 ofproto/ofproto.c (3790)- handle_openflow_() 函数中加入处理该消息的分支,

static enum ofperr

3793 handle_openflow__(struct ofconn *ofconn, const struct ofpbuf *msg)

3794 {

3795     const struct ofp_header *oh = msg->data;

3796     enum ofptype type;

3797     enum ofperr error;

3798

3799     error = ofptype_decode(&type, oh);

3800     if (error) {

3801         return error;

3802     }

3803

3804     switch (type) {

3805         /* OpenFlow requests. */

3806     case OFPTYPE_ECHO_REQUEST:

3807         return handle_echo_request(ofconn, oh);

3808

3809     case OFPTYPE_FEATURES_REQUEST:

3810         return handle_features_request(ofconn, oh);

3811

3812     case OFPTYPE_GET_CONFIG_REQUEST:

3813         return handle_get_config_request(ofconn, oh);

3814

3815     case OFPTYPE_SET_CONFIG:

3816         return handle_set_config(ofconn, oh);

3817

3818     case OFPTYPE_PACKET_OUT:

3819         return handle_packet_out(ofconn, oh);

3820     case OFPTYPE_FP_UPDATE:

3821         return handle_fp_update(ofconn, oh);

...............................................

}

处理函数自己依据须要来实现,这里只打印日志信息。

static enum ofperr

handle_fp_update(struct ofconn *ofconn, const struct ofp_header *oh)

{

//TODO

VLOG_INFO("=====I GOT FP FROM CONTROLLER ========");

return 0;

}

以上就是ovs上须要考虑的地方,细节处要注意。接下来是Floodlight上面通过构造一个该消息,而后当接收到packet in 后就将其发送到这个SW上。

5. 在org.openflow.protocol 包中添加我们的消息类,以及对应的方法。

public class OFFPUpdate extends OFMessage implements OFActionFactoryAware{

public static int MINIMUM_LENGTH = 16;

public static int BUFFER_ID_NONE = 0xffffffff;

protected OFActionFactory actionFactory;

protected int bufferId;

protected short inPort = 0x0000; // not used yet

protected short other;

protected List<OFAction> actions;

public OFFPUpdate(){

super();

this.type = OFType.FP_UPDATE;

this.length = U16.t(MINIMUM_LENGTH);

}

....................................

}

6.然后再OFType中标示这个新增的消息类。

public enum OFType {

HELLO               (0, OFHello.class, new Instantiable<OFMessage>() {

@Override

public OFMessage instantiate() {

return new OFHello();

}}),

...........................................................

FP_UPDATE          (20, OFFPUpdate.class, new Instantiable<OFMessage>() {

@Override

public OFMessage instantiate() {

return new OFFPUpdate();

}});

}

7. 在org.openflow.protocol.action中新增相应的action类,由于每次从SDN controller发送控制类消息时都会设置它的action list。

public class OFActionFPUpdate extends OFAction implements Cloneable {

public static int MINIMUM_LENGTH = 8;

protected int vector;

public OFActionFPUpdate() {

super.setType(OFActionType.FP_UPDATE);

super.setLength((short) MINIMUM_LENGTH);

}

public OFActionFPUpdate(int vector) {

super();

super.setType(OFActionType.FP_UPDATE);

super.setLength((short) MINIMUM_LENGTH);

this.vector = vector;

}

...................................

}

8. 最后写一个简单的module 来測试。这里的思路是当收到某个SW发来的packetin消息时,我们就给这个交换机下发一个FPUpdate的消息。主要代码:

OFFPUpdate fu =

(OFFPUpdate) floodlightProvider.getOFMessageFactory()

.getMessage(OFType.FP_UPDATE);

OFActionFPUpdate action = new OFActionFPUpdate();

action.setVector(0xffffffff); // just for testing

List<OFAction> actions = new ArrayList<OFAction>();

actions.add(action);

fu.setBufferId(OFFPUpdate.BUFFER_ID_NONE)

.setInPort((short)0)

.setActions(actions)

.setLengthU(OFFPUpdate.MINIMUM_LENGTH+OFActionFPUpdate.MINIMUM_LENGTH);

try {

sw.write(fu, cntx);

} catch (IOException e) {

logger.error("Failure writing fp update", e);

}

通过简单的測试,整个流程是正确的。

转载请注明出处:点击打开链接

对openflow 1.0协议的扩展相关推荐

  1. 用C#做短信CMPP2.0/3.0协议 支持扩展号支持物理网卡

    此程序为中国移动CMPP协议程序接口,适合在中国移动申请了短信发送端口的公司使用. 短信群发已经成为现在软件系统.网络营销等必不可少的应用工具.可应用在短信验证.信息群发.游戏虚拟商品购买.事件提醒. ...

  2. OAuth 2.0 协议学习笔记

    协议官网 在传统的客户端-服务器身份验证模型中,客户端通过使用资源所有者的凭据向服务器进行身份验证来请求服务器上的访问受限资源(受保护资源). 为了向第三方应用程序提供对受限资源的访问,资源所有者与第 ...

  3. OpenID Connect:OAuth 2.0协议之上的简单身份层

    OpenID Connect是什么?OpenID Connect(目前版本是1.0)是OAuth 2.0协议(可参考本人此篇:OAuth 2.0 / RCF6749 协议解读)之上的简单身份层,用 A ...

  4. 对Java的URL类支持的协议进行扩展的方法

    转载自   对Java的URL类支持的协议进行扩展的方法 JAVA默认提供了对file,ftp,gopher,http,https,jar,mailto,netdoc协议的支持.当我们要利用这些协议来 ...

  5. oauth2.0协议流程_正确的工作流程:我应该使用哪个OAuth 2.0流程?

    oauth2.0协议流程 什么是OAuth 2.0 OAuth 2.0是一个已被广泛采用的委托授权框架,已经存在了很多年,并且似乎已经存在. 如果您不熟悉OAuth 2.0的基本概念,可以使用 川崎孝 ...

  6. 深入理解OAuth2.0协议

    1. 引言 如果你开车去酒店赴宴,你经常会苦于找不到停车位而耽误很多时间.是否有好办法可以避免这个问题呢?有的,听说有一些豪车的车主就不担心这个问题.豪车一般配备两种钥匙:主钥匙和泊车钥匙.当你到酒店 ...

  7. 最近在学OAuth2.0协议,给大家分享一下

    1. 引言 如果你开车去酒店赴宴,你经常会苦于找不到停车位而耽误很多时间.是否有好办法可以避免这个问题呢?有的,听说有一些豪车的车主就不担心这个问题.豪车一般配备两种钥匙:主钥匙和泊车钥匙.当你到酒店 ...

  8. OpenFlow1.0协议解析

    *注:此笔记为在学习OpenFlow协议时的个人总结,如需转载麻烦表明出处,并附上连接(http://blog.csdn.net/sherkyoung/article/details/39159601 ...

  9. MW7299/PD+HUB2.0集成芯片上行支持PD3.0协议下行HUB支持BC1.2协议QFN32封装

    MW7299/PD+HUB2.0集成芯片上行支持PD3.0协议下行HUB支持BC1.2协议QFN32封装 特点:采用单 5V 电源供电,内置 5V转 3.3V LDO. ●内嵌一个类 RISC 处理器 ...

最新文章

  1. AMBA、AHB、APB、AXI总线介绍和对比
  2. hibenate5.1配置mysql_hibernate5.2的基本配置方法(详解)
  3. Diffie-Hellman密钥交换协议
  4. 看完99%的人都学会了!9次Android面试经验总结,我先收藏为敬
  5. GCD简介二:多核心的性能
  6. MyBatis-Plus_查询进阶01
  7. C++笔记-基于邻接矩阵的BFS(宽度优先遍历)
  8. 流量卡官网源码【全解无后门】 修复添加教程
  9. python selenium 文件上传_python-selenium -- 文件上传操作
  10. ubuntu安装mysql没反应_Ubuntu安装mysql三种安装方式
  11. 手绘流程图讲解spark是如何实现集群的高可用
  12. SpringBoot:javalist和set区别
  13. Linux netstat 命令详解
  14. 04 _ 可扩展架构案例(一):电商平台架构是如何演变的?
  15. 关于使用nRF Studio 批量烧写NRF52840 不成功的解决方案
  16. 苹果手机怎么投屏王者荣耀
  17. iOS实现炫酷悬停交互视图
  18. linux幸运字符,删好友后幸运字符怎么找回来
  19. 区块链以及蚂蚁开放联盟链初探索
  20. Qiime2+Origin绘制稀释曲线

热门文章

  1. java angularjs 跨域访问_AngularJS实现跨域请求
  2. linux httpd 自动启动,在Linux启动时让Apache也自动启动
  3. w ndows10电脑配置看哪里,win10对电脑配置要求及如何查看电脑配置
  4. jfinal结合freemarker,页面使用$符获取属性值报错原因解决
  5. 修改Launcher2欢迎页面字符重叠
  6. VB如何直接显示内存中的二进制图像数据
  7. 通用业务平台设计(一):概览
  8. 雷军100亿押注IoT,小米借AI两翼齐飞
  9. 据说,上海AI产业规模700亿,包揽全国1/3人才
  10. 我要学ASP.NET MVC 3.0(十三): MVC 3.0 防止跨站点请求伪造 (CSRF) 攻击