vif.c

来自「xen虚拟机源代码安装包」· C语言 代码 · 共 388 行

C
388
字号
/* * Copyright (C) 2004, 2005 Mike Wray <mike.wray@hp.com> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by the  * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. *  * 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., * 59 Temple Place, suite 330, Boston, MA 02111-1307 USA * */#ifdef __KERNEL__#include <linux/config.h>#include <linux/kernel.h>#include <linux/module.h>#include <linux/spinlock.h>#else#include "sys_kernel.h"#include "spinlock.h"#include "skbuff.h"#endif#include <vif.h>#include <varp.h>#include <varp_util.h>#include "allocate.h"#include "iostream.h"#include "hash_table.h"#include "timer_util.h"#define MODULE_NAME "VNET"#define DEBUG 1#undef DEBUG#include "debug.h"/** Vif table ttl - interval between sweeps of old vifs. */#define VIF_TABLE_TTL (60*HZ)/** Vif entry ttl - a vif entry older than this is removed. */#define VIF_ENTRY_TTL (60*HZ)/** Table of vifs indexed by VifKey. */HashTable *vif_table = NULL;rwlock_t vif_table_lock = RW_LOCK_UNLOCKED;struct timer_list vif_table_timer = {};int vif_table_sweeps = 0;#define vif_read_lock(flags)    read_lock_irqsave(&vif_table_lock, (flags))#define vif_read_unlock(flags)  read_unlock_irqrestore(&vif_table_lock, (flags))#define vif_write_lock(flags)   write_lock_irqsave(&vif_table_lock, (flags))#define vif_write_unlock(flags) write_unlock_irqrestore(&vif_table_lock, (flags))void vif_entry_print(Vif *vif, IOStream *io){    char vnetbuf[VNET_ID_BUF];    unsigned long now = jiffies;    IOStream_print(io, "(vif\n");    IOStream_print(io, " (vnet %s)\n", VnetId_ntoa(&vif->vnet, vnetbuf));    IOStream_print(io, " (vmac " MACFMT ")\n", MAC6TUPLE(vif->vmac.mac));    IOStream_print(io, " (age %u)\n", now - vif->timestamp);    IOStream_print(io, ")\n");}void vif_print(IOStream *io){    HashTable_for_decl(entry);    Vif *vif;    unsigned long flags;    vif_read_lock(flags);    IOStream_print(io, "(viftable\n");    IOStream_print(io, " (table_ttl %u)\n", VIF_TABLE_TTL);    IOStream_print(io, " (entry_ttl %u)\n", VIF_ENTRY_TTL);    IOStream_print(io, " (sweeps %d)\n", vif_table_sweeps);    IOStream_print(io, ")\n");        HashTable_for_each(entry, vif_table){        vif = entry->value;        vif_entry_print(vif, io);    }    vif_read_unlock(flags);}void vif_decref(Vif *vif){    if(!vif) return;    if(atomic_dec_and_test(&vif->refcount)){        kfree(vif);    }}void vif_incref(Vif *vif){    if(!vif) return;    atomic_inc(&vif->refcount);}/** Hash function for keys in the vif table. * Hashes the vnet id and mac. * * @param k key (VifKey) * @return hashcode */static Hashcode vif_key_hash_fn(void *k){    return hash_hvoid(0, k, sizeof(VifKey));}/** Test equality for keys in the vif table. * Compares vnet and mac. * * @param k1 key to compare (VifKey) * @param k2 key to compare (VifKey) * @return 1 if equal, 0 otherwise */static int vif_key_equal_fn(void *k1, void *k2){    return memcmp(k1, k2, sizeof(VifKey)) == 0;}/** Free an entry in the vif table. * * @param table containing table * @param entry entry to free */static void vif_entry_free_fn(HashTable *table, HTEntry *entry){    Vif *vif;    if(!entry) return;    vif = entry->value;    if(vif){        vif_decref(vif);    }    HTEntry_free(entry);}/** Lookup a vif. * Caller must hold vif lock. * * @param vnet vnet id * @param mac MAC address * @return 0 on success, -ENOENT otherwise */static int _vif_lookup(VnetId *vnet, Vmac *vmac, Vif **vif){    int err = 0;    VifKey key = { .vnet = *vnet, .vmac = *vmac };    HTEntry *entry = NULL;        entry = HashTable_get_entry(vif_table, &key);    if(entry){        *vif = entry->value;        vif_incref(*vif);    } else {        *vif = NULL;        err = -ENOENT;    }    return err;}/** Lookup a vif. * * @param vnet vnet id * @param mac MAC address * @return 0 on success, -ENOENT otherwise */int vif_lookup(VnetId *vnet, Vmac *vmac, Vif **vif){    unsigned long flags;        int err;    vif_read_lock(flags);    err = _vif_lookup(vnet, vmac, vif);    vif_read_unlock(flags);    return err;}/** Create a new vif. * Entry must not exist. * Caller must hold vif lock. * * @param vnet vnet id * @param mac MAC address * @return 0 on success, negative error code otherwise */static int _vif_add(VnetId *vnet, Vmac *vmac, Vif **val){    int err = 0;    Vif *vif = NULL;    HTEntry *entry;    unsigned long now = jiffies;    vif = ALLOCATE(Vif);    if(!vif){        err = -ENOMEM;        goto exit;    }    atomic_set(&vif->refcount, 1);    vif->vnet = *vnet;    vif->vmac = *vmac;    vif->timestamp = now;    entry = HashTable_add(vif_table, vif, vif);    if(!entry){        err = -ENOMEM;        deallocate(vif);        vif = NULL;        goto exit;    }    vif_incref(vif);  exit:    *val = (err ? NULL : vif);    return err;}/** Delete a vif entry. * * @param vnet vnet id * @param mac MAC address * @return number of entries deleted, or negative error code */int vif_remove(VnetId *vnet, Vmac *vmac){    int err = 0;    VifKey key = { .vnet = *vnet, .vmac = *vmac };    unsigned long flags;    vif_write_lock(flags);    err = HashTable_remove(vif_table, &key);    vif_write_unlock(flags);    return err;}/** Delete all vifs on a vnet. * * @param vnet vnet id * @return number of entries deleted */int vif_remove_vnet(VnetId *vnet){    int count = 0;    unsigned long flags;    HashTable_for_decl(entry);        vif_write_lock(flags);    HashTable_for_each(entry, vif_table){        Vif *vif = entry->value;        if(VnetId_eq(&vif->vnet, vnet)){            count += HashTable_remove(vif_table, vif);        }    }    vif_write_unlock(flags);    return count;}/** Purge the vif table. */void vif_purge(void){    unsigned long flags;    vif_write_lock(flags);    HashTable_clear(vif_table);    vif_write_unlock(flags);}/** Sweep old vif entries from the vif table. */void vif_sweep(void){    HashTable_for_decl(entry);    Vif *vif;    int vif_count = 0;    unsigned long now = jiffies;    unsigned long old = VIF_ENTRY_TTL;    unsigned long flags;    vif_write_lock(flags);    vif_table_sweeps++;    HashTable_for_each(entry, vif_table){        vif = entry->value;        vif_count++;        if(!(vif->flags & VIF_FLAG_PERSISTENT)           && (now - vif->timestamp > old)){            iprintf("> Sweeping:\n");            vif_entry_print(vif, iostdout);            HashTable_remove(vif_table, entry->key);        }    }    vif_write_unlock(flags);}/** Create a new vif if it does not exist. * Caller must hold vif lock. * * @param vnet vnet id * @param mac MAC address * @return 0 on success, negative error code otherwise */int _vif_create(VnetId *vnet, Vmac *vmac, Vif **vif){    int err = 0;    if(_vif_lookup(vnet, vmac, vif) == 0){        goto exit;    }    err = _vif_add(vnet, vmac, vif);  exit:    return err;}/** Create a new vif if it does not exist. * * @param vnet vnet id * @param mac MAC address * @return 0 on success, negative error code otherwise */int vif_create(VnetId *vnet, Vmac *vmac, int vflags, Vif **vif){    int err = 0;    unsigned long flags;    vif_write_lock(flags);    err = _vif_create(vnet, vmac, vif);    if(!err && *vif){        (*vif)->flags = vflags;    }    vif_write_unlock(flags);    return err;}/** Update the timestamp for a vif. * * @param vnet vnet id * @param mac MAC address * @return 0 on success, negative error code otherwise */int vif_update(VnetId *vnet, Vmac *vmac){    Vif *vif = NULL;    int err = 0;    unsigned long now = jiffies;    unsigned long flags;    vif_write_lock(flags);    err = _vif_create(vnet, vmac, &vif);    if(err) goto exit;    vif->timestamp = now;    vif_decref(vif);  exit:    vif_write_unlock(flags);    return err;}static void vif_table_timer_fn(unsigned long arg){    if(!vif_table) return;    vif_sweep();    timer_set(&vif_table_timer, VIF_TABLE_TTL);}    /** Initialize the vif table. * * @return 0 on success, error code otherwise */int vif_init(void){    int err = 0;    vif_table = HashTable_new(0);    if(!vif_table){        err = -ENOMEM;        goto exit;    }    vif_table->entry_free_fn = vif_entry_free_fn;    vif_table->key_size = sizeof(VifKey);    vif_table->key_hash_fn   = vif_key_hash_fn;    vif_table->key_equal_fn  = vif_key_equal_fn;    timer_init(&vif_table_timer, vif_table_timer_fn, 0);    timer_set(&vif_table_timer, VIF_TABLE_TTL);  exit:    if(err < 0){        eprintf("> vif_init err=%d\n", err);    }    return err;}void vif_exit(void){    timer_cancel(&vif_table_timer);    HashTable_free(vif_table);    vif_table = NULL;}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?