297 lines
5.9 KiB
C
297 lines
5.9 KiB
C
|
|
|||
|
#include <linux/kernel.h>
|
|||
|
#include <linux/module.h>
|
|||
|
#include <linux/init.h>
|
|||
|
#include <linux/timer.h>
|
|||
|
#include <linux/uaccess.h>
|
|||
|
#include <linux/sysctl.h>
|
|||
|
#include <linux/ip.h>
|
|||
|
#include <linux/ipv6.h>
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
* 三元组超时时间设置为120s
|
|||
|
*/
|
|||
|
int dpi_cache_node_timeout = 120;
|
|||
|
|
|||
|
/*
|
|||
|
* 三元组数据结构
|
|||
|
*/
|
|||
|
struct dpi_cache_node {
|
|||
|
struct list_head list;
|
|||
|
|
|||
|
union {
|
|||
|
struct in6_addr ip6;
|
|||
|
struct in_addr ip4;
|
|||
|
|
|||
|
}ip;
|
|||
|
|
|||
|
__u16 proto;
|
|||
|
__u16 port;
|
|||
|
__u32 appid;
|
|||
|
uint64_t uptime;
|
|||
|
};
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
* 初始化cache链表
|
|||
|
*/
|
|||
|
#define TRIPLE_CAHCE_HASH_SIZE (2048)
|
|||
|
struct list_head dpi_cache_head[TRIPLE_CAHCE_HASH_SIZE] = {0};
|
|||
|
|
|||
|
typedef struct pkt_info{
|
|||
|
unsigned int srcip;
|
|||
|
unsigned short sport;
|
|||
|
unsigned int dstip;
|
|||
|
unsigned short dport;
|
|||
|
unsigned short proto;
|
|||
|
}pkt_info_t;
|
|||
|
|
|||
|
#define POLY 0x01101 // CRC20生成多项式x^20+x^12+x^8+1即:01101 CRC32:04C11DB7L
|
|||
|
static unsigned int crc_table[256] = {0};
|
|||
|
|
|||
|
unsigned int get_sum_poly(unsigned char data)
|
|||
|
{
|
|||
|
unsigned int sum_poly = data;
|
|||
|
int j;
|
|||
|
sum_poly <<= 24;
|
|||
|
for(j = 0; j < 8; j++)
|
|||
|
{
|
|||
|
int hi = sum_poly&0x80000000; // 取得reg的最高位
|
|||
|
sum_poly <<= 1;
|
|||
|
if(hi) sum_poly = sum_poly^POLY;
|
|||
|
}
|
|||
|
return sum_poly;
|
|||
|
}
|
|||
|
|
|||
|
void create_crc_table(void) //在使用CRC20_key函数应该先建立crc表
|
|||
|
{
|
|||
|
int i;
|
|||
|
for(i = 0; i < 256; i++)
|
|||
|
{
|
|||
|
crc_table[i] = get_sum_poly(i&0xFF);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
unsigned int CRC20_key(unsigned char* data, int len)
|
|||
|
{
|
|||
|
int i;
|
|||
|
unsigned int reg = 0xFFFFFFFF;// 0xFFFFFFFF,见后面解释
|
|||
|
for(i = 0; i < len; i++)
|
|||
|
{
|
|||
|
reg = (reg<<8) ^ crc_table[((reg>>24)&0xFF) ^ data[i]];
|
|||
|
}
|
|||
|
return (reg&0XFFFFF);//得到的reg取后20作为key值
|
|||
|
}
|
|||
|
|
|||
|
/*
|
|||
|
* 初始化dpi三元组hash链表
|
|||
|
*/
|
|||
|
void dpi_cache_list_init(void)
|
|||
|
{
|
|||
|
int i = 0;
|
|||
|
for (i = 0; i < TRIPLE_CAHCE_HASH_SIZE; i++){
|
|||
|
INIT_LIST_HEAD(&dpi_cache_head[i]);
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
* 三元组hash值计算
|
|||
|
*/
|
|||
|
int dpi_cache_node_hash(struct dpi_cache_node node)
|
|||
|
{
|
|||
|
int i = 0;
|
|||
|
int dip = 0;
|
|||
|
|
|||
|
pkt_info_t info;
|
|||
|
info.dport = node.port;
|
|||
|
info.sport = 0;
|
|||
|
info.srcip = 0;
|
|||
|
info.proto = node.proto;
|
|||
|
if (node.ip.ip4.s_addr !=0 ) {
|
|||
|
info.dstip = node.ip.ip4.s_addr;
|
|||
|
} else {
|
|||
|
for (i = 0; i < 4; ++i) {
|
|||
|
dip ^= node.ip.ip6.s6_addr32[i];
|
|||
|
}
|
|||
|
info.dstip = dip;
|
|||
|
}
|
|||
|
|
|||
|
return CRC20_key((unsigned char *)&info, sizeof(pkt_info_t))&(TRIPLE_CAHCE_HASH_SIZE - 1);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
* src 和dst比较,并更新dst的appid,update_appid=true,update
|
|||
|
*/
|
|||
|
bool dpi_cache_node_compare(struct dpi_cache_node src,struct dpi_cache_node dst,bool update_appid)
|
|||
|
{
|
|||
|
int i = 0;
|
|||
|
if (src.port != dst.port){
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
if (src.proto != dst.proto){
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
if(src.ip.ip4.s_addr != 0){
|
|||
|
if(src.ip.ip4.s_addr != dst.ip.ip4.s_addr){
|
|||
|
return false;
|
|||
|
}
|
|||
|
}else{
|
|||
|
for (i = 0; i < 4; i++){
|
|||
|
if (src.ip.ip6.s6_addr32[i] != dst.ip.ip6.s6_addr32[i]){
|
|||
|
return false;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
if (update_appid){
|
|||
|
dst.appid = src.appid;
|
|||
|
dst.uptime = get_jiffies_64();
|
|||
|
}
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
*遍历hash链表,查找node
|
|||
|
*/
|
|||
|
bool dpi_cache_node_search(struct dpi_cache_node node,int flag)
|
|||
|
{
|
|||
|
struct list_head * pList;
|
|||
|
struct dpi_cache_node *pNode;
|
|||
|
int hash = dpi_cache_node_hash(node);
|
|||
|
list_for_each(pList,&dpi_cache_head[hash]){
|
|||
|
pNode = list_entry(pList,struct dpi_cache_node,list);
|
|||
|
if (dpi_cache_node_compare(node,*pNode,flag)){
|
|||
|
return true;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
* DPI 信息记录,外部接口调用
|
|||
|
*/
|
|||
|
bool dpi_cache_node_add(struct dpi_cache_node node)
|
|||
|
{
|
|||
|
int hash = 0;
|
|||
|
struct dpi_cache_node *pNode =NULL;
|
|||
|
|
|||
|
if (dpi_cache_node_search(node,true)) {
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
pNode = kmalloc(sizeof(struct dpi_cache_node), GFP_KERNEL);
|
|||
|
if (NULL == pNode) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
hash = dpi_cache_node_hash(node);
|
|||
|
pNode->uptime = get_jiffies_64();
|
|||
|
pNode->appid = node.appid;
|
|||
|
list_add_tail(&pNode->list,&dpi_cache_head[hash]);
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
* 释放hash链表上所有node节点
|
|||
|
*/
|
|||
|
void dpi_cache_list_release(void)
|
|||
|
{
|
|||
|
struct list_head * pList = NULL;
|
|||
|
struct dpi_cache_node *pNode = NULL;
|
|||
|
int i = 0;
|
|||
|
|
|||
|
for (i = 0; i < TRIPLE_CAHCE_HASH_SIZE; i++){
|
|||
|
list_for_each(pList,&dpi_cache_head[i]){
|
|||
|
pNode = list_entry(pList,struct dpi_cache_node,list);
|
|||
|
if(pNode){
|
|||
|
list_del(&pNode->list);
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*
|
|||
|
*定时器超时,删除超时的节点
|
|||
|
*/
|
|||
|
void dpi_cache_node_timeout_func(void)
|
|||
|
{
|
|||
|
struct list_head * pList = NULL;
|
|||
|
struct dpi_cache_node *pNode = NULL;
|
|||
|
int i = 0;
|
|||
|
|
|||
|
for (i = 0; i < TRIPLE_CAHCE_HASH_SIZE; i++){
|
|||
|
list_for_each(pList,&dpi_cache_head[i]){
|
|||
|
pNode = list_entry(pList,struct dpi_cache_node,list);
|
|||
|
if(pNode&&
|
|||
|
(get_jiffies_64() - pNode->uptime >= dpi_cache_node_timeout)){
|
|||
|
list_del(&pNode->list);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/*
|
|||
|
* 定时器操作
|
|||
|
*/
|
|||
|
struct timer_list gTimer;
|
|||
|
|
|||
|
void dpi_cache_timer_handler(unsigned long data) {
|
|||
|
printk(KERN_INFO"timer pending:%d\n", timer_pending(&gTimer));
|
|||
|
dpi_cache_node_timeout_func();
|
|||
|
mod_timer(&gTimer, jiffies+msecs_to_jiffies(dpi_cache_node_timeout*1000));
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
int dpi_cache_timer_init(void) {
|
|||
|
printk(KERN_INFO"%s jiffies:%ld\n", __func__, jiffies);
|
|||
|
printk(KERN_INFO"ji:%d,HZ:%d\n", jiffies_to_msecs(250), HZ);
|
|||
|
init_timer(&gTimer);
|
|||
|
gTimer.expires = jiffies + dpi_cache_node_timeout*HZ;
|
|||
|
gTimer.function = dpi_cache_timer_handler;
|
|||
|
add_timer(&gTimer);
|
|||
|
printk(KERN_INFO"timer pending:%d\n", timer_pending(&gTimer));
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
void dpi_cache_timer_exit(void) {
|
|||
|
printk(KERN_INFO"%s jiffies:%ld\n", __func__, jiffies);
|
|||
|
del_timer(&gTimer);
|
|||
|
}
|
|||
|
|
|||
|
int dpi_cahce_module_init(void)
|
|||
|
{
|
|||
|
dpi_cache_timer_init();
|
|||
|
dpi_cache_list_init();
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
int dpi_cache_module_exit(void)
|
|||
|
{
|
|||
|
dpi_cache_timer_exit();
|
|||
|
dpi_cache_list_release();
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
module_init(dpi_cahce_module_init);
|
|||
|
module_exit(dpi_cache_timer_exit);
|
|||
|
MODULE_LICENSE("GPL");
|
|||
|
|
|||
|
|