Skip to content

Commit

Permalink
ofproto-dpif-xlate: Expose xlate cache.
Browse files Browse the repository at this point in the history
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
Show file tree
Hide file tree
Showing 6 changed files with 413 additions and 298 deletions.
2 changes: 2 additions & 0 deletions ofproto/automake.mk
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,8 @@ ofproto_libofproto_la_SOURCES = \
ofproto/ofproto-dpif-xlate.c \
ofproto/ofproto-dpif-xlate.h \
ofproto/ofproto-provider.h \
ofproto/ofproto-dpif-xlate-cache.c \
ofproto/ofproto-dpif-xlate-cache.h \
ofproto/pinsched.c \
ofproto/pinsched.h \
ofproto/tunnel.c \
Expand Down
1 change: 1 addition & 0 deletions ofproto/ofproto-dpif-upcall.c
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@
#include "ofproto-dpif-ipfix.h"
#include "ofproto-dpif-sflow.h"
#include "ofproto-dpif-xlate.h"
#include "ofproto-dpif-xlate-cache.h"
#include "ovs-rcu.h"
#include "packets.h"
#include "poll-loop.h"
Expand Down
239 changes: 239 additions & 0 deletions ofproto/ofproto-dpif-xlate-cache.c
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);
}
132 changes: 132 additions & 0 deletions ofproto/ofproto-dpif-xlate-cache.h
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 */
Loading

0 comments on commit 901a517

Please sign in to comment.