forked from openvswitch/ovs
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
ofproto-dpif-xlate: Expose xlate cache.
Later patches will need to create xlate cache entries from different modules. This patch refactors the xlate cache code in preparation without any functional changes, so that the changes are clearly visible in the following patches. The definition of XC_ENTRY_FOR_EACH() iterator macro is changed so that it now does not take the xlate cache pointer to unify the usage accross all call sites. Signed-off-by: Jarno Rajahalme <[email protected]> Acked-by: Ben Pfaff <[email protected]>
- Loading branch information
Jarno Rajahalme
committed
Sep 14, 2016
1 parent
064799a
commit 901a517
Showing
6 changed files
with
413 additions
and
298 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,239 @@ | ||
/* Copyright (c) 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016 Nicira, Inc. | ||
* | ||
* 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 "ofproto/ofproto-dpif-xlate-cache.h" | ||
|
||
#include <errno.h> | ||
#include <arpa/inet.h> | ||
#include <net/if.h> | ||
#include <sys/socket.h> | ||
#include <netinet/in.h> | ||
|
||
#include "bfd.h" | ||
#include "bitmap.h" | ||
#include "bond.h" | ||
#include "bundle.h" | ||
#include "byte-order.h" | ||
#include "coverage.h" | ||
#include "dp-packet.h" | ||
#include "dpif.h" | ||
#include "learn.h" | ||
#include "mac-learning.h" | ||
#include "netdev-vport.h" | ||
#include "ofproto/ofproto-dpif-mirror.h" | ||
#include "ofproto/ofproto-dpif.h" | ||
#include "ofproto/ofproto-dpif-xlate.h" | ||
#include "ofproto/ofproto-provider.h" | ||
#include "openvswitch/dynamic-string.h" | ||
#include "openvswitch/vlog.h" | ||
#include "ovs-router.h" | ||
#include "packets.h" | ||
#include "tnl-neigh-cache.h" | ||
#include "util.h" | ||
|
||
VLOG_DEFINE_THIS_MODULE(ofproto_xlate_cache); | ||
|
||
struct xlate_cache * | ||
xlate_cache_new(void) | ||
{ | ||
struct xlate_cache *xcache = xmalloc(sizeof *xcache); | ||
|
||
ofpbuf_init(&xcache->entries, 512); | ||
return xcache; | ||
} | ||
|
||
struct xc_entry * | ||
xlate_cache_add_entry(struct xlate_cache *xcache, enum xc_type type) | ||
{ | ||
struct xc_entry *entry; | ||
|
||
entry = ofpbuf_put_zeros(&xcache->entries, sizeof *entry); | ||
entry->type = type; | ||
|
||
return entry; | ||
} | ||
|
||
static void | ||
xlate_cache_netdev(struct xc_entry *entry, const struct dpif_flow_stats *stats) | ||
{ | ||
if (entry->dev.tx) { | ||
netdev_vport_inc_tx(entry->dev.tx, stats); | ||
} | ||
if (entry->dev.rx) { | ||
netdev_vport_inc_rx(entry->dev.rx, stats); | ||
} | ||
if (entry->dev.bfd) { | ||
bfd_account_rx(entry->dev.bfd, stats); | ||
} | ||
} | ||
|
||
/* Push stats and perform side effects of flow translation. */ | ||
void | ||
xlate_push_stats_entry(struct xc_entry *entry, | ||
const struct dpif_flow_stats *stats) | ||
{ | ||
struct eth_addr dmac; | ||
|
||
switch (entry->type) { | ||
case XC_RULE: | ||
rule_dpif_credit_stats(entry->rule, stats); | ||
break; | ||
case XC_BOND: | ||
bond_account(entry->bond.bond, entry->bond.flow, | ||
entry->bond.vid, stats->n_bytes); | ||
break; | ||
case XC_NETDEV: | ||
xlate_cache_netdev(entry, stats); | ||
break; | ||
case XC_NETFLOW: | ||
netflow_flow_update(entry->nf.netflow, entry->nf.flow, | ||
entry->nf.iface, stats); | ||
break; | ||
case XC_MIRROR: | ||
mirror_update_stats(entry->mirror.mbridge, | ||
entry->mirror.mirrors, | ||
stats->n_packets, stats->n_bytes); | ||
break; | ||
case XC_LEARN: | ||
ofproto_dpif_flow_mod(entry->learn.ofproto, entry->learn.fm); | ||
break; | ||
case XC_NORMAL: | ||
xlate_mac_learning_update(entry->normal.ofproto, | ||
entry->normal.in_port, | ||
entry->normal.dl_src, | ||
entry->normal.vlan, | ||
entry->normal.is_gratuitous_arp); | ||
break; | ||
case XC_FIN_TIMEOUT: | ||
if (stats->tcp_flags & (TCP_FIN | TCP_RST)) { | ||
rule_dpif_reduce_timeouts(entry->fin.rule, entry->fin.idle, | ||
entry->fin.hard); | ||
} | ||
break; | ||
case XC_GROUP: | ||
group_dpif_credit_stats(entry->group.group, entry->group.bucket, | ||
stats); | ||
break; | ||
case XC_TNL_NEIGH: | ||
/* Lookup neighbor to avoid timeout. */ | ||
tnl_neigh_lookup(entry->tnl_neigh_cache.br_name, | ||
&entry->tnl_neigh_cache.d_ipv6, &dmac); | ||
break; | ||
default: | ||
OVS_NOT_REACHED(); | ||
} | ||
} | ||
|
||
void | ||
xlate_push_stats(struct xlate_cache *xcache, | ||
const struct dpif_flow_stats *stats) | ||
{ | ||
if (!stats->n_packets) { | ||
return; | ||
} | ||
|
||
struct xc_entry *entry; | ||
struct ofpbuf entries = xcache->entries; | ||
XC_ENTRY_FOR_EACH (entry, &entries) { | ||
xlate_push_stats_entry(entry, stats); | ||
} | ||
} | ||
|
||
static void | ||
xlate_dev_unref(struct xc_entry *entry) | ||
{ | ||
if (entry->dev.tx) { | ||
netdev_close(entry->dev.tx); | ||
} | ||
if (entry->dev.rx) { | ||
netdev_close(entry->dev.rx); | ||
} | ||
if (entry->dev.bfd) { | ||
bfd_unref(entry->dev.bfd); | ||
} | ||
} | ||
|
||
static void | ||
xlate_cache_clear_netflow(struct netflow *netflow, struct flow *flow) | ||
{ | ||
netflow_flow_clear(netflow, flow); | ||
netflow_unref(netflow); | ||
free(flow); | ||
} | ||
|
||
void | ||
xlate_cache_clear_entry(struct xc_entry *entry) | ||
{ | ||
switch (entry->type) { | ||
case XC_RULE: | ||
rule_dpif_unref(entry->rule); | ||
break; | ||
case XC_BOND: | ||
free(entry->bond.flow); | ||
bond_unref(entry->bond.bond); | ||
break; | ||
case XC_NETDEV: | ||
xlate_dev_unref(entry); | ||
break; | ||
case XC_NETFLOW: | ||
xlate_cache_clear_netflow(entry->nf.netflow, entry->nf.flow); | ||
break; | ||
case XC_MIRROR: | ||
mbridge_unref(entry->mirror.mbridge); | ||
break; | ||
case XC_LEARN: | ||
free(entry->learn.fm); | ||
ofpbuf_delete(entry->learn.ofpacts); | ||
break; | ||
case XC_NORMAL: | ||
break; | ||
case XC_FIN_TIMEOUT: | ||
/* 'u.fin.rule' is always already held as a XC_RULE, which | ||
* has already released it's reference above. */ | ||
break; | ||
case XC_GROUP: | ||
group_dpif_unref(entry->group.group); | ||
break; | ||
case XC_TNL_NEIGH: | ||
break; | ||
default: | ||
OVS_NOT_REACHED(); | ||
} | ||
} | ||
|
||
void | ||
xlate_cache_clear(struct xlate_cache *xcache) | ||
{ | ||
if (!xcache) { | ||
return; | ||
} | ||
|
||
struct xc_entry *entry; | ||
struct ofpbuf entries = xcache->entries; | ||
XC_ENTRY_FOR_EACH (entry, &entries) { | ||
xlate_cache_clear_entry(entry); | ||
} | ||
|
||
ofpbuf_clear(&xcache->entries); | ||
} | ||
|
||
void | ||
xlate_cache_delete(struct xlate_cache *xcache) | ||
{ | ||
xlate_cache_clear(xcache); | ||
ofpbuf_uninit(&xcache->entries); | ||
free(xcache); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,132 @@ | ||
/* Copyright (c) 2016 Nicira, Inc. | ||
* | ||
* 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. */ | ||
|
||
#ifndef OFPROTO_DPIF_XLATE_CACHE_H | ||
#define OFPROTO_DPIF_XLATE_CACHE_H 1 | ||
|
||
#include <net/if.h> | ||
#include <sys/socket.h> | ||
#include <netinet/in.h> | ||
|
||
#include "openvswitch/types.h" | ||
#include "dp-packet.h" | ||
#include "odp-util.h" | ||
#include "ofproto/ofproto-dpif-mirror.h" | ||
#include "openvswitch/ofpbuf.h" | ||
|
||
struct bfd; | ||
struct bond; | ||
struct dpif_flow_stats; | ||
struct flow; | ||
struct group_dpif; | ||
struct mbridge; | ||
struct netdev; | ||
struct netflow; | ||
struct ofpbuf; | ||
struct ofproto_dpif; | ||
struct ofputil_bucket; | ||
struct ofputil_flow_mod; | ||
struct rule_dpif; | ||
|
||
enum xc_type { | ||
XC_RULE, | ||
XC_BOND, | ||
XC_NETDEV, | ||
XC_NETFLOW, | ||
XC_MIRROR, | ||
XC_LEARN, | ||
XC_NORMAL, | ||
XC_FIN_TIMEOUT, | ||
XC_GROUP, | ||
XC_TNL_NEIGH, | ||
}; | ||
|
||
/* xlate_cache entries hold enough information to perform the side effects of | ||
* xlate_actions() for a rule, without needing to perform rule translation | ||
* from scratch. The primary usage of these is to submit statistics to objects | ||
* that a flow relates to, although they may be used for other effects as well | ||
* (for instance, refreshing hard timeouts for learned flows). | ||
* | ||
* An explicit reference is taken to all pointers other than the ones for | ||
* struct ofproto_dpif. ofproto_dpif pointers are explicitly protected by | ||
* destroying all xlate caches before the ofproto is destroyed. */ | ||
struct xc_entry { | ||
enum xc_type type; | ||
union { | ||
struct rule_dpif *rule; | ||
struct { | ||
struct netdev *tx; | ||
struct netdev *rx; | ||
struct bfd *bfd; | ||
} dev; | ||
struct { | ||
struct netflow *netflow; | ||
struct flow *flow; | ||
ofp_port_t iface; | ||
} nf; | ||
struct { | ||
struct mbridge *mbridge; | ||
mirror_mask_t mirrors; | ||
} mirror; | ||
struct { | ||
struct bond *bond; | ||
struct flow *flow; | ||
uint16_t vid; | ||
} bond; | ||
struct { | ||
struct ofproto_dpif *ofproto; | ||
struct ofputil_flow_mod *fm; | ||
struct ofpbuf *ofpacts; | ||
} learn; | ||
struct { | ||
struct ofproto_dpif *ofproto; | ||
ofp_port_t in_port; | ||
struct eth_addr dl_src; | ||
int vlan; | ||
bool is_gratuitous_arp; | ||
} normal; | ||
struct { | ||
struct rule_dpif *rule; | ||
uint16_t idle; | ||
uint16_t hard; | ||
} fin; | ||
struct { | ||
struct group_dpif *group; | ||
struct ofputil_bucket *bucket; | ||
} group; | ||
struct { | ||
char br_name[IFNAMSIZ]; | ||
struct in6_addr d_ipv6; | ||
} tnl_neigh_cache; | ||
}; | ||
}; | ||
|
||
#define XC_ENTRY_FOR_EACH(ENTRY, ENTRIES) \ | ||
for (ENTRY = ofpbuf_try_pull(ENTRIES, sizeof *ENTRY); \ | ||
ENTRY; \ | ||
ENTRY = ofpbuf_try_pull(ENTRIES, sizeof *ENTRY)) | ||
|
||
struct xlate_cache { | ||
struct ofpbuf entries; | ||
}; | ||
|
||
struct xlate_cache *xlate_cache_new(void); | ||
struct xc_entry *xlate_cache_add_entry(struct xlate_cache *, enum xc_type); | ||
void xlate_push_stats_entry(struct xc_entry *, const struct dpif_flow_stats *); | ||
void xlate_push_stats(struct xlate_cache *, const struct dpif_flow_stats *); | ||
void xlate_cache_clear_entry(struct xc_entry *); | ||
void xlate_cache_clear(struct xlate_cache *); | ||
void xlate_cache_delete(struct xlate_cache *); | ||
|
||
#endif /* ofproto-dpif-xlate-cache.h */ |
Oops, something went wrong.