2010-05-27 13:14:05 -07:00
|
|
|
|
/*
|
2011-01-12 13:42:50 -08:00
|
|
|
|
* Copyright (c) 2008, 2009, 2010, 2011 Nicira Networks.
|
2010-05-27 13:14:05 -07:00
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at:
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include "ofp-print.h"
|
2011-01-12 13:42:50 -08:00
|
|
|
|
#include <errno.h>
|
2010-05-27 13:14:05 -07:00
|
|
|
|
#include <inttypes.h>
|
2011-11-01 13:25:49 +01:00
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <netinet/in.h>
|
2011-05-09 16:25:48 -07:00
|
|
|
|
#include <netinet/icmp6.h>
|
2010-05-27 13:14:05 -07:00
|
|
|
|
#include <stdlib.h>
|
2011-04-05 12:37:52 -07:00
|
|
|
|
#include "autopath.h"
|
2011-06-10 17:45:45 -07:00
|
|
|
|
#include "bundle.h"
|
2010-10-28 17:13:18 -07:00
|
|
|
|
#include "byte-order.h"
|
2010-11-10 14:39:54 -08:00
|
|
|
|
#include "classifier.h"
|
2011-01-12 13:42:50 -08:00
|
|
|
|
#include "dynamic-string.h"
|
2011-09-12 16:19:57 -07:00
|
|
|
|
#include "learn.h"
|
2010-12-17 14:38:50 -08:00
|
|
|
|
#include "multipath.h"
|
2010-11-11 10:41:33 -08:00
|
|
|
|
#include "nx-match.h"
|
2011-01-12 13:42:50 -08:00
|
|
|
|
#include "ofp-errors.h"
|
2010-05-27 13:14:05 -07:00
|
|
|
|
#include "ofp-util.h"
|
|
|
|
|
#include "ofpbuf.h"
|
|
|
|
|
#include "packets.h"
|
|
|
|
|
#include "random.h"
|
2011-03-10 15:02:05 -08:00
|
|
|
|
#include "unaligned.h"
|
2010-12-09 10:41:32 -08:00
|
|
|
|
#include "type-props.h"
|
2010-07-16 11:02:49 -07:00
|
|
|
|
#include "vlog.h"
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
2010-10-19 14:47:01 -07:00
|
|
|
|
VLOG_DEFINE_THIS_MODULE(ofp_util);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
|
|
|
|
/* Rate limit for OpenFlow message parse errors. These always indicate a bug
|
|
|
|
|
* in the peer and so there's not much point in showing a lot of them. */
|
|
|
|
|
static struct vlog_rate_limit bad_ofmsg_rl = VLOG_RATE_LIMIT_INIT(1, 5);
|
|
|
|
|
|
2010-11-22 10:09:18 -08:00
|
|
|
|
/* Given the wildcard bit count in the least-significant 6 of 'wcbits', returns
|
|
|
|
|
* an IP netmask with a 1 in each bit that must match and a 0 in each bit that
|
|
|
|
|
* is wildcarded.
|
|
|
|
|
*
|
|
|
|
|
* The bits in 'wcbits' are in the format used in enum ofp_flow_wildcards: 0
|
|
|
|
|
* is exact match, 1 ignores the LSB, 2 ignores the 2 least-significant bits,
|
|
|
|
|
* ..., 32 and higher wildcard the entire field. This is the *opposite* of the
|
|
|
|
|
* usual convention where e.g. /24 indicates that 8 bits (not 24 bits) are
|
|
|
|
|
* wildcarded. */
|
|
|
|
|
ovs_be32
|
|
|
|
|
ofputil_wcbits_to_netmask(int wcbits)
|
|
|
|
|
{
|
|
|
|
|
wcbits &= 0x3f;
|
|
|
|
|
return wcbits < 32 ? htonl(~((1u << wcbits) - 1)) : 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Given the IP netmask 'netmask', returns the number of bits of the IP address
|
2011-08-17 10:55:15 -07:00
|
|
|
|
* that it wildcards, that is, the number of 0-bits in 'netmask'. 'netmask'
|
|
|
|
|
* must be a CIDR netmask (see ip_is_cidr()). */
|
2010-11-22 10:09:18 -08:00
|
|
|
|
int
|
|
|
|
|
ofputil_netmask_to_wcbits(ovs_be32 netmask)
|
|
|
|
|
{
|
2011-08-17 10:55:15 -07:00
|
|
|
|
return 32 - ip_count_cidr_bits(netmask);
|
2010-11-22 10:09:18 -08:00
|
|
|
|
}
|
|
|
|
|
|
2010-11-10 14:39:54 -08:00
|
|
|
|
/* A list of the FWW_* and OFPFW_ bits that have the same value, meaning, and
|
|
|
|
|
* name. */
|
|
|
|
|
#define WC_INVARIANT_LIST \
|
|
|
|
|
WC_INVARIANT_BIT(IN_PORT) \
|
|
|
|
|
WC_INVARIANT_BIT(DL_SRC) \
|
|
|
|
|
WC_INVARIANT_BIT(DL_DST) \
|
|
|
|
|
WC_INVARIANT_BIT(DL_TYPE) \
|
|
|
|
|
WC_INVARIANT_BIT(NW_PROTO) \
|
|
|
|
|
WC_INVARIANT_BIT(TP_SRC) \
|
|
|
|
|
WC_INVARIANT_BIT(TP_DST)
|
|
|
|
|
|
|
|
|
|
/* Verify that all of the invariant bits (as defined on WC_INVARIANT_LIST)
|
|
|
|
|
* actually have the same names and values. */
|
|
|
|
|
#define WC_INVARIANT_BIT(NAME) BUILD_ASSERT_DECL(FWW_##NAME == OFPFW_##NAME);
|
|
|
|
|
WC_INVARIANT_LIST
|
|
|
|
|
#undef WC_INVARIANT_BIT
|
|
|
|
|
|
|
|
|
|
/* WC_INVARIANTS is the invariant bits (as defined on WC_INVARIANT_LIST) all
|
|
|
|
|
* OR'd together. */
|
2011-05-06 12:38:20 -07:00
|
|
|
|
static const flow_wildcards_t WC_INVARIANTS = 0
|
2010-11-10 14:39:54 -08:00
|
|
|
|
#define WC_INVARIANT_BIT(NAME) | FWW_##NAME
|
|
|
|
|
WC_INVARIANT_LIST
|
|
|
|
|
#undef WC_INVARIANT_BIT
|
2011-05-06 12:38:20 -07:00
|
|
|
|
;
|
2010-11-10 14:39:54 -08:00
|
|
|
|
|
2011-05-31 16:15:45 -07:00
|
|
|
|
/* Converts the wildcard in 'ofpfw' into a flow_wildcards in 'wc' for use in
|
|
|
|
|
* struct cls_rule. It is the caller's responsibility to handle the special
|
|
|
|
|
* case where the flow match's dl_vlan is set to OFP_VLAN_NONE. */
|
2011-06-01 09:09:20 -07:00
|
|
|
|
void
|
2011-05-31 16:15:45 -07:00
|
|
|
|
ofputil_wildcard_from_openflow(uint32_t ofpfw, struct flow_wildcards *wc)
|
2010-11-10 14:39:54 -08:00
|
|
|
|
{
|
2011-11-09 23:39:16 -08:00
|
|
|
|
BUILD_ASSERT_DECL(FLOW_WC_SEQ == 7);
|
2011-07-29 13:15:09 -07:00
|
|
|
|
|
2010-11-10 14:39:54 -08:00
|
|
|
|
/* Initialize most of rule->wc. */
|
2010-12-06 16:11:55 -08:00
|
|
|
|
flow_wildcards_init_catchall(wc);
|
2011-05-06 12:38:20 -07:00
|
|
|
|
wc->wildcards = (OVS_FORCE flow_wildcards_t) ofpfw & WC_INVARIANTS;
|
2010-12-07 14:02:17 -08:00
|
|
|
|
|
|
|
|
|
/* Wildcard fields that aren't defined by ofp_match or tun_id. */
|
2011-11-09 23:39:16 -08:00
|
|
|
|
wc->wildcards |= (FWW_ARP_SHA | FWW_ARP_THA | FWW_NW_ECN | FWW_NW_TTL
|
2011-11-05 15:48:12 -07:00
|
|
|
|
| FWW_ND_TARGET | FWW_IPV6_LABEL);
|
2010-12-07 14:02:17 -08:00
|
|
|
|
|
2011-11-09 23:39:16 -08:00
|
|
|
|
if (ofpfw & OFPFW_NW_TOS) {
|
|
|
|
|
/* OpenFlow 1.0 defines a TOS wildcard, but it's much later in
|
|
|
|
|
* the enum than we can use. */
|
|
|
|
|
wc->wildcards |= FWW_NW_DSCP;
|
2010-11-10 14:39:54 -08:00
|
|
|
|
}
|
Implement new fragment handling policy.
Until now, OVS has handled IP fragments more awkwardly than necessary. It
has not been possible to match on L4 headers, even in fragments with offset
0 where they are actually present. This means that there was no way to
implement ACLs that treat, say, different TCP ports differently, on
fragmented traffic; instead, all decisions for fragment forwarding had to
be made on the basis of L2 and L3 headers alone.
This commit improves the situation significantly. It is still not possible
to match on L4 headers in fragments with nonzero offset, because that
information is simply not present in such fragments, but this commit adds
the ability to match on L4 headers for fragments with zero offset. This
means that it becomes possible to implement ACLs that drop such "first
fragments" on the basis of L4 headers. In practice, that effectively
blocks even fragmented traffic on an L4 basis, because the receiving IP
stack cannot reassemble a full packet when the first fragment is missing.
This commit works by adding a new "fragment type" to the kernel flow match
and making it available through OpenFlow as a new NXM field named
NXM_NX_IP_FRAG. Because OpenFlow 1.0 explicitly says that the L4 fields
are always 0 for IP fragments, it adds a new OpenFlow fragment handling
mode that fills in the L4 fields for "first fragments". It also enhances
ovs-ofctl to allow users to configure this new fragment handling mode and
to parse the new field.
Signed-off-by: Ben Pfaff <blp@nicira.com>
Bug #7557.
2011-10-19 21:33:44 -07:00
|
|
|
|
|
2010-11-10 14:39:54 -08:00
|
|
|
|
wc->nw_src_mask = ofputil_wcbits_to_netmask(ofpfw >> OFPFW_NW_SRC_SHIFT);
|
|
|
|
|
wc->nw_dst_mask = ofputil_wcbits_to_netmask(ofpfw >> OFPFW_NW_DST_SHIFT);
|
|
|
|
|
|
|
|
|
|
if (ofpfw & OFPFW_DL_DST) {
|
|
|
|
|
/* OpenFlow 1.0 OFPFW_DL_DST covers the whole Ethernet destination, but
|
|
|
|
|
* Open vSwitch breaks the Ethernet destination into bits as FWW_DL_DST
|
|
|
|
|
* and FWW_ETH_MCAST. */
|
|
|
|
|
wc->wildcards |= FWW_ETH_MCAST;
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-31 16:15:45 -07:00
|
|
|
|
/* VLAN TCI mask. */
|
|
|
|
|
if (!(ofpfw & OFPFW_DL_VLAN_PCP)) {
|
|
|
|
|
wc->vlan_tci_mask |= htons(VLAN_PCP_MASK | VLAN_CFI);
|
|
|
|
|
}
|
|
|
|
|
if (!(ofpfw & OFPFW_DL_VLAN)) {
|
|
|
|
|
wc->vlan_tci_mask |= htons(VLAN_VID_MASK | VLAN_CFI);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Converts the ofp_match in 'match' into a cls_rule in 'rule', with the given
|
|
|
|
|
* 'priority'. */
|
|
|
|
|
void
|
|
|
|
|
ofputil_cls_rule_from_match(const struct ofp_match *match,
|
|
|
|
|
unsigned int priority, struct cls_rule *rule)
|
|
|
|
|
{
|
|
|
|
|
uint32_t ofpfw = ntohl(match->wildcards) & OFPFW_ALL;
|
|
|
|
|
|
|
|
|
|
/* Initialize rule->priority, rule->wc. */
|
|
|
|
|
rule->priority = !ofpfw ? UINT16_MAX : priority;
|
|
|
|
|
ofputil_wildcard_from_openflow(ofpfw, &rule->wc);
|
|
|
|
|
|
2010-11-23 10:06:28 -08:00
|
|
|
|
/* Initialize most of rule->flow. */
|
2010-11-10 14:39:54 -08:00
|
|
|
|
rule->flow.nw_src = match->nw_src;
|
|
|
|
|
rule->flow.nw_dst = match->nw_dst;
|
2011-05-11 12:13:10 -07:00
|
|
|
|
rule->flow.in_port = ntohs(match->in_port);
|
2011-01-23 18:44:44 -08:00
|
|
|
|
rule->flow.dl_type = ofputil_dl_type_from_openflow(match->dl_type);
|
2010-11-10 14:39:54 -08:00
|
|
|
|
rule->flow.tp_src = match->tp_src;
|
|
|
|
|
rule->flow.tp_dst = match->tp_dst;
|
|
|
|
|
memcpy(rule->flow.dl_src, match->dl_src, ETH_ADDR_LEN);
|
|
|
|
|
memcpy(rule->flow.dl_dst, match->dl_dst, ETH_ADDR_LEN);
|
2011-11-09 17:10:27 -08:00
|
|
|
|
rule->flow.nw_tos = match->nw_tos & IP_DSCP_MASK;
|
2010-11-10 14:39:54 -08:00
|
|
|
|
rule->flow.nw_proto = match->nw_proto;
|
|
|
|
|
|
2010-11-23 10:06:28 -08:00
|
|
|
|
/* Translate VLANs. */
|
2011-05-31 16:15:44 -07:00
|
|
|
|
if (!(ofpfw & OFPFW_DL_VLAN) && match->dl_vlan == htons(OFP_VLAN_NONE)) {
|
|
|
|
|
/* Match only packets without 802.1Q header.
|
|
|
|
|
*
|
|
|
|
|
* When OFPFW_DL_VLAN_PCP is wildcarded, this is obviously correct.
|
|
|
|
|
*
|
|
|
|
|
* If OFPFW_DL_VLAN_PCP is matched, the flow match is contradictory,
|
|
|
|
|
* because we can't have a specific PCP without an 802.1Q header.
|
|
|
|
|
* However, older versions of OVS treated this as matching packets
|
|
|
|
|
* withut an 802.1Q header, so we do here too. */
|
2010-11-23 10:06:28 -08:00
|
|
|
|
rule->flow.vlan_tci = htons(0);
|
2011-05-31 16:15:45 -07:00
|
|
|
|
rule->wc.vlan_tci_mask = htons(0xffff);
|
2011-05-31 16:15:44 -07:00
|
|
|
|
} else {
|
|
|
|
|
ovs_be16 vid, pcp, tci;
|
|
|
|
|
|
|
|
|
|
vid = match->dl_vlan & htons(VLAN_VID_MASK);
|
|
|
|
|
pcp = htons((match->dl_vlan_pcp << VLAN_PCP_SHIFT) & VLAN_PCP_MASK);
|
|
|
|
|
tci = vid | pcp | htons(VLAN_CFI);
|
2011-05-31 16:15:45 -07:00
|
|
|
|
rule->flow.vlan_tci = tci & rule->wc.vlan_tci_mask;
|
2010-11-23 10:06:28 -08:00
|
|
|
|
}
|
|
|
|
|
|
2010-11-10 14:39:54 -08:00
|
|
|
|
/* Clean up. */
|
|
|
|
|
cls_rule_zero_wildcarded_fields(rule);
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
/* Convert 'rule' into the OpenFlow match structure 'match'. */
|
2010-11-10 14:39:54 -08:00
|
|
|
|
void
|
2011-04-26 09:42:18 -07:00
|
|
|
|
ofputil_cls_rule_to_match(const struct cls_rule *rule, struct ofp_match *match)
|
2010-11-10 14:39:54 -08:00
|
|
|
|
{
|
|
|
|
|
const struct flow_wildcards *wc = &rule->wc;
|
2011-05-06 12:38:20 -07:00
|
|
|
|
uint32_t ofpfw;
|
2010-11-10 14:39:54 -08:00
|
|
|
|
|
2010-11-23 10:06:28 -08:00
|
|
|
|
/* Figure out most OpenFlow wildcards. */
|
2011-05-06 12:38:20 -07:00
|
|
|
|
ofpfw = (OVS_FORCE uint32_t) (wc->wildcards & WC_INVARIANTS);
|
2010-11-10 14:39:54 -08:00
|
|
|
|
ofpfw |= ofputil_netmask_to_wcbits(wc->nw_src_mask) << OFPFW_NW_SRC_SHIFT;
|
|
|
|
|
ofpfw |= ofputil_netmask_to_wcbits(wc->nw_dst_mask) << OFPFW_NW_DST_SHIFT;
|
2011-11-09 23:39:16 -08:00
|
|
|
|
if (wc->wildcards & FWW_NW_DSCP) {
|
2010-11-10 14:39:54 -08:00
|
|
|
|
ofpfw |= OFPFW_NW_TOS;
|
|
|
|
|
}
|
2010-12-09 14:19:51 -08:00
|
|
|
|
|
2010-11-23 10:06:28 -08:00
|
|
|
|
/* Translate VLANs. */
|
|
|
|
|
match->dl_vlan = htons(0);
|
|
|
|
|
match->dl_vlan_pcp = 0;
|
|
|
|
|
if (rule->wc.vlan_tci_mask == htons(0)) {
|
|
|
|
|
ofpfw |= OFPFW_DL_VLAN | OFPFW_DL_VLAN_PCP;
|
|
|
|
|
} else if (rule->wc.vlan_tci_mask & htons(VLAN_CFI)
|
|
|
|
|
&& !(rule->flow.vlan_tci & htons(VLAN_CFI))) {
|
|
|
|
|
match->dl_vlan = htons(OFP_VLAN_NONE);
|
|
|
|
|
} else {
|
|
|
|
|
if (!(rule->wc.vlan_tci_mask & htons(VLAN_VID_MASK))) {
|
|
|
|
|
ofpfw |= OFPFW_DL_VLAN;
|
|
|
|
|
} else {
|
|
|
|
|
match->dl_vlan = htons(vlan_tci_to_vid(rule->flow.vlan_tci));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(rule->wc.vlan_tci_mask & htons(VLAN_PCP_MASK))) {
|
|
|
|
|
ofpfw |= OFPFW_DL_VLAN_PCP;
|
|
|
|
|
} else {
|
|
|
|
|
match->dl_vlan_pcp = vlan_tci_to_pcp(rule->flow.vlan_tci);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Compose most of the match structure. */
|
2010-11-10 14:39:54 -08:00
|
|
|
|
match->wildcards = htonl(ofpfw);
|
2011-05-11 12:13:10 -07:00
|
|
|
|
match->in_port = htons(rule->flow.in_port);
|
2010-11-10 14:39:54 -08:00
|
|
|
|
memcpy(match->dl_src, rule->flow.dl_src, ETH_ADDR_LEN);
|
|
|
|
|
memcpy(match->dl_dst, rule->flow.dl_dst, ETH_ADDR_LEN);
|
2011-01-23 18:44:44 -08:00
|
|
|
|
match->dl_type = ofputil_dl_type_to_openflow(rule->flow.dl_type);
|
2010-11-10 14:39:54 -08:00
|
|
|
|
match->nw_src = rule->flow.nw_src;
|
|
|
|
|
match->nw_dst = rule->flow.nw_dst;
|
2011-11-09 17:10:27 -08:00
|
|
|
|
match->nw_tos = rule->flow.nw_tos & IP_DSCP_MASK;
|
2010-11-10 14:39:54 -08:00
|
|
|
|
match->nw_proto = rule->flow.nw_proto;
|
|
|
|
|
match->tp_src = rule->flow.tp_src;
|
|
|
|
|
match->tp_dst = rule->flow.tp_dst;
|
|
|
|
|
memset(match->pad1, '\0', sizeof match->pad1);
|
|
|
|
|
memset(match->pad2, '\0', sizeof match->pad2);
|
|
|
|
|
}
|
|
|
|
|
|
2011-01-23 18:44:44 -08:00
|
|
|
|
/* Given a 'dl_type' value in the format used in struct flow, returns the
|
|
|
|
|
* corresponding 'dl_type' value for use in an OpenFlow ofp_match structure. */
|
|
|
|
|
ovs_be16
|
|
|
|
|
ofputil_dl_type_to_openflow(ovs_be16 flow_dl_type)
|
|
|
|
|
{
|
|
|
|
|
return (flow_dl_type == htons(FLOW_DL_TYPE_NONE)
|
|
|
|
|
? htons(OFP_DL_TYPE_NOT_ETH_TYPE)
|
|
|
|
|
: flow_dl_type);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Given a 'dl_type' value in the format used in an OpenFlow ofp_match
|
|
|
|
|
* structure, returns the corresponding 'dl_type' value for use in struct
|
|
|
|
|
* flow. */
|
|
|
|
|
ovs_be16
|
|
|
|
|
ofputil_dl_type_from_openflow(ovs_be16 ofp_dl_type)
|
|
|
|
|
{
|
|
|
|
|
return (ofp_dl_type == htons(OFP_DL_TYPE_NOT_ETH_TYPE)
|
|
|
|
|
? htons(FLOW_DL_TYPE_NONE)
|
|
|
|
|
: ofp_dl_type);
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-15 11:04:40 -08:00
|
|
|
|
/* Returns a transaction ID to use for an outgoing OpenFlow message. */
|
2010-11-16 11:00:25 -08:00
|
|
|
|
static ovs_be32
|
2010-05-27 13:14:05 -07:00
|
|
|
|
alloc_xid(void)
|
|
|
|
|
{
|
2010-11-15 11:04:40 -08:00
|
|
|
|
static uint32_t next_xid = 1;
|
2010-11-16 11:00:25 -08:00
|
|
|
|
return htonl(next_xid++);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
/* Basic parsing of OpenFlow messages. */
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
2010-12-06 10:20:20 -08:00
|
|
|
|
struct ofputil_msg_type {
|
|
|
|
|
enum ofputil_msg_code code; /* OFPUTIL_*. */
|
|
|
|
|
uint32_t value; /* OFPT_*, OFPST_*, NXT_*, or NXST_*. */
|
|
|
|
|
const char *name; /* e.g. "OFPT_FLOW_REMOVED". */
|
|
|
|
|
unsigned int min_size; /* Minimum total message size in bytes. */
|
|
|
|
|
/* 0 if 'min_size' is the exact size that the message must be. Otherwise,
|
|
|
|
|
* the message may exceed 'min_size' by an even multiple of this value. */
|
|
|
|
|
unsigned int extra_multiple;
|
|
|
|
|
};
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
/* Represents a malformed OpenFlow message. */
|
|
|
|
|
static const struct ofputil_msg_type ofputil_invalid_type = {
|
|
|
|
|
OFPUTIL_MSG_INVALID, 0, "OFPUTIL_MSG_INVALID", 0, 0
|
|
|
|
|
};
|
|
|
|
|
|
2010-12-06 10:20:20 -08:00
|
|
|
|
struct ofputil_msg_category {
|
|
|
|
|
const char *name; /* e.g. "OpenFlow message" */
|
|
|
|
|
const struct ofputil_msg_type *types;
|
|
|
|
|
size_t n_types;
|
|
|
|
|
int missing_error; /* ofp_mkerr() value for missing type. */
|
|
|
|
|
};
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
static int
|
|
|
|
|
ofputil_check_length(const struct ofputil_msg_type *type, unsigned int size)
|
2010-12-06 10:20:20 -08:00
|
|
|
|
{
|
|
|
|
|
switch (type->extra_multiple) {
|
|
|
|
|
case 0:
|
|
|
|
|
if (size != type->min_size) {
|
2011-10-27 12:54:44 -07:00
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "received %s with incorrect "
|
2010-12-06 10:20:20 -08:00
|
|
|
|
"length %u (expected length %u)",
|
2011-10-27 12:54:44 -07:00
|
|
|
|
type->name, size, type->min_size);
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
2011-10-27 12:54:44 -07:00
|
|
|
|
return 0;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
|
if (size < type->min_size) {
|
2011-10-27 12:54:44 -07:00
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "received %s with incorrect "
|
2010-12-06 10:20:20 -08:00
|
|
|
|
"length %u (expected length at least %u bytes)",
|
2011-10-27 12:54:44 -07:00
|
|
|
|
type->name, size, type->min_size);
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
2011-10-27 12:54:44 -07:00
|
|
|
|
return 0;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
if (size < type->min_size
|
|
|
|
|
|| (size - type->min_size) % type->extra_multiple) {
|
2011-10-27 12:54:44 -07:00
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "received %s with incorrect "
|
2010-12-06 10:20:20 -08:00
|
|
|
|
"length %u (must be exactly %u bytes or longer "
|
|
|
|
|
"by an integer multiple of %u bytes)",
|
2011-10-27 12:54:44 -07:00
|
|
|
|
type->name, size,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
type->min_size, type->extra_multiple);
|
2011-10-27 12:54:44 -07:00
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
2011-10-27 12:54:44 -07:00
|
|
|
|
return 0;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
ofputil_lookup_openflow_message(const struct ofputil_msg_category *cat,
|
2011-10-27 12:54:44 -07:00
|
|
|
|
uint32_t value,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
const struct ofputil_msg_type *type;
|
|
|
|
|
|
|
|
|
|
for (type = cat->types; type < &cat->types[cat->n_types]; type++) {
|
|
|
|
|
if (type->value == value) {
|
|
|
|
|
*typep = type;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-13 16:20:06 -08:00
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "received %s of unknown type %"PRIu32,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
cat->name, value);
|
|
|
|
|
return cat->missing_error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ofputil_decode_vendor(const struct ofp_header *oh, size_t length,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
static const struct ofputil_msg_type nxt_messages[] = {
|
|
|
|
|
{ OFPUTIL_NXT_ROLE_REQUEST,
|
|
|
|
|
NXT_ROLE_REQUEST, "NXT_ROLE_REQUEST",
|
|
|
|
|
sizeof(struct nx_role_request), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_NXT_ROLE_REPLY,
|
|
|
|
|
NXT_ROLE_REPLY, "NXT_ROLE_REPLY",
|
|
|
|
|
sizeof(struct nx_role_request), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_NXT_SET_FLOW_FORMAT,
|
|
|
|
|
NXT_SET_FLOW_FORMAT, "NXT_SET_FLOW_FORMAT",
|
|
|
|
|
sizeof(struct nxt_set_flow_format), 0 },
|
|
|
|
|
|
2011-12-09 15:48:26 -08:00
|
|
|
|
{ OFPUTIL_NXT_SET_PACKET_IN_FORMAT,
|
|
|
|
|
NXT_SET_PACKET_IN_FORMAT, "NXT_SET_PACKET_IN_FORMAT",
|
|
|
|
|
sizeof(struct nxt_set_packet_in_format), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_NXT_PACKET_IN,
|
|
|
|
|
NXT_PACKET_IN, "NXT_PACKET_IN",
|
|
|
|
|
sizeof(struct nxt_packet_in), 1 },
|
|
|
|
|
|
2010-12-06 10:20:20 -08:00
|
|
|
|
{ OFPUTIL_NXT_FLOW_MOD,
|
|
|
|
|
NXT_FLOW_MOD, "NXT_FLOW_MOD",
|
|
|
|
|
sizeof(struct nx_flow_mod), 8 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_NXT_FLOW_REMOVED,
|
|
|
|
|
NXT_FLOW_REMOVED, "NXT_FLOW_REMOVED",
|
|
|
|
|
sizeof(struct nx_flow_removed), 8 },
|
2011-06-09 15:18:21 -07:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_NXT_FLOW_MOD_TABLE_ID,
|
|
|
|
|
NXT_FLOW_MOD_TABLE_ID, "NXT_FLOW_MOD_TABLE_ID",
|
|
|
|
|
sizeof(struct nxt_flow_mod_table_id), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const struct ofputil_msg_category nxt_category = {
|
|
|
|
|
"Nicira extension message",
|
|
|
|
|
nxt_messages, ARRAY_SIZE(nxt_messages),
|
|
|
|
|
OFP_MKERR(OFPET_BAD_REQUEST, OFPBRC_BAD_SUBTYPE)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const struct ofp_vendor_header *ovh;
|
|
|
|
|
const struct nicira_header *nh;
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
if (length < sizeof(struct ofp_vendor_header)) {
|
|
|
|
|
if (length == ntohs(oh->length)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "truncated vendor message");
|
|
|
|
|
}
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-06 10:20:20 -08:00
|
|
|
|
ovh = (const struct ofp_vendor_header *) oh;
|
|
|
|
|
if (ovh->vendor != htonl(NX_VENDOR_ID)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "received vendor message for unknown "
|
|
|
|
|
"vendor %"PRIx32, ntohl(ovh->vendor));
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_VENDOR);
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
if (length < sizeof(struct nicira_header)) {
|
|
|
|
|
if (length == ntohs(oh->length)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "received Nicira vendor message of "
|
|
|
|
|
"length %u (expected at least %zu)",
|
|
|
|
|
ntohs(ovh->header.length),
|
|
|
|
|
sizeof(struct nicira_header));
|
|
|
|
|
}
|
2010-12-06 10:20:20 -08:00
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nh = (const struct nicira_header *) oh;
|
|
|
|
|
return ofputil_lookup_openflow_message(&nxt_category, ntohl(nh->subtype),
|
2011-10-27 12:54:44 -07:00
|
|
|
|
typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-10-27 12:54:44 -07:00
|
|
|
|
check_nxstats_msg(const struct ofp_header *oh, size_t length)
|
2010-12-06 10:20:20 -08:00
|
|
|
|
{
|
2011-05-26 09:30:25 -07:00
|
|
|
|
const struct ofp_stats_msg *osm = (const struct ofp_stats_msg *) oh;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
ovs_be32 vendor;
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
if (length < sizeof(struct ofp_vendor_stats_msg)) {
|
|
|
|
|
if (length == ntohs(oh->length)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "truncated vendor stats message");
|
|
|
|
|
}
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-26 09:34:05 -07:00
|
|
|
|
memcpy(&vendor, osm + 1, sizeof vendor);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
if (vendor != htonl(NX_VENDOR_ID)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "received vendor stats message for "
|
|
|
|
|
"unknown vendor %"PRIx32, ntohl(vendor));
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_VENDOR);
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
if (length < sizeof(struct nicira_stats_msg)) {
|
|
|
|
|
if (length == ntohs(osm->header.length)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "truncated Nicira stats message");
|
|
|
|
|
}
|
2010-12-06 10:20:20 -08:00
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ofputil_decode_nxst_request(const struct ofp_header *oh, size_t length,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
static const struct ofputil_msg_type nxst_requests[] = {
|
|
|
|
|
{ OFPUTIL_NXST_FLOW_REQUEST,
|
|
|
|
|
NXST_FLOW, "NXST_FLOW request",
|
|
|
|
|
sizeof(struct nx_flow_stats_request), 8 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_NXST_AGGREGATE_REQUEST,
|
|
|
|
|
NXST_AGGREGATE, "NXST_AGGREGATE request",
|
|
|
|
|
sizeof(struct nx_aggregate_stats_request), 8 },
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const struct ofputil_msg_category nxst_request_category = {
|
2010-12-13 16:20:24 -08:00
|
|
|
|
"Nicira extension statistics request",
|
2010-12-06 10:20:20 -08:00
|
|
|
|
nxst_requests, ARRAY_SIZE(nxst_requests),
|
|
|
|
|
OFP_MKERR(OFPET_BAD_REQUEST, OFPBRC_BAD_SUBTYPE)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const struct nicira_stats_msg *nsm;
|
|
|
|
|
int error;
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = check_nxstats_msg(oh, length);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nsm = (struct nicira_stats_msg *) oh;
|
|
|
|
|
return ofputil_lookup_openflow_message(&nxst_request_category,
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ntohl(nsm->subtype), typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ofputil_decode_nxst_reply(const struct ofp_header *oh, size_t length,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
static const struct ofputil_msg_type nxst_replies[] = {
|
|
|
|
|
{ OFPUTIL_NXST_FLOW_REPLY,
|
|
|
|
|
NXST_FLOW, "NXST_FLOW reply",
|
|
|
|
|
sizeof(struct nicira_stats_msg), 8 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_NXST_AGGREGATE_REPLY,
|
|
|
|
|
NXST_AGGREGATE, "NXST_AGGREGATE reply",
|
|
|
|
|
sizeof(struct nx_aggregate_stats_reply), 0 },
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const struct ofputil_msg_category nxst_reply_category = {
|
2010-12-13 16:20:24 -08:00
|
|
|
|
"Nicira extension statistics reply",
|
2010-12-06 10:20:20 -08:00
|
|
|
|
nxst_replies, ARRAY_SIZE(nxst_replies),
|
|
|
|
|
OFP_MKERR(OFPET_BAD_REQUEST, OFPBRC_BAD_SUBTYPE)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const struct nicira_stats_msg *nsm;
|
|
|
|
|
int error;
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = check_nxstats_msg(oh, length);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nsm = (struct nicira_stats_msg *) oh;
|
|
|
|
|
return ofputil_lookup_openflow_message(&nxst_reply_category,
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ntohl(nsm->subtype), typep);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
check_stats_msg(const struct ofp_header *oh, size_t length)
|
|
|
|
|
{
|
|
|
|
|
if (length < sizeof(struct ofp_stats_msg)) {
|
|
|
|
|
if (length == ntohs(oh->length)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "truncated stats message");
|
|
|
|
|
}
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ofputil_decode_ofpst_request(const struct ofp_header *oh, size_t length,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
static const struct ofputil_msg_type ofpst_requests[] = {
|
|
|
|
|
{ OFPUTIL_OFPST_DESC_REQUEST,
|
|
|
|
|
OFPST_DESC, "OFPST_DESC request",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_FLOW_REQUEST,
|
|
|
|
|
OFPST_FLOW, "OFPST_FLOW request",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_flow_stats_request), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_AGGREGATE_REQUEST,
|
|
|
|
|
OFPST_AGGREGATE, "OFPST_AGGREGATE request",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_flow_stats_request), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_TABLE_REQUEST,
|
|
|
|
|
OFPST_TABLE, "OFPST_TABLE request",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_PORT_REQUEST,
|
|
|
|
|
OFPST_PORT, "OFPST_PORT request",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_port_stats_request), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_QUEUE_REQUEST,
|
|
|
|
|
OFPST_QUEUE, "OFPST_QUEUE request",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_queue_stats_request), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ 0,
|
|
|
|
|
OFPST_VENDOR, "OFPST_VENDOR request",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_vendor_stats_msg), 1 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const struct ofputil_msg_category ofpst_request_category = {
|
|
|
|
|
"OpenFlow statistics",
|
|
|
|
|
ofpst_requests, ARRAY_SIZE(ofpst_requests),
|
|
|
|
|
OFP_MKERR(OFPET_BAD_REQUEST, OFPBRC_BAD_STAT)
|
|
|
|
|
};
|
|
|
|
|
|
2011-05-26 09:30:25 -07:00
|
|
|
|
const struct ofp_stats_msg *request = (const struct ofp_stats_msg *) oh;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
int error;
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = check_stats_msg(oh, length);
|
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-06 10:20:20 -08:00
|
|
|
|
error = ofputil_lookup_openflow_message(&ofpst_request_category,
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ntohs(request->type), typep);
|
2011-05-26 09:30:25 -07:00
|
|
|
|
if (!error && request->type == htons(OFPST_VENDOR)) {
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = ofputil_decode_nxst_request(oh, length, typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ofputil_decode_ofpst_reply(const struct ofp_header *oh, size_t length,
|
2010-12-06 10:20:20 -08:00
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
static const struct ofputil_msg_type ofpst_replies[] = {
|
|
|
|
|
{ OFPUTIL_OFPST_DESC_REPLY,
|
|
|
|
|
OFPST_DESC, "OFPST_DESC reply",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_desc_stats), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_FLOW_REPLY,
|
|
|
|
|
OFPST_FLOW, "OFPST_FLOW reply",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), 1 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_AGGREGATE_REPLY,
|
|
|
|
|
OFPST_AGGREGATE, "OFPST_AGGREGATE reply",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_aggregate_stats_reply), 0 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_TABLE_REPLY,
|
|
|
|
|
OFPST_TABLE, "OFPST_TABLE reply",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), sizeof(struct ofp_table_stats) },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_PORT_REPLY,
|
|
|
|
|
OFPST_PORT, "OFPST_PORT reply",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), sizeof(struct ofp_port_stats) },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPST_QUEUE_REPLY,
|
|
|
|
|
OFPST_QUEUE, "OFPST_QUEUE reply",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), sizeof(struct ofp_queue_stats) },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ 0,
|
|
|
|
|
OFPST_VENDOR, "OFPST_VENDOR reply",
|
2011-05-31 16:49:06 -07:00
|
|
|
|
sizeof(struct ofp_vendor_stats_msg), 1 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const struct ofputil_msg_category ofpst_reply_category = {
|
|
|
|
|
"OpenFlow statistics",
|
|
|
|
|
ofpst_replies, ARRAY_SIZE(ofpst_replies),
|
|
|
|
|
OFP_MKERR(OFPET_BAD_REQUEST, OFPBRC_BAD_STAT)
|
|
|
|
|
};
|
|
|
|
|
|
2011-05-26 09:30:25 -07:00
|
|
|
|
const struct ofp_stats_msg *reply = (const struct ofp_stats_msg *) oh;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
int error;
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = check_stats_msg(oh, length);
|
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-06 10:20:20 -08:00
|
|
|
|
error = ofputil_lookup_openflow_message(&ofpst_reply_category,
|
2011-10-27 12:54:44 -07:00
|
|
|
|
ntohs(reply->type), typep);
|
2011-05-26 09:30:25 -07:00
|
|
|
|
if (!error && reply->type == htons(OFPST_VENDOR)) {
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = ofputil_decode_nxst_reply(oh, length, typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
static int
|
|
|
|
|
ofputil_decode_msg_type__(const struct ofp_header *oh, size_t length,
|
|
|
|
|
const struct ofputil_msg_type **typep)
|
2010-12-06 10:20:20 -08:00
|
|
|
|
{
|
|
|
|
|
static const struct ofputil_msg_type ofpt_messages[] = {
|
|
|
|
|
{ OFPUTIL_OFPT_HELLO,
|
|
|
|
|
OFPT_HELLO, "OFPT_HELLO",
|
|
|
|
|
sizeof(struct ofp_hello), 1 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_ERROR,
|
|
|
|
|
OFPT_ERROR, "OFPT_ERROR",
|
|
|
|
|
sizeof(struct ofp_error_msg), 1 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_ECHO_REQUEST,
|
|
|
|
|
OFPT_ECHO_REQUEST, "OFPT_ECHO_REQUEST",
|
|
|
|
|
sizeof(struct ofp_header), 1 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_ECHO_REPLY,
|
|
|
|
|
OFPT_ECHO_REPLY, "OFPT_ECHO_REPLY",
|
|
|
|
|
sizeof(struct ofp_header), 1 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_FEATURES_REQUEST,
|
|
|
|
|
OFPT_FEATURES_REQUEST, "OFPT_FEATURES_REQUEST",
|
|
|
|
|
sizeof(struct ofp_header), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_FEATURES_REPLY,
|
|
|
|
|
OFPT_FEATURES_REPLY, "OFPT_FEATURES_REPLY",
|
|
|
|
|
sizeof(struct ofp_switch_features), sizeof(struct ofp_phy_port) },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_GET_CONFIG_REQUEST,
|
|
|
|
|
OFPT_GET_CONFIG_REQUEST, "OFPT_GET_CONFIG_REQUEST",
|
|
|
|
|
sizeof(struct ofp_header), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_GET_CONFIG_REPLY,
|
|
|
|
|
OFPT_GET_CONFIG_REPLY, "OFPT_GET_CONFIG_REPLY",
|
|
|
|
|
sizeof(struct ofp_switch_config), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_SET_CONFIG,
|
|
|
|
|
OFPT_SET_CONFIG, "OFPT_SET_CONFIG",
|
|
|
|
|
sizeof(struct ofp_switch_config), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_PACKET_IN,
|
|
|
|
|
OFPT_PACKET_IN, "OFPT_PACKET_IN",
|
|
|
|
|
offsetof(struct ofp_packet_in, data), 1 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_FLOW_REMOVED,
|
|
|
|
|
OFPT_FLOW_REMOVED, "OFPT_FLOW_REMOVED",
|
|
|
|
|
sizeof(struct ofp_flow_removed), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_PORT_STATUS,
|
|
|
|
|
OFPT_PORT_STATUS, "OFPT_PORT_STATUS",
|
|
|
|
|
sizeof(struct ofp_port_status), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_PACKET_OUT,
|
|
|
|
|
OFPT_PACKET_OUT, "OFPT_PACKET_OUT",
|
|
|
|
|
sizeof(struct ofp_packet_out), 1 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_FLOW_MOD,
|
|
|
|
|
OFPT_FLOW_MOD, "OFPT_FLOW_MOD",
|
|
|
|
|
sizeof(struct ofp_flow_mod), 1 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_PORT_MOD,
|
|
|
|
|
OFPT_PORT_MOD, "OFPT_PORT_MOD",
|
|
|
|
|
sizeof(struct ofp_port_mod), 0 },
|
|
|
|
|
|
|
|
|
|
{ 0,
|
|
|
|
|
OFPT_STATS_REQUEST, "OFPT_STATS_REQUEST",
|
2011-05-26 09:30:25 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), 1 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ 0,
|
|
|
|
|
OFPT_STATS_REPLY, "OFPT_STATS_REPLY",
|
2011-05-26 09:30:25 -07:00
|
|
|
|
sizeof(struct ofp_stats_msg), 1 },
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_BARRIER_REQUEST,
|
|
|
|
|
OFPT_BARRIER_REQUEST, "OFPT_BARRIER_REQUEST",
|
|
|
|
|
sizeof(struct ofp_header), 0 },
|
|
|
|
|
|
|
|
|
|
{ OFPUTIL_OFPT_BARRIER_REPLY,
|
|
|
|
|
OFPT_BARRIER_REPLY, "OFPT_BARRIER_REPLY",
|
|
|
|
|
sizeof(struct ofp_header), 0 },
|
|
|
|
|
|
|
|
|
|
{ 0,
|
|
|
|
|
OFPT_VENDOR, "OFPT_VENDOR",
|
|
|
|
|
sizeof(struct ofp_vendor_header), 1 },
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const struct ofputil_msg_category ofpt_category = {
|
|
|
|
|
"OpenFlow message",
|
|
|
|
|
ofpt_messages, ARRAY_SIZE(ofpt_messages),
|
|
|
|
|
OFP_MKERR(OFPET_BAD_REQUEST, OFPBRC_BAD_TYPE)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
int error;
|
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = ofputil_lookup_openflow_message(&ofpt_category, oh->type, typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
if (!error) {
|
|
|
|
|
switch (oh->type) {
|
|
|
|
|
case OFPT_VENDOR:
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = ofputil_decode_vendor(oh, length, typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case OFPT_STATS_REQUEST:
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = ofputil_decode_ofpst_request(oh, length, typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case OFPT_STATS_REPLY:
|
2011-10-27 12:54:44 -07:00
|
|
|
|
error = ofputil_decode_ofpst_reply(oh, length, typep);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-10-27 12:54:44 -07:00
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Decodes the message type represented by 'oh'. Returns 0 if successful or
|
|
|
|
|
* an OpenFlow error code constructed with ofp_mkerr() on failure. Either
|
|
|
|
|
* way, stores in '*typep' a type structure that can be inspected with the
|
|
|
|
|
* ofputil_msg_type_*() functions.
|
|
|
|
|
*
|
|
|
|
|
* oh->length must indicate the correct length of the message (and must be at
|
|
|
|
|
* least sizeof(struct ofp_header)).
|
|
|
|
|
*
|
|
|
|
|
* Success indicates that 'oh' is at least as long as the minimum-length
|
|
|
|
|
* message of its type. */
|
|
|
|
|
int
|
|
|
|
|
ofputil_decode_msg_type(const struct ofp_header *oh,
|
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
size_t length = ntohs(oh->length);
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
error = ofputil_decode_msg_type__(oh, length, typep);
|
|
|
|
|
if (!error) {
|
|
|
|
|
error = ofputil_check_length(*typep, length);
|
|
|
|
|
}
|
2010-12-06 10:20:20 -08:00
|
|
|
|
if (error) {
|
2011-10-27 12:54:44 -07:00
|
|
|
|
*typep = &ofputil_invalid_type;
|
|
|
|
|
}
|
|
|
|
|
return error;
|
|
|
|
|
}
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
2011-10-27 12:54:44 -07:00
|
|
|
|
/* Decodes the message type represented by 'oh', of which only the first
|
|
|
|
|
* 'length' bytes are available. Returns 0 if successful or an OpenFlow error
|
|
|
|
|
* code constructed with ofp_mkerr() on failure. Either way, stores in
|
|
|
|
|
* '*typep' a type structure that can be inspected with the
|
|
|
|
|
* ofputil_msg_type_*() functions. */
|
|
|
|
|
int
|
|
|
|
|
ofputil_decode_msg_type_partial(const struct ofp_header *oh, size_t length,
|
|
|
|
|
const struct ofputil_msg_type **typep)
|
|
|
|
|
{
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
error = (length >= sizeof *oh
|
|
|
|
|
? ofputil_decode_msg_type__(oh, length, typep)
|
|
|
|
|
: ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN));
|
|
|
|
|
if (error) {
|
2010-12-06 10:20:20 -08:00
|
|
|
|
*typep = &ofputil_invalid_type;
|
|
|
|
|
}
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Returns an OFPUTIL_* message type code for 'type'. */
|
|
|
|
|
enum ofputil_msg_code
|
|
|
|
|
ofputil_msg_type_code(const struct ofputil_msg_type *type)
|
|
|
|
|
{
|
|
|
|
|
return type->code;
|
|
|
|
|
}
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
2010-12-07 13:22:46 -08:00
|
|
|
|
/* Flow formats. */
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
ofputil_flow_format_is_valid(enum nx_flow_format flow_format)
|
|
|
|
|
{
|
|
|
|
|
switch (flow_format) {
|
|
|
|
|
case NXFF_OPENFLOW10:
|
|
|
|
|
case NXFF_NXM:
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
|
ofputil_flow_format_to_string(enum nx_flow_format flow_format)
|
|
|
|
|
{
|
|
|
|
|
switch (flow_format) {
|
|
|
|
|
case NXFF_OPENFLOW10:
|
|
|
|
|
return "openflow10";
|
|
|
|
|
case NXFF_NXM:
|
|
|
|
|
return "nxm";
|
|
|
|
|
default:
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-07 13:32:01 -08:00
|
|
|
|
int
|
|
|
|
|
ofputil_flow_format_from_string(const char *s)
|
|
|
|
|
{
|
|
|
|
|
return (!strcmp(s, "openflow10") ? NXFF_OPENFLOW10
|
|
|
|
|
: !strcmp(s, "nxm") ? NXFF_NXM
|
|
|
|
|
: -1);
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-09 15:48:26 -08:00
|
|
|
|
bool
|
|
|
|
|
ofputil_packet_in_format_is_valid(enum nx_packet_in_format packet_in_format)
|
|
|
|
|
{
|
|
|
|
|
switch (packet_in_format) {
|
|
|
|
|
case NXPIF_OPENFLOW10:
|
|
|
|
|
case NXPIF_NXM:
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
|
ofputil_packet_in_format_to_string(enum nx_packet_in_format packet_in_format)
|
|
|
|
|
{
|
|
|
|
|
switch (packet_in_format) {
|
|
|
|
|
case NXPIF_OPENFLOW10:
|
|
|
|
|
return "openflow10";
|
|
|
|
|
case NXPIF_NXM:
|
|
|
|
|
return "nxm";
|
|
|
|
|
default:
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
ofputil_packet_in_format_from_string(const char *s)
|
|
|
|
|
{
|
|
|
|
|
return (!strcmp(s, "openflow10") ? NXPIF_OPENFLOW10
|
|
|
|
|
: !strcmp(s, "nxm") ? NXPIF_NXM
|
|
|
|
|
: -1);
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-07 13:32:01 -08:00
|
|
|
|
static bool
|
|
|
|
|
regs_fully_wildcarded(const struct flow_wildcards *wc)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < FLOW_N_REGS; i++) {
|
|
|
|
|
if (wc->reg_masks[i] != 0) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
/* Returns the minimum nx_flow_format to use for sending 'rule' to a switch
|
|
|
|
|
* (e.g. to add or remove a flow). Only NXM can handle tunnel IDs, registers,
|
|
|
|
|
* or fixing the Ethernet multicast bit. Otherwise, it's better to use
|
|
|
|
|
* NXFF_OPENFLOW10 for backward compatibility. */
|
|
|
|
|
enum nx_flow_format
|
|
|
|
|
ofputil_min_flow_format(const struct cls_rule *rule)
|
2011-01-20 15:29:00 -08:00
|
|
|
|
{
|
|
|
|
|
const struct flow_wildcards *wc = &rule->wc;
|
|
|
|
|
|
2011-11-09 23:39:16 -08:00
|
|
|
|
BUILD_ASSERT_DECL(FLOW_WC_SEQ == 7);
|
2011-07-29 13:15:09 -07:00
|
|
|
|
|
2011-01-20 15:29:00 -08:00
|
|
|
|
/* Only NXM supports separately wildcards the Ethernet multicast bit. */
|
|
|
|
|
if (!(wc->wildcards & FWW_DL_DST) != !(wc->wildcards & FWW_ETH_MCAST)) {
|
2011-04-26 09:42:18 -07:00
|
|
|
|
return NXFF_NXM;
|
2011-01-20 15:29:00 -08:00
|
|
|
|
}
|
|
|
|
|
|
2010-12-07 14:02:17 -08:00
|
|
|
|
/* Only NXM supports matching ARP hardware addresses. */
|
|
|
|
|
if (!(wc->wildcards & FWW_ARP_SHA) || !(wc->wildcards & FWW_ARP_THA)) {
|
2011-04-26 09:42:18 -07:00
|
|
|
|
return NXFF_NXM;
|
2010-12-07 14:02:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
2010-12-29 19:03:46 -08:00
|
|
|
|
/* Only NXM supports matching IPv6 traffic. */
|
|
|
|
|
if (!(wc->wildcards & FWW_DL_TYPE)
|
|
|
|
|
&& (rule->flow.dl_type == htons(ETH_TYPE_IPV6))) {
|
2011-04-26 09:42:18 -07:00
|
|
|
|
return NXFF_NXM;
|
2010-12-29 19:03:46 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-01-20 15:29:00 -08:00
|
|
|
|
/* Only NXM supports matching registers. */
|
|
|
|
|
if (!regs_fully_wildcarded(wc)) {
|
2011-04-26 09:42:18 -07:00
|
|
|
|
return NXFF_NXM;
|
2011-01-20 15:29:00 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
/* Only NXM supports matching tun_id. */
|
|
|
|
|
if (wc->tun_id_mask != htonll(0)) {
|
|
|
|
|
return NXFF_NXM;
|
2011-01-20 15:29:00 -08:00
|
|
|
|
}
|
|
|
|
|
|
Implement new fragment handling policy.
Until now, OVS has handled IP fragments more awkwardly than necessary. It
has not been possible to match on L4 headers, even in fragments with offset
0 where they are actually present. This means that there was no way to
implement ACLs that treat, say, different TCP ports differently, on
fragmented traffic; instead, all decisions for fragment forwarding had to
be made on the basis of L2 and L3 headers alone.
This commit improves the situation significantly. It is still not possible
to match on L4 headers in fragments with nonzero offset, because that
information is simply not present in such fragments, but this commit adds
the ability to match on L4 headers for fragments with zero offset. This
means that it becomes possible to implement ACLs that drop such "first
fragments" on the basis of L4 headers. In practice, that effectively
blocks even fragmented traffic on an L4 basis, because the receiving IP
stack cannot reassemble a full packet when the first fragment is missing.
This commit works by adding a new "fragment type" to the kernel flow match
and making it available through OpenFlow as a new NXM field named
NXM_NX_IP_FRAG. Because OpenFlow 1.0 explicitly says that the L4 fields
are always 0 for IP fragments, it adds a new OpenFlow fragment handling
mode that fills in the L4 fields for "first fragments". It also enhances
ovs-ofctl to allow users to configure this new fragment handling mode and
to parse the new field.
Signed-off-by: Ben Pfaff <blp@nicira.com>
Bug #7557.
2011-10-19 21:33:44 -07:00
|
|
|
|
/* Only NXM supports matching fragments. */
|
2011-11-09 17:10:27 -08:00
|
|
|
|
if (wc->nw_frag_mask) {
|
Implement new fragment handling policy.
Until now, OVS has handled IP fragments more awkwardly than necessary. It
has not been possible to match on L4 headers, even in fragments with offset
0 where they are actually present. This means that there was no way to
implement ACLs that treat, say, different TCP ports differently, on
fragmented traffic; instead, all decisions for fragment forwarding had to
be made on the basis of L2 and L3 headers alone.
This commit improves the situation significantly. It is still not possible
to match on L4 headers in fragments with nonzero offset, because that
information is simply not present in such fragments, but this commit adds
the ability to match on L4 headers for fragments with zero offset. This
means that it becomes possible to implement ACLs that drop such "first
fragments" on the basis of L4 headers. In practice, that effectively
blocks even fragmented traffic on an L4 basis, because the receiving IP
stack cannot reassemble a full packet when the first fragment is missing.
This commit works by adding a new "fragment type" to the kernel flow match
and making it available through OpenFlow as a new NXM field named
NXM_NX_IP_FRAG. Because OpenFlow 1.0 explicitly says that the L4 fields
are always 0 for IP fragments, it adds a new OpenFlow fragment handling
mode that fills in the L4 fields for "first fragments". It also enhances
ovs-ofctl to allow users to configure this new fragment handling mode and
to parse the new field.
Signed-off-by: Ben Pfaff <blp@nicira.com>
Bug #7557.
2011-10-19 21:33:44 -07:00
|
|
|
|
return NXFF_NXM;
|
|
|
|
|
}
|
|
|
|
|
|
2011-11-01 15:57:56 -07:00
|
|
|
|
/* Only NXM supports matching IPv6 flow label. */
|
|
|
|
|
if (!(wc->wildcards & FWW_IPV6_LABEL)) {
|
|
|
|
|
return NXFF_NXM;
|
|
|
|
|
}
|
|
|
|
|
|
2011-11-02 23:34:15 -07:00
|
|
|
|
/* Only NXM supports matching IP ECN bits. */
|
2011-11-09 23:39:16 -08:00
|
|
|
|
if (!(wc->wildcards & FWW_NW_ECN)) {
|
2011-11-02 23:34:15 -07:00
|
|
|
|
return NXFF_NXM;
|
|
|
|
|
}
|
|
|
|
|
|
2011-11-05 15:48:12 -07:00
|
|
|
|
/* Only NXM supports matching IP TTL/hop limit. */
|
|
|
|
|
if (!(wc->wildcards & FWW_NW_TTL)) {
|
|
|
|
|
return NXFF_NXM;
|
|
|
|
|
}
|
|
|
|
|
|
2011-01-20 15:29:00 -08:00
|
|
|
|
/* Other formats can express this rule. */
|
2011-04-26 09:42:18 -07:00
|
|
|
|
return NXFF_OPENFLOW10;
|
2010-12-07 13:32:01 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Returns an OpenFlow message that can be used to set the flow format to
|
|
|
|
|
* 'flow_format'. */
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
ofputil_make_set_flow_format(enum nx_flow_format flow_format)
|
|
|
|
|
{
|
2011-04-26 09:42:18 -07:00
|
|
|
|
struct nxt_set_flow_format *sff;
|
2010-12-07 13:32:01 -08:00
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
sff = make_nxmsg(sizeof *sff, NXT_SET_FLOW_FORMAT, &msg);
|
|
|
|
|
sff->format = htonl(flow_format);
|
2010-12-07 13:32:01 -08:00
|
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-09 15:48:26 -08:00
|
|
|
|
struct ofpbuf *
|
|
|
|
|
ofputil_make_set_packet_in_format(enum nx_packet_in_format packet_in_format)
|
|
|
|
|
{
|
|
|
|
|
struct nxt_set_packet_in_format *spif;
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
|
|
|
|
spif = make_nxmsg(sizeof *spif, NXT_SET_PACKET_IN_FORMAT, &msg);
|
|
|
|
|
spif->format = htonl(packet_in_format);
|
|
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-12 09:58:01 -07:00
|
|
|
|
/* Returns an OpenFlow message that can be used to turn the flow_mod_table_id
|
|
|
|
|
* extension on or off (according to 'flow_mod_table_id'). */
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
ofputil_make_flow_mod_table_id(bool flow_mod_table_id)
|
|
|
|
|
{
|
|
|
|
|
struct nxt_flow_mod_table_id *nfmti;
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
|
|
|
|
nfmti = make_nxmsg(sizeof *nfmti, NXT_FLOW_MOD_TABLE_ID, &msg);
|
|
|
|
|
nfmti->set = flow_mod_table_id;
|
|
|
|
|
return msg;
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-07 13:22:46 -08:00
|
|
|
|
/* Converts an OFPT_FLOW_MOD or NXT_FLOW_MOD message 'oh' into an abstract
|
|
|
|
|
* flow_mod in 'fm'. Returns 0 if successful, otherwise an OpenFlow error
|
|
|
|
|
* code.
|
|
|
|
|
*
|
2011-05-12 09:58:01 -07:00
|
|
|
|
* 'flow_mod_table_id' should be true if the NXT_FLOW_MOD_TABLE_ID extension is
|
|
|
|
|
* enabled, false otherwise.
|
|
|
|
|
*
|
2010-12-07 12:45:24 -08:00
|
|
|
|
* Does not validate the flow_mod actions. */
|
|
|
|
|
int
|
2011-08-08 14:46:38 -07:00
|
|
|
|
ofputil_decode_flow_mod(struct ofputil_flow_mod *fm,
|
|
|
|
|
const struct ofp_header *oh, bool flow_mod_table_id)
|
2010-12-07 12:45:24 -08:00
|
|
|
|
{
|
|
|
|
|
const struct ofputil_msg_type *type;
|
2011-05-12 09:58:01 -07:00
|
|
|
|
uint16_t command;
|
2010-12-07 12:45:24 -08:00
|
|
|
|
struct ofpbuf b;
|
|
|
|
|
|
2010-12-07 15:47:19 -08:00
|
|
|
|
ofpbuf_use_const(&b, oh, ntohs(oh->length));
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
ofputil_decode_msg_type(oh, &type);
|
|
|
|
|
if (ofputil_msg_type_code(type) == OFPUTIL_OFPT_FLOW_MOD) {
|
|
|
|
|
/* Standard OpenFlow flow_mod. */
|
|
|
|
|
const struct ofp_flow_mod *ofm;
|
ofp-util: Revise OpenFlow 1.0 ofp_match normalization.
For a long time, Open vSwitch has "normalized" OpenFlow 1.0 flows in a
funny way: it tries to change fields that are wildcarded into fields
that are exact-match. For example, the normalize_match() function
knows that if dl_type is wildcarded, then all of the L3 and L4 fields
will always be extracted as 0, so it sets those fields to exact-match
and their values to 0.
The reason for this was originally that exact-match flows were much
cheaper for Open vSwitch to implement, because they could be implemented
with a hash table, whereas other kinds of flows had to be implemented
with an expensive linear search. But these days Open vSwitch has a
smarter classifier in which wildcarded flows have minimal cost. Also,
it is no longer possible for OpenFlow 1.0 to specify truly exact-match
flows, because Open vSwitch supports fields for which OpenFlow 1.0
cannot specify values and therefore will always be wildcarded.
Now, it no longer makes sense to do this transformation, so this commit
removes it. Presumably, this will be less surprising for users.
Reviewed-by: Simon Horman <horms@verge.net.au>
2011-05-02 11:04:33 -07:00
|
|
|
|
uint16_t priority;
|
2010-12-07 12:45:24 -08:00
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
/* Dissect the message. */
|
2010-12-07 15:49:36 -08:00
|
|
|
|
ofm = ofpbuf_pull(&b, sizeof *ofm);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
error = ofputil_pull_actions(&b, b.size, &fm->actions, &fm->n_actions);
|
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
ofp-util: Revise OpenFlow 1.0 ofp_match normalization.
For a long time, Open vSwitch has "normalized" OpenFlow 1.0 flows in a
funny way: it tries to change fields that are wildcarded into fields
that are exact-match. For example, the normalize_match() function
knows that if dl_type is wildcarded, then all of the L3 and L4 fields
will always be extracted as 0, so it sets those fields to exact-match
and their values to 0.
The reason for this was originally that exact-match flows were much
cheaper for Open vSwitch to implement, because they could be implemented
with a hash table, whereas other kinds of flows had to be implemented
with an expensive linear search. But these days Open vSwitch has a
smarter classifier in which wildcarded flows have minimal cost. Also,
it is no longer possible for OpenFlow 1.0 to specify truly exact-match
flows, because Open vSwitch supports fields for which OpenFlow 1.0
cannot specify values and therefore will always be wildcarded.
Now, it no longer makes sense to do this transformation, so this commit
removes it. Presumably, this will be less surprising for users.
Reviewed-by: Simon Horman <horms@verge.net.au>
2011-05-02 11:04:33 -07:00
|
|
|
|
/* Set priority based on original wildcards. Normally we'd allow
|
|
|
|
|
* ofputil_cls_rule_from_match() to do this for us, but
|
2011-05-09 16:25:48 -07:00
|
|
|
|
* ofputil_normalize_rule() can put wildcards where the original flow
|
ofp-util: Revise OpenFlow 1.0 ofp_match normalization.
For a long time, Open vSwitch has "normalized" OpenFlow 1.0 flows in a
funny way: it tries to change fields that are wildcarded into fields
that are exact-match. For example, the normalize_match() function
knows that if dl_type is wildcarded, then all of the L3 and L4 fields
will always be extracted as 0, so it sets those fields to exact-match
and their values to 0.
The reason for this was originally that exact-match flows were much
cheaper for Open vSwitch to implement, because they could be implemented
with a hash table, whereas other kinds of flows had to be implemented
with an expensive linear search. But these days Open vSwitch has a
smarter classifier in which wildcarded flows have minimal cost. Also,
it is no longer possible for OpenFlow 1.0 to specify truly exact-match
flows, because Open vSwitch supports fields for which OpenFlow 1.0
cannot specify values and therefore will always be wildcarded.
Now, it no longer makes sense to do this transformation, so this commit
removes it. Presumably, this will be less surprising for users.
Reviewed-by: Simon Horman <horms@verge.net.au>
2011-05-02 11:04:33 -07:00
|
|
|
|
* didn't have them. */
|
|
|
|
|
priority = ntohs(ofm->priority);
|
|
|
|
|
if (!(ofm->match.wildcards & htonl(OFPFW_ALL))) {
|
|
|
|
|
priority = UINT16_MAX;
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-09 16:25:48 -07:00
|
|
|
|
/* Translate the rule. */
|
|
|
|
|
ofputil_cls_rule_from_match(&ofm->match, priority, &fm->cr);
|
|
|
|
|
ofputil_normalize_rule(&fm->cr, NXFF_OPENFLOW10);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
/* Translate the message. */
|
|
|
|
|
fm->cookie = ofm->cookie;
|
2011-12-23 12:23:24 -08:00
|
|
|
|
fm->cookie_mask = htonll(UINT64_MAX);
|
2011-05-12 09:58:01 -07:00
|
|
|
|
command = ntohs(ofm->command);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
fm->idle_timeout = ntohs(ofm->idle_timeout);
|
|
|
|
|
fm->hard_timeout = ntohs(ofm->hard_timeout);
|
|
|
|
|
fm->buffer_id = ntohl(ofm->buffer_id);
|
|
|
|
|
fm->out_port = ntohs(ofm->out_port);
|
|
|
|
|
fm->flags = ntohs(ofm->flags);
|
|
|
|
|
} else if (ofputil_msg_type_code(type) == OFPUTIL_NXT_FLOW_MOD) {
|
|
|
|
|
/* Nicira extended flow_mod. */
|
|
|
|
|
const struct nx_flow_mod *nfm;
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
/* Dissect the message. */
|
2010-12-07 15:49:36 -08:00
|
|
|
|
nfm = ofpbuf_pull(&b, sizeof *nfm);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
error = nx_pull_match(&b, ntohs(nfm->match_len), ntohs(nfm->priority),
|
2011-12-23 12:23:24 -08:00
|
|
|
|
&fm->cr, &fm->cookie, &fm->cookie_mask);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
error = ofputil_pull_actions(&b, b.size, &fm->actions, &fm->n_actions);
|
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Translate the message. */
|
2011-05-12 09:58:01 -07:00
|
|
|
|
command = ntohs(nfm->command);
|
2011-12-23 12:23:24 -08:00
|
|
|
|
if (command == OFPFC_ADD) {
|
|
|
|
|
if (fm->cookie_mask) {
|
|
|
|
|
/* The "NXM_NX_COOKIE*" matches are not valid for flow
|
|
|
|
|
* additions. Additions must use the "cookie" field of
|
|
|
|
|
* the "nx_flow_mod" structure. */
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, NXBRC_NXM_INVALID);
|
|
|
|
|
} else {
|
|
|
|
|
fm->cookie = nfm->cookie;
|
|
|
|
|
fm->cookie_mask = htonll(UINT64_MAX);
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-12-07 12:45:24 -08:00
|
|
|
|
fm->idle_timeout = ntohs(nfm->idle_timeout);
|
|
|
|
|
fm->hard_timeout = ntohs(nfm->hard_timeout);
|
|
|
|
|
fm->buffer_id = ntohl(nfm->buffer_id);
|
|
|
|
|
fm->out_port = ntohs(nfm->out_port);
|
|
|
|
|
fm->flags = ntohs(nfm->flags);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-12 09:58:01 -07:00
|
|
|
|
if (flow_mod_table_id) {
|
|
|
|
|
fm->command = command & 0xff;
|
|
|
|
|
fm->table_id = command >> 8;
|
|
|
|
|
} else {
|
|
|
|
|
fm->command = command;
|
|
|
|
|
fm->table_id = 0xff;
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-07 12:45:24 -08:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Converts 'fm' into an OFPT_FLOW_MOD or NXT_FLOW_MOD message according to
|
2011-05-12 09:58:01 -07:00
|
|
|
|
* 'flow_format' and returns the message.
|
|
|
|
|
*
|
|
|
|
|
* 'flow_mod_table_id' should be true if the NXT_FLOW_MOD_TABLE_ID extension is
|
|
|
|
|
* enabled, false otherwise. */
|
2010-12-07 12:45:24 -08:00
|
|
|
|
struct ofpbuf *
|
2011-08-08 14:46:38 -07:00
|
|
|
|
ofputil_encode_flow_mod(const struct ofputil_flow_mod *fm,
|
2011-05-12 09:58:01 -07:00
|
|
|
|
enum nx_flow_format flow_format,
|
|
|
|
|
bool flow_mod_table_id)
|
2010-12-07 12:45:24 -08:00
|
|
|
|
{
|
|
|
|
|
size_t actions_len = fm->n_actions * sizeof *fm->actions;
|
|
|
|
|
struct ofpbuf *msg;
|
2011-05-12 09:58:01 -07:00
|
|
|
|
uint16_t command;
|
|
|
|
|
|
|
|
|
|
command = (flow_mod_table_id
|
|
|
|
|
? (fm->command & 0xff) | (fm->table_id << 8)
|
|
|
|
|
: fm->command);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
if (flow_format == NXFF_OPENFLOW10) {
|
2010-12-07 12:45:24 -08:00
|
|
|
|
struct ofp_flow_mod *ofm;
|
|
|
|
|
|
|
|
|
|
msg = ofpbuf_new(sizeof *ofm + actions_len);
|
|
|
|
|
ofm = put_openflow(sizeof *ofm, OFPT_FLOW_MOD, msg);
|
2011-04-26 09:42:18 -07:00
|
|
|
|
ofputil_cls_rule_to_match(&fm->cr, &ofm->match);
|
|
|
|
|
ofm->cookie = fm->cookie;
|
2011-06-15 09:43:03 -07:00
|
|
|
|
ofm->command = htons(command);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
ofm->idle_timeout = htons(fm->idle_timeout);
|
|
|
|
|
ofm->hard_timeout = htons(fm->hard_timeout);
|
|
|
|
|
ofm->priority = htons(fm->cr.priority);
|
|
|
|
|
ofm->buffer_id = htonl(fm->buffer_id);
|
|
|
|
|
ofm->out_port = htons(fm->out_port);
|
|
|
|
|
ofm->flags = htons(fm->flags);
|
|
|
|
|
} else if (flow_format == NXFF_NXM) {
|
|
|
|
|
struct nx_flow_mod *nfm;
|
|
|
|
|
int match_len;
|
|
|
|
|
|
|
|
|
|
msg = ofpbuf_new(sizeof *nfm + NXM_TYPICAL_LEN + actions_len);
|
|
|
|
|
put_nxmsg(sizeof *nfm, NXT_FLOW_MOD, msg);
|
|
|
|
|
nfm = msg->data;
|
2011-05-12 09:58:01 -07:00
|
|
|
|
nfm->command = htons(command);
|
2011-12-23 12:23:24 -08:00
|
|
|
|
if (command == OFPFC_ADD) {
|
|
|
|
|
nfm->cookie = fm->cookie;
|
|
|
|
|
match_len = nx_put_match(msg, &fm->cr, 0, 0);
|
|
|
|
|
} else {
|
|
|
|
|
nfm->cookie = 0;
|
|
|
|
|
match_len = nx_put_match(msg, &fm->cr,
|
|
|
|
|
fm->cookie, fm->cookie_mask);
|
|
|
|
|
}
|
2010-12-07 12:45:24 -08:00
|
|
|
|
nfm->idle_timeout = htons(fm->idle_timeout);
|
|
|
|
|
nfm->hard_timeout = htons(fm->hard_timeout);
|
|
|
|
|
nfm->priority = htons(fm->cr.priority);
|
|
|
|
|
nfm->buffer_id = htonl(fm->buffer_id);
|
|
|
|
|
nfm->out_port = htons(fm->out_port);
|
|
|
|
|
nfm->flags = htons(fm->flags);
|
|
|
|
|
nfm->match_len = htons(match_len);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ofpbuf_put(msg, fm->actions, actions_len);
|
|
|
|
|
update_openflow_length(msg);
|
|
|
|
|
return msg;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-08-08 14:48:48 -07:00
|
|
|
|
ofputil_decode_ofpst_flow_request(struct ofputil_flow_stats_request *fsr,
|
2010-12-07 12:45:24 -08:00
|
|
|
|
const struct ofp_header *oh,
|
|
|
|
|
bool aggregate)
|
|
|
|
|
{
|
2011-05-31 16:49:06 -07:00
|
|
|
|
const struct ofp_flow_stats_request *ofsr =
|
|
|
|
|
(const struct ofp_flow_stats_request *) oh;
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
fsr->aggregate = aggregate;
|
2011-04-26 09:42:18 -07:00
|
|
|
|
ofputil_cls_rule_from_match(&ofsr->match, 0, &fsr->match);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
fsr->out_port = ntohs(ofsr->out_port);
|
|
|
|
|
fsr->table_id = ofsr->table_id;
|
2011-12-23 12:23:24 -08:00
|
|
|
|
fsr->cookie = fsr->cookie_mask = htonll(0);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2011-08-08 14:48:48 -07:00
|
|
|
|
ofputil_decode_nxst_flow_request(struct ofputil_flow_stats_request *fsr,
|
2010-12-07 12:45:24 -08:00
|
|
|
|
const struct ofp_header *oh,
|
|
|
|
|
bool aggregate)
|
|
|
|
|
{
|
|
|
|
|
const struct nx_flow_stats_request *nfsr;
|
|
|
|
|
struct ofpbuf b;
|
|
|
|
|
int error;
|
|
|
|
|
|
2010-12-07 15:47:19 -08:00
|
|
|
|
ofpbuf_use_const(&b, oh, ntohs(oh->length));
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
2010-12-07 15:49:36 -08:00
|
|
|
|
nfsr = ofpbuf_pull(&b, sizeof *nfsr);
|
2011-12-23 12:23:24 -08:00
|
|
|
|
error = nx_pull_match(&b, ntohs(nfsr->match_len), 0, &fsr->match,
|
|
|
|
|
&fsr->cookie, &fsr->cookie_mask);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
if (b.size) {
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fsr->aggregate = aggregate;
|
|
|
|
|
fsr->out_port = ntohs(nfsr->out_port);
|
|
|
|
|
fsr->table_id = nfsr->table_id;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Converts an OFPST_FLOW, OFPST_AGGREGATE, NXST_FLOW, or NXST_AGGREGATE
|
2011-04-26 09:42:18 -07:00
|
|
|
|
* request 'oh', into an abstract flow_stats_request in 'fsr'. Returns 0 if
|
|
|
|
|
* successful, otherwise an OpenFlow error code. */
|
2010-12-07 12:45:24 -08:00
|
|
|
|
int
|
2011-08-08 14:48:48 -07:00
|
|
|
|
ofputil_decode_flow_stats_request(struct ofputil_flow_stats_request *fsr,
|
2011-04-26 09:42:18 -07:00
|
|
|
|
const struct ofp_header *oh)
|
2010-12-07 12:45:24 -08:00
|
|
|
|
{
|
|
|
|
|
const struct ofputil_msg_type *type;
|
|
|
|
|
struct ofpbuf b;
|
|
|
|
|
int code;
|
|
|
|
|
|
2010-12-07 15:47:19 -08:00
|
|
|
|
ofpbuf_use_const(&b, oh, ntohs(oh->length));
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
ofputil_decode_msg_type(oh, &type);
|
|
|
|
|
code = ofputil_msg_type_code(type);
|
|
|
|
|
switch (code) {
|
|
|
|
|
case OFPUTIL_OFPST_FLOW_REQUEST:
|
2011-04-26 09:42:18 -07:00
|
|
|
|
return ofputil_decode_ofpst_flow_request(fsr, oh, false);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
case OFPUTIL_OFPST_AGGREGATE_REQUEST:
|
2011-04-26 09:42:18 -07:00
|
|
|
|
return ofputil_decode_ofpst_flow_request(fsr, oh, true);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
case OFPUTIL_NXST_FLOW_REQUEST:
|
|
|
|
|
return ofputil_decode_nxst_flow_request(fsr, oh, false);
|
|
|
|
|
|
|
|
|
|
case OFPUTIL_NXST_AGGREGATE_REQUEST:
|
|
|
|
|
return ofputil_decode_nxst_flow_request(fsr, oh, true);
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
/* Hey, the caller lied. */
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Converts abstract flow_stats_request 'fsr' into an OFPST_FLOW,
|
2011-03-10 15:02:05 -08:00
|
|
|
|
* OFPST_AGGREGATE, NXST_FLOW, or NXST_AGGREGATE request 'oh' according to
|
2010-12-07 12:45:24 -08:00
|
|
|
|
* 'flow_format', and returns the message. */
|
|
|
|
|
struct ofpbuf *
|
2011-08-08 14:48:48 -07:00
|
|
|
|
ofputil_encode_flow_stats_request(const struct ofputil_flow_stats_request *fsr,
|
2010-12-07 12:45:24 -08:00
|
|
|
|
enum nx_flow_format flow_format)
|
|
|
|
|
{
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
if (flow_format == NXFF_OPENFLOW10) {
|
2010-12-07 12:45:24 -08:00
|
|
|
|
struct ofp_flow_stats_request *ofsr;
|
|
|
|
|
int type;
|
|
|
|
|
|
|
|
|
|
type = fsr->aggregate ? OFPST_AGGREGATE : OFPST_FLOW;
|
2011-05-31 16:49:06 -07:00
|
|
|
|
ofsr = ofputil_make_stats_request(sizeof *ofsr, type, 0, &msg);
|
2011-04-26 09:42:18 -07:00
|
|
|
|
ofputil_cls_rule_to_match(&fsr->match, &ofsr->match);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
ofsr->table_id = fsr->table_id;
|
|
|
|
|
ofsr->out_port = htons(fsr->out_port);
|
|
|
|
|
} else if (flow_format == NXFF_NXM) {
|
|
|
|
|
struct nx_flow_stats_request *nfsr;
|
|
|
|
|
int match_len;
|
2010-12-13 16:20:54 -08:00
|
|
|
|
int subtype;
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
2010-12-13 16:20:54 -08:00
|
|
|
|
subtype = fsr->aggregate ? NXST_AGGREGATE : NXST_FLOW;
|
2011-05-31 16:49:06 -07:00
|
|
|
|
ofputil_make_stats_request(sizeof *nfsr, OFPST_VENDOR, subtype, &msg);
|
2011-12-23 12:23:24 -08:00
|
|
|
|
match_len = nx_put_match(msg, &fsr->match,
|
|
|
|
|
fsr->cookie, fsr->cookie_mask);
|
2010-12-07 12:45:24 -08:00
|
|
|
|
|
|
|
|
|
nfsr = msg->data;
|
|
|
|
|
nfsr->out_port = htons(fsr->out_port);
|
|
|
|
|
nfsr->match_len = htons(match_len);
|
|
|
|
|
nfsr->table_id = fsr->table_id;
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
|
}
|
2010-12-06 10:20:20 -08:00
|
|
|
|
|
2011-03-10 15:02:05 -08:00
|
|
|
|
/* Converts an OFPST_FLOW or NXST_FLOW reply in 'msg' into an abstract
|
2011-04-26 09:42:18 -07:00
|
|
|
|
* ofputil_flow_stats in 'fs'.
|
2011-03-10 15:02:05 -08:00
|
|
|
|
*
|
|
|
|
|
* Multiple OFPST_FLOW or NXST_FLOW replies can be packed into a single
|
|
|
|
|
* OpenFlow message. Calling this function multiple times for a single 'msg'
|
|
|
|
|
* iterates through the replies. The caller must initially leave 'msg''s layer
|
|
|
|
|
* pointers null and not modify them between calls.
|
|
|
|
|
*
|
|
|
|
|
* Returns 0 if successful, EOF if no replies were left in this 'msg',
|
|
|
|
|
* otherwise a positive errno value. */
|
|
|
|
|
int
|
|
|
|
|
ofputil_decode_flow_stats_reply(struct ofputil_flow_stats *fs,
|
2011-04-26 09:42:18 -07:00
|
|
|
|
struct ofpbuf *msg)
|
2011-03-10 15:02:05 -08:00
|
|
|
|
{
|
|
|
|
|
const struct ofputil_msg_type *type;
|
|
|
|
|
int code;
|
|
|
|
|
|
|
|
|
|
ofputil_decode_msg_type(msg->l2 ? msg->l2 : msg->data, &type);
|
|
|
|
|
code = ofputil_msg_type_code(type);
|
|
|
|
|
if (!msg->l2) {
|
|
|
|
|
msg->l2 = msg->data;
|
|
|
|
|
if (code == OFPUTIL_OFPST_FLOW_REPLY) {
|
2011-05-26 09:30:25 -07:00
|
|
|
|
ofpbuf_pull(msg, sizeof(struct ofp_stats_msg));
|
2011-03-10 15:02:05 -08:00
|
|
|
|
} else if (code == OFPUTIL_NXST_FLOW_REPLY) {
|
|
|
|
|
ofpbuf_pull(msg, sizeof(struct nicira_stats_msg));
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!msg->size) {
|
|
|
|
|
return EOF;
|
|
|
|
|
} else if (code == OFPUTIL_OFPST_FLOW_REPLY) {
|
|
|
|
|
const struct ofp_flow_stats *ofs;
|
|
|
|
|
size_t length;
|
|
|
|
|
|
|
|
|
|
ofs = ofpbuf_try_pull(msg, sizeof *ofs);
|
|
|
|
|
if (!ofs) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "OFPST_FLOW reply has %zu leftover "
|
|
|
|
|
"bytes at end", msg->size);
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
length = ntohs(ofs->length);
|
|
|
|
|
if (length < sizeof *ofs) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "OFPST_FLOW reply claims invalid "
|
|
|
|
|
"length %zu", length);
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ofputil_pull_actions(msg, length - sizeof *ofs,
|
|
|
|
|
&fs->actions, &fs->n_actions)) {
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fs->cookie = get_32aligned_be64(&ofs->cookie);
|
|
|
|
|
ofputil_cls_rule_from_match(&ofs->match, ntohs(ofs->priority),
|
2011-04-26 09:42:18 -07:00
|
|
|
|
&fs->rule);
|
2011-03-10 15:02:05 -08:00
|
|
|
|
fs->table_id = ofs->table_id;
|
|
|
|
|
fs->duration_sec = ntohl(ofs->duration_sec);
|
|
|
|
|
fs->duration_nsec = ntohl(ofs->duration_nsec);
|
|
|
|
|
fs->idle_timeout = ntohs(ofs->idle_timeout);
|
|
|
|
|
fs->hard_timeout = ntohs(ofs->hard_timeout);
|
|
|
|
|
fs->packet_count = ntohll(get_32aligned_be64(&ofs->packet_count));
|
|
|
|
|
fs->byte_count = ntohll(get_32aligned_be64(&ofs->byte_count));
|
|
|
|
|
} else if (code == OFPUTIL_NXST_FLOW_REPLY) {
|
|
|
|
|
const struct nx_flow_stats *nfs;
|
|
|
|
|
size_t match_len, length;
|
|
|
|
|
|
|
|
|
|
nfs = ofpbuf_try_pull(msg, sizeof *nfs);
|
|
|
|
|
if (!nfs) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "NXST_FLOW reply has %zu leftover "
|
|
|
|
|
"bytes at end", msg->size);
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
length = ntohs(nfs->length);
|
|
|
|
|
match_len = ntohs(nfs->match_len);
|
|
|
|
|
if (length < sizeof *nfs + ROUND_UP(match_len, 8)) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "NXST_FLOW reply with match_len=%zu "
|
|
|
|
|
"claims invalid length %zu", match_len, length);
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
2011-12-23 12:23:24 -08:00
|
|
|
|
if (nx_pull_match(msg, match_len, ntohs(nfs->priority), &fs->rule,
|
|
|
|
|
NULL, NULL)) {
|
2011-03-10 15:02:05 -08:00
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ofputil_pull_actions(msg,
|
|
|
|
|
length - sizeof *nfs - ROUND_UP(match_len, 8),
|
|
|
|
|
&fs->actions, &fs->n_actions)) {
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fs->cookie = nfs->cookie;
|
|
|
|
|
fs->table_id = nfs->table_id;
|
|
|
|
|
fs->duration_sec = ntohl(nfs->duration_sec);
|
|
|
|
|
fs->duration_nsec = ntohl(nfs->duration_nsec);
|
|
|
|
|
fs->idle_timeout = ntohs(nfs->idle_timeout);
|
|
|
|
|
fs->hard_timeout = ntohs(nfs->hard_timeout);
|
|
|
|
|
fs->packet_count = ntohll(nfs->packet_count);
|
|
|
|
|
fs->byte_count = ntohll(nfs->byte_count);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-27 15:16:08 -07:00
|
|
|
|
/* Returns 'count' unchanged except that UINT64_MAX becomes 0.
|
|
|
|
|
*
|
|
|
|
|
* We use this in situations where OVS internally uses UINT64_MAX to mean
|
|
|
|
|
* "value unknown" but OpenFlow 1.0 does not define any unknown value. */
|
|
|
|
|
static uint64_t
|
|
|
|
|
unknown_to_zero(uint64_t count)
|
|
|
|
|
{
|
|
|
|
|
return count != UINT64_MAX ? count : 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-31 16:55:02 -07:00
|
|
|
|
/* Appends an OFPST_FLOW or NXST_FLOW reply that contains the data in 'fs' to
|
|
|
|
|
* those already present in the list of ofpbufs in 'replies'. 'replies' should
|
|
|
|
|
* have been initialized with ofputil_start_stats_reply(). */
|
|
|
|
|
void
|
|
|
|
|
ofputil_append_flow_stats_reply(const struct ofputil_flow_stats *fs,
|
|
|
|
|
struct list *replies)
|
|
|
|
|
{
|
|
|
|
|
size_t act_len = fs->n_actions * sizeof *fs->actions;
|
|
|
|
|
const struct ofp_stats_msg *osm;
|
|
|
|
|
|
|
|
|
|
osm = ofpbuf_from_list(list_back(replies))->data;
|
|
|
|
|
if (osm->type == htons(OFPST_FLOW)) {
|
|
|
|
|
size_t len = offsetof(struct ofp_flow_stats, actions) + act_len;
|
|
|
|
|
struct ofp_flow_stats *ofs;
|
|
|
|
|
|
|
|
|
|
ofs = ofputil_append_stats_reply(len, replies);
|
|
|
|
|
ofs->length = htons(len);
|
|
|
|
|
ofs->table_id = fs->table_id;
|
|
|
|
|
ofs->pad = 0;
|
|
|
|
|
ofputil_cls_rule_to_match(&fs->rule, &ofs->match);
|
|
|
|
|
ofs->duration_sec = htonl(fs->duration_sec);
|
|
|
|
|
ofs->duration_nsec = htonl(fs->duration_nsec);
|
|
|
|
|
ofs->priority = htons(fs->rule.priority);
|
|
|
|
|
ofs->idle_timeout = htons(fs->idle_timeout);
|
|
|
|
|
ofs->hard_timeout = htons(fs->hard_timeout);
|
|
|
|
|
memset(ofs->pad2, 0, sizeof ofs->pad2);
|
|
|
|
|
put_32aligned_be64(&ofs->cookie, fs->cookie);
|
2011-05-27 15:16:08 -07:00
|
|
|
|
put_32aligned_be64(&ofs->packet_count,
|
|
|
|
|
htonll(unknown_to_zero(fs->packet_count)));
|
|
|
|
|
put_32aligned_be64(&ofs->byte_count,
|
|
|
|
|
htonll(unknown_to_zero(fs->byte_count)));
|
2011-05-31 16:55:02 -07:00
|
|
|
|
memcpy(ofs->actions, fs->actions, act_len);
|
|
|
|
|
} else if (osm->type == htons(OFPST_VENDOR)) {
|
|
|
|
|
struct nx_flow_stats *nfs;
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
size_t start_len;
|
|
|
|
|
|
|
|
|
|
msg = ofputil_reserve_stats_reply(
|
|
|
|
|
sizeof *nfs + NXM_MAX_LEN + act_len, replies);
|
|
|
|
|
start_len = msg->size;
|
|
|
|
|
|
|
|
|
|
nfs = ofpbuf_put_uninit(msg, sizeof *nfs);
|
|
|
|
|
nfs->table_id = fs->table_id;
|
|
|
|
|
nfs->pad = 0;
|
|
|
|
|
nfs->duration_sec = htonl(fs->duration_sec);
|
|
|
|
|
nfs->duration_nsec = htonl(fs->duration_nsec);
|
|
|
|
|
nfs->priority = htons(fs->rule.priority);
|
|
|
|
|
nfs->idle_timeout = htons(fs->idle_timeout);
|
|
|
|
|
nfs->hard_timeout = htons(fs->hard_timeout);
|
2011-12-23 12:23:24 -08:00
|
|
|
|
nfs->match_len = htons(nx_put_match(msg, &fs->rule, 0, 0));
|
2011-05-31 16:55:02 -07:00
|
|
|
|
memset(nfs->pad2, 0, sizeof nfs->pad2);
|
|
|
|
|
nfs->cookie = fs->cookie;
|
|
|
|
|
nfs->packet_count = htonll(fs->packet_count);
|
|
|
|
|
nfs->byte_count = htonll(fs->byte_count);
|
|
|
|
|
ofpbuf_put(msg, fs->actions, act_len);
|
|
|
|
|
nfs->length = htons(msg->size - start_len);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-26 16:02:56 -07:00
|
|
|
|
/* Converts abstract ofputil_aggregate_stats 'stats' into an OFPST_AGGREGATE or
|
|
|
|
|
* NXST_AGGREGATE reply according to 'flow_format', and returns the message. */
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
ofputil_encode_aggregate_stats_reply(
|
|
|
|
|
const struct ofputil_aggregate_stats *stats,
|
|
|
|
|
const struct ofp_stats_msg *request)
|
|
|
|
|
{
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
|
|
|
|
if (request->type == htons(OFPST_AGGREGATE)) {
|
|
|
|
|
struct ofp_aggregate_stats_reply *asr;
|
|
|
|
|
|
|
|
|
|
asr = ofputil_make_stats_reply(sizeof *asr, request, &msg);
|
2011-05-27 15:16:08 -07:00
|
|
|
|
put_32aligned_be64(&asr->packet_count,
|
|
|
|
|
htonll(unknown_to_zero(stats->packet_count)));
|
|
|
|
|
put_32aligned_be64(&asr->byte_count,
|
|
|
|
|
htonll(unknown_to_zero(stats->byte_count)));
|
2011-05-26 16:02:56 -07:00
|
|
|
|
asr->flow_count = htonl(stats->flow_count);
|
|
|
|
|
} else if (request->type == htons(OFPST_VENDOR)) {
|
|
|
|
|
struct nx_aggregate_stats_reply *nasr;
|
|
|
|
|
|
|
|
|
|
nasr = ofputil_make_stats_reply(sizeof *nasr, request, &msg);
|
|
|
|
|
assert(nasr->nsm.subtype == htonl(NXST_AGGREGATE));
|
|
|
|
|
nasr->packet_count = htonll(stats->packet_count);
|
|
|
|
|
nasr->byte_count = htonll(stats->byte_count);
|
|
|
|
|
nasr->flow_count = htonl(stats->flow_count);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
/* Converts an OFPT_FLOW_REMOVED or NXT_FLOW_REMOVED message 'oh' into an
|
|
|
|
|
* abstract ofputil_flow_removed in 'fr'. Returns 0 if successful, otherwise
|
|
|
|
|
* an OpenFlow error code. */
|
2010-12-09 10:31:49 -08:00
|
|
|
|
int
|
|
|
|
|
ofputil_decode_flow_removed(struct ofputil_flow_removed *fr,
|
2011-04-26 09:42:18 -07:00
|
|
|
|
const struct ofp_header *oh)
|
2010-12-09 10:31:49 -08:00
|
|
|
|
{
|
|
|
|
|
const struct ofputil_msg_type *type;
|
|
|
|
|
enum ofputil_msg_code code;
|
|
|
|
|
|
|
|
|
|
ofputil_decode_msg_type(oh, &type);
|
|
|
|
|
code = ofputil_msg_type_code(type);
|
|
|
|
|
if (code == OFPUTIL_OFPT_FLOW_REMOVED) {
|
|
|
|
|
const struct ofp_flow_removed *ofr;
|
|
|
|
|
|
|
|
|
|
ofr = (const struct ofp_flow_removed *) oh;
|
|
|
|
|
ofputil_cls_rule_from_match(&ofr->match, ntohs(ofr->priority),
|
2011-04-26 09:42:18 -07:00
|
|
|
|
&fr->rule);
|
2010-12-09 10:31:49 -08:00
|
|
|
|
fr->cookie = ofr->cookie;
|
|
|
|
|
fr->reason = ofr->reason;
|
|
|
|
|
fr->duration_sec = ntohl(ofr->duration_sec);
|
|
|
|
|
fr->duration_nsec = ntohl(ofr->duration_nsec);
|
|
|
|
|
fr->idle_timeout = ntohs(ofr->idle_timeout);
|
|
|
|
|
fr->packet_count = ntohll(ofr->packet_count);
|
|
|
|
|
fr->byte_count = ntohll(ofr->byte_count);
|
|
|
|
|
} else if (code == OFPUTIL_NXT_FLOW_REMOVED) {
|
|
|
|
|
struct nx_flow_removed *nfr;
|
|
|
|
|
struct ofpbuf b;
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
ofpbuf_use_const(&b, oh, ntohs(oh->length));
|
|
|
|
|
|
|
|
|
|
nfr = ofpbuf_pull(&b, sizeof *nfr);
|
|
|
|
|
error = nx_pull_match(&b, ntohs(nfr->match_len), ntohs(nfr->priority),
|
2011-12-23 12:23:24 -08:00
|
|
|
|
&fr->rule, NULL, NULL);
|
2010-12-09 10:31:49 -08:00
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
if (b.size) {
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fr->cookie = nfr->cookie;
|
|
|
|
|
fr->reason = nfr->reason;
|
|
|
|
|
fr->duration_sec = ntohl(nfr->duration_sec);
|
|
|
|
|
fr->duration_nsec = ntohl(nfr->duration_nsec);
|
|
|
|
|
fr->idle_timeout = ntohs(nfr->idle_timeout);
|
|
|
|
|
fr->packet_count = ntohll(nfr->packet_count);
|
|
|
|
|
fr->byte_count = ntohll(nfr->byte_count);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-15 09:49:14 -07:00
|
|
|
|
/* Converts abstract ofputil_flow_removed 'fr' into an OFPT_FLOW_REMOVED or
|
|
|
|
|
* NXT_FLOW_REMOVED message 'oh' according to 'flow_format', and returns the
|
|
|
|
|
* message. */
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
ofputil_encode_flow_removed(const struct ofputil_flow_removed *fr,
|
|
|
|
|
enum nx_flow_format flow_format)
|
|
|
|
|
{
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
2011-04-26 09:42:18 -07:00
|
|
|
|
if (flow_format == NXFF_OPENFLOW10) {
|
2011-03-15 09:49:14 -07:00
|
|
|
|
struct ofp_flow_removed *ofr;
|
|
|
|
|
|
|
|
|
|
ofr = make_openflow_xid(sizeof *ofr, OFPT_FLOW_REMOVED, htonl(0),
|
|
|
|
|
&msg);
|
2011-04-26 09:42:18 -07:00
|
|
|
|
ofputil_cls_rule_to_match(&fr->rule, &ofr->match);
|
2011-05-20 16:42:40 -07:00
|
|
|
|
ofr->cookie = fr->cookie;
|
2011-03-15 09:49:14 -07:00
|
|
|
|
ofr->priority = htons(fr->rule.priority);
|
|
|
|
|
ofr->reason = fr->reason;
|
|
|
|
|
ofr->duration_sec = htonl(fr->duration_sec);
|
|
|
|
|
ofr->duration_nsec = htonl(fr->duration_nsec);
|
|
|
|
|
ofr->idle_timeout = htons(fr->idle_timeout);
|
2011-05-27 15:16:08 -07:00
|
|
|
|
ofr->packet_count = htonll(unknown_to_zero(fr->packet_count));
|
|
|
|
|
ofr->byte_count = htonll(unknown_to_zero(fr->byte_count));
|
2011-03-15 09:49:14 -07:00
|
|
|
|
} else if (flow_format == NXFF_NXM) {
|
|
|
|
|
struct nx_flow_removed *nfr;
|
|
|
|
|
int match_len;
|
|
|
|
|
|
|
|
|
|
make_nxmsg_xid(sizeof *nfr, NXT_FLOW_REMOVED, htonl(0), &msg);
|
2011-12-23 12:23:24 -08:00
|
|
|
|
match_len = nx_put_match(msg, &fr->rule, 0, 0);
|
2011-03-15 09:49:14 -07:00
|
|
|
|
|
|
|
|
|
nfr = msg->data;
|
|
|
|
|
nfr->cookie = fr->cookie;
|
|
|
|
|
nfr->priority = htons(fr->rule.priority);
|
|
|
|
|
nfr->reason = fr->reason;
|
|
|
|
|
nfr->duration_sec = htonl(fr->duration_sec);
|
|
|
|
|
nfr->duration_nsec = htonl(fr->duration_nsec);
|
|
|
|
|
nfr->idle_timeout = htons(fr->idle_timeout);
|
|
|
|
|
nfr->match_len = htons(match_len);
|
|
|
|
|
nfr->packet_count = htonll(fr->packet_count);
|
|
|
|
|
nfr->byte_count = htonll(fr->byte_count);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-22 16:35:23 -08:00
|
|
|
|
int
|
|
|
|
|
ofputil_decode_packet_in(struct ofputil_packet_in *pin,
|
|
|
|
|
const struct ofp_header *oh)
|
|
|
|
|
{
|
|
|
|
|
const struct ofputil_msg_type *type;
|
|
|
|
|
enum ofputil_msg_code code;
|
|
|
|
|
|
|
|
|
|
ofputil_decode_msg_type(oh, &type);
|
|
|
|
|
code = ofputil_msg_type_code(type);
|
|
|
|
|
memset(pin, 0, sizeof *pin);
|
|
|
|
|
|
|
|
|
|
if (code == OFPUTIL_OFPT_PACKET_IN) {
|
|
|
|
|
const struct ofp_packet_in *opi = (const struct ofp_packet_in *) oh;
|
|
|
|
|
|
|
|
|
|
pin->packet = opi->data;
|
|
|
|
|
pin->packet_len = ntohs(opi->header.length)
|
|
|
|
|
- offsetof(struct ofp_packet_in, data);
|
|
|
|
|
|
2012-01-04 16:40:13 -08:00
|
|
|
|
pin->fmd.in_port = ntohs(opi->in_port);
|
2011-12-22 16:35:23 -08:00
|
|
|
|
pin->reason = opi->reason;
|
|
|
|
|
pin->buffer_id = ntohl(opi->buffer_id);
|
|
|
|
|
pin->total_len = ntohs(opi->total_len);
|
2011-12-09 15:48:26 -08:00
|
|
|
|
} else if (code == OFPUTIL_NXT_PACKET_IN) {
|
|
|
|
|
const struct nxt_packet_in *npi;
|
|
|
|
|
struct cls_rule rule;
|
|
|
|
|
struct ofpbuf b;
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
ofpbuf_use_const(&b, oh, ntohs(oh->length));
|
|
|
|
|
|
|
|
|
|
npi = ofpbuf_pull(&b, sizeof *npi);
|
|
|
|
|
error = nx_pull_match_loose(&b, ntohs(npi->match_len), 0, &rule, NULL,
|
|
|
|
|
NULL);
|
|
|
|
|
if (error) {
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!ofpbuf_try_pull(&b, 2)) {
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pin->packet = b.data;
|
|
|
|
|
pin->packet_len = b.size;
|
|
|
|
|
pin->reason = npi->reason;
|
|
|
|
|
pin->table_id = npi->table_id;
|
|
|
|
|
pin->cookie = npi->cookie;
|
|
|
|
|
|
|
|
|
|
pin->fmd.in_port = rule.flow.in_port;
|
|
|
|
|
|
|
|
|
|
pin->fmd.tun_id = rule.flow.tun_id;
|
|
|
|
|
pin->fmd.tun_id_mask = rule.wc.tun_id_mask;
|
|
|
|
|
|
|
|
|
|
memcpy(pin->fmd.regs, rule.flow.regs, sizeof pin->fmd.regs);
|
|
|
|
|
memcpy(pin->fmd.reg_masks, rule.wc.reg_masks,
|
|
|
|
|
sizeof pin->fmd.reg_masks);
|
|
|
|
|
|
|
|
|
|
pin->buffer_id = ntohl(npi->buffer_id);
|
|
|
|
|
pin->total_len = ntohs(npi->total_len);
|
2011-12-22 16:35:23 -08:00
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-09 15:48:26 -08:00
|
|
|
|
/* Converts abstract ofputil_packet_in 'pin' into a PACKET_IN message
|
|
|
|
|
* in the format specified by 'packet_in_format'. */
|
2011-03-22 09:13:02 -07:00
|
|
|
|
struct ofpbuf *
|
2011-12-09 15:48:26 -08:00
|
|
|
|
ofputil_encode_packet_in(const struct ofputil_packet_in *pin,
|
|
|
|
|
enum nx_packet_in_format packet_in_format)
|
2011-03-22 09:13:02 -07:00
|
|
|
|
{
|
2011-12-09 15:48:26 -08:00
|
|
|
|
size_t send_len = MIN(pin->send_len, pin->packet_len);
|
|
|
|
|
struct ofpbuf *packet;
|
2011-03-22 09:13:02 -07:00
|
|
|
|
|
|
|
|
|
/* Add OFPT_PACKET_IN. */
|
2011-12-09 15:48:26 -08:00
|
|
|
|
if (packet_in_format == NXPIF_OPENFLOW10) {
|
|
|
|
|
size_t header_len = offsetof(struct ofp_packet_in, data);
|
|
|
|
|
struct ofp_packet_in *opi;
|
|
|
|
|
|
|
|
|
|
packet = ofpbuf_new(send_len + header_len);
|
|
|
|
|
opi = ofpbuf_put_zeros(packet, header_len);
|
|
|
|
|
opi->header.version = OFP_VERSION;
|
|
|
|
|
opi->header.type = OFPT_PACKET_IN;
|
|
|
|
|
opi->total_len = htons(pin->total_len);
|
|
|
|
|
opi->in_port = htons(pin->fmd.in_port);
|
|
|
|
|
opi->reason = pin->reason;
|
|
|
|
|
opi->buffer_id = htonl(pin->buffer_id);
|
|
|
|
|
|
|
|
|
|
ofpbuf_put(packet, pin->packet, send_len);
|
|
|
|
|
} else if (packet_in_format == NXPIF_NXM) {
|
|
|
|
|
struct nxt_packet_in *npi;
|
|
|
|
|
struct cls_rule rule;
|
|
|
|
|
size_t match_len;
|
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
|
|
/* Estimate of required PACKET_IN length includes the NPI header, space
|
|
|
|
|
* for the match (2 times sizeof the metadata seems like enough), 2
|
|
|
|
|
* bytes for padding, and the packet length. */
|
|
|
|
|
packet = ofpbuf_new(sizeof *npi + sizeof(struct flow_metadata) * 2
|
|
|
|
|
+ 2 + send_len);
|
|
|
|
|
|
|
|
|
|
cls_rule_init_catchall(&rule, 0);
|
|
|
|
|
cls_rule_set_tun_id_masked(&rule, pin->fmd.tun_id,
|
|
|
|
|
pin->fmd.tun_id_mask);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < FLOW_N_REGS; i++) {
|
|
|
|
|
cls_rule_set_reg_masked(&rule, i, pin->fmd.regs[i],
|
|
|
|
|
pin->fmd.reg_masks[i]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cls_rule_set_in_port(&rule, pin->fmd.in_port);
|
|
|
|
|
|
|
|
|
|
ofpbuf_put_zeros(packet, sizeof *npi);
|
|
|
|
|
match_len = nx_put_match(packet, &rule, 0, 0);
|
|
|
|
|
ofpbuf_put_zeros(packet, 2);
|
|
|
|
|
ofpbuf_put(packet, pin->packet, send_len);
|
|
|
|
|
|
|
|
|
|
npi = packet->data;
|
|
|
|
|
npi->nxh.header.version = OFP_VERSION;
|
|
|
|
|
npi->nxh.header.type = OFPT_VENDOR;
|
|
|
|
|
npi->nxh.vendor = htonl(NX_VENDOR_ID);
|
|
|
|
|
npi->nxh.subtype = htonl(NXT_PACKET_IN);
|
|
|
|
|
|
|
|
|
|
npi->buffer_id = htonl(pin->buffer_id);
|
|
|
|
|
npi->total_len = htons(pin->total_len);
|
|
|
|
|
npi->reason = pin->reason;
|
|
|
|
|
npi->table_id = pin->table_id;
|
|
|
|
|
npi->cookie = pin->cookie;
|
|
|
|
|
npi->match_len = htons(match_len);
|
|
|
|
|
} else {
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
update_openflow_length(packet);
|
|
|
|
|
|
|
|
|
|
return packet;
|
2011-03-22 09:13:02 -07:00
|
|
|
|
}
|
|
|
|
|
|
2010-12-06 10:20:20 -08:00
|
|
|
|
/* Returns a string representing the message type of 'type'. The string is the
|
|
|
|
|
* enumeration constant for the type, e.g. "OFPT_HELLO". For statistics
|
|
|
|
|
* messages, the constant is followed by "request" or "reply",
|
|
|
|
|
* e.g. "OFPST_AGGREGATE reply". */
|
|
|
|
|
const char *
|
|
|
|
|
ofputil_msg_type_name(const struct ofputil_msg_type *type)
|
|
|
|
|
{
|
|
|
|
|
return type->name;
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-27 13:14:05 -07:00
|
|
|
|
/* Allocates and stores in '*bufferp' a new ofpbuf with a size of
|
|
|
|
|
* 'openflow_len', starting with an OpenFlow header with the given 'type' and
|
|
|
|
|
* an arbitrary transaction id. Allocated bytes beyond the header, if any, are
|
|
|
|
|
* zeroed.
|
|
|
|
|
*
|
|
|
|
|
* The caller is responsible for freeing '*bufferp' when it is no longer
|
|
|
|
|
* needed.
|
|
|
|
|
*
|
|
|
|
|
* The OpenFlow header length is initially set to 'openflow_len'; if the
|
|
|
|
|
* message is later extended, the length should be updated with
|
|
|
|
|
* update_openflow_length() before sending.
|
|
|
|
|
*
|
|
|
|
|
* Returns the header. */
|
|
|
|
|
void *
|
|
|
|
|
make_openflow(size_t openflow_len, uint8_t type, struct ofpbuf **bufferp)
|
|
|
|
|
{
|
|
|
|
|
*bufferp = ofpbuf_new(openflow_len);
|
|
|
|
|
return put_openflow_xid(openflow_len, type, alloc_xid(), *bufferp);
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-18 16:25:52 -07:00
|
|
|
|
/* Similar to make_openflow() but creates a Nicira vendor extension message
|
|
|
|
|
* with the specific 'subtype'. 'subtype' should be in host byte order. */
|
|
|
|
|
void *
|
|
|
|
|
make_nxmsg(size_t openflow_len, uint32_t subtype, struct ofpbuf **bufferp)
|
|
|
|
|
{
|
|
|
|
|
return make_nxmsg_xid(openflow_len, subtype, alloc_xid(), bufferp);
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-27 13:14:05 -07:00
|
|
|
|
/* Allocates and stores in '*bufferp' a new ofpbuf with a size of
|
|
|
|
|
* 'openflow_len', starting with an OpenFlow header with the given 'type' and
|
|
|
|
|
* transaction id 'xid'. Allocated bytes beyond the header, if any, are
|
|
|
|
|
* zeroed.
|
|
|
|
|
*
|
|
|
|
|
* The caller is responsible for freeing '*bufferp' when it is no longer
|
|
|
|
|
* needed.
|
|
|
|
|
*
|
|
|
|
|
* The OpenFlow header length is initially set to 'openflow_len'; if the
|
|
|
|
|
* message is later extended, the length should be updated with
|
|
|
|
|
* update_openflow_length() before sending.
|
|
|
|
|
*
|
|
|
|
|
* Returns the header. */
|
|
|
|
|
void *
|
2010-11-16 11:00:25 -08:00
|
|
|
|
make_openflow_xid(size_t openflow_len, uint8_t type, ovs_be32 xid,
|
2010-05-27 13:14:05 -07:00
|
|
|
|
struct ofpbuf **bufferp)
|
|
|
|
|
{
|
|
|
|
|
*bufferp = ofpbuf_new(openflow_len);
|
|
|
|
|
return put_openflow_xid(openflow_len, type, xid, *bufferp);
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-18 16:25:52 -07:00
|
|
|
|
/* Similar to make_openflow_xid() but creates a Nicira vendor extension message
|
|
|
|
|
* with the specific 'subtype'. 'subtype' should be in host byte order. */
|
|
|
|
|
void *
|
2010-11-16 11:00:25 -08:00
|
|
|
|
make_nxmsg_xid(size_t openflow_len, uint32_t subtype, ovs_be32 xid,
|
2010-10-18 16:25:52 -07:00
|
|
|
|
struct ofpbuf **bufferp)
|
|
|
|
|
{
|
2010-11-23 13:31:07 -08:00
|
|
|
|
*bufferp = ofpbuf_new(openflow_len);
|
|
|
|
|
return put_nxmsg_xid(openflow_len, subtype, xid, *bufferp);
|
2010-10-18 16:25:52 -07:00
|
|
|
|
}
|
|
|
|
|
|
2010-05-27 13:14:05 -07:00
|
|
|
|
/* Appends 'openflow_len' bytes to 'buffer', starting with an OpenFlow header
|
|
|
|
|
* with the given 'type' and an arbitrary transaction id. Allocated bytes
|
|
|
|
|
* beyond the header, if any, are zeroed.
|
|
|
|
|
*
|
|
|
|
|
* The OpenFlow header length is initially set to 'openflow_len'; if the
|
|
|
|
|
* message is later extended, the length should be updated with
|
|
|
|
|
* update_openflow_length() before sending.
|
|
|
|
|
*
|
|
|
|
|
* Returns the header. */
|
|
|
|
|
void *
|
|
|
|
|
put_openflow(size_t openflow_len, uint8_t type, struct ofpbuf *buffer)
|
|
|
|
|
{
|
|
|
|
|
return put_openflow_xid(openflow_len, type, alloc_xid(), buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Appends 'openflow_len' bytes to 'buffer', starting with an OpenFlow header
|
|
|
|
|
* with the given 'type' and an transaction id 'xid'. Allocated bytes beyond
|
|
|
|
|
* the header, if any, are zeroed.
|
|
|
|
|
*
|
|
|
|
|
* The OpenFlow header length is initially set to 'openflow_len'; if the
|
|
|
|
|
* message is later extended, the length should be updated with
|
|
|
|
|
* update_openflow_length() before sending.
|
|
|
|
|
*
|
|
|
|
|
* Returns the header. */
|
|
|
|
|
void *
|
2010-11-16 11:00:25 -08:00
|
|
|
|
put_openflow_xid(size_t openflow_len, uint8_t type, ovs_be32 xid,
|
2010-05-27 13:14:05 -07:00
|
|
|
|
struct ofpbuf *buffer)
|
|
|
|
|
{
|
|
|
|
|
struct ofp_header *oh;
|
|
|
|
|
|
|
|
|
|
assert(openflow_len >= sizeof *oh);
|
|
|
|
|
assert(openflow_len <= UINT16_MAX);
|
|
|
|
|
|
|
|
|
|
oh = ofpbuf_put_uninit(buffer, openflow_len);
|
|
|
|
|
oh->version = OFP_VERSION;
|
|
|
|
|
oh->type = type;
|
|
|
|
|
oh->length = htons(openflow_len);
|
|
|
|
|
oh->xid = xid;
|
|
|
|
|
memset(oh + 1, 0, openflow_len - sizeof *oh);
|
|
|
|
|
return oh;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-23 13:31:07 -08:00
|
|
|
|
/* Similar to put_openflow() but append a Nicira vendor extension message with
|
|
|
|
|
* the specific 'subtype'. 'subtype' should be in host byte order. */
|
|
|
|
|
void *
|
|
|
|
|
put_nxmsg(size_t openflow_len, uint32_t subtype, struct ofpbuf *buffer)
|
|
|
|
|
{
|
|
|
|
|
return put_nxmsg_xid(openflow_len, subtype, alloc_xid(), buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Similar to put_openflow_xid() but append a Nicira vendor extension message
|
|
|
|
|
* with the specific 'subtype'. 'subtype' should be in host byte order. */
|
|
|
|
|
void *
|
|
|
|
|
put_nxmsg_xid(size_t openflow_len, uint32_t subtype, ovs_be32 xid,
|
|
|
|
|
struct ofpbuf *buffer)
|
|
|
|
|
{
|
|
|
|
|
struct nicira_header *nxh;
|
|
|
|
|
|
|
|
|
|
nxh = put_openflow_xid(openflow_len, OFPT_VENDOR, xid, buffer);
|
|
|
|
|
nxh->vendor = htonl(NX_VENDOR_ID);
|
|
|
|
|
nxh->subtype = htonl(subtype);
|
|
|
|
|
return nxh;
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-27 13:14:05 -07:00
|
|
|
|
/* Updates the 'length' field of the OpenFlow message in 'buffer' to
|
|
|
|
|
* 'buffer->size'. */
|
|
|
|
|
void
|
2010-08-30 00:24:53 -07:00
|
|
|
|
update_openflow_length(struct ofpbuf *buffer)
|
2010-05-27 13:14:05 -07:00
|
|
|
|
{
|
|
|
|
|
struct ofp_header *oh = ofpbuf_at_assert(buffer, 0, sizeof *oh);
|
2010-08-30 00:24:53 -07:00
|
|
|
|
oh->length = htons(buffer->size);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
2011-05-31 16:49:06 -07:00
|
|
|
|
static void
|
|
|
|
|
put_stats__(ovs_be32 xid, uint8_t ofp_type,
|
|
|
|
|
ovs_be16 ofpst_type, ovs_be32 nxst_subtype,
|
|
|
|
|
struct ofpbuf *msg)
|
|
|
|
|
{
|
|
|
|
|
if (ofpst_type == htons(OFPST_VENDOR)) {
|
|
|
|
|
struct nicira_stats_msg *nsm;
|
|
|
|
|
|
|
|
|
|
nsm = put_openflow_xid(sizeof *nsm, ofp_type, xid, msg);
|
|
|
|
|
nsm->vsm.osm.type = ofpst_type;
|
|
|
|
|
nsm->vsm.vendor = htonl(NX_VENDOR_ID);
|
|
|
|
|
nsm->subtype = nxst_subtype;
|
|
|
|
|
} else {
|
|
|
|
|
struct ofp_stats_msg *osm;
|
|
|
|
|
|
|
|
|
|
osm = put_openflow_xid(sizeof *osm, ofp_type, xid, msg);
|
|
|
|
|
osm->type = ofpst_type;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Creates a statistics request message with total length 'openflow_len'
|
|
|
|
|
* (including all headers) and the given 'ofpst_type', and stores the buffer
|
|
|
|
|
* containing the new message in '*bufferp'. If 'ofpst_type' is OFPST_VENDOR
|
|
|
|
|
* then 'nxst_subtype' is used as the Nicira vendor extension statistics
|
|
|
|
|
* subtype (otherwise 'nxst_subtype' is ignored).
|
|
|
|
|
*
|
|
|
|
|
* Initializes bytes following the headers to all-bits-zero.
|
|
|
|
|
*
|
|
|
|
|
* Returns the first byte of the new message. */
|
2010-11-23 13:31:07 -08:00
|
|
|
|
void *
|
2011-05-31 16:49:06 -07:00
|
|
|
|
ofputil_make_stats_request(size_t openflow_len, uint16_t ofpst_type,
|
|
|
|
|
uint32_t nxst_subtype, struct ofpbuf **bufferp)
|
2010-11-23 13:31:07 -08:00
|
|
|
|
{
|
2011-05-31 16:49:06 -07:00
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
|
|
|
|
msg = *bufferp = ofpbuf_new(openflow_len);
|
|
|
|
|
put_stats__(alloc_xid(), OFPT_STATS_REQUEST,
|
|
|
|
|
htons(ofpst_type), htonl(nxst_subtype), msg);
|
|
|
|
|
ofpbuf_padto(msg, openflow_len);
|
|
|
|
|
|
|
|
|
|
return msg->data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
put_stats_reply__(const struct ofp_stats_msg *request, struct ofpbuf *msg)
|
|
|
|
|
{
|
|
|
|
|
assert(request->header.type == OFPT_STATS_REQUEST ||
|
|
|
|
|
request->header.type == OFPT_STATS_REPLY);
|
|
|
|
|
put_stats__(request->header.xid, OFPT_STATS_REPLY, request->type,
|
|
|
|
|
(request->type != htons(OFPST_VENDOR)
|
|
|
|
|
? htonl(0)
|
|
|
|
|
: ((const struct nicira_stats_msg *) request)->subtype),
|
|
|
|
|
msg);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Creates a statistics reply message with total length 'openflow_len'
|
|
|
|
|
* (including all headers) and the same type (either a standard OpenFlow
|
|
|
|
|
* statistics type or a Nicira extension type and subtype) as 'request', and
|
|
|
|
|
* stores the buffer containing the new message in '*bufferp'.
|
|
|
|
|
*
|
|
|
|
|
* Initializes bytes following the headers to all-bits-zero.
|
|
|
|
|
*
|
|
|
|
|
* Returns the first byte of the new message. */
|
|
|
|
|
void *
|
|
|
|
|
ofputil_make_stats_reply(size_t openflow_len,
|
|
|
|
|
const struct ofp_stats_msg *request,
|
|
|
|
|
struct ofpbuf **bufferp)
|
|
|
|
|
{
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
|
|
|
|
msg = *bufferp = ofpbuf_new(openflow_len);
|
|
|
|
|
put_stats_reply__(request, msg);
|
|
|
|
|
ofpbuf_padto(msg, openflow_len);
|
|
|
|
|
|
|
|
|
|
return msg->data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Initializes 'replies' as a list of ofpbufs that will contain a series of
|
|
|
|
|
* replies to 'request', which should be an OpenFlow or Nicira extension
|
|
|
|
|
* statistics request. Initially 'replies' will have a single reply message
|
|
|
|
|
* that has only a header. The functions ofputil_reserve_stats_reply() and
|
|
|
|
|
* ofputil_append_stats_reply() may be used to add to the reply. */
|
|
|
|
|
void
|
|
|
|
|
ofputil_start_stats_reply(const struct ofp_stats_msg *request,
|
|
|
|
|
struct list *replies)
|
|
|
|
|
{
|
|
|
|
|
struct ofpbuf *msg;
|
|
|
|
|
|
|
|
|
|
msg = ofpbuf_new(1024);
|
|
|
|
|
put_stats_reply__(request, msg);
|
|
|
|
|
|
|
|
|
|
list_init(replies);
|
|
|
|
|
list_push_back(replies, &msg->list_node);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Prepares to append up to 'len' bytes to the series of statistics replies in
|
|
|
|
|
* 'replies', which should have been initialized with
|
|
|
|
|
* ofputil_start_stats_reply(). Returns an ofpbuf with at least 'len' bytes of
|
|
|
|
|
* tailroom. (The 'len' bytes have not actually be allocated; the caller must
|
|
|
|
|
* do so with e.g. ofpbuf_put_uninit().) */
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
ofputil_reserve_stats_reply(size_t len, struct list *replies)
|
|
|
|
|
{
|
|
|
|
|
struct ofpbuf *msg = ofpbuf_from_list(list_back(replies));
|
|
|
|
|
struct ofp_stats_msg *osm = msg->data;
|
|
|
|
|
|
|
|
|
|
if (msg->size + len <= UINT16_MAX) {
|
|
|
|
|
ofpbuf_prealloc_tailroom(msg, len);
|
|
|
|
|
} else {
|
|
|
|
|
osm->flags |= htons(OFPSF_REPLY_MORE);
|
|
|
|
|
|
|
|
|
|
msg = ofpbuf_new(MAX(1024, sizeof(struct nicira_stats_msg) + len));
|
|
|
|
|
put_stats_reply__(osm, msg);
|
|
|
|
|
list_push_back(replies, &msg->list_node);
|
|
|
|
|
}
|
|
|
|
|
return msg;
|
2010-11-23 13:31:07 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-05-31 16:49:06 -07:00
|
|
|
|
/* Appends 'len' bytes to the series of statistics replies in 'replies', and
|
|
|
|
|
* returns the first byte. */
|
2010-11-23 13:31:07 -08:00
|
|
|
|
void *
|
2011-05-31 16:49:06 -07:00
|
|
|
|
ofputil_append_stats_reply(size_t len, struct list *replies)
|
2010-11-23 13:31:07 -08:00
|
|
|
|
{
|
2011-05-31 16:49:06 -07:00
|
|
|
|
return ofpbuf_put_uninit(ofputil_reserve_stats_reply(len, replies), len);
|
2010-11-23 13:31:07 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-05-26 09:34:05 -07:00
|
|
|
|
/* Returns the first byte past the ofp_stats_msg header in 'oh'. */
|
2010-12-06 10:20:20 -08:00
|
|
|
|
const void *
|
|
|
|
|
ofputil_stats_body(const struct ofp_header *oh)
|
|
|
|
|
{
|
|
|
|
|
assert(oh->type == OFPT_STATS_REQUEST || oh->type == OFPT_STATS_REPLY);
|
2011-05-26 09:30:25 -07:00
|
|
|
|
return (const struct ofp_stats_msg *) oh + 1;
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-05-26 09:34:05 -07:00
|
|
|
|
/* Returns the number of bytes past the ofp_stats_msg header in 'oh'. */
|
2010-12-06 10:20:20 -08:00
|
|
|
|
size_t
|
|
|
|
|
ofputil_stats_body_len(const struct ofp_header *oh)
|
|
|
|
|
{
|
|
|
|
|
assert(oh->type == OFPT_STATS_REQUEST || oh->type == OFPT_STATS_REPLY);
|
2011-05-26 09:30:25 -07:00
|
|
|
|
return ntohs(oh->length) - sizeof(struct ofp_stats_msg);
|
2010-12-06 10:20:20 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-05-26 09:34:05 -07:00
|
|
|
|
/* Returns the first byte past the nicira_stats_msg header in 'oh'. */
|
2010-12-07 14:21:38 -08:00
|
|
|
|
const void *
|
|
|
|
|
ofputil_nxstats_body(const struct ofp_header *oh)
|
|
|
|
|
{
|
|
|
|
|
assert(oh->type == OFPT_STATS_REQUEST || oh->type == OFPT_STATS_REPLY);
|
|
|
|
|
return ((const struct nicira_stats_msg *) oh) + 1;
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-26 09:34:05 -07:00
|
|
|
|
/* Returns the number of bytes past the nicira_stats_msg header in 'oh'. */
|
2010-12-07 14:21:38 -08:00
|
|
|
|
size_t
|
|
|
|
|
ofputil_nxstats_body_len(const struct ofp_header *oh)
|
|
|
|
|
{
|
|
|
|
|
assert(oh->type == OFPT_STATS_REQUEST || oh->type == OFPT_STATS_REPLY);
|
|
|
|
|
return ntohs(oh->length) - sizeof(struct nicira_stats_msg);
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-27 13:14:05 -07:00
|
|
|
|
struct ofpbuf *
|
2010-11-10 14:51:49 -08:00
|
|
|
|
make_flow_mod(uint16_t command, const struct cls_rule *rule,
|
|
|
|
|
size_t actions_len)
|
2010-05-27 13:14:05 -07:00
|
|
|
|
{
|
|
|
|
|
struct ofp_flow_mod *ofm;
|
|
|
|
|
size_t size = sizeof *ofm + actions_len;
|
|
|
|
|
struct ofpbuf *out = ofpbuf_new(size);
|
|
|
|
|
ofm = ofpbuf_put_zeros(out, sizeof *ofm);
|
|
|
|
|
ofm->header.version = OFP_VERSION;
|
|
|
|
|
ofm->header.type = OFPT_FLOW_MOD;
|
|
|
|
|
ofm->header.length = htons(size);
|
|
|
|
|
ofm->cookie = 0;
|
2010-11-10 14:51:49 -08:00
|
|
|
|
ofm->priority = htons(MIN(rule->priority, UINT16_MAX));
|
2011-04-26 09:42:18 -07:00
|
|
|
|
ofputil_cls_rule_to_match(rule, &ofm->match);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
ofm->command = htons(command);
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *
|
2010-11-10 14:51:49 -08:00
|
|
|
|
make_add_flow(const struct cls_rule *rule, uint32_t buffer_id,
|
2010-05-27 13:14:05 -07:00
|
|
|
|
uint16_t idle_timeout, size_t actions_len)
|
|
|
|
|
{
|
2010-11-10 14:51:49 -08:00
|
|
|
|
struct ofpbuf *out = make_flow_mod(OFPFC_ADD, rule, actions_len);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
struct ofp_flow_mod *ofm = out->data;
|
|
|
|
|
ofm->idle_timeout = htons(idle_timeout);
|
|
|
|
|
ofm->hard_timeout = htons(OFP_FLOW_PERMANENT);
|
|
|
|
|
ofm->buffer_id = htonl(buffer_id);
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *
|
2010-11-10 14:51:49 -08:00
|
|
|
|
make_del_flow(const struct cls_rule *rule)
|
2010-05-27 13:14:05 -07:00
|
|
|
|
{
|
2010-11-10 14:51:49 -08:00
|
|
|
|
struct ofpbuf *out = make_flow_mod(OFPFC_DELETE_STRICT, rule, 0);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
struct ofp_flow_mod *ofm = out->data;
|
|
|
|
|
ofm->out_port = htons(OFPP_NONE);
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *
|
2010-11-10 14:51:49 -08:00
|
|
|
|
make_add_simple_flow(const struct cls_rule *rule,
|
2010-05-27 13:14:05 -07:00
|
|
|
|
uint32_t buffer_id, uint16_t out_port,
|
|
|
|
|
uint16_t idle_timeout)
|
|
|
|
|
{
|
2010-07-20 11:10:45 -07:00
|
|
|
|
if (out_port != OFPP_NONE) {
|
|
|
|
|
struct ofp_action_output *oao;
|
|
|
|
|
struct ofpbuf *buffer;
|
|
|
|
|
|
2010-11-10 14:51:49 -08:00
|
|
|
|
buffer = make_add_flow(rule, buffer_id, idle_timeout, sizeof *oao);
|
2011-08-16 16:30:57 -07:00
|
|
|
|
ofputil_put_OFPAT_OUTPUT(buffer)->port = htons(out_port);
|
2010-07-20 11:10:45 -07:00
|
|
|
|
return buffer;
|
|
|
|
|
} else {
|
2010-11-10 14:51:49 -08:00
|
|
|
|
return make_add_flow(rule, buffer_id, idle_timeout, 0);
|
2010-07-20 11:10:45 -07:00
|
|
|
|
}
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
make_packet_in(uint32_t buffer_id, uint16_t in_port, uint8_t reason,
|
|
|
|
|
const struct ofpbuf *payload, int max_send_len)
|
|
|
|
|
{
|
|
|
|
|
struct ofp_packet_in *opi;
|
|
|
|
|
struct ofpbuf *buf;
|
|
|
|
|
int send_len;
|
|
|
|
|
|
|
|
|
|
send_len = MIN(max_send_len, payload->size);
|
|
|
|
|
buf = ofpbuf_new(sizeof *opi + send_len);
|
|
|
|
|
opi = put_openflow_xid(offsetof(struct ofp_packet_in, data),
|
|
|
|
|
OFPT_PACKET_IN, 0, buf);
|
|
|
|
|
opi->buffer_id = htonl(buffer_id);
|
|
|
|
|
opi->total_len = htons(payload->size);
|
|
|
|
|
opi->in_port = htons(in_port);
|
|
|
|
|
opi->reason = reason;
|
|
|
|
|
ofpbuf_put(buf, payload->data, send_len);
|
|
|
|
|
update_openflow_length(buf);
|
|
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
make_packet_out(const struct ofpbuf *packet, uint32_t buffer_id,
|
|
|
|
|
uint16_t in_port,
|
|
|
|
|
const struct ofp_action_header *actions, size_t n_actions)
|
|
|
|
|
{
|
|
|
|
|
size_t actions_len = n_actions * sizeof *actions;
|
|
|
|
|
struct ofp_packet_out *opo;
|
|
|
|
|
size_t size = sizeof *opo + actions_len + (packet ? packet->size : 0);
|
|
|
|
|
struct ofpbuf *out = ofpbuf_new(size);
|
|
|
|
|
|
|
|
|
|
opo = ofpbuf_put_uninit(out, sizeof *opo);
|
|
|
|
|
opo->header.version = OFP_VERSION;
|
|
|
|
|
opo->header.type = OFPT_PACKET_OUT;
|
|
|
|
|
opo->header.length = htons(size);
|
|
|
|
|
opo->header.xid = htonl(0);
|
|
|
|
|
opo->buffer_id = htonl(buffer_id);
|
2011-09-29 13:30:07 -07:00
|
|
|
|
opo->in_port = htons(in_port);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
opo->actions_len = htons(actions_len);
|
|
|
|
|
ofpbuf_put(out, actions, actions_len);
|
|
|
|
|
if (packet) {
|
|
|
|
|
ofpbuf_put(out, packet->data, packet->size);
|
|
|
|
|
}
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
make_unbuffered_packet_out(const struct ofpbuf *packet,
|
|
|
|
|
uint16_t in_port, uint16_t out_port)
|
|
|
|
|
{
|
|
|
|
|
struct ofp_action_output action;
|
|
|
|
|
action.type = htons(OFPAT_OUTPUT);
|
|
|
|
|
action.len = htons(sizeof action);
|
|
|
|
|
action.port = htons(out_port);
|
|
|
|
|
return make_packet_out(packet, UINT32_MAX, in_port,
|
|
|
|
|
(struct ofp_action_header *) &action, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
make_buffered_packet_out(uint32_t buffer_id,
|
|
|
|
|
uint16_t in_port, uint16_t out_port)
|
|
|
|
|
{
|
2010-07-20 11:10:45 -07:00
|
|
|
|
if (out_port != OFPP_NONE) {
|
|
|
|
|
struct ofp_action_output action;
|
|
|
|
|
action.type = htons(OFPAT_OUTPUT);
|
|
|
|
|
action.len = htons(sizeof action);
|
|
|
|
|
action.port = htons(out_port);
|
|
|
|
|
return make_packet_out(NULL, buffer_id, in_port,
|
|
|
|
|
(struct ofp_action_header *) &action, 1);
|
|
|
|
|
} else {
|
|
|
|
|
return make_packet_out(NULL, buffer_id, in_port, NULL, 0);
|
|
|
|
|
}
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Creates and returns an OFPT_ECHO_REQUEST message with an empty payload. */
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
make_echo_request(void)
|
|
|
|
|
{
|
|
|
|
|
struct ofp_header *rq;
|
|
|
|
|
struct ofpbuf *out = ofpbuf_new(sizeof *rq);
|
|
|
|
|
rq = ofpbuf_put_uninit(out, sizeof *rq);
|
|
|
|
|
rq->version = OFP_VERSION;
|
|
|
|
|
rq->type = OFPT_ECHO_REQUEST;
|
|
|
|
|
rq->length = htons(sizeof *rq);
|
2010-11-16 11:00:25 -08:00
|
|
|
|
rq->xid = htonl(0);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Creates and returns an OFPT_ECHO_REPLY message matching the
|
|
|
|
|
* OFPT_ECHO_REQUEST message in 'rq'. */
|
|
|
|
|
struct ofpbuf *
|
|
|
|
|
make_echo_reply(const struct ofp_header *rq)
|
|
|
|
|
{
|
|
|
|
|
size_t size = ntohs(rq->length);
|
|
|
|
|
struct ofpbuf *out = ofpbuf_new(size);
|
|
|
|
|
struct ofp_header *reply = ofpbuf_put(out, rq, size);
|
|
|
|
|
reply->type = OFPT_ECHO_REPLY;
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
Implement new fragment handling policy.
Until now, OVS has handled IP fragments more awkwardly than necessary. It
has not been possible to match on L4 headers, even in fragments with offset
0 where they are actually present. This means that there was no way to
implement ACLs that treat, say, different TCP ports differently, on
fragmented traffic; instead, all decisions for fragment forwarding had to
be made on the basis of L2 and L3 headers alone.
This commit improves the situation significantly. It is still not possible
to match on L4 headers in fragments with nonzero offset, because that
information is simply not present in such fragments, but this commit adds
the ability to match on L4 headers for fragments with zero offset. This
means that it becomes possible to implement ACLs that drop such "first
fragments" on the basis of L4 headers. In practice, that effectively
blocks even fragmented traffic on an L4 basis, because the receiving IP
stack cannot reassemble a full packet when the first fragment is missing.
This commit works by adding a new "fragment type" to the kernel flow match
and making it available through OpenFlow as a new NXM field named
NXM_NX_IP_FRAG. Because OpenFlow 1.0 explicitly says that the L4 fields
are always 0 for IP fragments, it adds a new OpenFlow fragment handling
mode that fills in the L4 fields for "first fragments". It also enhances
ovs-ofctl to allow users to configure this new fragment handling mode and
to parse the new field.
Signed-off-by: Ben Pfaff <blp@nicira.com>
Bug #7557.
2011-10-19 21:33:44 -07:00
|
|
|
|
const char *
|
|
|
|
|
ofputil_frag_handling_to_string(enum ofp_config_flags flags)
|
|
|
|
|
{
|
|
|
|
|
switch (flags & OFPC_FRAG_MASK) {
|
|
|
|
|
case OFPC_FRAG_NORMAL: return "normal";
|
|
|
|
|
case OFPC_FRAG_DROP: return "drop";
|
|
|
|
|
case OFPC_FRAG_REASM: return "reassemble";
|
|
|
|
|
case OFPC_FRAG_NX_MATCH: return "nx-match";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
ofputil_frag_handling_from_string(const char *s, enum ofp_config_flags *flags)
|
|
|
|
|
{
|
|
|
|
|
if (!strcasecmp(s, "normal")) {
|
|
|
|
|
*flags = OFPC_FRAG_NORMAL;
|
|
|
|
|
} else if (!strcasecmp(s, "drop")) {
|
|
|
|
|
*flags = OFPC_FRAG_DROP;
|
|
|
|
|
} else if (!strcasecmp(s, "reassemble")) {
|
|
|
|
|
*flags = OFPC_FRAG_REASM;
|
|
|
|
|
} else if (!strcasecmp(s, "nx-match")) {
|
|
|
|
|
*flags = OFPC_FRAG_NX_MATCH;
|
|
|
|
|
} else {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-06-17 15:04:12 -07:00
|
|
|
|
/* Checks that 'port' is a valid output port for the OFPAT_OUTPUT action, given
|
|
|
|
|
* that the switch will never have more than 'max_ports' ports. Returns 0 if
|
|
|
|
|
* 'port' is valid, otherwise an ofp_mkerr() return code. */
|
2011-07-14 14:23:02 -07:00
|
|
|
|
int
|
|
|
|
|
ofputil_check_output_port(uint16_t port, int max_ports)
|
2010-05-27 13:14:05 -07:00
|
|
|
|
{
|
|
|
|
|
switch (port) {
|
|
|
|
|
case OFPP_IN_PORT:
|
|
|
|
|
case OFPP_TABLE:
|
|
|
|
|
case OFPP_NORMAL:
|
|
|
|
|
case OFPP_FLOOD:
|
|
|
|
|
case OFPP_ALL:
|
|
|
|
|
case OFPP_CONTROLLER:
|
|
|
|
|
case OFPP_LOCAL:
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
default:
|
2010-06-17 15:04:12 -07:00
|
|
|
|
if (port < max_ports) {
|
2010-05-27 13:14:05 -07:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_OUT_PORT);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-16 15:26:18 -07:00
|
|
|
|
#define OFPUTIL_NAMED_PORTS \
|
|
|
|
|
OFPUTIL_NAMED_PORT(IN_PORT) \
|
|
|
|
|
OFPUTIL_NAMED_PORT(TABLE) \
|
|
|
|
|
OFPUTIL_NAMED_PORT(NORMAL) \
|
|
|
|
|
OFPUTIL_NAMED_PORT(FLOOD) \
|
|
|
|
|
OFPUTIL_NAMED_PORT(ALL) \
|
|
|
|
|
OFPUTIL_NAMED_PORT(CONTROLLER) \
|
|
|
|
|
OFPUTIL_NAMED_PORT(LOCAL) \
|
|
|
|
|
OFPUTIL_NAMED_PORT(NONE)
|
|
|
|
|
|
|
|
|
|
/* Checks whether 's' is the string representation of an OpenFlow port number,
|
|
|
|
|
* either as an integer or a string name (e.g. "LOCAL"). If it is, stores the
|
|
|
|
|
* number in '*port' and returns true. Otherwise, returns false. */
|
|
|
|
|
bool
|
|
|
|
|
ofputil_port_from_string(const char *name, uint16_t *port)
|
|
|
|
|
{
|
|
|
|
|
struct pair {
|
|
|
|
|
const char *name;
|
|
|
|
|
uint16_t value;
|
|
|
|
|
};
|
|
|
|
|
static const struct pair pairs[] = {
|
|
|
|
|
#define OFPUTIL_NAMED_PORT(NAME) {#NAME, OFPP_##NAME},
|
|
|
|
|
OFPUTIL_NAMED_PORTS
|
|
|
|
|
#undef OFPUTIL_NAMED_PORT
|
|
|
|
|
};
|
|
|
|
|
static const int n_pairs = ARRAY_SIZE(pairs);
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
if (str_to_int(name, 0, &i) && i >= 0 && i < UINT16_MAX) {
|
|
|
|
|
*port = i;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < n_pairs; i++) {
|
|
|
|
|
if (!strcasecmp(name, pairs[i].name)) {
|
|
|
|
|
*port = pairs[i].value;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Appends to 's' a string representation of the OpenFlow port number 'port'.
|
|
|
|
|
* Most ports' string representation is just the port number, but for special
|
|
|
|
|
* ports, e.g. OFPP_LOCAL, it is the name, e.g. "LOCAL". */
|
|
|
|
|
void
|
|
|
|
|
ofputil_format_port(uint16_t port, struct ds *s)
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
|
|
switch (port) {
|
|
|
|
|
#define OFPUTIL_NAMED_PORT(NAME) case OFPP_##NAME: name = #NAME; break;
|
|
|
|
|
OFPUTIL_NAMED_PORTS
|
|
|
|
|
#undef OFPUTIL_NAMED_PORT
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
ds_put_format(s, "%"PRIu16, port);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
ds_put_cstr(s, name);
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-09 09:24:18 -07:00
|
|
|
|
static int
|
|
|
|
|
check_resubmit_table(const struct nx_action_resubmit *nar)
|
|
|
|
|
{
|
|
|
|
|
if (nar->pad[0] || nar->pad[1] || nar->pad[2]) {
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_ARGUMENT);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-10 13:05:17 -07:00
|
|
|
|
static int
|
|
|
|
|
check_output_reg(const struct nx_action_output_reg *naor,
|
|
|
|
|
const struct flow *flow)
|
|
|
|
|
{
|
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < sizeof naor->zero; i++) {
|
|
|
|
|
if (naor->zero[i]) {
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_ARGUMENT);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nxm_src_check(naor->src, nxm_decode_ofs(naor->ofs_nbits),
|
|
|
|
|
nxm_decode_n_bits(naor->ofs_nbits), flow);
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
int
|
|
|
|
|
validate_actions(const union ofp_action *actions, size_t n_actions,
|
|
|
|
|
const struct flow *flow, int max_ports)
|
2010-06-17 15:04:12 -07:00
|
|
|
|
{
|
2011-06-30 10:05:52 -07:00
|
|
|
|
const union ofp_action *a;
|
|
|
|
|
size_t left;
|
2010-06-17 15:04:12 -07:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
OFPUTIL_ACTION_FOR_EACH (a, left, actions, n_actions) {
|
|
|
|
|
uint16_t port;
|
|
|
|
|
int error;
|
|
|
|
|
int code;
|
2010-06-17 15:04:12 -07:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
code = ofputil_decode_action(a);
|
|
|
|
|
if (code < 0) {
|
|
|
|
|
char *msg;
|
2010-06-17 15:04:12 -07:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
error = -code;
|
|
|
|
|
msg = ofputil_error_to_string(error);
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl,
|
|
|
|
|
"action decoding error at offset %td (%s)",
|
|
|
|
|
(a - actions) * sizeof *a, msg);
|
|
|
|
|
free(msg);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
return error;
|
|
|
|
|
}
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
error = 0;
|
|
|
|
|
switch ((enum ofputil_action_code) code) {
|
|
|
|
|
case OFPUTIL_OFPAT_OUTPUT:
|
2011-07-14 14:23:02 -07:00
|
|
|
|
error = ofputil_check_output_port(ntohs(a->output.port),
|
|
|
|
|
max_ports);
|
2011-06-30 10:05:52 -07:00
|
|
|
|
break;
|
2010-12-09 10:41:32 -08:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
case OFPUTIL_OFPAT_SET_VLAN_VID:
|
|
|
|
|
if (a->vlan_vid.vlan_vid & ~htons(0xfff)) {
|
|
|
|
|
error = ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_ARGUMENT);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2010-11-12 16:23:26 -08:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
case OFPUTIL_OFPAT_SET_VLAN_PCP:
|
|
|
|
|
if (a->vlan_pcp.vlan_pcp & ~7) {
|
|
|
|
|
error = ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_ARGUMENT);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2010-11-12 16:23:26 -08:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
case OFPUTIL_OFPAT_ENQUEUE:
|
|
|
|
|
port = ntohs(((const struct ofp_action_enqueue *) a)->port);
|
2011-11-18 19:00:34 -08:00
|
|
|
|
if (port >= max_ports && port != OFPP_IN_PORT
|
|
|
|
|
&& port != OFPP_LOCAL) {
|
2011-06-30 10:05:52 -07:00
|
|
|
|
error = ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_OUT_PORT);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2010-11-12 16:23:26 -08:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
case OFPUTIL_NXAST_REG_MOVE:
|
|
|
|
|
error = nxm_check_reg_move((const struct nx_action_reg_move *) a,
|
|
|
|
|
flow);
|
|
|
|
|
break;
|
2010-11-12 16:23:26 -08:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
case OFPUTIL_NXAST_REG_LOAD:
|
|
|
|
|
error = nxm_check_reg_load((const struct nx_action_reg_load *) a,
|
|
|
|
|
flow);
|
|
|
|
|
break;
|
2010-12-10 10:42:42 -08:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
case OFPUTIL_NXAST_MULTIPATH:
|
2011-07-19 15:47:02 -07:00
|
|
|
|
error = multipath_check((const struct nx_action_multipath *) a,
|
|
|
|
|
flow);
|
2011-06-30 10:05:52 -07:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case OFPUTIL_NXAST_AUTOPATH:
|
2011-07-19 15:47:02 -07:00
|
|
|
|
error = autopath_check((const struct nx_action_autopath *) a,
|
|
|
|
|
flow);
|
2011-06-30 10:05:52 -07:00
|
|
|
|
break;
|
|
|
|
|
|
2011-06-10 17:45:45 -07:00
|
|
|
|
case OFPUTIL_NXAST_BUNDLE:
|
2011-07-20 15:07:46 -07:00
|
|
|
|
case OFPUTIL_NXAST_BUNDLE_LOAD:
|
2011-06-10 17:45:45 -07:00
|
|
|
|
error = bundle_check((const struct nx_action_bundle *) a,
|
2011-07-20 15:07:46 -07:00
|
|
|
|
max_ports, flow);
|
2011-06-10 17:45:45 -07:00
|
|
|
|
break;
|
|
|
|
|
|
2011-08-10 13:05:17 -07:00
|
|
|
|
case OFPUTIL_NXAST_OUTPUT_REG:
|
|
|
|
|
error = check_output_reg((const struct nx_action_output_reg *) a,
|
|
|
|
|
flow);
|
|
|
|
|
break;
|
|
|
|
|
|
2011-08-09 09:24:18 -07:00
|
|
|
|
case OFPUTIL_NXAST_RESUBMIT_TABLE:
|
|
|
|
|
error = check_resubmit_table(
|
|
|
|
|
(const struct nx_action_resubmit *) a);
|
|
|
|
|
break;
|
|
|
|
|
|
2011-09-12 16:19:57 -07:00
|
|
|
|
case OFPUTIL_NXAST_LEARN:
|
|
|
|
|
error = learn_check((const struct nx_action_learn *) a, flow);
|
|
|
|
|
break;
|
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
case OFPUTIL_OFPAT_STRIP_VLAN:
|
|
|
|
|
case OFPUTIL_OFPAT_SET_NW_SRC:
|
|
|
|
|
case OFPUTIL_OFPAT_SET_NW_DST:
|
|
|
|
|
case OFPUTIL_OFPAT_SET_NW_TOS:
|
|
|
|
|
case OFPUTIL_OFPAT_SET_TP_SRC:
|
|
|
|
|
case OFPUTIL_OFPAT_SET_TP_DST:
|
|
|
|
|
case OFPUTIL_OFPAT_SET_DL_SRC:
|
|
|
|
|
case OFPUTIL_OFPAT_SET_DL_DST:
|
|
|
|
|
case OFPUTIL_NXAST_RESUBMIT:
|
|
|
|
|
case OFPUTIL_NXAST_SET_TUNNEL:
|
|
|
|
|
case OFPUTIL_NXAST_SET_QUEUE:
|
|
|
|
|
case OFPUTIL_NXAST_POP_QUEUE:
|
|
|
|
|
case OFPUTIL_NXAST_NOTE:
|
|
|
|
|
case OFPUTIL_NXAST_SET_TUNNEL64:
|
2011-10-28 14:46:42 -07:00
|
|
|
|
case OFPUTIL_NXAST_EXIT:
|
2011-06-30 10:05:52 -07:00
|
|
|
|
break;
|
2010-12-17 14:38:50 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-05 12:37:52 -07:00
|
|
|
|
if (error) {
|
2011-06-30 10:05:52 -07:00
|
|
|
|
char *msg = ofputil_error_to_string(error);
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "bad action at offset %td (%s)",
|
|
|
|
|
(a - actions) * sizeof *a, msg);
|
|
|
|
|
free(msg);
|
2011-04-05 12:37:52 -07:00
|
|
|
|
return error;
|
|
|
|
|
}
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
2011-06-30 10:05:52 -07:00
|
|
|
|
if (left) {
|
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "bad action format at offset %zu",
|
|
|
|
|
(n_actions - left) * sizeof *a);
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
2011-08-05 15:48:45 -07:00
|
|
|
|
struct ofputil_action {
|
|
|
|
|
int code;
|
2011-06-30 10:05:52 -07:00
|
|
|
|
unsigned int min_len;
|
|
|
|
|
unsigned int max_len;
|
|
|
|
|
};
|
2010-10-18 11:18:10 -07:00
|
|
|
|
|
2011-08-05 15:48:45 -07:00
|
|
|
|
static const struct ofputil_action action_bad_type
|
|
|
|
|
= { -OFP_MKERR(OFPET_BAD_ACTION, OFPBAC_BAD_TYPE), 0, UINT_MAX };
|
|
|
|
|
static const struct ofputil_action action_bad_len
|
|
|
|
|
= { -OFP_MKERR(OFPET_BAD_ACTION, OFPBAC_BAD_LEN), 0, UINT_MAX };
|
|
|
|
|
static const struct ofputil_action action_bad_vendor
|
|
|
|
|
= { -OFP_MKERR(OFPET_BAD_ACTION, OFPBAC_BAD_VENDOR), 0, UINT_MAX };
|
2010-10-18 11:18:10 -07:00
|
|
|
|
|
2011-08-05 15:48:45 -07:00
|
|
|
|
static const struct ofputil_action *
|
2011-06-30 10:05:52 -07:00
|
|
|
|
ofputil_decode_ofpat_action(const union ofp_action *a)
|
|
|
|
|
{
|
2011-08-05 15:48:45 -07:00
|
|
|
|
enum ofp_action_type type = ntohs(a->type);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
2011-08-05 15:48:45 -07:00
|
|
|
|
switch (type) {
|
2011-08-17 11:01:17 -07:00
|
|
|
|
#define OFPAT_ACTION(ENUM, STRUCT, NAME) \
|
2011-08-05 15:48:45 -07:00
|
|
|
|
case ENUM: { \
|
|
|
|
|
static const struct ofputil_action action = { \
|
2011-08-17 11:01:17 -07:00
|
|
|
|
OFPUTIL_##ENUM, \
|
|
|
|
|
sizeof(struct STRUCT), \
|
|
|
|
|
sizeof(struct STRUCT) \
|
2011-08-05 15:48:45 -07:00
|
|
|
|
}; \
|
|
|
|
|
return &action; \
|
|
|
|
|
}
|
2011-08-17 11:01:17 -07:00
|
|
|
|
#include "ofp-util.def"
|
2011-08-05 15:48:45 -07:00
|
|
|
|
|
|
|
|
|
case OFPAT_VENDOR:
|
|
|
|
|
default:
|
|
|
|
|
return &action_bad_type;
|
2011-06-30 10:05:52 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
2011-08-05 15:48:45 -07:00
|
|
|
|
static const struct ofputil_action *
|
2011-06-30 10:05:52 -07:00
|
|
|
|
ofputil_decode_nxast_action(const union ofp_action *a)
|
|
|
|
|
{
|
2011-08-05 15:48:45 -07:00
|
|
|
|
const struct nx_action_header *nah = (const struct nx_action_header *) a;
|
|
|
|
|
enum nx_action_subtype subtype = ntohs(nah->subtype);
|
|
|
|
|
|
|
|
|
|
switch (subtype) {
|
2011-08-17 11:01:17 -07:00
|
|
|
|
#define NXAST_ACTION(ENUM, STRUCT, EXTENSIBLE, NAME) \
|
|
|
|
|
case ENUM: { \
|
|
|
|
|
static const struct ofputil_action action = { \
|
|
|
|
|
OFPUTIL_##ENUM, \
|
|
|
|
|
sizeof(struct STRUCT), \
|
|
|
|
|
EXTENSIBLE ? UINT_MAX : sizeof(struct STRUCT) \
|
|
|
|
|
}; \
|
|
|
|
|
return &action; \
|
2011-06-30 10:05:52 -07:00
|
|
|
|
}
|
2011-08-17 11:01:17 -07:00
|
|
|
|
#include "ofp-util.def"
|
2011-08-05 15:48:45 -07:00
|
|
|
|
|
|
|
|
|
case NXAST_SNAT__OBSOLETE:
|
|
|
|
|
case NXAST_DROP_SPOOFED_ARP__OBSOLETE:
|
|
|
|
|
default:
|
|
|
|
|
return &action_bad_type;
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
/* Parses 'a' to determine its type. Returns a nonnegative OFPUTIL_OFPAT_* or
|
|
|
|
|
* OFPUTIL_NXAST_* constant if successful, otherwise a negative OpenFlow error
|
|
|
|
|
* code (as returned by ofp_mkerr()).
|
|
|
|
|
*
|
|
|
|
|
* The caller must have already verified that 'a''s length is correct (that is,
|
|
|
|
|
* a->header.len is nonzero and a multiple of sizeof(union ofp_action) and no
|
|
|
|
|
* longer than the amount of space allocated to 'a').
|
|
|
|
|
*
|
|
|
|
|
* This function verifies that 'a''s length is correct for the type of action
|
|
|
|
|
* that it represents. */
|
2010-05-27 13:14:05 -07:00
|
|
|
|
int
|
2011-06-30 10:05:52 -07:00
|
|
|
|
ofputil_decode_action(const union ofp_action *a)
|
2010-05-27 13:14:05 -07:00
|
|
|
|
{
|
2011-08-05 15:48:45 -07:00
|
|
|
|
const struct ofputil_action *action;
|
|
|
|
|
uint16_t len = ntohs(a->header.len);
|
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
if (a->type != htons(OFPAT_VENDOR)) {
|
2011-08-05 15:48:45 -07:00
|
|
|
|
action = ofputil_decode_ofpat_action(a);
|
2011-06-30 10:05:52 -07:00
|
|
|
|
} else {
|
2011-08-05 15:48:45 -07:00
|
|
|
|
switch (ntohl(a->vendor.vendor)) {
|
|
|
|
|
case NX_VENDOR_ID:
|
|
|
|
|
if (len < sizeof(struct nx_action_header)) {
|
|
|
|
|
return -ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_LEN);
|
|
|
|
|
}
|
|
|
|
|
action = ofputil_decode_nxast_action(a);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
action = &action_bad_vendor;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2011-06-30 10:05:52 -07:00
|
|
|
|
}
|
2011-08-05 15:48:45 -07:00
|
|
|
|
|
|
|
|
|
return (len >= action->min_len && len <= action->max_len
|
|
|
|
|
? action->code
|
|
|
|
|
: -ofp_mkerr(OFPET_BAD_ACTION, OFPBAC_BAD_LEN));
|
2011-06-30 10:05:52 -07:00
|
|
|
|
}
|
2010-05-27 13:14:05 -07:00
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
/* Parses 'a' and returns its type as an OFPUTIL_OFPAT_* or OFPUTIL_NXAST_*
|
|
|
|
|
* constant. The caller must have already validated that 'a' is a valid action
|
|
|
|
|
* understood by Open vSwitch (e.g. by a previous successful call to
|
|
|
|
|
* ofputil_decode_action()). */
|
|
|
|
|
enum ofputil_action_code
|
|
|
|
|
ofputil_decode_action_unsafe(const union ofp_action *a)
|
|
|
|
|
{
|
2011-08-05 15:48:45 -07:00
|
|
|
|
const struct ofputil_action *action;
|
|
|
|
|
|
2011-06-30 10:05:52 -07:00
|
|
|
|
if (a->type != htons(OFPAT_VENDOR)) {
|
2011-08-05 15:48:45 -07:00
|
|
|
|
action = ofputil_decode_ofpat_action(a);
|
2011-06-30 10:05:52 -07:00
|
|
|
|
} else {
|
2011-08-05 15:48:45 -07:00
|
|
|
|
action = ofputil_decode_nxast_action(a);
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
2011-08-05 15:48:45 -07:00
|
|
|
|
|
|
|
|
|
return action->code;
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
2011-08-17 11:01:17 -07:00
|
|
|
|
/* Returns the 'enum ofputil_action_code' corresponding to 'name' (e.g. if
|
|
|
|
|
* 'name' is "output" then the return value is OFPUTIL_OFPAT_OUTPUT), or -1 if
|
|
|
|
|
* 'name' is not the name of any action.
|
|
|
|
|
*
|
|
|
|
|
* ofp-util.def lists the mapping from names to action. */
|
|
|
|
|
int
|
|
|
|
|
ofputil_action_code_from_name(const char *name)
|
|
|
|
|
{
|
|
|
|
|
static const char *names[OFPUTIL_N_ACTIONS] = {
|
|
|
|
|
#define OFPAT_ACTION(ENUM, STRUCT, NAME) NAME,
|
|
|
|
|
#define NXAST_ACTION(ENUM, STRUCT, EXTENSIBLE, NAME) NAME,
|
|
|
|
|
#include "ofp-util.def"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const char **p;
|
|
|
|
|
|
|
|
|
|
for (p = names; p < &names[ARRAY_SIZE(names)]; p++) {
|
|
|
|
|
if (*p && !strcasecmp(name, *p)) {
|
|
|
|
|
return p - names;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-16 16:30:57 -07:00
|
|
|
|
/* Appends an action of the type specified by 'code' to 'buf' and returns the
|
|
|
|
|
* action. Initializes the parts of 'action' that identify it as having type
|
|
|
|
|
* <ENUM> and length 'sizeof *action' and zeros the rest. For actions that
|
|
|
|
|
* have variable length, the length used and cleared is that of struct
|
|
|
|
|
* <STRUCT>. */
|
|
|
|
|
void *
|
|
|
|
|
ofputil_put_action(enum ofputil_action_code code, struct ofpbuf *buf)
|
|
|
|
|
{
|
|
|
|
|
switch (code) {
|
|
|
|
|
#define OFPAT_ACTION(ENUM, STRUCT, NAME) \
|
|
|
|
|
case OFPUTIL_##ENUM: return ofputil_put_##ENUM(buf);
|
|
|
|
|
#define NXAST_ACTION(ENUM, STRUCT, EXTENSIBLE, NAME) \
|
|
|
|
|
case OFPUTIL_##ENUM: return ofputil_put_##ENUM(buf);
|
|
|
|
|
#include "ofp-util.def"
|
|
|
|
|
}
|
|
|
|
|
NOT_REACHED();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define OFPAT_ACTION(ENUM, STRUCT, NAME) \
|
|
|
|
|
void \
|
|
|
|
|
ofputil_init_##ENUM(struct STRUCT *s) \
|
|
|
|
|
{ \
|
|
|
|
|
memset(s, 0, sizeof *s); \
|
|
|
|
|
s->type = htons(ENUM); \
|
|
|
|
|
s->len = htons(sizeof *s); \
|
|
|
|
|
} \
|
|
|
|
|
\
|
|
|
|
|
struct STRUCT * \
|
|
|
|
|
ofputil_put_##ENUM(struct ofpbuf *buf) \
|
|
|
|
|
{ \
|
|
|
|
|
struct STRUCT *s = ofpbuf_put_uninit(buf, sizeof *s); \
|
|
|
|
|
ofputil_init_##ENUM(s); \
|
|
|
|
|
return s; \
|
|
|
|
|
}
|
|
|
|
|
#define NXAST_ACTION(ENUM, STRUCT, EXTENSIBLE, NAME) \
|
|
|
|
|
void \
|
|
|
|
|
ofputil_init_##ENUM(struct STRUCT *s) \
|
|
|
|
|
{ \
|
|
|
|
|
memset(s, 0, sizeof *s); \
|
|
|
|
|
s->type = htons(OFPAT_VENDOR); \
|
|
|
|
|
s->len = htons(sizeof *s); \
|
|
|
|
|
s->vendor = htonl(NX_VENDOR_ID); \
|
|
|
|
|
s->subtype = htons(ENUM); \
|
|
|
|
|
} \
|
|
|
|
|
\
|
|
|
|
|
struct STRUCT * \
|
|
|
|
|
ofputil_put_##ENUM(struct ofpbuf *buf) \
|
|
|
|
|
{ \
|
|
|
|
|
struct STRUCT *s = ofpbuf_put_uninit(buf, sizeof *s); \
|
|
|
|
|
ofputil_init_##ENUM(s); \
|
|
|
|
|
return s; \
|
|
|
|
|
}
|
|
|
|
|
#include "ofp-util.def"
|
|
|
|
|
|
2011-03-29 14:42:20 -07:00
|
|
|
|
/* Returns true if 'action' outputs to 'port', false otherwise. */
|
2010-06-17 15:04:12 -07:00
|
|
|
|
bool
|
2011-03-29 14:42:20 -07:00
|
|
|
|
action_outputs_to_port(const union ofp_action *action, ovs_be16 port)
|
2010-06-17 15:04:12 -07:00
|
|
|
|
{
|
|
|
|
|
switch (ntohs(action->type)) {
|
|
|
|
|
case OFPAT_OUTPUT:
|
|
|
|
|
return action->output.port == port;
|
|
|
|
|
case OFPAT_ENQUEUE:
|
|
|
|
|
return ((const struct ofp_action_enqueue *) action)->port == port;
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-09 16:25:48 -07:00
|
|
|
|
/* "Normalizes" the wildcards in 'rule'. That means:
|
|
|
|
|
*
|
|
|
|
|
* 1. If the type of level N is known, then only the valid fields for that
|
|
|
|
|
* level may be specified. For example, ARP does not have a TOS field,
|
|
|
|
|
* so nw_tos must be wildcarded if 'rule' specifies an ARP flow.
|
|
|
|
|
* Similarly, IPv4 does not have any IPv6 addresses, so ipv6_src and
|
|
|
|
|
* ipv6_dst (and other fields) must be wildcarded if 'rule' specifies an
|
|
|
|
|
* IPv4 flow.
|
|
|
|
|
*
|
|
|
|
|
* 2. If the type of level N is not known (or not understood by Open
|
|
|
|
|
* vSwitch), then no fields at all for that level may be specified. For
|
|
|
|
|
* example, Open vSwitch does not understand SCTP, an L4 protocol, so the
|
|
|
|
|
* L4 fields tp_src and tp_dst must be wildcarded if 'rule' specifies an
|
|
|
|
|
* SCTP flow.
|
|
|
|
|
*
|
|
|
|
|
* 'flow_format' specifies the format of the flow as received or as intended to
|
|
|
|
|
* be sent. This is important for IPv6 and ARP, for which NXM supports more
|
|
|
|
|
* detailed matching. */
|
|
|
|
|
void
|
|
|
|
|
ofputil_normalize_rule(struct cls_rule *rule, enum nx_flow_format flow_format)
|
|
|
|
|
{
|
|
|
|
|
enum {
|
|
|
|
|
MAY_NW_ADDR = 1 << 0, /* nw_src, nw_dst */
|
|
|
|
|
MAY_TP_ADDR = 1 << 1, /* tp_src, tp_dst */
|
|
|
|
|
MAY_NW_PROTO = 1 << 2, /* nw_proto */
|
2011-11-05 15:48:12 -07:00
|
|
|
|
MAY_IPVx = 1 << 3, /* tos, frag, ttl */
|
2011-05-09 16:25:48 -07:00
|
|
|
|
MAY_ARP_SHA = 1 << 4, /* arp_sha */
|
|
|
|
|
MAY_ARP_THA = 1 << 5, /* arp_tha */
|
2011-11-09 17:49:49 -08:00
|
|
|
|
MAY_IPV6 = 1 << 6, /* ipv6_src, ipv6_dst, ipv6_label */
|
2011-05-09 16:25:48 -07:00
|
|
|
|
MAY_ND_TARGET = 1 << 7 /* nd_target */
|
|
|
|
|
} may_match;
|
|
|
|
|
|
|
|
|
|
struct flow_wildcards wc;
|
|
|
|
|
|
|
|
|
|
/* Figure out what fields may be matched. */
|
|
|
|
|
if (rule->flow.dl_type == htons(ETH_TYPE_IP)) {
|
2011-11-02 18:17:36 -07:00
|
|
|
|
may_match = MAY_NW_PROTO | MAY_IPVx | MAY_NW_ADDR;
|
2011-05-09 16:25:48 -07:00
|
|
|
|
if (rule->flow.nw_proto == IPPROTO_TCP ||
|
|
|
|
|
rule->flow.nw_proto == IPPROTO_UDP ||
|
|
|
|
|
rule->flow.nw_proto == IPPROTO_ICMP) {
|
|
|
|
|
may_match |= MAY_TP_ADDR;
|
|
|
|
|
}
|
|
|
|
|
} else if (rule->flow.dl_type == htons(ETH_TYPE_IPV6)
|
|
|
|
|
&& flow_format == NXFF_NXM) {
|
2011-11-09 17:49:49 -08:00
|
|
|
|
may_match = MAY_NW_PROTO | MAY_IPVx | MAY_IPV6;
|
2011-05-09 16:25:48 -07:00
|
|
|
|
if (rule->flow.nw_proto == IPPROTO_TCP ||
|
|
|
|
|
rule->flow.nw_proto == IPPROTO_UDP) {
|
|
|
|
|
may_match |= MAY_TP_ADDR;
|
|
|
|
|
} else if (rule->flow.nw_proto == IPPROTO_ICMPV6) {
|
|
|
|
|
may_match |= MAY_TP_ADDR;
|
|
|
|
|
if (rule->flow.tp_src == htons(ND_NEIGHBOR_SOLICIT)) {
|
|
|
|
|
may_match |= MAY_ND_TARGET | MAY_ARP_SHA;
|
|
|
|
|
} else if (rule->flow.tp_src == htons(ND_NEIGHBOR_ADVERT)) {
|
|
|
|
|
may_match |= MAY_ND_TARGET | MAY_ARP_THA;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (rule->flow.dl_type == htons(ETH_TYPE_ARP)) {
|
|
|
|
|
may_match = MAY_NW_PROTO | MAY_NW_ADDR;
|
|
|
|
|
if (flow_format == NXFF_NXM) {
|
|
|
|
|
may_match |= MAY_ARP_SHA | MAY_ARP_THA;
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
ofp-util: Revise OpenFlow 1.0 ofp_match normalization.
For a long time, Open vSwitch has "normalized" OpenFlow 1.0 flows in a
funny way: it tries to change fields that are wildcarded into fields
that are exact-match. For example, the normalize_match() function
knows that if dl_type is wildcarded, then all of the L3 and L4 fields
will always be extracted as 0, so it sets those fields to exact-match
and their values to 0.
The reason for this was originally that exact-match flows were much
cheaper for Open vSwitch to implement, because they could be implemented
with a hash table, whereas other kinds of flows had to be implemented
with an expensive linear search. But these days Open vSwitch has a
smarter classifier in which wildcarded flows have minimal cost. Also,
it is no longer possible for OpenFlow 1.0 to specify truly exact-match
flows, because Open vSwitch supports fields for which OpenFlow 1.0
cannot specify values and therefore will always be wildcarded.
Now, it no longer makes sense to do this transformation, so this commit
removes it. Presumably, this will be less surprising for users.
Reviewed-by: Simon Horman <horms@verge.net.au>
2011-05-02 11:04:33 -07:00
|
|
|
|
} else {
|
2011-05-09 16:25:48 -07:00
|
|
|
|
may_match = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Clear the fields that may not be matched. */
|
|
|
|
|
wc = rule->wc;
|
|
|
|
|
if (!(may_match & MAY_NW_ADDR)) {
|
|
|
|
|
wc.nw_src_mask = wc.nw_dst_mask = htonl(0);
|
|
|
|
|
}
|
|
|
|
|
if (!(may_match & MAY_TP_ADDR)) {
|
|
|
|
|
wc.wildcards |= FWW_TP_SRC | FWW_TP_DST;
|
|
|
|
|
}
|
|
|
|
|
if (!(may_match & MAY_NW_PROTO)) {
|
|
|
|
|
wc.wildcards |= FWW_NW_PROTO;
|
|
|
|
|
}
|
2011-11-02 18:17:36 -07:00
|
|
|
|
if (!(may_match & MAY_IPVx)) {
|
2011-11-09 23:39:16 -08:00
|
|
|
|
wc.wildcards |= FWW_NW_DSCP;
|
|
|
|
|
wc.wildcards |= FWW_NW_ECN;
|
2011-11-05 15:48:12 -07:00
|
|
|
|
wc.wildcards |= FWW_NW_TTL;
|
2011-05-09 16:25:48 -07:00
|
|
|
|
}
|
|
|
|
|
if (!(may_match & MAY_ARP_SHA)) {
|
|
|
|
|
wc.wildcards |= FWW_ARP_SHA;
|
|
|
|
|
}
|
|
|
|
|
if (!(may_match & MAY_ARP_THA)) {
|
|
|
|
|
wc.wildcards |= FWW_ARP_THA;
|
|
|
|
|
}
|
2011-11-09 17:49:49 -08:00
|
|
|
|
if (!(may_match & MAY_IPV6)) {
|
2011-05-09 16:25:48 -07:00
|
|
|
|
wc.ipv6_src_mask = wc.ipv6_dst_mask = in6addr_any;
|
2011-11-01 15:57:56 -07:00
|
|
|
|
wc.wildcards |= FWW_IPV6_LABEL;
|
2011-05-09 16:25:48 -07:00
|
|
|
|
}
|
|
|
|
|
if (!(may_match & MAY_ND_TARGET)) {
|
|
|
|
|
wc.wildcards |= FWW_ND_TARGET;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Log any changes. */
|
|
|
|
|
if (!flow_wildcards_equal(&wc, &rule->wc)) {
|
|
|
|
|
bool log = !VLOG_DROP_INFO(&bad_ofmsg_rl);
|
|
|
|
|
char *pre = log ? cls_rule_to_string(rule) : NULL;
|
|
|
|
|
|
|
|
|
|
rule->wc = wc;
|
|
|
|
|
cls_rule_zero_wildcarded_fields(rule);
|
|
|
|
|
|
|
|
|
|
if (log) {
|
|
|
|
|
char *post = cls_rule_to_string(rule);
|
|
|
|
|
VLOG_INFO("normalization changed ofp_match, details:");
|
|
|
|
|
VLOG_INFO(" pre: %s", pre);
|
|
|
|
|
VLOG_INFO("post: %s", post);
|
|
|
|
|
free(pre);
|
|
|
|
|
free(post);
|
|
|
|
|
}
|
2010-05-27 13:14:05 -07:00
|
|
|
|
}
|
2010-06-24 14:20:45 -07:00
|
|
|
|
}
|
2010-08-20 10:29:03 -07:00
|
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
|
vendor_code_to_id(uint8_t code)
|
|
|
|
|
{
|
|
|
|
|
switch (code) {
|
|
|
|
|
#define OFPUTIL_VENDOR(NAME, VENDOR_ID) case NAME: return VENDOR_ID;
|
2010-11-04 10:33:08 -07:00
|
|
|
|
OFPUTIL_VENDORS
|
|
|
|
|
#undef OFPUTIL_VENDOR
|
2010-08-20 10:29:03 -07:00
|
|
|
|
default:
|
|
|
|
|
return UINT32_MAX;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-01-12 13:42:50 -08:00
|
|
|
|
static int
|
|
|
|
|
vendor_id_to_code(uint32_t id)
|
|
|
|
|
{
|
|
|
|
|
switch (id) {
|
|
|
|
|
#define OFPUTIL_VENDOR(NAME, VENDOR_ID) case VENDOR_ID: return NAME;
|
|
|
|
|
OFPUTIL_VENDORS
|
|
|
|
|
#undef OFPUTIL_VENDOR
|
|
|
|
|
default:
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-20 10:29:03 -07:00
|
|
|
|
/* Creates and returns an OpenFlow message of type OFPT_ERROR with the error
|
|
|
|
|
* information taken from 'error', whose encoding must be as described in the
|
|
|
|
|
* large comment in ofp-util.h. If 'oh' is nonnull, then the error will use
|
|
|
|
|
* oh->xid as its transaction ID, and it will include up to the first 64 bytes
|
|
|
|
|
* of 'oh'.
|
|
|
|
|
*
|
|
|
|
|
* Returns NULL if 'error' is not an OpenFlow error code. */
|
|
|
|
|
struct ofpbuf *
|
2011-01-12 13:42:50 -08:00
|
|
|
|
ofputil_encode_error_msg(int error, const struct ofp_header *oh)
|
2010-08-20 10:29:03 -07:00
|
|
|
|
{
|
|
|
|
|
static struct vlog_rate_limit rl = VLOG_RATE_LIMIT_INIT(1, 5);
|
|
|
|
|
|
|
|
|
|
struct ofpbuf *buf;
|
|
|
|
|
const void *data;
|
|
|
|
|
size_t len;
|
|
|
|
|
uint8_t vendor;
|
|
|
|
|
uint16_t type;
|
|
|
|
|
uint16_t code;
|
2010-11-16 11:00:25 -08:00
|
|
|
|
ovs_be32 xid;
|
2010-08-20 10:29:03 -07:00
|
|
|
|
|
|
|
|
|
if (!is_ofp_error(error)) {
|
|
|
|
|
/* We format 'error' with strerror() here since it seems likely to be
|
|
|
|
|
* a system errno value. */
|
|
|
|
|
VLOG_WARN_RL(&rl, "invalid OpenFlow error code %d (%s)",
|
|
|
|
|
error, strerror(error));
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (oh) {
|
|
|
|
|
xid = oh->xid;
|
|
|
|
|
data = oh;
|
|
|
|
|
len = ntohs(oh->length);
|
|
|
|
|
if (len > 64) {
|
|
|
|
|
len = 64;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
xid = 0;
|
|
|
|
|
data = NULL;
|
|
|
|
|
len = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vendor = get_ofp_err_vendor(error);
|
|
|
|
|
type = get_ofp_err_type(error);
|
|
|
|
|
code = get_ofp_err_code(error);
|
|
|
|
|
if (vendor == OFPUTIL_VENDOR_OPENFLOW) {
|
|
|
|
|
struct ofp_error_msg *oem;
|
|
|
|
|
|
|
|
|
|
oem = make_openflow_xid(len + sizeof *oem, OFPT_ERROR, xid, &buf);
|
|
|
|
|
oem->type = htons(type);
|
|
|
|
|
oem->code = htons(code);
|
|
|
|
|
} else {
|
|
|
|
|
struct ofp_error_msg *oem;
|
2010-11-04 10:33:28 -07:00
|
|
|
|
struct nx_vendor_error *nve;
|
2010-08-20 10:29:03 -07:00
|
|
|
|
uint32_t vendor_id;
|
|
|
|
|
|
|
|
|
|
vendor_id = vendor_code_to_id(vendor);
|
|
|
|
|
if (vendor_id == UINT32_MAX) {
|
|
|
|
|
VLOG_WARN_RL(&rl, "error %x contains invalid vendor code %d",
|
|
|
|
|
error, vendor);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-04 10:33:28 -07:00
|
|
|
|
oem = make_openflow_xid(len + sizeof *oem + sizeof *nve,
|
2010-08-20 10:29:03 -07:00
|
|
|
|
OFPT_ERROR, xid, &buf);
|
|
|
|
|
oem->type = htons(NXET_VENDOR);
|
|
|
|
|
oem->code = htons(NXVC_VENDOR_ERROR);
|
|
|
|
|
|
2010-12-07 21:58:30 -08:00
|
|
|
|
nve = (struct nx_vendor_error *)oem->data;
|
2010-11-04 10:33:28 -07:00
|
|
|
|
nve->vendor = htonl(vendor_id);
|
|
|
|
|
nve->type = htons(type);
|
|
|
|
|
nve->code = htons(code);
|
2010-08-20 10:29:03 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (len) {
|
2010-12-08 15:35:54 -08:00
|
|
|
|
buf->size -= len;
|
2010-08-20 10:29:03 -07:00
|
|
|
|
ofpbuf_put(buf, data, len);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
|
}
|
2010-11-08 10:43:19 -08:00
|
|
|
|
|
2011-01-12 13:42:50 -08:00
|
|
|
|
/* Decodes 'oh', which should be an OpenFlow OFPT_ERROR message, and returns an
|
|
|
|
|
* Open vSwitch internal error code in the format described in the large
|
|
|
|
|
* comment in ofp-util.h.
|
|
|
|
|
*
|
|
|
|
|
* If 'payload_ofs' is nonnull, on success '*payload_ofs' is set to the offset
|
|
|
|
|
* to the payload starting from 'oh' and on failure it is set to 0. */
|
|
|
|
|
int
|
|
|
|
|
ofputil_decode_error_msg(const struct ofp_header *oh, size_t *payload_ofs)
|
|
|
|
|
{
|
|
|
|
|
static struct vlog_rate_limit rl = VLOG_RATE_LIMIT_INIT(1, 5);
|
|
|
|
|
|
|
|
|
|
const struct ofp_error_msg *oem;
|
|
|
|
|
uint16_t type, code;
|
|
|
|
|
struct ofpbuf b;
|
|
|
|
|
int vendor;
|
|
|
|
|
|
|
|
|
|
if (payload_ofs) {
|
|
|
|
|
*payload_ofs = 0;
|
|
|
|
|
}
|
|
|
|
|
if (oh->type != OFPT_ERROR) {
|
|
|
|
|
return EPROTO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ofpbuf_use_const(&b, oh, ntohs(oh->length));
|
|
|
|
|
oem = ofpbuf_try_pull(&b, sizeof *oem);
|
|
|
|
|
if (!oem) {
|
|
|
|
|
return EPROTO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type = ntohs(oem->type);
|
|
|
|
|
code = ntohs(oem->code);
|
|
|
|
|
if (type == NXET_VENDOR && code == NXVC_VENDOR_ERROR) {
|
|
|
|
|
const struct nx_vendor_error *nve = ofpbuf_try_pull(&b, sizeof *nve);
|
|
|
|
|
if (!nve) {
|
|
|
|
|
return EPROTO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vendor = vendor_id_to_code(ntohl(nve->vendor));
|
|
|
|
|
if (vendor < 0) {
|
|
|
|
|
VLOG_WARN_RL(&rl, "error contains unknown vendor ID %#"PRIx32,
|
|
|
|
|
ntohl(nve->vendor));
|
|
|
|
|
return EPROTO;
|
|
|
|
|
}
|
|
|
|
|
type = ntohs(nve->type);
|
|
|
|
|
code = ntohs(nve->code);
|
|
|
|
|
} else {
|
|
|
|
|
vendor = OFPUTIL_VENDOR_OPENFLOW;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (type >= 1024) {
|
|
|
|
|
VLOG_WARN_RL(&rl, "error contains type %"PRIu16" greater than "
|
|
|
|
|
"supported maximum value 1023", type);
|
|
|
|
|
return EPROTO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (payload_ofs) {
|
|
|
|
|
*payload_ofs = (uint8_t *) b.data - (uint8_t *) oh;
|
|
|
|
|
}
|
|
|
|
|
return ofp_mkerr_vendor(vendor, type, code);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
ofputil_format_error(struct ds *s, int error)
|
|
|
|
|
{
|
|
|
|
|
if (is_errno(error)) {
|
|
|
|
|
ds_put_cstr(s, strerror(error));
|
|
|
|
|
} else {
|
|
|
|
|
uint16_t type = get_ofp_err_type(error);
|
|
|
|
|
uint16_t code = get_ofp_err_code(error);
|
|
|
|
|
const char *type_s = ofp_error_type_to_string(type);
|
|
|
|
|
const char *code_s = ofp_error_code_to_string(type, code);
|
|
|
|
|
|
|
|
|
|
ds_put_format(s, "type ");
|
|
|
|
|
if (type_s) {
|
|
|
|
|
ds_put_cstr(s, type_s);
|
|
|
|
|
} else {
|
|
|
|
|
ds_put_format(s, "%"PRIu16, type);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ds_put_cstr(s, ", code ");
|
|
|
|
|
if (code_s) {
|
|
|
|
|
ds_put_cstr(s, code_s);
|
|
|
|
|
} else {
|
|
|
|
|
ds_put_format(s, "%"PRIu16, code);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
ofputil_error_to_string(int error)
|
|
|
|
|
{
|
|
|
|
|
struct ds s = DS_EMPTY_INITIALIZER;
|
|
|
|
|
ofputil_format_error(&s, error);
|
|
|
|
|
return ds_steal_cstr(&s);
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-08 10:43:19 -08:00
|
|
|
|
/* Attempts to pull 'actions_len' bytes from the front of 'b'. Returns 0 if
|
|
|
|
|
* successful, otherwise an OpenFlow error.
|
|
|
|
|
*
|
|
|
|
|
* If successful, the first action is stored in '*actionsp' and the number of
|
|
|
|
|
* "union ofp_action" size elements into '*n_actionsp'. Otherwise NULL and 0
|
|
|
|
|
* are stored, respectively.
|
|
|
|
|
*
|
|
|
|
|
* This function does not check that the actions are valid (the caller should
|
|
|
|
|
* do so, with validate_actions()). The caller is also responsible for making
|
|
|
|
|
* sure that 'b->data' is initially aligned appropriately for "union
|
|
|
|
|
* ofp_action". */
|
|
|
|
|
int
|
|
|
|
|
ofputil_pull_actions(struct ofpbuf *b, unsigned int actions_len,
|
|
|
|
|
union ofp_action **actionsp, size_t *n_actionsp)
|
|
|
|
|
{
|
2010-11-10 16:22:18 -08:00
|
|
|
|
if (actions_len % OFP_ACTION_ALIGN != 0) {
|
2010-12-23 10:36:55 -08:00
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "OpenFlow message actions length %u "
|
|
|
|
|
"is not a multiple of %d", actions_len, OFP_ACTION_ALIGN);
|
2010-11-08 10:43:19 -08:00
|
|
|
|
goto error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*actionsp = ofpbuf_try_pull(b, actions_len);
|
|
|
|
|
if (*actionsp == NULL) {
|
2010-12-23 10:36:55 -08:00
|
|
|
|
VLOG_WARN_RL(&bad_ofmsg_rl, "OpenFlow message actions length %u "
|
|
|
|
|
"exceeds remaining message length (%zu)",
|
|
|
|
|
actions_len, b->size);
|
2010-11-08 10:43:19 -08:00
|
|
|
|
goto error;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-10 16:22:18 -08:00
|
|
|
|
*n_actionsp = actions_len / OFP_ACTION_ALIGN;
|
2010-11-08 10:43:19 -08:00
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
error:
|
|
|
|
|
*actionsp = NULL;
|
|
|
|
|
*n_actionsp = 0;
|
|
|
|
|
return ofp_mkerr(OFPET_BAD_REQUEST, OFPBRC_BAD_LEN);
|
|
|
|
|
}
|
2011-05-26 16:49:10 -07:00
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
ofputil_actions_equal(const union ofp_action *a, size_t n_a,
|
|
|
|
|
const union ofp_action *b, size_t n_b)
|
|
|
|
|
{
|
|
|
|
|
return n_a == n_b && (!n_a || !memcmp(a, b, n_a * sizeof *a));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
union ofp_action *
|
|
|
|
|
ofputil_actions_clone(const union ofp_action *actions, size_t n)
|
|
|
|
|
{
|
|
|
|
|
return n ? xmemdup(actions, n * sizeof *actions) : NULL;
|
|
|
|
|
}
|
2011-08-17 13:39:02 -07:00
|
|
|
|
|
|
|
|
|
/* Parses a key or a key-value pair from '*stringp'.
|
|
|
|
|
*
|
|
|
|
|
* On success: Stores the key into '*keyp'. Stores the value, if present, into
|
|
|
|
|
* '*valuep', otherwise an empty string. Advances '*stringp' past the end of
|
|
|
|
|
* the key-value pair, preparing it for another call. '*keyp' and '*valuep'
|
|
|
|
|
* are substrings of '*stringp' created by replacing some of its bytes by null
|
|
|
|
|
* terminators. Returns true.
|
|
|
|
|
*
|
|
|
|
|
* If '*stringp' is just white space or commas, sets '*keyp' and '*valuep' to
|
|
|
|
|
* NULL and returns false. */
|
|
|
|
|
bool
|
|
|
|
|
ofputil_parse_key_value(char **stringp, char **keyp, char **valuep)
|
|
|
|
|
{
|
|
|
|
|
char *pos, *key, *value;
|
|
|
|
|
size_t key_len;
|
|
|
|
|
|
|
|
|
|
pos = *stringp;
|
|
|
|
|
pos += strspn(pos, ", \t\r\n");
|
|
|
|
|
if (*pos == '\0') {
|
|
|
|
|
*keyp = *valuep = NULL;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
key = pos;
|
|
|
|
|
key_len = strcspn(pos, ":=(, \t\r\n");
|
|
|
|
|
if (key[key_len] == ':' || key[key_len] == '=') {
|
|
|
|
|
/* The value can be separated by a colon. */
|
|
|
|
|
size_t value_len;
|
|
|
|
|
|
|
|
|
|
value = key + key_len + 1;
|
|
|
|
|
value_len = strcspn(value, ", \t\r\n");
|
|
|
|
|
pos = value + value_len + (value[value_len] != '\0');
|
|
|
|
|
value[value_len] = '\0';
|
|
|
|
|
} else if (key[key_len] == '(') {
|
|
|
|
|
/* The value can be surrounded by balanced parentheses. The outermost
|
|
|
|
|
* set of parentheses is removed. */
|
|
|
|
|
int level = 1;
|
|
|
|
|
size_t value_len;
|
|
|
|
|
|
|
|
|
|
value = key + key_len + 1;
|
|
|
|
|
for (value_len = 0; level > 0; value_len++) {
|
|
|
|
|
switch (value[value_len]) {
|
|
|
|
|
case '\0':
|
|
|
|
|
ovs_fatal(0, "unbalanced parentheses in argument to %s", key);
|
|
|
|
|
|
|
|
|
|
case '(':
|
|
|
|
|
level++;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ')':
|
|
|
|
|
level--;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
value[value_len - 1] = '\0';
|
|
|
|
|
pos = value + value_len;
|
|
|
|
|
} else {
|
|
|
|
|
/* There might be no value at all. */
|
|
|
|
|
value = key + key_len; /* Will become the empty string below. */
|
|
|
|
|
pos = key + key_len + (key[key_len] != '\0');
|
|
|
|
|
}
|
|
|
|
|
key[key_len] = '\0';
|
|
|
|
|
|
|
|
|
|
*stringp = pos;
|
|
|
|
|
*keyp = key;
|
|
|
|
|
*valuep = value;
|
|
|
|
|
return true;
|
|
|
|
|
}
|