summaryrefslogtreecommitdiff
path: root/src/wy_singleflow_broadcast.cpp
blob: 3cc86c4e0d35ab3f14be00a572a4d25b38fa91d4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h> 
#include <stdio.h> 
#include <stdlib.h>
#include <assert.h>
#include <errno.h>
#include <pthread.h>
#include <string.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <arpa/inet.h>
#include <sys/prctl.h>

#include "wy_singleflow_entry.h"


extern struct wysf_global_info g_wysf_global_info;

static int udp_broadcast_server_socket_init(void *log_runtime, u_int16_t port)
{
	struct sockaddr_in servaddr;
	int on = 1, socket_fd;
	
	if((socket_fd = socket(AF_INET,SOCK_DGRAM,0))<0)//AF_INET:IPV4 AF_INET6:IPV6
	{
		return -1;
	}
	if(setsockopt(socket_fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on))<0)
	{
		MESA_RUNTIME_LOGV3(log_runtime, RLOG_LV_FATAL, "setsockopt for udp_port %u failed, because:%s!!", port, strerror(errno));
	}
	on = 5242880;
	if(setsockopt(socket_fd, SOL_SOCKET, SO_RCVBUF, &on, sizeof(on))<0)
	{
		MESA_RUNTIME_LOGV3(log_runtime, RLOG_LV_FATAL, "setsockopt for udp_port %u failed, because:%s!!", port, strerror(errno));
	}

	bzero(&servaddr,sizeof(servaddr));
	servaddr.sin_family = AF_INET;
	servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
	servaddr.sin_port = htons(port);
	
	if(bind(socket_fd,(struct sockaddr *)&servaddr,sizeof(servaddr))<0)
	{
		MESA_RUNTIME_LOGV3(log_runtime, RLOG_LV_FATAL, "bind udp socket port %u failed, because: %s!!", port, strerror(errno));
		close(socket_fd);
		return -2;
	}
	evutil_make_socket_nonblocking(socket_fd);
	return socket_fd;
}

static long broadcast_stream_tuple_htable_cb(void *data, const uchar *key, uint size, void *user_arg)
{
	int32_t clj_ip=*(int32_t *)user_arg;
	std::map<int32_t, struct judian_as_group*>::iterator iter;
	struct judian_as_group *group;

	if((iter = g_wysf_global_info.forwardip2judian->find(clj_ip)) == g_wysf_global_info.forwardip2judian->end())
	{
		char ipbuffer[32];
		inet_ntop(AF_INET, &clj_ip, ipbuffer, 32);
		MESA_RUNTIME_LOGV3(g_wysf_global_info.log_runtime, RLOG_LV_FATAL, "lookup group for clj_ip: %s not found!!", ipbuffer);
		return -1;
	}
	group = iter->second;

	if(data != NULL)
	{
		MESA_htable_del(g_wysf_global_info.stream_tuple_mapping, key, size, NULL);
	}
	MESA_htable_add(g_wysf_global_info.stream_tuple_mapping, key, size, group);
	return 0;
}

void udp_server_read_broadcast_cb(evutil_socket_t fd, short events, void *arg)
{
	struct udp_broadcast_msg *broadmsg;
	char buffer[4096];
	int from_len = 0, buf_len, clj_ip;
	struct sockaddr_in saddr;
	long cb_ret;

	from_len = sizeof (struct sockaddr);
	while((buf_len = recvfrom (fd, buffer, 4096, 0, (struct sockaddr*)&saddr, (socklen_t *)&from_len))>0)
	{
		broadmsg = (struct udp_broadcast_msg *)buffer;
		if((u_int32_t)buf_len!=sizeof(struct udp_broadcast_msg) || broadmsg->header.magic_number != WYSF_MAGIC_NUMBER ||
			broadmsg->header.total_len!=sizeof(struct broadcast_message))
		{
			continue;
		}

		clj_ip = broadmsg->msg.clj_ip;
		broadmsg->msg.clj_ip = 0;
		MESA_htable_search_cb(g_wysf_global_info.stream_tuple_mapping, (uchar*)&broadmsg->msg, sizeof(struct broadcast_message),
			broadcast_stream_tuple_htable_cb, &clj_ip, &cb_ret);

		from_len = sizeof (struct sockaddr);
	}
}

void *thread_fwdnode_breoadcast(void *arg)
{
	struct event_base *evbase;
	struct event udp_server_event;

	evbase = event_base_new();
	
	if((g_wysf_global_info.broadcast_udp_servert_sockfd = udp_broadcast_server_socket_init(g_wysf_global_info.log_runtime, g_wysf_global_info.broadcast_udp_server_port)) < 0)
	{
		assert(0);return NULL;
	}
	event_assign(&udp_server_event, evbase, g_wysf_global_info.broadcast_udp_servert_sockfd, 
		EV_READ|EV_PERSIST, udp_server_read_broadcast_cb, NULL);
	event_add(&udp_server_event, NULL);

	event_base_dispatch(evbase);
	printf("Libevent dispath error, should not run here.\n");
	MESA_RUNTIME_LOGV3(g_wysf_global_info.log_runtime, RLOG_LV_FATAL, "Libevent dispath error, should not run here.");
	assert(0);return NULL;
}