2013-10-03 18:16:47 -07:00
/*
* Copyright ( c ) 2007 - 2013 Nicira , Inc .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of version 2 of the GNU General Public
* License as published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA
* 02110 - 1301 , USA
*/
# include "flow.h"
# include "datapath.h"
# include <linux/uaccess.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <linux/if_ether.h>
# include <linux/if_vlan.h>
# include <net/llc_pdu.h>
# include <linux/kernel.h>
2013-12-12 16:09:06 +01:00
# include <linux/hash.h>
2013-10-03 18:16:47 -07:00
# include <linux/jiffies.h>
# include <linux/llc.h>
# include <linux/module.h>
# include <linux/in.h>
# include <linux/rcupdate.h>
# include <linux/if_arp.h>
# include <linux/ip.h>
# include <linux/ipv6.h>
# include <linux/sctp.h>
# include <linux/tcp.h>
# include <linux/udp.h>
# include <linux/icmp.h>
# include <linux/icmpv6.h>
# include <linux/rculist.h>
# include <net/ip.h>
# include <net/ipv6.h>
# include <net/ndisc.h>
2013-10-04 00:14:23 -07:00
# define TBL_MIN_BUCKETS 1024
# define REHASH_INTERVAL (10 * 60 * HZ)
2013-10-03 18:16:47 -07:00
static struct kmem_cache * flow_cache ;
static u16 range_n_bytes ( const struct sw_flow_key_range * range )
{
return range - > end - range - > start ;
}
void ovs_flow_mask_key ( struct sw_flow_key * dst , const struct sw_flow_key * src ,
const struct sw_flow_mask * mask )
{
const long * m = ( long * ) ( ( u8 * ) & mask - > key + mask - > range . start ) ;
const long * s = ( long * ) ( ( u8 * ) src + mask - > range . start ) ;
long * d = ( long * ) ( ( u8 * ) dst + mask - > range . start ) ;
int i ;
/* The memory outside of the 'mask->range' are not set since
* further operations on ' dst ' only uses contents within
* ' mask - > range ' .
*/
for ( i = 0 ; i < range_n_bytes ( & mask - > range ) ; i + = sizeof ( long ) )
* d + + = * s + + & * m + + ;
}
2013-10-29 17:22:21 -07:00
struct sw_flow * ovs_flow_alloc ( bool percpu_stats )
2013-10-03 18:16:47 -07:00
{
struct sw_flow * flow ;
2013-10-29 17:22:21 -07:00
int cpu ;
2013-10-03 18:16:47 -07:00
flow = kmem_cache_alloc ( flow_cache , GFP_KERNEL ) ;
if ( ! flow )
return ERR_PTR ( - ENOMEM ) ;
flow - > sf_acts = NULL ;
flow - > mask = NULL ;
2013-10-29 17:22:21 -07:00
flow - > stats . is_percpu = percpu_stats ;
if ( ! percpu_stats ) {
flow - > stats . stat = kzalloc ( sizeof ( * flow - > stats . stat ) , GFP_KERNEL ) ;
if ( ! flow - > stats . stat )
goto err ;
spin_lock_init ( & flow - > stats . stat - > lock ) ;
} else {
flow - > stats . cpu_stats = alloc_percpu ( struct flow_stats ) ;
if ( ! flow - > stats . cpu_stats )
goto err ;
for_each_possible_cpu ( cpu ) {
struct flow_stats * cpu_stats ;
cpu_stats = per_cpu_ptr ( flow - > stats . cpu_stats , cpu ) ;
spin_lock_init ( & cpu_stats - > lock ) ;
}
}
2013-10-03 18:16:47 -07:00
return flow ;
2013-10-29 17:22:21 -07:00
err :
2014-01-08 18:13:14 +08:00
kmem_cache_free ( flow_cache , flow ) ;
2013-10-29 17:22:21 -07:00
return ERR_PTR ( - ENOMEM ) ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
int ovs_flow_tbl_count ( struct flow_table * table )
{
return table - > count ;
}
2013-10-03 18:16:47 -07:00
static struct flex_array * alloc_buckets ( unsigned int n_buckets )
{
struct flex_array * buckets ;
int i , err ;
buckets = flex_array_alloc ( sizeof ( struct hlist_head ) ,
n_buckets , GFP_KERNEL ) ;
if ( ! buckets )
return NULL ;
err = flex_array_prealloc ( buckets , 0 , n_buckets , GFP_KERNEL ) ;
if ( err ) {
flex_array_free ( buckets ) ;
return NULL ;
}
for ( i = 0 ; i < n_buckets ; i + + )
INIT_HLIST_HEAD ( ( struct hlist_head * )
flex_array_get ( buckets , i ) ) ;
return buckets ;
}
static void flow_free ( struct sw_flow * flow )
{
kfree ( ( struct sf_flow_acts __force * ) flow - > sf_acts ) ;
2013-10-29 17:22:21 -07:00
if ( flow - > stats . is_percpu )
free_percpu ( flow - > stats . cpu_stats ) ;
else
kfree ( flow - > stats . stat ) ;
2013-10-03 18:16:47 -07:00
kmem_cache_free ( flow_cache , flow ) ;
}
static void rcu_free_flow_callback ( struct rcu_head * rcu )
{
struct sw_flow * flow = container_of ( rcu , struct sw_flow , rcu ) ;
flow_free ( flow ) ;
}
2013-10-04 00:17:42 -07:00
static void flow_mask_del_ref ( struct sw_flow_mask * mask , bool deferred )
{
if ( ! mask )
return ;
BUG_ON ( ! mask - > ref_count ) ;
mask - > ref_count - - ;
if ( ! mask - > ref_count ) {
list_del_rcu ( & mask - > list ) ;
if ( deferred )
2013-12-10 12:02:03 +01:00
kfree_rcu ( mask , rcu ) ;
2013-10-04 00:17:42 -07:00
else
kfree ( mask ) ;
}
}
2013-10-03 18:16:47 -07:00
void ovs_flow_free ( struct sw_flow * flow , bool deferred )
{
if ( ! flow )
return ;
2013-10-04 00:17:42 -07:00
flow_mask_del_ref ( flow - > mask , deferred ) ;
2013-10-03 18:16:47 -07:00
if ( deferred )
call_rcu ( & flow - > rcu , rcu_free_flow_callback ) ;
else
flow_free ( flow ) ;
}
static void free_buckets ( struct flex_array * buckets )
{
flex_array_free ( buckets ) ;
}
2013-10-04 00:14:23 -07:00
static void __table_instance_destroy ( struct table_instance * ti )
2013-10-03 18:16:47 -07:00
{
int i ;
2013-10-04 00:14:23 -07:00
if ( ti - > keep_flows )
2013-10-03 18:16:47 -07:00
goto skip_flows ;
2013-10-04 00:14:23 -07:00
for ( i = 0 ; i < ti - > n_buckets ; i + + ) {
2013-10-03 18:16:47 -07:00
struct sw_flow * flow ;
2013-10-04 00:14:23 -07:00
struct hlist_head * head = flex_array_get ( ti - > buckets , i ) ;
2013-10-03 18:16:47 -07:00
struct hlist_node * n ;
2013-10-04 00:14:23 -07:00
int ver = ti - > node_ver ;
2013-10-03 18:16:47 -07:00
hlist_for_each_entry_safe ( flow , n , head , hash_node [ ver ] ) {
hlist_del ( & flow - > hash_node [ ver ] ) ;
ovs_flow_free ( flow , false ) ;
}
}
skip_flows :
2013-10-04 00:14:23 -07:00
free_buckets ( ti - > buckets ) ;
kfree ( ti ) ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
static struct table_instance * table_instance_alloc ( int new_size )
2013-10-03 18:16:47 -07:00
{
2013-10-04 00:14:23 -07:00
struct table_instance * ti = kmalloc ( sizeof ( * ti ) , GFP_KERNEL ) ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
if ( ! ti )
2013-10-03 18:16:47 -07:00
return NULL ;
2013-10-04 00:14:23 -07:00
ti - > buckets = alloc_buckets ( new_size ) ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
if ( ! ti - > buckets ) {
kfree ( ti ) ;
2013-10-03 18:16:47 -07:00
return NULL ;
}
2013-10-04 00:14:23 -07:00
ti - > n_buckets = new_size ;
ti - > node_ver = 0 ;
ti - > keep_flows = false ;
get_random_bytes ( & ti - > hash_seed , sizeof ( u32 ) ) ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
return ti ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
int ovs_flow_tbl_init ( struct flow_table * table )
2013-10-03 18:16:47 -07:00
{
2013-10-04 00:14:23 -07:00
struct table_instance * ti ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
ti = table_instance_alloc ( TBL_MIN_BUCKETS ) ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
if ( ! ti )
return - ENOMEM ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
rcu_assign_pointer ( table - > ti , ti ) ;
INIT_LIST_HEAD ( & table - > mask_list ) ;
table - > last_rehash = jiffies ;
table - > count = 0 ;
return 0 ;
2013-10-03 18:16:47 -07:00
}
static void flow_tbl_destroy_rcu_cb ( struct rcu_head * rcu )
{
2013-10-04 00:14:23 -07:00
struct table_instance * ti = container_of ( rcu , struct table_instance , rcu ) ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
__table_instance_destroy ( ti ) ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
static void table_instance_destroy ( struct table_instance * ti , bool deferred )
2013-10-03 18:16:47 -07:00
{
2013-10-04 00:14:23 -07:00
if ( ! ti )
2013-10-03 18:16:47 -07:00
return ;
if ( deferred )
2013-10-04 00:14:23 -07:00
call_rcu ( & ti - > rcu , flow_tbl_destroy_rcu_cb ) ;
2013-10-03 18:16:47 -07:00
else
2013-10-04 00:14:23 -07:00
__table_instance_destroy ( ti ) ;
}
2013-10-04 00:17:42 -07:00
void ovs_flow_tbl_destroy ( struct flow_table * table )
2013-10-04 00:14:23 -07:00
{
struct table_instance * ti = ovsl_dereference ( table - > ti ) ;
2013-10-04 00:17:42 -07:00
table_instance_destroy ( ti , false ) ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
struct sw_flow * ovs_flow_tbl_dump_next ( struct table_instance * ti ,
2013-10-03 18:16:47 -07:00
u32 * bucket , u32 * last )
{
struct sw_flow * flow ;
struct hlist_head * head ;
int ver ;
int i ;
2013-10-04 00:14:23 -07:00
ver = ti - > node_ver ;
while ( * bucket < ti - > n_buckets ) {
2013-10-03 18:16:47 -07:00
i = 0 ;
2013-10-04 00:14:23 -07:00
head = flex_array_get ( ti - > buckets , * bucket ) ;
2013-10-03 18:16:47 -07:00
hlist_for_each_entry_rcu ( flow , head , hash_node [ ver ] ) {
if ( i < * last ) {
i + + ;
continue ;
}
* last = i + 1 ;
return flow ;
}
( * bucket ) + + ;
* last = 0 ;
}
return NULL ;
}
2013-10-04 00:14:23 -07:00
static struct hlist_head * find_bucket ( struct table_instance * ti , u32 hash )
2013-10-03 18:16:47 -07:00
{
2013-10-04 00:14:23 -07:00
hash = jhash_1word ( hash , ti - > hash_seed ) ;
return flex_array_get ( ti - > buckets ,
( hash & ( ti - > n_buckets - 1 ) ) ) ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
static void table_instance_insert ( struct table_instance * ti , struct sw_flow * flow )
2013-10-03 18:16:47 -07:00
{
struct hlist_head * head ;
2013-10-04 00:14:23 -07:00
head = find_bucket ( ti , flow - > hash ) ;
hlist_add_head_rcu ( & flow - > hash_node [ ti - > node_ver ] , head ) ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
static void flow_table_copy_flows ( struct table_instance * old ,
struct table_instance * new )
2013-10-03 18:16:47 -07:00
{
int old_ver ;
int i ;
old_ver = old - > node_ver ;
new - > node_ver = ! old_ver ;
/* Insert in new table. */
for ( i = 0 ; i < old - > n_buckets ; i + + ) {
struct sw_flow * flow ;
struct hlist_head * head ;
head = flex_array_get ( old - > buckets , i ) ;
hlist_for_each_entry ( flow , head , hash_node [ old_ver ] )
2013-10-04 00:14:23 -07:00
table_instance_insert ( new , flow ) ;
2013-10-03 18:16:47 -07:00
}
old - > keep_flows = true ;
}
2013-10-04 00:14:23 -07:00
static struct table_instance * table_instance_rehash ( struct table_instance * ti ,
2013-10-03 18:16:47 -07:00
int n_buckets )
{
2013-10-04 00:14:23 -07:00
struct table_instance * new_ti ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
new_ti = table_instance_alloc ( n_buckets ) ;
if ( ! new_ti )
2013-10-04 00:17:42 -07:00
return NULL ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
flow_table_copy_flows ( ti , new_ti ) ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
return new_ti ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
int ovs_flow_tbl_flush ( struct flow_table * flow_table )
2013-10-03 18:16:47 -07:00
{
2013-10-04 00:14:23 -07:00
struct table_instance * old_ti ;
struct table_instance * new_ti ;
2013-10-03 18:16:47 -07:00
2013-10-04 00:14:23 -07:00
old_ti = ovsl_dereference ( flow_table - > ti ) ;
new_ti = table_instance_alloc ( TBL_MIN_BUCKETS ) ;
if ( ! new_ti )
return - ENOMEM ;
rcu_assign_pointer ( flow_table - > ti , new_ti ) ;
flow_table - > last_rehash = jiffies ;
flow_table - > count = 0 ;
table_instance_destroy ( old_ti , true ) ;
return 0 ;
2013-10-03 18:16:47 -07:00
}
static u32 flow_hash ( const struct sw_flow_key * key , int key_start ,
int key_end )
{
u32 * hash_key = ( u32 * ) ( ( u8 * ) key + key_start ) ;
int hash_u32s = ( key_end - key_start ) > > 2 ;
/* Make sure number of hash bytes are multiple of u32. */
BUILD_BUG_ON ( sizeof ( long ) % sizeof ( u32 ) ) ;
2013-12-12 16:09:06 +01:00
return arch_fast_hash2 ( hash_key , hash_u32s , 0 ) ;
2013-10-03 18:16:47 -07:00
}
static int flow_key_start ( const struct sw_flow_key * key )
{
if ( key - > tun_key . ipv4_dst )
return 0 ;
else
return rounddown ( offsetof ( struct sw_flow_key , phy ) ,
sizeof ( long ) ) ;
}
static bool cmp_key ( const struct sw_flow_key * key1 ,
const struct sw_flow_key * key2 ,
int key_start , int key_end )
{
const long * cp1 = ( long * ) ( ( u8 * ) key1 + key_start ) ;
const long * cp2 = ( long * ) ( ( u8 * ) key2 + key_start ) ;
long diffs = 0 ;
int i ;
for ( i = key_start ; i < key_end ; i + = sizeof ( long ) )
diffs | = * cp1 + + ^ * cp2 + + ;
return diffs = = 0 ;
}
static bool flow_cmp_masked_key ( const struct sw_flow * flow ,
const struct sw_flow_key * key ,
int key_start , int key_end )
{
return cmp_key ( & flow - > key , key , key_start , key_end ) ;
}
bool ovs_flow_cmp_unmasked_key ( const struct sw_flow * flow ,
struct sw_flow_match * match )
{
struct sw_flow_key * key = match - > key ;
int key_start = flow_key_start ( key ) ;
int key_end = match - > range . end ;
return cmp_key ( & flow - > unmasked_key , key , key_start , key_end ) ;
}
2013-10-04 00:14:23 -07:00
static struct sw_flow * masked_flow_lookup ( struct table_instance * ti ,
2013-10-03 18:16:47 -07:00
const struct sw_flow_key * unmasked ,
struct sw_flow_mask * mask )
{
struct sw_flow * flow ;
struct hlist_head * head ;
int key_start = mask - > range . start ;
int key_end = mask - > range . end ;
u32 hash ;
struct sw_flow_key masked_key ;
ovs_flow_mask_key ( & masked_key , unmasked , mask ) ;
hash = flow_hash ( & masked_key , key_start , key_end ) ;
2013-10-04 00:14:23 -07:00
head = find_bucket ( ti , hash ) ;
hlist_for_each_entry_rcu ( flow , head , hash_node [ ti - > node_ver ] ) {
2013-10-29 23:10:58 -07:00
if ( flow - > mask = = mask & & flow - > hash = = hash & &
2013-10-03 18:16:47 -07:00
flow_cmp_masked_key ( flow , & masked_key ,
key_start , key_end ) )
return flow ;
}
return NULL ;
}
2013-11-25 10:42:46 -08:00
struct sw_flow * ovs_flow_tbl_lookup_stats ( struct flow_table * tbl ,
2013-10-22 10:42:46 -07:00
const struct sw_flow_key * key ,
u32 * n_mask_hit )
2013-10-03 18:16:47 -07:00
{
2013-12-03 10:58:53 -08:00
struct table_instance * ti = rcu_dereference_ovsl ( tbl - > ti ) ;
2013-10-03 18:16:47 -07:00
struct sw_flow_mask * mask ;
2013-10-04 00:14:23 -07:00
struct sw_flow * flow ;
2013-10-03 18:16:47 -07:00
2013-10-22 10:42:46 -07:00
* n_mask_hit = 0 ;
2013-10-04 00:14:23 -07:00
list_for_each_entry_rcu ( mask , & tbl - > mask_list , list ) {
2013-10-22 10:42:46 -07:00
( * n_mask_hit ) + + ;
2013-10-04 00:14:23 -07:00
flow = masked_flow_lookup ( ti , key , mask ) ;
2013-10-03 18:16:47 -07:00
if ( flow ) /* Found */
2013-10-04 00:14:23 -07:00
return flow ;
2013-10-03 18:16:47 -07:00
}
2013-10-04 00:14:23 -07:00
return NULL ;
}
2013-10-03 18:16:47 -07:00
2013-11-25 10:42:46 -08:00
struct sw_flow * ovs_flow_tbl_lookup ( struct flow_table * tbl ,
const struct sw_flow_key * key )
{
u32 __always_unused n_mask_hit ;
return ovs_flow_tbl_lookup_stats ( tbl , key , & n_mask_hit ) ;
}
2013-10-22 10:42:46 -07:00
int ovs_flow_tbl_num_masks ( const struct flow_table * table )
{
struct sw_flow_mask * mask ;
int num = 0 ;
list_for_each_entry ( mask , & table - > mask_list , list )
num + + ;
return num ;
}
2013-10-04 00:14:23 -07:00
static struct table_instance * table_instance_expand ( struct table_instance * ti )
{
return table_instance_rehash ( ti , ti - > n_buckets * 2 ) ;
2013-10-03 18:16:47 -07:00
}
void ovs_flow_tbl_remove ( struct flow_table * table , struct sw_flow * flow )
{
2013-10-04 00:14:23 -07:00
struct table_instance * ti = ovsl_dereference ( table - > ti ) ;
2013-10-03 18:16:47 -07:00
BUG_ON ( table - > count = = 0 ) ;
2013-10-04 00:14:23 -07:00
hlist_del_rcu ( & flow - > hash_node [ ti - > node_ver ] ) ;
2013-10-03 18:16:47 -07:00
table - > count - - ;
}
2013-10-04 00:17:42 -07:00
static struct sw_flow_mask * mask_alloc ( void )
2013-10-03 18:16:47 -07:00
{
struct sw_flow_mask * mask ;
mask = kmalloc ( sizeof ( * mask ) , GFP_KERNEL ) ;
if ( mask )
mask - > ref_count = 0 ;
return mask ;
}
2013-10-04 00:17:42 -07:00
static void mask_add_ref ( struct sw_flow_mask * mask )
2013-10-03 18:16:47 -07:00
{
mask - > ref_count + + ;
}
static bool mask_equal ( const struct sw_flow_mask * a ,
const struct sw_flow_mask * b )
{
u8 * a_ = ( u8 * ) & a - > key + a - > range . start ;
u8 * b_ = ( u8 * ) & b - > key + b - > range . start ;
return ( a - > range . end = = b - > range . end )
& & ( a - > range . start = = b - > range . start )
& & ( memcmp ( a_ , b_ , range_n_bytes ( & a - > range ) ) = = 0 ) ;
}
2013-10-04 00:17:42 -07:00
static struct sw_flow_mask * flow_mask_find ( const struct flow_table * tbl ,
2013-10-03 18:16:47 -07:00
const struct sw_flow_mask * mask )
{
struct list_head * ml ;
2013-10-04 00:14:23 -07:00
list_for_each ( ml , & tbl - > mask_list ) {
2013-10-03 18:16:47 -07:00
struct sw_flow_mask * m ;
m = container_of ( ml , struct sw_flow_mask , list ) ;
if ( mask_equal ( mask , m ) )
return m ;
}
return NULL ;
}
2013-11-25 10:40:51 -08:00
/* Add 'mask' into the mask list, if it is not already there. */
2013-10-04 00:17:42 -07:00
static int flow_mask_insert ( struct flow_table * tbl , struct sw_flow * flow ,
struct sw_flow_mask * new )
{
struct sw_flow_mask * mask ;
mask = flow_mask_find ( tbl , new ) ;
if ( ! mask ) {
/* Allocate a new mask if none exsits. */
mask = mask_alloc ( ) ;
if ( ! mask )
return - ENOMEM ;
mask - > key = new - > key ;
mask - > range = new - > range ;
list_add_rcu ( & mask - > list , & tbl - > mask_list ) ;
}
mask_add_ref ( mask ) ;
flow - > mask = mask ;
return 0 ;
}
int ovs_flow_tbl_insert ( struct flow_table * table , struct sw_flow * flow ,
struct sw_flow_mask * mask )
2013-10-03 18:16:47 -07:00
{
2013-10-04 00:17:42 -07:00
struct table_instance * new_ti = NULL ;
struct table_instance * ti ;
int err ;
err = flow_mask_insert ( table , flow , mask ) ;
if ( err )
return err ;
flow - > hash = flow_hash ( & flow - > key , flow - > mask - > range . start ,
flow - > mask - > range . end ) ;
ti = ovsl_dereference ( table - > ti ) ;
table_instance_insert ( ti , flow ) ;
table - > count + + ;
/* Expand table, if necessary, to make room. */
if ( table - > count > ti - > n_buckets )
new_ti = table_instance_expand ( ti ) ;
else if ( time_after ( jiffies , table - > last_rehash + REHASH_INTERVAL ) )
new_ti = table_instance_rehash ( ti , ti - > n_buckets ) ;
if ( new_ti ) {
rcu_assign_pointer ( table - > ti , new_ti ) ;
table_instance_destroy ( ti , true ) ;
table - > last_rehash = jiffies ;
}
return 0 ;
2013-10-03 18:16:47 -07:00
}
/* Initializes the flow module.
* Returns zero if successful or a negative error code . */
int ovs_flow_init ( void )
{
BUILD_BUG_ON ( __alignof__ ( struct sw_flow_key ) % __alignof__ ( long ) ) ;
BUILD_BUG_ON ( sizeof ( struct sw_flow_key ) % sizeof ( long ) ) ;
flow_cache = kmem_cache_create ( " sw_flow " , sizeof ( struct sw_flow ) , 0 ,
0 , NULL ) ;
if ( flow_cache = = NULL )
return - ENOMEM ;
return 0 ;
}
/* Uninitializes the flow module. */
void ovs_flow_exit ( void )
{
kmem_cache_destroy ( flow_cache ) ;
}