2017-07-29 20:32:26 +08:00
/*
* network . cpp
*
* Created on : Jul 29 , 2017
* Author : wangyu
*/
# include "common.h"
# include "network.h"
# include "log.h"
2017-09-24 03:14:08 -05:00
# include "misc.h"
2017-07-29 20:32:26 +08:00
int raw_recv_fd = - 1 ;
int raw_send_fd = - 1 ;
2017-08-04 17:12:23 +08:00
u32_t link_level_header_len = 0 ; //set it to 14 if SOCK_RAW is used in socket(PF_PACKET, SOCK_RAW, htons(ETH_P_IP));
2018-08-29 05:25:44 -05:00
int use_tcp_dummy_socket = 0 ;
2017-07-29 20:32:26 +08:00
2017-09-06 05:31:29 -05:00
int seq_mode = 3 ;
int max_seq_mode = 4 ;
2017-10-11 09:10:38 -05:00
int random_drop = 0 ;
2017-07-29 20:32:26 +08:00
int filter_port = - 1 ;
int disable_bpf_filter = 0 ; //for test only,most time no need to disable this
2018-07-17 11:33:02 -05:00
//u32_t bind_address_uint32=0;
2017-07-29 20:32:26 +08:00
2017-08-07 13:58:42 +08:00
int lower_level = 0 ;
2017-08-21 20:26:55 +08:00
int lower_level_manual = 0 ;
2017-08-07 13:58:42 +08:00
int ifindex = - 1 ;
char if_name [ 100 ] = " " ;
2017-07-29 20:32:26 +08:00
2018-07-18 02:02:30 -05:00
char dev [ 100 ] = " " ;
2017-08-13 08:18:23 +08:00
unsigned short g_ip_id_counter = 0 ;
2017-08-21 20:26:55 +08:00
unsigned char dest_hw_addr [ sizeof ( sockaddr_ll : : sll_addr ) ] =
{ 0xff , 0xff , 0xff , 0xff , 0xff , 0xff , 0 , 0 } ;
2017-08-07 13:58:42 +08:00
//{0x00,0x23,0x45,0x67,0x89,0xb9};
2017-07-29 20:32:26 +08:00
2017-09-06 05:31:29 -05:00
const u32_t receive_window_lower_bound = 40960 ;
const u32_t receive_window_random_range = 512 ;
const unsigned char wscale = 0x05 ;
2018-07-21 14:23:13 -05:00
char g_packet_buf [ buf_len ] ; //looks dirty but works well
int g_packet_buf_len = - 1 ;
int g_packet_buf_cnt = 0 ;
union
{
sockaddr_ll ll ;
sockaddr_in ipv4 ;
sockaddr_in6 ipv6 ;
} g_sockaddr ;
socklen_t g_sockaddr_len = - 1 ;
2017-07-29 20:32:26 +08:00
struct sock_filter code_tcp_old [ ] = {
{ 0x28 , 0 , 0 , 0x0000000c } , //0
{ 0x15 , 0 , 10 , 0x00000800 } , //1
{ 0x30 , 0 , 0 , 0x00000017 } , //2
{ 0x15 , 0 , 8 , 0x00000006 } , //3
{ 0x28 , 0 , 0 , 0x00000014 } , //4
{ 0x45 , 6 , 0 , 0x00001fff } , //5
{ 0xb1 , 0 , 0 , 0x0000000e } , //6
{ 0x48 , 0 , 0 , 0x0000000e } , //7
{ 0x15 , 2 , 0 , 0x0000ef32 } , //8
{ 0x48 , 0 , 0 , 0x00000010 } , //9
{ 0x15 , 0 , 1 , 0x0000ef32 } , //10
{ 0x6 , 0 , 0 , 0x0000ffff } , //11
{ 0x6 , 0 , 0 , 0x00000000 } , //12
} ;
struct sock_filter code_tcp [ ] = {
2017-08-18 18:23:40 +08:00
//{ 0x5, 0, 0, 0x00000001 },//0 //jump to 2,dirty hack from tcpdump -d's output
//{ 0x5, 0, 0, 0x00000000 },//1
2017-07-29 20:32:26 +08:00
{ 0x30 , 0 , 0 , 0x00000009 } , //2
{ 0x15 , 0 , 6 , 0x00000006 } , //3
{ 0x28 , 0 , 0 , 0x00000006 } , //4
{ 0x45 , 4 , 0 , 0x00001fff } , //5
{ 0xb1 , 0 , 0 , 0x00000000 } , //6
{ 0x48 , 0 , 0 , 0x00000002 } , //7
2017-08-05 20:44:48 +08:00
{ 0x15 , 0 , 1 , 0x0000fffe } , //8 //modify this fffe to the port you listen on
2017-07-29 20:32:26 +08:00
{ 0x6 , 0 , 0 , 0x0000ffff } , //9
{ 0x6 , 0 , 0 , 0x00000000 } , //10
} ;
2018-07-24 05:36:56 -05:00
/*
{ 0x28 , 0 , 0 , 0x0000000c } ,
{ 0x15 , 0 , 8 , 0x00000800 } ,
{ 0x30 , 0 , 0 , 0x00000017 } ,
{ 0x15 , 0 , 6 , 0x00000006 } ,
{ 0x28 , 0 , 0 , 0x00000014 } ,
{ 0x45 , 4 , 0 , 0x00001fff } ,
{ 0xb1 , 0 , 0 , 0x0000000e } ,
{ 0x48 , 0 , 0 , 0x00000010 } ,
{ 0x15 , 0 , 1 , 0x0000fffe } ,
{ 0x6 , 0 , 0 , 0x0000ffff } ,
{ 0x6 , 0 , 0 , 0x00000000 } ,
*/
2017-08-18 18:23:40 +08:00
int code_tcp_port_index = 6 ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
//tcpdump -i ens33 ip6 and tcp and dst port 65534 -dd
struct sock_filter code_tcp6 [ ] = {
//{ 0x28, 0, 0, 0x0000000c },//0
//{ 0x15, 0, 5, 0x000086dd },//1
{ 0x30 , 0 , 0 , 0x00000006 } , //2
{ 0x15 , 0 , 3 , 0x00000006 } , //3
{ 0x28 , 0 , 0 , 0x0000002a } , //4
{ 0x15 , 0 , 1 , 0x0000fffe } , //5
{ 0x6 , 0 , 0 , 0x00040000 } , //6
{ 0x6 , 0 , 0 , 0x00000000 } , //7
} ; //note: this filter doesnt support extension headers
/*
{ 0x30 , 0 , 0 , 0x00000014 } , //2
{ 0x15 , 0 , 3 , 0x00000006 } , //3
{ 0x28 , 0 , 0 , 0x00000038 } , //4
{ 0x15 , 0 , 1 , 0x0000fffe } , //5
{ 0x6 , 0 , 0 , 0x00040000 } , //6
{ 0x6 , 0 , 0 , 0x00000000 } , //7
*/
int code_tcp6_port_index = 3 ;
2017-07-29 20:32:26 +08:00
struct sock_filter code_udp [ ] = {
2017-08-18 18:23:40 +08:00
//{ 0x5, 0, 0, 0x00000001 },
//{ 0x5, 0, 0, 0x00000000 },
2017-07-29 20:32:26 +08:00
{ 0x30 , 0 , 0 , 0x00000009 } ,
{ 0x15 , 0 , 6 , 0x00000011 } ,
{ 0x28 , 0 , 0 , 0x00000006 } ,
{ 0x45 , 4 , 0 , 0x00001fff } ,
{ 0xb1 , 0 , 0 , 0x00000000 } ,
{ 0x48 , 0 , 0 , 0x00000002 } ,
2017-08-05 20:44:48 +08:00
{ 0x15 , 0 , 1 , 0x0000fffe } , //modify this fffe to the port you listen on
2017-07-29 20:32:26 +08:00
{ 0x6 , 0 , 0 , 0x0000ffff } ,
{ 0x6 , 0 , 0 , 0x00000000 } ,
} ;
2017-08-18 18:23:40 +08:00
int code_udp_port_index = 6 ;
2018-07-24 05:36:56 -05:00
struct sock_filter code_udp6 [ ] = {
// { 0x28, 0, 0, 0x0000000c },
// { 0x15, 0, 5, 0x000086dd },
{ 0x30 , 0 , 0 , 0x00000006 } ,
{ 0x15 , 0 , 3 , 0x00000011 } ,
{ 0x28 , 0 , 0 , 0x0000002a } ,
{ 0x15 , 0 , 1 , 0x0000fffe } ,
{ 0x6 , 0 , 0 , 0x00040000 } ,
{ 0x6 , 0 , 0 , 0x00000000 } ,
} ;
int code_udp6_port_index = 3 ;
2017-07-29 20:32:26 +08:00
struct sock_filter code_icmp [ ] = {
2017-08-18 18:23:40 +08:00
//{ 0x5, 0, 0, 0x00000001 },
//{ 0x5, 0, 0, 0x00000000 },
2017-07-29 20:32:26 +08:00
{ 0x30 , 0 , 0 , 0x00000009 } ,
{ 0x15 , 0 , 1 , 0x00000001 } ,
{ 0x6 , 0 , 0 , 0x0000ffff } ,
{ 0x6 , 0 , 0 , 0x00000000 } ,
} ;
2018-07-24 08:33:42 -05:00
struct sock_filter code_icmp6 [ ] = {
// { 0x28, 0, 0, 0x0000000c },
// { 0x15, 0, 6, 0x000086dd },
{ 0x30 , 0 , 0 , 0x00000006 } ,
{ 0x15 , 3 , 0 , 0x0000003a } ,
{ 0x15 , 0 , 3 , 0x0000002c } ,
{ 0x30 , 0 , 0 , 0x00000028 } ,
{ 0x15 , 0 , 1 , 0x0000003a } ,
{ 0x6 , 0 , 0 , 0x00040000 } ,
{ 0x6 , 0 , 0 , 0x00000000 } ,
} ;
2017-07-29 20:32:26 +08:00
/*
tcpdump - i eth1 ip and icmp - d
( 000 ) ldh [ 12 ]
( 001 ) jeq # 0x800 jt 2 jf 5
( 002 ) ldb [ 23 ]
( 003 ) jeq # 0x1 jt 4 jf 5
( 004 ) ret # 65535
( 005 ) ret # 0
tcpdump - i eth1 ip and icmp - dd
{ 0x28 , 0 , 0 , 0x0000000c } ,
{ 0x15 , 0 , 3 , 0x00000800 } ,
{ 0x30 , 0 , 0 , 0x00000017 } ,
{ 0x15 , 0 , 1 , 0x00000001 } ,
{ 0x6 , 0 , 0 , 0x0000ffff } ,
{ 0x6 , 0 , 0 , 0x00000000 } ,
*/
/*
tcpdump - i eth1 ip and tcp and dst port 65534 - dd
{ 0x28 , 0 , 0 , 0x0000000c } ,
{ 0x15 , 0 , 8 , 0x00000800 } ,
{ 0x30 , 0 , 0 , 0x00000017 } ,
{ 0x15 , 0 , 6 , 0x00000006 } ,
{ 0x28 , 0 , 0 , 0x00000014 } ,
{ 0x45 , 4 , 0 , 0x00001fff } ,
{ 0xb1 , 0 , 0 , 0x0000000e } ,
{ 0x48 , 0 , 0 , 0x00000010 } ,
{ 0x15 , 0 , 1 , 0x0000fffe } ,
{ 0x6 , 0 , 0 , 0x0000ffff } ,
{ 0x6 , 0 , 0 , 0x00000000 } ,
( 000 ) ldh [ 12 ]
( 001 ) jeq # 0x800 jt 2 jf 10
( 002 ) ldb [ 23 ]
( 003 ) jeq # 0x6 jt 4 jf 10
( 004 ) ldh [ 20 ]
( 005 ) jset # 0x1fff jt 10 jf 6
( 006 ) ldxb 4 * ( [ 14 ] & 0xf )
( 007 ) ldh [ x + 16 ]
( 00 8 ) jeq # 0xfffe jt 9 jf 10
( 00 9 ) ret # 65535
( 010 ) ret # 0
*/
2017-07-30 03:46:28 +08:00
packet_info_t : : packet_info_t ( )
{
2017-07-31 19:01:50 +08:00
src_port = 0 ;
dst_port = 0 ;
if ( raw_mode = = mode_faketcp )
{
protocol = IPPROTO_TCP ;
ack_seq = get_true_random_number ( ) ;
seq = get_true_random_number ( ) ;
2017-08-02 18:57:27 +08:00
has_ts = 0 ;
2017-08-03 23:32:19 +08:00
ts_ack = 0 ;
2017-08-02 19:10:38 +08:00
syn = 0 ;
ack = 1 ;
2017-09-06 05:31:29 -05:00
ack_seq_counter = 0 ;
2017-08-04 03:29:53 +08:00
//mylog(log_info,"<cons ,ts_ack= %u>\n",ts_ack);
2017-07-31 19:01:50 +08:00
}
else if ( raw_mode = = mode_udp )
{
protocol = IPPROTO_UDP ;
}
else if ( raw_mode = = mode_icmp )
{
2018-07-24 08:33:42 -05:00
if ( raw_ip_version = = AF_INET )
{
protocol = IPPROTO_ICMP ;
}
else
{
assert ( raw_ip_version = = AF_INET6 ) ;
protocol = IPPROTO_ICMPV6 ;
}
2018-07-26 12:41:55 -05:00
my_icmp_seq = 0 ;
2017-07-31 19:01:50 +08:00
}
2017-07-29 20:32:26 +08:00
2017-07-30 03:46:28 +08:00
}
2017-07-29 20:32:26 +08:00
int init_raw_socket ( )
{
2018-07-24 05:36:56 -05:00
assert ( raw_ip_version = = AF_INET | | raw_ip_version = = AF_INET6 ) ;
2017-07-29 20:32:26 +08:00
2017-08-13 08:18:23 +08:00
g_ip_id_counter = get_true_random_number ( ) % 65535 ;
2017-08-07 13:58:42 +08:00
if ( lower_level = = 0 )
{
2018-07-23 13:52:36 -05:00
raw_send_fd = socket ( raw_ip_version , SOCK_RAW , IPPROTO_RAW ) ; // IPPROTO_TCP??
2017-07-29 20:32:26 +08:00
2017-08-07 13:58:42 +08:00
if ( raw_send_fd = = - 1 ) {
mylog ( log_fatal , " Failed to create raw_send_fd \n " ) ;
//perror("Failed to create raw_send_fd");
myexit ( 1 ) ;
}
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
/*ETH_P_IP
2017-08-07 13:58:42 +08:00
int one = 1 ;
const int * val = & one ;
if ( setsockopt ( raw_send_fd , IPPROTO_IP , IP_HDRINCL , val , sizeof ( one ) ) < 0 ) {
mylog ( log_fatal , " Error setting IP_HDRINCL %d \n " , errno ) ;
//perror("Error setting IP_HDRINCL");
myexit ( 2 ) ;
2018-07-21 14:24:55 -05:00
} */
2017-08-07 13:58:42 +08:00
}
else
{
2018-07-21 14:23:13 -05:00
raw_send_fd = socket ( PF_PACKET , SOCK_DGRAM , htons ( ETH_P_IP ) ) ; // todo how to create a recv only raw socket?
2017-08-07 13:58:42 +08:00
if ( raw_send_fd = = - 1 ) {
mylog ( log_fatal , " Failed to create raw_send_fd \n " ) ;
//perror("Failed to create raw_send_fd");
myexit ( 1 ) ;
}
2017-08-21 20:26:55 +08:00
//init_ifindex(if_name);
2017-08-07 13:58:42 +08:00
}
2017-07-29 20:32:26 +08:00
2018-07-22 03:23:53 -05:00
int opt = 0 ;
assert ( setsockopt ( raw_send_fd , SOL_SOCKET , SO_RCVBUF , & opt , sizeof ( opt ) ) = = 0 ) ; // raw_send_fd is for send only, set its recv buffer to zero
2017-09-04 03:21:34 -05:00
if ( force_socket_buf )
{
if ( setsockopt ( raw_send_fd , SOL_SOCKET , SO_SNDBUFFORCE , & socket_buf_size , sizeof ( socket_buf_size ) ) < 0 )
{
mylog ( log_fatal , " SO_SNDBUFFORCE fail socket_buf_size=%d errno=%s \n " , socket_buf_size , strerror ( errno ) ) ;
myexit ( 1 ) ;
}
}
else
{
if ( setsockopt ( raw_send_fd , SOL_SOCKET , SO_SNDBUF , & socket_buf_size , sizeof ( socket_buf_size ) ) < 0 )
{
mylog ( log_fatal , " SO_SNDBUF fail socket_buf_size=%d errno=%s \n " , socket_buf_size , strerror ( errno ) ) ;
myexit ( 1 ) ;
}
}
2017-08-07 13:58:42 +08:00
2017-07-29 20:32:26 +08:00
//raw_fd = socket(AF_PACKET, SOCK_DGRAM, htons(ETH_P_ALL));
2018-07-24 05:36:56 -05:00
if ( raw_ip_version = = AF_INET )
raw_recv_fd = socket ( PF_PACKET , SOCK_DGRAM , htons ( ETH_P_IP ) ) ;
else
raw_recv_fd = socket ( PF_PACKET , SOCK_DGRAM , htons ( ETH_P_IPV6 ) ) ;
2018-07-21 14:23:13 -05:00
//ETH_P_IP doesnt read outgoing packets
// https://stackoverflow.com/questions/20264895/eth-p-ip-is-not-working-as-expected-i-can-only-receive-incoming-packets
// to capture both incoming and outgoing packets use ETH_P_ALL
2017-07-29 20:32:26 +08:00
if ( raw_recv_fd = = - 1 ) {
mylog ( log_fatal , " Failed to create raw_recv_fd \n " ) ;
//perror("");
2017-08-04 11:51:39 +08:00
myexit ( 1 ) ;
2017-07-29 20:32:26 +08:00
}
2018-07-18 02:02:30 -05:00
if ( strlen ( dev ) ! = 0 )
{
struct sockaddr_ll bind_address ;
memset ( & bind_address , 0 , sizeof ( bind_address ) ) ;
int index = - 1 ;
assert ( init_ifindex ( dev , raw_recv_fd , index ) = = 0 ) ;
bind_address . sll_family = AF_PACKET ;
2018-07-24 05:36:56 -05:00
if ( raw_ip_version = = AF_INET )
bind_address . sll_protocol = htons ( ETH_P_IP ) ;
else
bind_address . sll_protocol = htons ( ETH_P_IPV6 ) ;
2018-07-18 02:02:30 -05:00
bind_address . sll_ifindex = index ;
if ( bind ( raw_recv_fd , ( struct sockaddr * ) & bind_address , sizeof ( bind_address ) ) = = - 1 )
{
mylog ( log_fatal , " bind to dev [%s] failed \n " , dev ) ;
myexit ( 1 ) ;
}
}
2017-07-29 20:32:26 +08:00
2017-09-04 03:21:34 -05:00
if ( force_socket_buf )
{
if ( setsockopt ( raw_recv_fd , SOL_SOCKET , SO_RCVBUFFORCE , & socket_buf_size , sizeof ( socket_buf_size ) ) < 0 )
{
mylog ( log_fatal , " SO_RCVBUFFORCE fail socket_buf_size=%d errno=%s \n " , socket_buf_size , strerror ( errno ) ) ;
myexit ( 1 ) ;
}
}
else
{
if ( setsockopt ( raw_recv_fd , SOL_SOCKET , SO_RCVBUF , & socket_buf_size , sizeof ( socket_buf_size ) ) < 0 )
{
mylog ( log_fatal , " SO_RCVBUF fail socket_buf_size=%d errno=%s \n " , socket_buf_size , strerror ( errno ) ) ;
myexit ( 1 ) ;
}
}
2017-07-29 20:32:26 +08:00
//IP_HDRINCL to tell the kernel that headers are included in the packet
2017-08-07 13:58:42 +08:00
2017-07-29 20:32:26 +08:00
setnonblocking ( raw_send_fd ) ; //not really necessary
setnonblocking ( raw_recv_fd ) ;
return 0 ;
}
void init_filter ( int port )
{
sock_fprog bpf ;
2018-07-24 05:36:56 -05:00
assert ( raw_ip_version = = AF_INET | | raw_ip_version = = AF_INET6 ) ;
2017-07-31 19:01:50 +08:00
if ( raw_mode = = mode_faketcp | | raw_mode = = mode_udp )
{
filter_port = port ;
}
2017-07-29 20:32:26 +08:00
if ( disable_bpf_filter ) return ;
//if(raw_mode==mode_icmp) return ;
//code_tcp[8].k=code_tcp[10].k=port;
if ( raw_mode = = mode_faketcp )
{
2018-07-24 05:36:56 -05:00
if ( raw_ip_version = = AF_INET )
{
bpf . len = sizeof ( code_tcp ) / sizeof ( code_tcp [ 0 ] ) ;
code_tcp [ code_tcp_port_index ] . k = port ;
bpf . filter = code_tcp ;
} else
{
bpf . len = sizeof ( code_tcp6 ) / sizeof ( code_tcp6 [ 0 ] ) ;
code_tcp6 [ code_tcp6_port_index ] . k = port ;
bpf . filter = code_tcp6 ;
}
2017-07-29 20:32:26 +08:00
}
else if ( raw_mode = = mode_udp )
{
2018-07-24 05:36:56 -05:00
if ( raw_ip_version = = AF_INET )
{
bpf . len = sizeof ( code_udp ) / sizeof ( code_udp [ 0 ] ) ;
code_udp [ code_udp_port_index ] . k = port ;
bpf . filter = code_udp ;
} else
{
bpf . len = sizeof ( code_udp6 ) / sizeof ( code_udp6 [ 0 ] ) ;
code_udp6 [ code_udp6_port_index ] . k = port ;
bpf . filter = code_udp6 ;
}
2017-07-29 20:32:26 +08:00
}
else if ( raw_mode = = mode_icmp )
{
2018-07-24 08:33:42 -05:00
if ( raw_ip_version = = AF_INET )
{
bpf . len = sizeof ( code_icmp ) / sizeof ( code_icmp [ 0 ] ) ;
bpf . filter = code_icmp ;
} else
{
bpf . len = sizeof ( code_icmp6 ) / sizeof ( code_icmp6 [ 0 ] ) ;
bpf . filter = code_icmp6 ;
}
2017-07-29 20:32:26 +08:00
}
int dummy ;
int ret = setsockopt ( raw_recv_fd , SOL_SOCKET , SO_DETACH_FILTER , & dummy , sizeof ( dummy ) ) ; //in case i forgot to remove
if ( ret ! = 0 )
{
mylog ( log_debug , " error remove fiter \n " ) ;
//perror("filter");
//exit(-1);
}
ret = setsockopt ( raw_recv_fd , SOL_SOCKET , SO_ATTACH_FILTER , & bpf , sizeof ( bpf ) ) ;
if ( ret ! = 0 )
{
mylog ( log_fatal , " error set fiter \n " ) ;
//perror("filter");
2017-08-04 11:51:39 +08:00
myexit ( - 1 ) ;
2017-07-29 20:32:26 +08:00
}
}
void remove_filter ( )
{
filter_port = 0 ;
int dummy ;
int ret = setsockopt ( raw_recv_fd , SOL_SOCKET , SO_DETACH_FILTER , & dummy , sizeof ( dummy ) ) ;
if ( ret ! = 0 )
{
mylog ( log_debug , " error remove fiter \n " ) ;
//perror("filter");
//exit(-1);
}
}
2018-08-29 20:00:37 -05:00
2018-07-18 02:02:30 -05:00
int init_ifindex ( const char * if_name , int fd , int & index )
2017-08-07 13:58:42 +08:00
{
struct ifreq ifr ;
size_t if_name_len = strlen ( if_name ) ;
if ( if_name_len < sizeof ( ifr . ifr_name ) ) {
memcpy ( ifr . ifr_name , if_name , if_name_len ) ;
ifr . ifr_name [ if_name_len ] = 0 ;
} else {
mylog ( log_fatal , " interface name is too long \n " ) ;
myexit ( - 1 ) ;
}
2018-07-18 02:02:30 -05:00
if ( ioctl ( fd , SIOCGIFINDEX , & ifr ) = = - 1 ) {
2017-07-29 20:32:26 +08:00
2017-08-07 13:58:42 +08:00
mylog ( log_fatal , " SIOCGIFINDEX fail ,%s \n " , strerror ( errno ) ) ;
myexit ( - 1 ) ;
}
2017-08-26 05:38:33 -05:00
index = ifr . ifr_ifindex ;
mylog ( log_info , " ifname:%s ifindex:%d \n " , if_name , index ) ;
2017-08-07 13:58:42 +08:00
return 0 ;
}
2017-08-26 05:38:33 -05:00
bool interface_has_arp ( const char * interface ) {
struct ifreq ifr ;
// int sock = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
int sock = raw_send_fd ;
memset ( & ifr , 0 , sizeof ( ifr ) ) ;
strcpy ( ifr . ifr_name , interface ) ;
if ( ioctl ( sock , SIOCGIFFLAGS , & ifr ) < 0 ) {
//perror("SIOCGIFFLAGS");
mylog ( log_fatal , " ioctl(sock, SIOCGIFFLAGS, &ifr) failed for interface %s,errno %s \n " , interface , strerror ( errno ) ) ;
myexit ( - 1 ) ;
}
//close(sock);
return ! ( ifr . ifr_flags & IFF_NOARP ) ;
}
struct route_info_t
{
string if_name ;
u32_t dest ;
u32_t mask ;
u32_t gw ;
u32_t flag ;
} ;
int dest_idx = 1 ;
int gw_idx = 2 ;
int if_idx = 0 ;
int mask_idx = 7 ;
int flag_idx = 3 ;
vector < int > find_route_entry ( const vector < route_info_t > & route_info_vec , u32_t ip )
{
vector < int > res ;
for ( u32_t i = 0 ; i < = 32 ; i + + )
{
u32_t mask = 0xffffffff ;
//mask >>=i;
//if(i==32) mask=0; //why 0xffffffff>>32 equals 0xffffffff??
mask < < = i ;
if ( i = = 32 ) mask = 0 ;
log_bare ( log_debug , " (mask:%x) " , mask ) ;
for ( u32_t j = 0 ; j < route_info_vec . size ( ) ; j + + )
{
const route_info_t & info = route_info_vec [ j ] ;
if ( info . mask ! = mask )
continue ;
log_bare ( log_debug , " <<%d,%d>> " , i , j ) ;
if ( ( info . dest & mask ) = = ( ip & mask ) )
{
log_bare ( log_debug , " found! " ) ;
res . push_back ( j ) ;
}
}
if ( res . size ( ) ! = 0 )
{
return res ;
}
}
return res ;
}
int find_direct_dest ( const vector < route_info_t > & route_info_vec , u32_t ip , u32_t & dest_ip , string & if_name )
{
vector < int > res ;
for ( int i = 0 ; i < 1000 ; i + + )
{
res = find_route_entry ( route_info_vec , ip ) ;
log_bare ( log_debug , " <entry:%u> " , ( u32_t ) res . size ( ) ) ;
if ( res . size ( ) = = 0 )
{
mylog ( log_error , " cant find route entry \n " ) ;
return - 1 ;
}
if ( res . size ( ) > 1 )
{
mylog ( log_error , " found duplicated entries \n " ) ;
return - 1 ;
}
if ( ( route_info_vec [ res [ 0 ] ] . flag & 2 ) = = 0 )
{
dest_ip = ip ;
if_name = route_info_vec [ res [ 0 ] ] . if_name ;
return 0 ;
}
else
{
ip = route_info_vec [ res [ 0 ] ] . gw ;
}
}
mylog ( log_error , " dead loop in find_direct_dest \n " ) ;
return - 1 ;
}
struct arp_info_t
{
u32_t ip ;
string hw ;
string if_name ;
} ;
int arp_ip_idx = 0 ;
int arp_hw_idx = 3 ;
int arp_if_idx = 5 ;
int find_arp ( const vector < arp_info_t > & arp_info_vec , u32_t ip , string if_name , string & hw )
{
int pos = - 1 ;
int count = 0 ;
for ( u32_t i = 0 ; i < arp_info_vec . size ( ) ; i + + )
{
const arp_info_t & info = arp_info_vec [ i ] ;
if ( info . if_name ! = if_name ) continue ;
if ( info . ip = = ip )
{
count + + ;
pos = i ;
}
}
if ( count = = 0 )
{
//mylog(log_warn,"cant find arp entry for %s %s,using 00:00:00:00:00:00\n",my_ntoa(ip),if_name.c_str());
//hw="00:00:00:00:00:00";
mylog ( log_error , " cant find arp entry for %s %s \n " , my_ntoa ( ip ) , if_name . c_str ( ) ) ;
return - 1 ;
}
if ( count > 1 )
{
mylog ( log_error , " find multiple arp entry for %s %s \n " , my_ntoa ( ip ) , if_name . c_str ( ) ) ;
return - 1 ;
}
hw = arp_info_vec [ pos ] . hw ;
return 0 ;
}
int find_lower_level_info ( u32_t ip , u32_t & dest_ip , string & if_name , string & hw )
{
ip = htonl ( ip ) ;
if ( ip = = htonl ( inet_addr ( " 127.0.0.1 " ) ) )
{
dest_ip = ntohl ( ip ) ;
if_name = " lo " ;
hw = " 00:00:00:00:00:00 " ;
return 0 ;
}
string route_file ;
if ( read_file ( " /proc/net/route " , route_file ) ! = 0 ) return - 1 ;
string arp_file ;
if ( read_file ( " /proc/net/arp " , arp_file ) ! = 0 ) return - 1 ;
log_bare ( log_debug , " /proc/net/route:<<%s>> \n " , route_file . c_str ( ) ) ;
log_bare ( log_debug , " /proc/net/arp:<<%s>> \n " , route_file . c_str ( ) ) ;
auto route_vec2 = string_to_vec2 ( route_file . c_str ( ) ) ;
vector < route_info_t > route_info_vec ;
for ( u32_t i = 1 ; i < route_vec2 . size ( ) ; i + + )
{
log_bare ( log_debug , " <size:%u> " , ( u32_t ) route_vec2 [ i ] . size ( ) ) ;
if ( route_vec2 [ i ] . size ( ) ! = 11 )
{
mylog ( log_error , " route coloum %d !=11 \n " , int ( route_vec2 [ i ] . size ( ) ) ) ;
return - 1 ;
}
route_info_t tmp ;
tmp . if_name = route_vec2 [ i ] [ if_idx ] ;
if ( hex_to_u32_with_endian ( route_vec2 [ i ] [ dest_idx ] , tmp . dest ) ! = 0 ) return - 1 ;
if ( hex_to_u32_with_endian ( route_vec2 [ i ] [ gw_idx ] , tmp . gw ) ! = 0 ) return - 1 ;
if ( hex_to_u32_with_endian ( route_vec2 [ i ] [ mask_idx ] , tmp . mask ) ! = 0 ) return - 1 ;
if ( hex_to_u32 ( route_vec2 [ i ] [ flag_idx ] , tmp . flag ) ! = 0 ) return - 1 ;
route_info_vec . push_back ( tmp ) ;
for ( u32_t j = 0 ; j < route_vec2 [ i ] . size ( ) ; j + + )
{
log_bare ( log_debug , " <%s> " , route_vec2 [ i ] [ j ] . c_str ( ) ) ;
}
log_bare ( log_debug , " %s dest:%x mask:%x gw:%x flag:%x " , tmp . if_name . c_str ( ) , tmp . dest , tmp . mask , tmp . gw , tmp . flag ) ;
log_bare ( log_debug , " \n " ) ;
}
if ( find_direct_dest ( route_info_vec , ip , dest_ip , if_name ) ! = 0 )
{
mylog ( log_error , " find_direct_dest failed for ip %s \n " , my_ntoa ( ntohl ( ip ) ) ) ;
return - 1 ;
}
log_bare ( log_debug , " ======== \n " ) ;
auto arp_vec2 = string_to_vec2 ( arp_file . c_str ( ) ) ;
vector < arp_info_t > arp_info_vec ;
for ( u32_t i = 1 ; i < arp_vec2 . size ( ) ; i + + )
{
log_bare ( log_debug , " <<arp_vec2[i].size(): %d>> " , ( int ) arp_vec2 [ i ] . size ( ) ) ;
for ( u32_t j = 0 ; j < arp_vec2 [ i ] . size ( ) ; j + + )
{
log_bare ( log_debug , " <%s> " , arp_vec2 [ i ] [ j ] . c_str ( ) ) ;
}
if ( arp_vec2 [ i ] . size ( ) ! = 6 )
{
mylog ( log_error , " arp coloum %d !=11 \n " , int ( arp_vec2 [ i ] . size ( ) ) ) ;
return - 1 ;
}
arp_info_t tmp ;
tmp . if_name = arp_vec2 [ i ] [ arp_if_idx ] ;
tmp . hw = arp_vec2 [ i ] [ arp_hw_idx ] ;
tmp . ip = htonl ( inet_addr ( arp_vec2 [ i ] [ arp_ip_idx ] . c_str ( ) ) ) ;
arp_info_vec . push_back ( tmp ) ;
log_bare ( log_debug , " \n " ) ;
}
if ( ! interface_has_arp ( if_name . c_str ( ) ) )
{
mylog ( log_info , " %s is a noarp interface,using 00:00:00:00:00:00 \n " , if_name . c_str ( ) ) ;
hw = " 00:00:00:00:00:00 " ;
}
else if ( find_arp ( arp_info_vec , dest_ip , if_name , hw ) ! = 0 )
{
mylog ( log_error , " find_arp failed for dest_ip %s ,if_name %s \n " , my_ntoa ( ntohl ( ip ) ) , if_name . c_str ( ) ) ;
return - 1 ;
}
//printf("%s\n",hw.c_str());
dest_ip = ntohl ( dest_ip ) ;
return 0 ;
}
2018-07-23 09:50:21 -05:00
int send_raw_packet ( raw_info_t & raw_info , const char * packet , int len )
{
const packet_info_t & send_info = raw_info . send_info ;
const packet_info_t & recv_info = raw_info . recv_info ;
int ret ;
if ( lower_level = = 0 )
{
2018-07-23 13:52:36 -05:00
if ( raw_ip_version = = AF_INET )
{
struct sockaddr_in sin = { 0 } ;
sin . sin_family = raw_ip_version ;
//sin.sin_port = htons(info.dst_port); //dont need this
sin . sin_addr . s_addr = send_info . new_dst_ip . v4 ;
ret = sendto ( raw_send_fd , packet , len , 0 , ( struct sockaddr * ) & sin , sizeof ( sin ) ) ;
}
else if ( raw_ip_version = = AF_INET6 )
{
struct sockaddr_in6 sin = { 0 } ;
sin . sin6_family = raw_ip_version ;
//sin.sin_port = htons(info.dst_port); //dont need this
sin . sin6_addr = send_info . new_dst_ip . v6 ;
ret = sendto ( raw_send_fd , packet , len , 0 , ( struct sockaddr * ) & sin , sizeof ( sin ) ) ;
} else
{
assert ( 0 = = 1 ) ;
}
2018-07-23 09:50:21 -05:00
}
else
{
struct sockaddr_ll addr = { 0 } ; //={0} not necessary
memcpy ( & addr , & send_info . addr_ll , sizeof ( addr ) ) ;
ret = sendto ( raw_send_fd , packet , len , 0 , ( struct sockaddr * ) & addr , sizeof ( addr ) ) ;
}
if ( ret = = - 1 )
{
2017-07-29 20:32:26 +08:00
2018-07-23 09:50:21 -05:00
mylog ( log_trace , " sendto failed \n " ) ;
//perror("why?");
return - 1 ;
}
else
{
//mylog(log_info,"sendto succ\n");
}
return 0 ;
}
2017-07-29 20:32:26 +08:00
int send_raw_ip ( raw_info_t & raw_info , const char * payload , int payloadlen )
{
2017-07-30 23:18:37 +08:00
const packet_info_t & send_info = raw_info . send_info ;
const packet_info_t & recv_info = raw_info . recv_info ;
2017-07-29 20:32:26 +08:00
char send_raw_ip_buf [ buf_len ] ;
2018-02-24 17:26:29 -06:00
if ( raw_info . disabled )
{
2018-07-22 08:59:09 -05:00
mylog ( log_debug , " [%s,%d]connection disabled, no packet will be sent \n " , recv_info . new_src_ip . get_str1 ( ) , recv_info . src_port ) ;
2018-02-24 18:07:16 -06:00
assert ( max_rst_allowed > = 0 ) ;
2018-02-24 17:26:29 -06:00
return 0 ;
}
2018-07-24 05:36:56 -05:00
uint16_t ip_tot_len ;
if ( raw_ip_version = = AF_INET )
{
2018-07-26 11:25:36 -05:00
struct my_iphdr * iph = ( struct my_iphdr * ) send_raw_ip_buf ;
memset ( iph , 0 , sizeof ( my_iphdr ) ) ;
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
iph - > ihl = sizeof ( my_iphdr ) / 4 ; //we dont use ip options,so the length is just sizeof(iphdr)
2018-07-24 05:36:56 -05:00
iph - > version = 4 ;
iph - > tos = 0 ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
if ( lower_level )
{
//iph->id=0;
iph - > id = htons ( g_ip_id_counter + + ) ; //Id of this packet
}
else //no need to else?
{
iph - > id = htons ( g_ip_id_counter + + ) ; //Id of this packet
//iph->id = 0; //Id of this packet ,kernel will auto fill this if id is zero ,or really?????// todo //seems like there is a problem
}
2017-08-07 13:58:42 +08:00
2018-07-24 05:36:56 -05:00
iph - > frag_off = htons ( 0x4000 ) ; //DF set,others are zero
// iph->frag_off = htons(0x0000); //DF set,others are zero
iph - > ttl = ( unsigned char ) ttl_value ;
iph - > protocol = send_info . protocol ;
iph - > check = 0 ; //Set to 0 before calculating checksum
iph - > saddr = send_info . new_src_ip . v4 ; //Spoof the source ip address
iph - > daddr = send_info . new_dst_ip . v4 ;
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
ip_tot_len = sizeof ( struct my_iphdr ) + payloadlen ;
2018-07-24 05:36:56 -05:00
if ( lower_level ) iph - > tot_len = htons ( ip_tot_len ) ; //this is not necessary ,kernel will always auto fill this //http://man7.org/linux/man-pages/man7/raw.7.html
else
iph - > tot_len = 0 ;
2017-08-07 13:58:42 +08:00
2018-07-26 11:25:36 -05:00
memcpy ( send_raw_ip_buf + sizeof ( my_iphdr ) , payload , payloadlen ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
if ( lower_level ) iph - > check =
csum ( ( unsigned short * ) send_raw_ip_buf , iph - > ihl * 4 ) ; //this is not necessary ,kernel will always auto fill this
else
iph - > check = 0 ;
}
2017-08-07 13:58:42 +08:00
else
2018-07-24 05:36:56 -05:00
{
assert ( raw_ip_version = = AF_INET6 ) ;
struct my_ip6hdr * ip6h = ( struct my_ip6hdr * ) send_raw_ip_buf ;
memset ( ip6h , 0 , sizeof ( my_ip6hdr ) ) ;
ip6h - > version = 6 ;
ip6h - > payload_len = htons ( payloadlen ) ;
ip6h - > next_header = send_info . protocol ;
ip6h - > hop_limit = ( unsigned char ) ttl_value ;
ip6h - > src = send_info . new_src_ip . v6 ;
ip6h - > dst = send_info . new_dst_ip . v6 ;
ip_tot_len = sizeof ( struct my_ip6hdr ) + payloadlen ;
memcpy ( send_raw_ip_buf + sizeof ( my_ip6hdr ) , payload , payloadlen ) ;
}
2017-07-29 20:32:26 +08:00
2018-07-23 09:50:21 -05:00
return send_raw_packet ( raw_info , send_raw_ip_buf , ip_tot_len ) ;
2017-07-29 20:32:26 +08:00
}
2018-07-21 14:23:13 -05:00
int pre_recv_raw_packet ( )
2017-07-29 20:32:26 +08:00
{
2018-07-21 14:23:13 -05:00
assert ( g_packet_buf_cnt = = 0 ) ;
2017-07-29 20:32:26 +08:00
2018-07-21 14:23:13 -05:00
g_sockaddr_len = sizeof ( g_sockaddr . ll ) ;
g_packet_buf_len = recvfrom ( raw_recv_fd , g_packet_buf , max_data_len + 1 , 0 , ( sockaddr * ) & g_sockaddr , & g_sockaddr_len ) ;
//assert(g_sockaddr_len==sizeof(g_sockaddr.ll)); //g_sockaddr_len=18, sizeof(g_sockaddr.ll)=20, why its not equal? maybe its bc sll_halen is 6?
2017-07-29 20:32:26 +08:00
2018-07-21 14:23:13 -05:00
//assert(g_addr_ll_size==sizeof(g_addr_ll));
2018-02-20 06:10:18 -06:00
2018-07-21 14:23:13 -05:00
if ( g_packet_buf_len = = max_data_len + 1 )
2018-02-20 06:10:18 -06:00
{
mylog ( log_warn , " huge packet, data_len > %d,dropped \n " , max_data_len ) ;
return - 1 ;
}
2017-07-29 20:32:26 +08:00
2018-07-21 14:23:13 -05:00
if ( g_packet_buf_len < 0 )
2017-07-29 20:32:26 +08:00
{
2018-07-21 14:23:13 -05:00
mylog ( log_trace , " recv_len %d \n " , g_packet_buf_len ) ;
2017-07-29 20:32:26 +08:00
return - 1 ;
}
2018-07-21 14:23:13 -05:00
g_packet_buf_cnt + + ;
return 0 ;
}
int discard_raw_packet ( )
{
assert ( g_packet_buf_cnt = = 1 ) ;
g_packet_buf_cnt - - ;
return 0 ;
}
2018-07-23 13:52:36 -05:00
int recv_raw_packet ( char * & packet , int & len , int peek )
2018-07-21 14:23:13 -05:00
{
assert ( g_packet_buf_cnt = = 1 ) ;
2018-07-23 13:52:36 -05:00
if ( ! peek )
g_packet_buf_cnt - - ;
2018-07-21 14:23:13 -05:00
2018-07-23 09:50:21 -05:00
if ( g_packet_buf_len < int ( link_level_header_len ) )
{
2018-07-23 13:52:36 -05:00
mylog ( log_trace , " packet len %d shorter than link_level_header_len %d \n " , g_packet_buf_len , int ( link_level_header_len ) ) ;
2018-07-23 09:50:21 -05:00
return - 1 ;
}
2018-07-23 13:52:36 -05:00
if ( link_level_header_len = = 14 )
2018-07-23 09:50:21 -05:00
{
2018-07-23 13:52:36 -05:00
unsigned char a = g_packet_buf [ 12 ] ;
unsigned char b = g_packet_buf [ 13 ] ;
if ( ! ( ( a = = 0x08 & & b = = 0x00 ) | | ( a = = 0x86 & & b = = 0xdd ) ) )
{
mylog ( log_trace , " not an ipv4 or ipv6 packet! \n " ) ;
return - 1 ;
}
2018-07-23 09:50:21 -05:00
}
packet = g_packet_buf + int ( link_level_header_len ) ;
len = g_packet_buf_len - int ( link_level_header_len ) ;
return 0 ;
}
int recv_raw_ip ( raw_info_t & raw_info , char * & payload , int & payloadlen )
{
char * raw_packet_buf ;
2018-07-21 14:23:13 -05:00
//static char recv_raw_ip_buf[buf_len];
2018-07-23 09:50:21 -05:00
int raw_packet_len ;
2018-07-21 14:23:13 -05:00
2018-07-23 13:52:36 -05:00
if ( recv_raw_packet ( raw_packet_buf , raw_packet_len , raw_info . peek ) ! = 0 ) return - 1 ;
2018-07-21 14:23:13 -05:00
2018-07-23 13:52:36 -05:00
//const packet_info_t &send_info=raw_info.send_info;
2018-07-21 14:23:13 -05:00
packet_info_t & recv_info = raw_info . recv_info ;
2018-07-26 11:25:36 -05:00
my_iphdr * iph ;
2018-07-24 05:36:56 -05:00
my_ip6hdr * ip6h ;
2018-07-21 14:23:13 -05:00
int flag = 0 ;
//int recv_len = recvfrom(raw_recv_fd, recv_raw_ip_buf, max_data_len+1, flag ,(sockaddr*)&saddr , &saddr_size);
2018-07-23 09:50:21 -05:00
char * ip_begin = raw_packet_buf ; //14 is eth net header
2017-07-29 20:32:26 +08:00
2018-07-23 13:52:36 -05:00
if ( raw_packet_len < 1 )
{
mylog ( log_trace , " raw_packet_len <1, dropped \n " ) ;
return - 1 ;
}
2018-07-26 11:25:36 -05:00
iph = ( struct my_iphdr * ) ( ip_begin ) ;
2018-07-24 05:36:56 -05:00
ip6h = ( struct my_ip6hdr * ) ( ip_begin ) ;
2018-07-23 13:52:36 -05:00
if ( raw_ip_version = = AF_INET )
{
if ( iph - > version ! = 4 )
{
mylog ( log_trace , " expect ipv4 packet, but got something else: %02x \n " , iph - > version ) ;
return - 1 ;
}
2018-07-26 11:25:36 -05:00
if ( raw_packet_len < ( int ) sizeof ( my_iphdr ) )
2018-07-23 13:52:36 -05:00
{
mylog ( log_trace , " raw_packet_len<sizeof(iphdr) \n " ) ;
return - 1 ;
}
}
else
{
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 05:36:56 -05:00
if ( ip6h - > version ! = 6 )
2018-07-23 13:52:36 -05:00
{
2018-07-24 05:36:56 -05:00
mylog ( log_trace , " expect ipv6 packet, but got something else: %02x \n " , ip6h - > version ) ;
2018-07-23 13:52:36 -05:00
return - 1 ;
}
2018-07-24 05:36:56 -05:00
if ( raw_packet_len < ( int ) sizeof ( my_ip6hdr ) )
2018-07-23 13:52:36 -05:00
{
mylog ( log_trace , " raw_packet_len<sizeof(ip6_hdr) \n " ) ;
return - 1 ;
}
}
2017-07-29 20:32:26 +08:00
2017-08-21 20:26:55 +08:00
if ( lower_level )
{
2018-07-21 14:23:13 -05:00
memcpy ( & recv_info . addr_ll , & g_sockaddr . ll , sizeof ( recv_info . addr_ll ) ) ;
2017-08-21 20:26:55 +08:00
}
2017-07-29 20:32:26 +08:00
2018-07-24 08:33:42 -05:00
2017-07-29 20:32:26 +08:00
2018-07-23 13:52:36 -05:00
unsigned short iphdrlen ;
int ip_len ;
if ( raw_ip_version = = AF_INET )
{
recv_info . new_src_ip . v4 = iph - > saddr ;
recv_info . new_dst_ip . v4 = iph - > daddr ;
recv_info . protocol = iph - > protocol ;
iphdrlen = iph - > ihl * 4 ;
ip_len = ntohs ( iph - > tot_len ) ;
}
else
{
//todo flow id
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 05:36:56 -05:00
recv_info . new_src_ip . v6 = ip6h - > src ;
recv_info . new_dst_ip . v6 = ip6h - > dst ;
2018-07-23 13:52:36 -05:00
iphdrlen = 40 ;
2018-07-24 05:36:56 -05:00
recv_info . protocol = ip6h - > next_header ; //todo handle extension headers;
ip_len = ntohs ( ip6h - > payload_len ) + iphdrlen ;
2018-07-23 13:52:36 -05:00
}
2017-07-29 20:32:26 +08:00
2018-07-24 08:33:42 -05:00
if ( bind_addr_used & & ! recv_info . new_dst_ip . equal ( bind_addr ) )
{
mylog ( log_trace , " bind adress doenst match %s %s, dropped \n " , recv_info . new_dst_ip . get_str1 ( ) , bind_addr . get_str2 ( ) ) ;
//printf(" bind adress doenst match, dropped\n");
return - 1 ;
}
2017-07-29 20:32:26 +08:00
2018-07-23 13:52:36 -05:00
//if (!(iph->ihl > 0 && iph->ihl <=60)) {
// mylog(log_trace,"iph ihl error\n");
// return -1;
// }
2017-07-29 20:32:26 +08:00
2018-07-23 09:50:21 -05:00
if ( raw_packet_len < ip_len )
2017-07-29 20:32:26 +08:00
{
mylog ( log_debug , " incomplete packet \n " ) ;
return - 1 ;
}
2018-07-23 13:52:36 -05:00
if ( raw_ip_version = = AF_INET )
{
if ( raw_info . peek = = 0 ) // avoid cal it twice
{
u32_t ip_chk = csum ( ( unsigned short * ) ip_begin , iphdrlen ) ;
2017-07-29 20:32:26 +08:00
2018-07-23 13:52:36 -05:00
if ( ip_chk ! = 0 )
{
mylog ( log_debug , " ip header error %x \n " , ip_chk ) ;
return - 1 ;
}
}
}
else
{
2018-08-29 16:48:45 -05:00
//do nothing
2018-07-23 13:52:36 -05:00
}
2017-07-29 20:32:26 +08:00
payload = ip_begin + iphdrlen ;
payloadlen = ip_len - iphdrlen ;
if ( payloadlen < 0 )
{
mylog ( log_warn , " error payload len \n " ) ;
return - 1 ;
}
return 0 ;
}
2018-07-23 13:52:36 -05:00
int peek_raw ( raw_info_t & raw_info )
{
//static char peek_raw_buf[buf_len];
//assert(g_packet_buf_cnt==1);
//g_packet_buf_cnt--;
//char * peek_raw_buf=g_packet_buf;
//int recv_len=g_packet_buf_len;
//char *ip_begin=peek_raw_buf+link_level_header_len;
//struct sockaddr saddr={0};
//socklen_t saddr_size=sizeof(saddr);
//int recv_len = recvfrom(raw_recv_fd, peek_raw_buf,max_data_len, MSG_PEEK ,&saddr , &saddr_size);//change max_data_len to something smaller,we only need header here
//iphdr * iph = (struct iphdr *) (ip_begin);
//mylog(log_info,"recv_len %d\n",recv_len);
//if(recv_len<int(sizeof(iphdr)))
//{
// mylog(log_trace,"failed here %d %d\n",recv_len,int(sizeof(iphdr)));
// mylog(log_trace,"%s\n ",strerror(errno));
// return -1;
//}
//peek_info.new_src_ip.v4=iph->saddr;
//unsigned short iphdrlen =iph->ihl*4;
//char *payload=ip_begin+iphdrlen;
packet_info_t & recv_info = raw_info . recv_info ;
2017-07-29 20:32:26 +08:00
2018-07-23 13:52:36 -05:00
char * payload ; int payload_len ;
if ( recv_raw_ip ( raw_info , payload , payload_len ) ! = 0 )
return - 1 ;
//mylog(log_info,"protocol %d\n",iph->protocol);
switch ( raw_mode )
{
case mode_faketcp :
{
if ( recv_info . protocol ! = IPPROTO_TCP )
{
mylog ( log_trace , " failed here " ) ;
return - 1 ;
}
2018-07-26 11:25:36 -05:00
struct my_tcphdr * tcph = ( my_tcphdr * ) payload ;
if ( payload_len < int ( sizeof ( my_tcphdr ) ) )
2018-07-23 13:52:36 -05:00
{
mylog ( log_trace , " failed here " ) ;
return - 1 ;
}
recv_info . src_port = ntohs ( tcph - > source ) ;
recv_info . syn = tcph - > syn ;
break ;
}
case mode_udp :
{
if ( recv_info . protocol ! = IPPROTO_UDP ) return - 1 ;
2018-07-26 11:25:36 -05:00
struct my_udphdr * udph = ( my_udphdr * ) payload ;
if ( payload_len < int ( sizeof ( my_udphdr ) ) )
2018-07-23 13:52:36 -05:00
return - 1 ;
recv_info . src_port = ntohs ( udph - > source ) ;
break ;
}
case mode_icmp :
{
2018-07-24 08:33:42 -05:00
if ( raw_ip_version = = AF_INET )
{
if ( recv_info . protocol ! = IPPROTO_ICMP ) return - 1 ;
} else
{
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 08:33:42 -05:00
if ( recv_info . protocol ! = IPPROTO_ICMPV6 ) return - 1 ;
}
2018-07-26 11:25:36 -05:00
struct my_icmphdr * icmph = ( my_icmphdr * ) payload ;
if ( payload_len < int ( sizeof ( my_udphdr ) ) )
2018-07-23 13:52:36 -05:00
return - 1 ;
recv_info . src_port = ntohs ( icmph - > id ) ;
break ;
}
default : return - 1 ;
}
return 0 ;
}
2017-07-29 20:32:26 +08:00
int send_raw_icmp ( raw_info_t & raw_info , const char * payload , int payloadlen )
{
2017-08-03 23:32:19 +08:00
const packet_info_t & send_info = raw_info . send_info ;
2017-07-30 23:18:37 +08:00
const packet_info_t & recv_info = raw_info . recv_info ;
2017-07-29 20:32:26 +08:00
char send_raw_icmp_buf [ buf_len ] ;
2018-07-26 11:25:36 -05:00
my_icmphdr * icmph = ( struct my_icmphdr * ) ( send_raw_icmp_buf ) ;
memset ( icmph , 0 , sizeof ( my_icmphdr ) ) ;
2018-07-24 08:33:42 -05:00
if ( raw_ip_version = = AF_INET )
2017-07-29 20:32:26 +08:00
{
2018-07-24 08:33:42 -05:00
if ( program_mode = = client_mode )
{
icmph - > type = 8 ;
}
else
{
icmph - > type = 0 ;
}
2017-07-29 20:32:26 +08:00
}
else
{
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 08:33:42 -05:00
if ( program_mode = = client_mode )
{
icmph - > type = 128 ;
}
else
{
icmph - > type = 129 ;
}
2017-07-29 20:32:26 +08:00
}
icmph - > code = 0 ;
icmph - > id = htons ( send_info . src_port ) ;
2018-07-26 12:41:55 -05:00
icmph - > seq = htons ( send_info . my_icmp_seq ) ; /////////////modify
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
memcpy ( send_raw_icmp_buf + sizeof ( my_icmphdr ) , payload , payloadlen ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 08:33:42 -05:00
if ( raw_ip_version = = AF_INET )
{
2018-07-26 11:25:36 -05:00
icmph - > check_sum = csum ( ( unsigned short * ) send_raw_icmp_buf , sizeof ( my_icmphdr ) + payloadlen ) ;
2018-07-24 08:33:42 -05:00
}
else
{
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 08:33:42 -05:00
pseudo_header6 v6 ;
struct pseudo_header6 * psh = & v6 ;
psh - > src = send_info . new_src_ip . v6 ;
psh - > dst = send_info . new_dst_ip . v6 ;
psh - > next_header = IPPROTO_ICMPV6 ;
2018-07-26 11:25:36 -05:00
psh - > tcp_length = htons ( sizeof ( my_icmphdr ) + payloadlen ) ;
2018-07-24 08:33:42 -05:00
psh - > placeholder1 = 0 ;
psh - > placeholder2 = 0 ;
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
icmph - > check_sum = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header6 ) , ( unsigned short * ) send_raw_icmp_buf , sizeof ( my_icmphdr ) + payloadlen ) ;
2018-07-24 08:33:42 -05:00
}
2018-07-26 11:25:36 -05:00
if ( send_raw_ip ( raw_info , send_raw_icmp_buf , sizeof ( my_icmphdr ) + payloadlen ) ! = 0 )
2017-07-29 20:32:26 +08:00
{
return - 1 ;
}
2017-08-03 23:32:19 +08:00
/*if(program_mode==client_mode)
2017-07-31 19:01:50 +08:00
{
send_info . icmp_seq + + ;
2017-08-03 23:32:19 +08:00
} */
2017-07-31 19:01:50 +08:00
2017-07-29 20:32:26 +08:00
return 0 ;
}
int send_raw_udp ( raw_info_t & raw_info , const char * payload , int payloadlen )
{
2017-07-30 23:18:37 +08:00
const packet_info_t & send_info = raw_info . send_info ;
const packet_info_t & recv_info = raw_info . recv_info ;
2017-07-29 20:32:26 +08:00
char send_raw_udp_buf [ buf_len ] ;
2018-07-27 02:26:24 -05:00
my_udphdr * udph = ( struct my_udphdr * ) ( send_raw_udp_buf ) ;
2017-07-29 20:32:26 +08:00
2018-07-27 02:26:24 -05:00
memset ( udph , 0 , sizeof ( my_udphdr ) ) ;
2017-07-29 20:32:26 +08:00
udph - > source = htons ( send_info . src_port ) ;
udph - > dest = htons ( send_info . dst_port ) ;
2018-07-27 02:26:24 -05:00
int udp_tot_len = payloadlen + sizeof ( my_udphdr ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
2017-07-29 20:32:26 +08:00
if ( udp_tot_len > 65535 )
{
mylog ( log_debug , " invalid len \n " ) ;
return - 1 ;
}
2017-07-31 20:09:45 +08:00
mylog ( log_trace , " udp_len:%d %d \n " , udp_tot_len , udph - > len ) ;
2017-07-29 20:32:26 +08:00
udph - > len = htons ( uint16_t ( udp_tot_len ) ) ;
2018-07-27 02:26:24 -05:00
memcpy ( send_raw_udp_buf + sizeof ( my_udphdr ) , payload , payloadlen ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
if ( raw_ip_version = = AF_INET )
{
pseudo_header v4 ;
struct pseudo_header * psh = & v4 ;
psh - > source_address = send_info . new_src_ip . v4 ;
psh - > dest_address = send_info . new_dst_ip . v4 ;
psh - > placeholder = 0 ;
psh - > protocol = IPPROTO_UDP ;
psh - > tcp_length = htons ( udp_tot_len ) ;
udph - > check = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header ) , ( unsigned short * ) send_raw_udp_buf , udp_tot_len ) ;
}
else
{
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 05:36:56 -05:00
pseudo_header6 v6 ;
struct pseudo_header6 * psh = & v6 ;
psh - > src = send_info . new_src_ip . v6 ;
psh - > dst = send_info . new_dst_ip . v6 ;
psh - > next_header = IPPROTO_UDP ;
psh - > tcp_length = htons ( udp_tot_len ) ;
psh - > placeholder1 = 0 ;
psh - > placeholder2 = 0 ;
udph - > check = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header6 ) , ( unsigned short * ) send_raw_udp_buf , udp_tot_len ) ;
}
2017-07-29 20:32:26 +08:00
2018-07-22 03:23:53 -05:00
if ( send_raw_ip ( raw_info , send_raw_udp_buf , udp_tot_len ) ! = 0 )
2017-07-29 20:32:26 +08:00
{
return - 1 ;
}
return 0 ;
}
int send_raw_tcp ( raw_info_t & raw_info , const char * payload , int payloadlen ) { //TODO seq increase
2017-08-03 23:32:19 +08:00
const packet_info_t & send_info = raw_info . send_info ;
const packet_info_t & recv_info = raw_info . recv_info ;
2017-07-29 20:32:26 +08:00
2017-07-30 05:53:30 +08:00
//mylog(log_debug,"syn %d\n",send_info.syn);
2017-07-29 20:32:26 +08:00
2017-08-31 10:36:33 -05:00
char send_raw_tcp_buf [ buf_len ] ;
//char *send_raw_tcp_buf=send_raw_tcp_buf0;
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
struct my_tcphdr * tcph = ( struct my_tcphdr * ) ( send_raw_tcp_buf ) ;
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
memset ( tcph , 0 , sizeof ( my_tcphdr ) ) ;
2017-07-29 20:32:26 +08:00
//TCP Header
tcph - > source = htons ( send_info . src_port ) ;
tcph - > dest = htons ( send_info . dst_port ) ;
tcph - > seq = htonl ( send_info . seq ) ;
tcph - > ack_seq = htonl ( send_info . ack_seq ) ;
tcph - > fin = 0 ;
tcph - > syn = send_info . syn ;
tcph - > rst = 0 ;
tcph - > psh = send_info . psh ;
tcph - > ack = send_info . ack ;
if ( tcph - > syn = = 1 ) {
tcph - > doff = 10 ; //tcp header size
2018-07-26 11:25:36 -05:00
int i = sizeof ( my_tcphdr ) ;
2017-07-29 20:32:26 +08:00
send_raw_tcp_buf [ i + + ] = 0x02 ; //mss
send_raw_tcp_buf [ i + + ] = 0x04 ;
send_raw_tcp_buf [ i + + ] = 0x05 ;
send_raw_tcp_buf [ i + + ] = ( char ) 0xb4 ;
//raw_send_buf[i++]=0x01;
//raw_send_buf[i++]=0x01;
send_raw_tcp_buf [ i + + ] = 0x04 ; //sack ok
send_raw_tcp_buf [ i + + ] = 0x02 ; //sack ok
2017-08-04 03:29:53 +08:00
send_raw_tcp_buf [ i + + ] = 0x08 ; //ts i=6
send_raw_tcp_buf [ i + + ] = 0x0a ; //i=7
2017-07-29 20:32:26 +08:00
2017-08-31 10:36:33 -05:00
//*(u32_t*) (&send_raw_tcp_buf[i]) = htonl(
// (u32_t) get_current_time());
u32_t ts = htonl ( ( u32_t ) get_current_time ( ) ) ;
memcpy ( & send_raw_tcp_buf [ i ] , & ts , sizeof ( ts ) ) ;
2017-07-29 20:32:26 +08:00
i + = 4 ;
2017-08-04 03:29:53 +08:00
//mylog(log_info,"[syn]<send_info.ts_ack= %u>\n",send_info.ts_ack);
2017-08-31 10:36:33 -05:00
//*(u32_t*) (&send_raw_tcp_buf[i]) = htonl(send_info.ts_ack);
u32_t ts_ack = htonl ( send_info . ts_ack ) ;
memcpy ( & send_raw_tcp_buf [ i ] , & ts_ack , sizeof ( ts_ack ) ) ;
2017-07-29 20:32:26 +08:00
i + = 4 ;
send_raw_tcp_buf [ i + + ] = 0x01 ;
send_raw_tcp_buf [ i + + ] = 0x03 ;
send_raw_tcp_buf [ i + + ] = 0x03 ;
2017-09-06 05:31:29 -05:00
send_raw_tcp_buf [ i + + ] = wscale ;
2017-07-29 20:32:26 +08:00
} else {
tcph - > doff = 8 ;
2018-07-26 11:25:36 -05:00
int i = sizeof ( my_tcphdr ) ;
2017-07-29 20:32:26 +08:00
send_raw_tcp_buf [ i + + ] = 0x01 ;
send_raw_tcp_buf [ i + + ] = 0x01 ;
2017-08-04 03:29:53 +08:00
send_raw_tcp_buf [ i + + ] = 0x08 ; //ts //i=2
send_raw_tcp_buf [ i + + ] = 0x0a ; //i=3;
2017-07-29 20:32:26 +08:00
2017-08-31 10:36:33 -05:00
//*(u32_t*) (&send_raw_tcp_buf[i]) = htonl(
// (u32_t) get_current_time());
u32_t ts = htonl ( ( u32_t ) get_current_time ( ) ) ;
memcpy ( & send_raw_tcp_buf [ i ] , & ts , sizeof ( ts ) ) ;
2017-07-29 20:32:26 +08:00
i + = 4 ;
2017-08-04 03:29:53 +08:00
//mylog(log_info,"<send_info.ts_ack= %u>\n",send_info.ts_ack);
2017-08-31 10:36:33 -05:00
//*(u32_t*) (&send_raw_tcp_buf[i]) = htonl(send_info.ts_ack);
u32_t ts_ack = htonl ( send_info . ts_ack ) ;
memcpy ( & send_raw_tcp_buf [ i ] , & ts_ack , sizeof ( ts_ack ) ) ;
2017-07-29 20:32:26 +08:00
i + = 4 ;
}
tcph - > urg = 0 ;
//tcph->window = htons((uint16_t)(1024));
2018-08-29 20:00:37 -05:00
tcph - > window = htons ( ( uint16_t ) ( receive_window_lower_bound + get_true_random_number ( ) % receive_window_random_range ) ) ;
2017-07-29 20:32:26 +08:00
tcph - > check = 0 ; //leave checksum 0 now, filled later by pseudo header
tcph - > urg_ptr = 0 ;
2018-07-20 09:29:10 -05:00
char * tcp_data = send_raw_tcp_buf + + tcph - > doff * 4 ;
2017-07-29 20:32:26 +08:00
memcpy ( tcp_data , payload , payloadlen ) ;
2018-07-24 05:36:56 -05:00
int tcp_totlen = tcph - > doff * 4 + payloadlen ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
if ( raw_ip_version = = AF_INET )
{
pseudo_header v4 ;
struct pseudo_header * psh = & v4 ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
psh - > source_address = send_info . new_src_ip . v4 ;
psh - > dest_address = send_info . new_dst_ip . v4 ;
psh - > placeholder = 0 ;
psh - > protocol = IPPROTO_TCP ;
2018-07-24 08:33:42 -05:00
psh - > tcp_length = htons ( tcp_totlen ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
tcph - > check = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header ) , ( unsigned short * ) send_raw_tcp_buf , tcp_totlen ) ;
}
else
{
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 05:36:56 -05:00
pseudo_header6 v6 ;
struct pseudo_header6 * psh = & v6 ;
psh - > src = send_info . new_src_ip . v6 ;
psh - > dst = send_info . new_dst_ip . v6 ;
psh - > next_header = IPPROTO_TCP ;
2018-07-24 08:33:42 -05:00
psh - > tcp_length = htons ( tcp_totlen ) ;
2018-07-24 05:36:56 -05:00
psh - > placeholder1 = 0 ;
psh - > placeholder2 = 0 ;
tcph - > check = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header6 ) , ( unsigned short * ) send_raw_tcp_buf , tcp_totlen ) ;
}
2017-07-29 20:32:26 +08:00
2018-07-20 09:29:10 -05:00
if ( send_raw_ip ( raw_info , send_raw_tcp_buf , tcp_totlen ) ! = 0 )
2017-07-29 20:32:26 +08:00
{
return - 1 ;
}
2017-09-06 01:31:29 -05:00
raw_info . send_info . data_len = payloadlen ;
2017-07-29 20:32:26 +08:00
return 0 ;
}
/*
int send_raw_tcp_deprecated ( const packet_info_t & info , const char * payload , int payloadlen )
{
static uint16_t ip_id = 1 ;
char raw_send_buf [ buf_len ] ;
char raw_send_buf2 [ buf_len ] ;
//if((prog_mode==client_mode&& payloadlen!=9) ||(prog_mode==server_mode&& payloadlen!=5 ) )
mylog ( log_trace , " send raw from to %d %d %d %d \n " , info . src_ip , info . src_port , info . dst_ip , info . dst_port ) ;
char * data ;
memset ( raw_send_buf , 0 , payloadlen + 100 ) ;
struct iphdr * iph = ( struct iphdr * ) raw_send_buf ;
//TCP header
struct tcphdr * tcph = ( struct tcphdr * ) ( raw_send_buf + sizeof ( struct ip ) ) ;
struct sockaddr_in sin ;
struct pseudo_header psh ;
//some address resolution
sin . sin_family = AF_INET ;
sin . sin_port = htons ( info . dst_port ) ;
sin . sin_addr . s_addr = info . dst_ip ;
//Fill in the IP Header
iph - > ihl = 5 ;
iph - > version = 4 ;
iph - > tos = 0 ;
iph - > id = htonl ( ip_id + + ) ; //Id of this packet
iph - > frag_off = htons ( 0x4000 ) ; //DF set,others are zero
2018-06-05 11:40:09 -05:00
iph - > ttl = ( unsigned char ) ttl_value ;
2017-07-29 20:32:26 +08:00
iph - > protocol = IPPROTO_TCP ;
iph - > check = 0 ; //Set to 0 before calculating checksum
iph - > saddr = info . src_ip ; //Spoof the source ip address
iph - > daddr = info . dst_ip ;
//TCP Header
tcph - > source = htons ( info . src_port ) ;
tcph - > dest = htons ( info . dst_port ) ;
tcph - > seq = htonl ( info . seq ) ;
tcph - > ack_seq = htonl ( info . ack_seq ) ;
tcph - > fin = 0 ;
tcph - > syn = info . syn ;
tcph - > rst = 0 ;
tcph - > psh = info . psh ;
tcph - > ack = info . ack ;
if ( tcph - > syn = = 1 )
{
tcph - > doff = 10 ; //tcp header size
int i = sizeof ( struct iphdr ) + 20 ;
raw_send_buf [ i + + ] = 0x02 ; //mss
raw_send_buf [ i + + ] = 0x04 ;
raw_send_buf [ i + + ] = 0x05 ;
raw_send_buf [ i + + ] = 0xb4 ;
//raw_send_buf[i++]=0x01;
//raw_send_buf[i++]=0x01;
raw_send_buf [ i + + ] = 0x04 ; //sack ok
raw_send_buf [ i + + ] = 0x02 ; //sack ok
raw_send_buf [ i + + ] = 0x08 ; //i=6;
raw_send_buf [ i + + ] = 0x0a ;
* ( uint32_t * ) ( & raw_send_buf [ i ] ) = htonl ( ( uint32_t ) get_current_time ( ) ) ;
i + = 4 ;
* ( uint32_t * ) ( & raw_send_buf [ i ] ) = htonl ( info . ts_ack ) ;
i + = 4 ;
raw_send_buf [ i + + ] = 0x01 ;
raw_send_buf [ i + + ] = 0x03 ;
raw_send_buf [ i + + ] = 0x03 ;
raw_send_buf [ i + + ] = 0x05 ;
}
else
{
tcph - > doff = 8 ;
int i = sizeof ( struct iphdr ) + 20 ;
raw_send_buf [ i + + ] = 0x01 ;
raw_send_buf [ i + + ] = 0x01 ;
raw_send_buf [ i + + ] = 0x08 ; //i=0;
raw_send_buf [ i + + ] = 0x0a ;
* ( uint32_t * ) ( & raw_send_buf [ i ] ) = htonl ( ( uint32_t ) get_current_time ( ) ) ;
i + = 4 ;
* ( uint32_t * ) ( & raw_send_buf [ i ] ) = htonl ( info . ts_ack ) ;
i + = 4 ;
}
tcph - > urg = 0 ;
//tcph->window = htons((uint16_t)(1024));
tcph - > window = htons ( ( uint16_t ) ( 10240 + random ( ) % 100 ) ) ;
tcph - > check = 0 ; //leave checksum 0 now, filled later by pseudo header
tcph - > urg_ptr = 0 ;
//Data part
data = raw_send_buf + sizeof ( struct iphdr ) + tcph - > doff * 4 ;
iph - > tot_len = sizeof ( struct iphdr ) + tcph - > doff * 4 + payloadlen ;
memcpy ( data , payload , payloadlen ) ;
psh . source_address = info . src_ip ;
psh . dest_address = sin . sin_addr . s_addr ;
psh . placeholder = 0 ;
psh . protocol = IPPROTO_TCP ;
psh . tcp_length = htons ( tcph - > doff * 4 + payloadlen ) ;
int psize = sizeof ( struct pseudo_header ) + tcph - > doff * 4 + payloadlen ;
memcpy ( raw_send_buf2 , ( char * ) & psh , sizeof ( struct pseudo_header ) ) ;
memcpy ( raw_send_buf2 + sizeof ( struct pseudo_header ) , tcph , tcph - > doff * 4 + payloadlen ) ;
tcph - > check = csum ( ( unsigned short * ) raw_send_buf2 , psize ) ;
//Ip checksum
iph - > check = csum ( ( unsigned short * ) raw_send_buf , iph - > tot_len ) ;
mylog ( log_trace , " sent seq ack_seq len<%u %u %d> \n " , g_packet_info_send . seq , g_packet_info_send . ack_seq , payloadlen ) ;
int ret = sendto ( raw_send_fd , raw_send_buf , iph - > tot_len , 0 , ( struct sockaddr * ) & sin , sizeof ( sin ) ) ;
if ( g_packet_info_send . syn = = 0 & & g_packet_info_send . ack = = 1 & & payloadlen ! = 0 )
{
if ( seq_mode = = 0 )
{
}
else if ( seq_mode = = 1 )
{
g_packet_info_send . seq + = payloadlen ;
}
else if ( seq_mode = = 2 )
{
if ( random ( ) % 5 = = 3 )
g_packet_info_send . seq + = payloadlen ;
}
}
mylog ( log_trace , " <ret:%d> \n " , ret ) ;
if ( ret < 0 )
{
mylog ( log_fatal , " " ) ;
perror ( " raw send error \n " ) ;
//printf("send error\n");
}
return 0 ;
}
*/
int recv_raw_icmp ( raw_info_t & raw_info , char * & payload , int & payloadlen )
{
2017-08-03 23:32:19 +08:00
const packet_info_t & send_info = raw_info . send_info ;
2017-07-29 20:32:26 +08:00
packet_info_t & recv_info = raw_info . recv_info ;
static char recv_raw_icmp_buf [ buf_len ] ;
char * ip_payload ;
int ip_payloadlen ;
if ( recv_raw_ip ( raw_info , ip_payload , ip_payloadlen ) ! = 0 )
{
mylog ( log_debug , " recv_raw_ip error \n " ) ;
return - 1 ;
}
2018-08-31 17:10:46 +00:00
mylog ( log_trace , " ip_payloadlen=%d \n " , ip_payloadlen ) ;
2018-07-24 08:33:42 -05:00
if ( raw_ip_version = = AF_INET )
2017-07-29 20:32:26 +08:00
{
2018-07-24 08:33:42 -05:00
if ( recv_info . protocol ! = IPPROTO_ICMP )
{
//printf("not udp protocol\n");
return - 1 ;
}
}
else
{
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 08:33:42 -05:00
if ( recv_info . protocol ! = IPPROTO_ICMPV6 )
{
//printf("not udp protocol\n");
return - 1 ;
}
2017-07-29 20:32:26 +08:00
}
2018-08-31 17:10:46 +00:00
if ( ip_payloadlen < int ( sizeof ( my_icmphdr ) ) )
{
mylog ( log_debug , " too short to hold icmp header \n " ) ;
return - 1 ;
}
2017-08-04 03:29:53 +08:00
2018-07-26 11:25:36 -05:00
my_icmphdr * icmph = ( struct my_icmphdr * ) ( ip_payload ) ;
2017-07-29 20:32:26 +08:00
2017-07-31 19:01:50 +08:00
if ( ntohs ( icmph - > id ) ! = send_info . src_port )
{
2017-08-02 18:57:27 +08:00
mylog ( log_debug , " icmp id mis-match,ignored \n " ) ;
2017-07-31 19:01:50 +08:00
return - 1 ;
}
2017-07-29 20:32:26 +08:00
recv_info . src_port = recv_info . dst_port = ntohs ( icmph - > id ) ;
2018-07-26 12:41:55 -05:00
recv_info . my_icmp_seq = ntohs ( icmph - > seq ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 08:33:42 -05:00
if ( icmph - > code ! = 0 )
return - 1 ;
2017-07-31 19:01:50 +08:00
2018-07-24 08:33:42 -05:00
unsigned short check ;
if ( raw_ip_version = = AF_INET )
2017-07-29 20:32:26 +08:00
{
2018-07-24 08:33:42 -05:00
if ( program_mode = = client_mode )
{
if ( icmph - > type ! = 0 )
return - 1 ;
}
else
{
if ( icmph - > type ! = 8 )
return - 1 ;
}
check = csum ( ( unsigned short * ) ip_payload , ip_payloadlen ) ;
2017-07-29 20:32:26 +08:00
}
else
{
2018-08-29 16:48:45 -05:00
assert ( raw_ip_version = = AF_INET6 ) ;
2018-07-24 08:33:42 -05:00
if ( program_mode = = client_mode )
{
if ( icmph - > type ! = 129 )
return - 1 ;
}
else
{
if ( icmph - > type ! = 128 )
return - 1 ;
}
2017-07-31 19:01:50 +08:00
2018-07-24 08:33:42 -05:00
pseudo_header6 tmp_header ;
struct pseudo_header6 * psh = & tmp_header ;
2017-07-29 20:32:26 +08:00
2018-07-24 08:33:42 -05:00
psh - > src = recv_info . new_src_ip . v6 ;
psh - > dst = recv_info . new_dst_ip . v6 ;
psh - > placeholder1 = 0 ;
psh - > placeholder2 = 0 ;
psh - > next_header = IPPROTO_ICMPV6 ;
psh - > tcp_length = htons ( ip_payloadlen ) ;
check = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header6 ) , ( unsigned short * ) ip_payload , ip_payloadlen ) ;
}
2017-07-29 20:32:26 +08:00
if ( check ! = 0 )
{
mylog ( log_debug , " icmp checksum fail %x \n " , check ) ;
return - 1 ;
}
2017-07-31 19:01:50 +08:00
//mylog(log_info,"program_mode=%d\n",program_mode);
2017-08-03 23:32:19 +08:00
/*
2017-07-31 19:01:50 +08:00
if ( program_mode = = server_mode )
{
send_info . icmp_seq = ntohs ( icmph - > seq ) ;
//mylog(log_info,"send_info.seq=%d\n",send_info.seq);
2017-08-03 23:32:19 +08:00
} */
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
payload = ip_payload + sizeof ( my_icmphdr ) ;
payloadlen = ip_payloadlen - sizeof ( my_icmphdr ) ;
2017-07-31 20:09:45 +08:00
mylog ( log_trace , " get a packet len=%d \n " , payloadlen ) ;
2017-07-29 20:32:26 +08:00
return 0 ;
}
int recv_raw_udp ( raw_info_t & raw_info , char * & payload , int & payloadlen )
{
2017-07-30 23:18:37 +08:00
const packet_info_t & send_info = raw_info . send_info ;
2017-07-29 20:32:26 +08:00
packet_info_t & recv_info = raw_info . recv_info ;
2018-07-22 03:23:53 -05:00
//static char recv_raw_udp_buf[buf_len];
2017-07-29 20:32:26 +08:00
char * ip_payload ;
int ip_payloadlen ;
if ( recv_raw_ip ( raw_info , ip_payload , ip_payloadlen ) ! = 0 )
{
mylog ( log_debug , " recv_raw_ip error \n " ) ;
return - 1 ;
}
if ( recv_info . protocol ! = IPPROTO_UDP )
{
//printf("not udp protocol\n");
return - 1 ;
}
2018-07-27 02:26:24 -05:00
if ( ip_payloadlen < int ( sizeof ( my_udphdr ) ) )
2017-07-29 20:32:26 +08:00
{
mylog ( log_debug , " too short to hold udpheader \n " ) ;
return - 1 ;
}
2018-07-27 02:26:24 -05:00
my_udphdr * udph = ( struct my_udphdr * ) ip_payload ;
2017-07-29 20:32:26 +08:00
2017-08-13 02:14:50 +08:00
if ( int ( ntohs ( udph - > len ) ) ! = ip_payloadlen )
2017-07-29 20:32:26 +08:00
{
mylog ( log_debug , " udp length error %d %d \n " , ntohs ( udph - > len ) , ip_payloadlen ) ;
return - 1 ;
}
if ( udph - > dest ! = ntohs ( uint16_t ( filter_port ) ) )
{
//printf("%x %x",tcph->dest,);
return - 1 ;
}
2018-07-22 03:23:53 -05:00
//memcpy(recv_raw_udp_buf+ sizeof(struct pseudo_header) , ip_payload , ip_payloadlen);
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
/*
2018-07-22 03:23:53 -05:00
pseudo_header tmp_header = { 0 } ;
struct pseudo_header * psh = & tmp_header ;
2017-07-29 20:32:26 +08:00
2018-07-22 08:59:09 -05:00
psh - > source_address = recv_info . new_src_ip . v4 ;
psh - > dest_address = recv_info . new_dst_ip . v4 ;
2017-07-29 20:32:26 +08:00
psh - > placeholder = 0 ;
psh - > protocol = IPPROTO_UDP ;
psh - > tcp_length = htons ( ip_payloadlen ) ;
2018-07-22 03:23:53 -05:00
int csum_len = ip_payloadlen ;
uint16_t udp_chk = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header ) , ( unsigned short * ) ip_payload , csum_len ) ;
2018-07-24 05:36:56 -05:00
*/
uint16_t udp_chk ;
int csum_len = ip_payloadlen ;
if ( raw_ip_version = = AF_INET )
{
pseudo_header tmp_header ;
struct pseudo_header * psh = & tmp_header ;
psh - > source_address = recv_info . new_src_ip . v4 ;
psh - > dest_address = recv_info . new_dst_ip . v4 ;
psh - > placeholder = 0 ;
psh - > protocol = IPPROTO_UDP ;
psh - > tcp_length = htons ( ip_payloadlen ) ;
udp_chk = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header ) , ( unsigned short * ) ip_payload , csum_len ) ;
}
else
{
assert ( raw_ip_version = = AF_INET6 ) ;
pseudo_header6 tmp_header ;
struct pseudo_header6 * psh = & tmp_header ;
psh - > src = recv_info . new_src_ip . v6 ;
psh - > dst = recv_info . new_dst_ip . v6 ;
psh - > placeholder1 = 0 ;
psh - > placeholder2 = 0 ;
psh - > next_header = IPPROTO_UDP ;
psh - > tcp_length = htons ( ip_payloadlen ) ;
udp_chk = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header6 ) , ( unsigned short * ) ip_payload , csum_len ) ;
}
2017-07-29 20:32:26 +08:00
if ( udp_chk ! = 0 )
{
mylog ( log_debug , " udp_chk:%x \n " , udp_chk ) ;
mylog ( log_debug , " udp header error \n " ) ;
return - 1 ;
}
2018-07-22 03:23:53 -05:00
char * udp_begin = ip_payload ;
2017-07-29 20:32:26 +08:00
recv_info . src_port = ntohs ( udph - > source ) ;
recv_info . dst_port = ntohs ( udph - > dest ) ;
2018-07-27 02:26:24 -05:00
payloadlen = ip_payloadlen - sizeof ( my_udphdr ) ;
2017-07-29 20:32:26 +08:00
2018-07-27 02:26:24 -05:00
payload = udp_begin + sizeof ( my_udphdr ) ;
2017-07-29 20:32:26 +08:00
return 0 ;
}
2018-07-20 09:29:10 -05:00
int parse_tcp_option ( char * option_begin , char * option_end , packet_info_t & recv_info )
{
recv_info . has_ts = 0 ;
recv_info . ts = 0 ;
char * ptr = option_begin ;
//char *option_end=tcp_begin+tcp_hdr_len;
while ( ptr < option_end )
{
if ( * ptr = = 0 )
{
return 0 ;
}
else if ( * ptr = = 1 )
{
ptr + + ;
}
else if ( * ptr = = 8 )
{
if ( ptr + 1 > = option_end )
{
2018-07-25 05:22:07 -05:00
mylog ( log_trace , " invaild option ptr+1==option_end,for ts \n " ) ;
2018-07-20 09:29:10 -05:00
return - 1 ;
}
if ( * ( ptr + 1 ) ! = 10 )
{
2018-07-25 05:22:07 -05:00
mylog ( log_trace , " invaild ts len \n " ) ;
2018-07-20 09:29:10 -05:00
return - 1 ;
}
if ( ptr + 10 > option_end )
{
2018-07-25 05:22:07 -05:00
mylog ( log_trace , " ptr+10>option_end for ts \n " ) ;
2018-07-20 09:29:10 -05:00
return - 1 ;
}
2017-07-29 20:32:26 +08:00
2018-07-20 09:29:10 -05:00
recv_info . has_ts = 1 ;
recv_info . ts = read_u32 ( ptr + 2 ) ;
recv_info . ts_ack = read_u32 ( ptr + 6 ) ;
//printf("<%d %d>!\n",recv_info.ts,recv_info.ts_ack);
//return 0;//we currently only parse ts, so just return after its found
ptr + = 8 ;
}
else
{
if ( ptr + 1 > = option_end )
{
2018-07-25 05:22:07 -05:00
mylog ( log_trace , " invaild option ptr+1==option_end \n " ) ;
2018-07-20 09:29:10 -05:00
return - 1 ;
}
else
{
2018-07-20 11:54:57 -05:00
int len = ( unsigned char ) * ( ptr + 1 ) ;
if ( len < = 1 )
{
2018-07-25 05:22:07 -05:00
mylog ( log_trace , " invaild option len %d \n " , len ) ;
2018-07-20 11:54:57 -05:00
return - 1 ;
}
2018-07-20 09:29:10 -05:00
//omit check
2018-07-20 11:54:57 -05:00
ptr + = len ;
2018-07-20 09:29:10 -05:00
}
}
2018-07-20 11:54:57 -05:00
//printf("!");
2018-07-20 09:29:10 -05:00
}
2018-07-20 11:54:57 -05:00
//printf("\n");
2018-07-20 09:29:10 -05:00
return 0 ;
}
2017-07-29 20:32:26 +08:00
int recv_raw_tcp ( raw_info_t & raw_info , char * & payload , int & payloadlen )
{
2017-08-03 23:32:19 +08:00
const packet_info_t & send_info = raw_info . send_info ;
2017-07-29 20:32:26 +08:00
packet_info_t & recv_info = raw_info . recv_info ;
2018-07-20 09:29:10 -05:00
//static char recv_raw_tcp_buf[buf_len];
2017-07-29 20:32:26 +08:00
char * ip_payload ;
int ip_payloadlen ;
if ( recv_raw_ip ( raw_info , ip_payload , ip_payloadlen ) ! = 0 )
{
mylog ( log_debug , " recv_raw_ip error \n " ) ;
return - 1 ;
}
if ( recv_info . protocol ! = IPPROTO_TCP )
{
//printf("not tcp protocol\n");
return - 1 ;
}
2018-07-26 11:25:36 -05:00
my_tcphdr * tcph = ( struct my_tcphdr * ) ip_payload ;
2017-07-29 20:32:26 +08:00
unsigned short tcphdrlen = tcph - > doff * 4 ;
2017-08-04 03:29:53 +08:00
if ( ! ( tcphdrlen > 0 & & tcphdrlen < = 60 ) ) {
2017-07-29 20:32:26 +08:00
mylog ( log_debug , " tcph error \n " ) ;
return 0 ;
}
2018-07-20 11:54:57 -05:00
if ( tcphdrlen > ip_payloadlen ) {
mylog ( log_debug , " error,tcphdrlen >ip_payloadlen \n " ) ;
return 0 ;
}
2017-07-29 20:32:26 +08:00
if ( tcph - > dest ! = ntohs ( uint16_t ( filter_port ) ) )
{
//printf("%x %x",tcph->dest,);
return - 1 ;
}
2018-07-20 09:29:10 -05:00
// memcpy(recv_raw_tcp_buf+ sizeof(struct pseudo_header) , ip_payload , ip_payloadlen);
2018-07-24 05:36:56 -05:00
uint16_t tcp_chk ;
int csum_len = ip_payloadlen ;
if ( raw_ip_version = = AF_INET )
{
pseudo_header tmp_header ;
struct pseudo_header * psh = & tmp_header ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
psh - > source_address = recv_info . new_src_ip . v4 ;
psh - > dest_address = recv_info . new_dst_ip . v4 ;
psh - > placeholder = 0 ;
psh - > protocol = IPPROTO_TCP ;
psh - > tcp_length = htons ( ip_payloadlen ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
tcp_chk = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header ) , ( unsigned short * ) ip_payload , csum_len ) ;
}
else
{
assert ( raw_ip_version = = AF_INET6 ) ;
2017-07-29 20:32:26 +08:00
2018-07-24 05:36:56 -05:00
pseudo_header6 tmp_header ;
struct pseudo_header6 * psh = & tmp_header ;
psh - > src = recv_info . new_src_ip . v6 ;
psh - > dst = recv_info . new_dst_ip . v6 ;
psh - > placeholder1 = 0 ;
psh - > placeholder2 = 0 ;
psh - > next_header = IPPROTO_TCP ;
psh - > tcp_length = htons ( ip_payloadlen ) ;
2018-07-20 09:29:10 -05:00
2018-07-24 05:36:56 -05:00
tcp_chk = csum_with_header ( ( char * ) psh , sizeof ( pseudo_header6 ) , ( unsigned short * ) ip_payload , csum_len ) ;
}
2018-07-20 09:29:10 -05:00
/*for(int i=0;i<csum_len;i++)
{
printf ( " <%d> " , int ( ip_payload [ i ] ) ) ;
}
printf ( " \n " ) ; */
2017-07-29 20:32:26 +08:00
if ( tcp_chk ! = 0 )
{
mylog ( log_debug , " tcp_chk:%x \n " , tcp_chk ) ;
mylog ( log_debug , " tcp header error \n " ) ;
return - 1 ;
}
2018-07-20 09:29:10 -05:00
char * tcp_begin = ip_payload ; //ip packet's data part
2017-07-29 20:32:26 +08:00
2018-07-26 11:25:36 -05:00
char * tcp_option = ip_payload + sizeof ( my_tcphdr ) ;
2018-07-20 11:54:57 -05:00
char * option_end = ip_payload + tcphdrlen ;
2017-07-29 20:32:26 +08:00
2018-07-20 09:29:10 -05:00
/*
//old ts parse code
2017-07-29 20:32:26 +08:00
recv_info . has_ts = 0 ;
2017-08-04 03:29:53 +08:00
recv_info . ts = 0 ;
2017-07-29 20:32:26 +08:00
if ( tcph - > doff = = 10 )
{
if ( tcp_option [ 6 ] = = 0x08 & & tcp_option [ 7 ] = = 0x0a )
{
recv_info . has_ts = 1 ;
2017-08-31 10:36:33 -05:00
//recv_info.ts=ntohl(*(u32_t*)(&tcp_option[8]));
memcpy ( & recv_info . ts , & tcp_option [ 8 ] , sizeof ( recv_info . ts ) ) ;
recv_info . ts = ntohl ( recv_info . ts ) ;
//recv_info.ts_ack=ntohl(*(u32_t*)(&tcp_option[12]));
memcpy ( & recv_info . ts_ack , & tcp_option [ 12 ] , sizeof ( recv_info . ts_ack ) ) ;
recv_info . ts_ack = ntohl ( recv_info . ts_ack ) ;
2017-07-29 20:32:26 +08:00
//g_packet_info_send.ts_ack= ntohl(*(uint32_t*)(&tcp_option[8]));
}
2017-08-04 03:29:53 +08:00
else
{
2017-08-04 17:12:23 +08:00
// mylog(log_info,"\n");
2017-08-04 03:29:53 +08:00
}
2017-07-29 20:32:26 +08:00
}
else if ( tcph - > doff = = 8 )
{
2017-08-04 03:29:53 +08:00
if ( tcp_option [ 2 ] = = 0x08 & & tcp_option [ 3 ] = = 0x0a )
2017-07-29 20:32:26 +08:00
{
recv_info . has_ts = 1 ;
2017-08-31 10:36:33 -05:00
//recv_info.ts=ntohl(*(u32_t*)(&tcp_option[4]));
memcpy ( & recv_info . ts , & tcp_option [ 4 ] , sizeof ( recv_info . ts ) ) ;
recv_info . ts = ntohl ( recv_info . ts ) ;
//recv_info.ts_ack=ntohl(*(u32_t*)(&tcp_option[8]));
memcpy ( & recv_info . ts_ack , & tcp_option [ 8 ] , sizeof ( recv_info . ts_ack ) ) ;
recv_info . ts_ack = ntohl ( recv_info . ts_ack ) ;
2017-07-29 20:32:26 +08:00
//g_packet_info_send.ts_ack= ntohl(*(uint32_t*)(&tcp_option[0]));
}
2017-08-04 03:29:53 +08:00
else
{
2017-08-04 17:12:23 +08:00
//mylog(log_info,"!!!\n");
2017-08-04 03:29:53 +08:00
}
}
else
{
2017-08-04 21:46:49 +08:00
//mylog(log_info,"tcph->doff= %u\n",tcph->doff);
2017-07-29 20:32:26 +08:00
}
2018-07-20 09:29:10 -05:00
printf ( " <%d %d> \n " , recv_info . ts , recv_info . ts_ack ) ;
*/
2018-07-20 11:54:57 -05:00
parse_tcp_option ( tcp_option , option_end , recv_info ) ;
2017-08-04 21:46:49 +08:00
2017-07-29 20:32:26 +08:00
recv_info . ack = tcph - > ack ;
recv_info . syn = tcph - > syn ;
recv_info . rst = tcph - > rst ;
recv_info . src_port = ntohs ( tcph - > source ) ;
recv_info . dst_port = ntohs ( tcph - > dest ) ;
recv_info . seq = ntohl ( tcph - > seq ) ;
2017-09-06 05:31:29 -05:00
// recv_info.last_last_ack_seq=recv_info.last_ack_seq;
//recv_info.last_ack_seq=recv_info.ack_seq;
u32_t last_ack_seq = recv_info . ack_seq ;
2017-07-29 20:32:26 +08:00
recv_info . ack_seq = ntohl ( tcph - > ack_seq ) ;
2017-09-06 05:31:29 -05:00
if ( recv_info . ack_seq = = last_ack_seq )
{
recv_info . ack_seq_counter + + ;
}
else
{
recv_info . ack_seq_counter = 0 ;
}
2017-07-29 20:32:26 +08:00
recv_info . psh = tcph - > psh ;
2017-08-04 21:46:49 +08:00
if ( tcph - > rst = = 1 )
{
2018-02-24 17:26:29 -06:00
raw_info . rst_received + + ;
2018-02-24 18:07:16 -06:00
2018-02-24 17:26:29 -06:00
if ( max_rst_to_show > 0 )
{
if ( raw_info . rst_received < max_rst_to_show )
{
2018-07-22 08:59:09 -05:00
mylog ( log_warn , " [%s,%d]rst==1,cnt=%d \n " , recv_info . new_src_ip . get_str1 ( ) , recv_info . src_port , ( int ) raw_info . rst_received ) ;
2018-02-24 17:26:29 -06:00
}
else if ( raw_info . rst_received = = max_rst_to_show )
{
2018-07-22 08:59:09 -05:00
mylog ( log_warn , " [%s,%d]rst==1,cnt=%d >=max_rst_to_show, this log will be muted for current connection \n " , recv_info . new_src_ip . get_str1 ( ) , recv_info . src_port , ( int ) raw_info . rst_received ) ;
2018-02-24 17:26:29 -06:00
}
else
{
2018-07-22 08:59:09 -05:00
mylog ( log_debug , " [%s,%d]rst==1,cnt=%d \n " , recv_info . new_src_ip . get_str1 ( ) , recv_info . src_port , ( int ) raw_info . rst_received ) ;
2018-02-24 17:26:29 -06:00
}
}
2018-02-24 18:07:16 -06:00
else if ( max_rst_to_show = = 0 )
2018-02-24 17:26:29 -06:00
{
2018-07-22 08:59:09 -05:00
mylog ( log_debug , " [%s,%d]rst==1,cnt=%d \n " , recv_info . new_src_ip . get_str1 ( ) , recv_info . src_port , ( int ) raw_info . rst_received ) ;
2018-02-24 17:26:29 -06:00
}
else
{
2018-07-22 08:59:09 -05:00
mylog ( log_warn , " [%s,%d]rst==1,cnt=%d \n " , recv_info . new_src_ip . get_str1 ( ) , recv_info . src_port , ( int ) raw_info . rst_received ) ;
2018-02-24 17:26:29 -06:00
}
if ( max_rst_allowed > = 0 & & raw_info . rst_received = = max_rst_allowed + 1 )
{
2018-07-22 08:59:09 -05:00
mylog ( log_warn , " [%s,%d]connection disabled because of rst_received=%d > max_rst_allow=%d \n " , recv_info . new_src_ip . get_str1 ( ) , recv_info . src_port , ( int ) raw_info . rst_received , ( int ) max_rst_allowed ) ;
2018-02-24 17:26:29 -06:00
raw_info . disabled = 1 ;
}
2017-08-04 21:46:49 +08:00
}
2017-08-03 23:32:19 +08:00
/* if(recv_info.has_ts)
2017-07-29 20:32:26 +08:00
{
2017-07-30 23:18:37 +08:00
send_info . ts_ack = recv_info . ts ; //////////////////////////////////////////////modify
2017-08-03 23:32:19 +08:00
} */
2017-07-29 20:32:26 +08:00
payloadlen = ip_payloadlen - tcphdrlen ;
payload = tcp_begin + tcphdrlen ;
2017-07-31 19:50:05 +08:00
2017-08-03 23:32:19 +08:00
/*if (recv_info.syn == 0 && recv_info.ack == 1&& payloadlen != 0) //only modify send_info when the packet is not part of handshake
2017-07-31 19:50:05 +08:00
{
send_info . ack_seq = recv_info . seq ;
2017-08-03 23:32:19 +08:00
} */
2017-09-06 01:31:29 -05:00
raw_info . recv_info . data_len = payloadlen ;
2017-07-29 20:32:26 +08:00
return 0 ;
}
/*
int recv_raw_tcp_deprecated ( packet_info_t & info , char * & payload , int & payloadlen )
{
static char buf [ buf_len ] ;
char raw_recv_buf [ buf_len ] ;
char raw_recv_buf2 [ buf_len ] ;
char raw_recv_buf3 [ buf_len ] ;
iphdr * iph ;
tcphdr * tcph ;
int size ;
struct sockaddr saddr ;
socklen_t saddr_size ;
saddr_size = sizeof ( saddr ) ;
mylog ( log_trace , " raw! \n " ) ;
2017-08-12 19:21:24 +08:00
size = recvfrom ( raw_recv_fd , buf , max_data_len , 0 , & saddr , & saddr_size ) ;
2017-07-29 20:32:26 +08:00
if ( buf [ 12 ] ! = 8 | | buf [ 13 ] ! = 0 )
{
mylog ( log_debug , " not an ipv4 packet! \n " ) ;
return - 1 ;
}
char * ip_begin = buf + 14 ;
iph = ( struct iphdr * ) ( ip_begin ) ;
if ( ! ( iph - > ihl > 0 & & iph - > ihl < = 60 ) ) {
mylog ( log_debug , " iph ihl error " ) ;
return - 1 ;
}
if ( iph - > protocol ! = IPPROTO_TCP ) {
mylog ( log_debug , " iph protocal != tcp \n " ) ;
return - 1 ;
}
int ip_len = ntohs ( iph - > tot_len ) ;
unsigned short iphdrlen = iph - > ihl * 4 ;
tcph = ( struct tcphdr * ) ( ip_begin + iphdrlen ) ;
unsigned short tcphdrlen = tcph - > doff * 4 ;
if ( ! ( tcph - > doff > 0 & & tcph - > doff < = 60 ) ) {
mylog ( log_debug , " tcph error " ) ;
return 0 ;
}
if ( tcph - > dest ! = ntohs ( uint16_t ( filter_port ) ) )
{
//printf("%x %x",tcph->dest,);
return - 1 ;
}
/////ip
uint32_t ip_chk = csum ( ( unsigned short * ) ip_begin , iphdrlen ) ;
int psize = sizeof ( struct pseudo_header ) + ip_len - iphdrlen ;
/////ip end
///tcp
struct pseudo_header psh ;
psh . source_address = iph - > saddr ;
psh . dest_address = iph - > daddr ;
psh . placeholder = 0 ;
psh . protocol = IPPROTO_TCP ;
psh . tcp_length = htons ( ip_len - iphdrlen ) ;
memcpy ( raw_recv_buf2 , ( char * ) & psh , sizeof ( struct pseudo_header ) ) ;
memcpy ( raw_recv_buf2 + sizeof ( struct pseudo_header ) , ip_begin + iphdrlen , ip_len - iphdrlen ) ;
uint16_t tcp_chk = csum ( ( unsigned short * ) raw_recv_buf2 , psize ) ;
if ( ip_chk ! = 0 )
{
mylog ( log_debug , " ip header error %d \n " , ip_chk ) ;
return - 1 ;
}
if ( tcp_chk ! = 0 )
{
mylog ( log_debug , " tcp_chk:%x \n " , tcp_chk ) ;
mylog ( log_debug , " tcp header error \n " ) ;
return - 1 ;
}
char * tcp_begin = raw_recv_buf2 + sizeof ( struct pseudo_header ) ; //ip packet's data part
char * tcp_option = raw_recv_buf2 + sizeof ( struct pseudo_header ) + sizeof ( tcphdr ) ;
info . has_ts = 0 ;
if ( tcph - > doff = = 10 )
{
if ( tcp_option [ 6 ] = = 0x08 & & tcp_option [ 7 ] = = 0x0a )
{
info . has_ts = 1 ;
info . ts = ntohl ( * ( uint32_t * ) ( & tcp_option [ 8 ] ) ) ;
info . ts_ack = ntohl ( * ( uint32_t * ) ( & tcp_option [ 12 ] ) ) ;
//g_packet_info_send.ts_ack= ntohl(*(uint32_t*)(&tcp_option[8]));
}
}
else if ( tcph - > doff = = 8 )
{
if ( tcp_option [ 3 ] = = 0x08 & & tcp_option [ 4 ] = = 0x0a )
{
info . has_ts = 1 ;
info . ts = ntohl ( * ( uint32_t * ) ( & tcp_option [ 0 ] ) ) ;
info . ts_ack = ntohl ( * ( uint32_t * ) ( & tcp_option [ 4 ] ) ) ;
//g_packet_info_send.ts_ack= ntohl(*(uint32_t*)(&tcp_option[0]));
}
}
if ( tcph - > rst = = 1 )
{
mylog ( log_warn , " %%%%%%%%%%rst==1%%%%%%%%%%%%% \n " ) ;
}
info . ack = tcph - > ack ;
info . syn = tcph - > syn ;
info . rst = tcph - > rst ;
info . src_port = ntohs ( tcph - > source ) ;
info . src_ip = iph - > saddr ;
info . seq = ntohl ( tcph - > seq ) ;
info . ack_seq = ntohl ( tcph - > ack_seq ) ;
info . psh = tcph - > psh ;
if ( info . has_ts )
{
g_packet_info_send . ts_ack = info . ts ;
}
////tcp end
payloadlen = ip_len - tcphdrlen - iphdrlen ;
payload = ip_begin + tcphdrlen + iphdrlen ;
if ( payloadlen > 0 & & payload [ 0 ] = = ' h ' )
{
mylog ( log_debug , " recvd <%u %u %d> \n " , ntohl ( tcph - > seq ) , ntohl ( tcph - > ack_seq ) , payloadlen ) ;
}
if ( payloadlen > 0 & & tcph - > syn = = 0 & & tcph - > ack = = 1 )
{
//if(seq_increse)
g_packet_info_send . ack_seq = ntohl ( tcph - > seq ) + ( uint32_t ) payloadlen ;
}
//printf("%d\n",ip_len);
mylog ( log_trace , " <%u,%u,%u,%u,%d> \n " , ( unsigned int ) iphdrlen , ( unsigned int ) tcphdrlen , ( unsigned int ) tcph - > syn , ( unsigned int ) tcph - > ack , payloadlen ) ;
return 0 ;
} */
2017-08-03 23:32:19 +08:00
int send_raw0 ( raw_info_t & raw_info , const char * payload , int payloadlen )
2017-07-29 20:32:26 +08:00
{
2017-10-11 09:10:38 -05:00
if ( random_drop ! = 0 ) {
if ( get_true_random_number ( ) % 10000 < ( u32_t ) random_drop ) {
return 0 ;
}
}
2017-07-29 20:32:26 +08:00
packet_info_t & send_info = raw_info . send_info ;
packet_info_t & recv_info = raw_info . recv_info ;
2018-07-22 08:59:09 -05:00
mylog ( log_trace , " send_raw : from %s %d to %s %d \n " , send_info . new_src_ip . get_str1 ( ) , send_info . src_port , send_info . new_dst_ip . get_str2 ( ) , send_info . dst_port ) ;
2017-07-29 20:32:26 +08:00
switch ( raw_mode )
{
case mode_faketcp : return send_raw_tcp ( raw_info , payload , payloadlen ) ;
case mode_udp : return send_raw_udp ( raw_info , payload , payloadlen ) ;
case mode_icmp : return send_raw_icmp ( raw_info , payload , payloadlen ) ;
default : return - 1 ;
}
}
2017-08-03 23:32:19 +08:00
int recv_raw0 ( raw_info_t & raw_info , char * & payload , int & payloadlen )
2017-07-29 20:32:26 +08:00
{
packet_info_t & send_info = raw_info . send_info ;
packet_info_t & recv_info = raw_info . recv_info ;
switch ( raw_mode )
{
case mode_faketcp : return recv_raw_tcp ( raw_info , payload , payloadlen ) ;
case mode_udp : return recv_raw_udp ( raw_info , payload , payloadlen ) ;
case mode_icmp : return recv_raw_icmp ( raw_info , payload , payloadlen ) ;
default : return - 1 ;
}
}
2017-08-03 23:32:19 +08:00
int after_send_raw0 ( raw_info_t & raw_info )
{
packet_info_t & send_info = raw_info . send_info ;
packet_info_t & recv_info = raw_info . recv_info ;
if ( raw_mode = = mode_faketcp )
{
2017-09-06 01:31:29 -05:00
if ( send_info . syn = = 0 & & send_info . ack = = 1 & & raw_info . send_info . data_len ! = 0 ) //only modify send_info when the packet is not part of handshake
2017-08-03 23:32:19 +08:00
{
if ( seq_mode = = 0 )
{
} else if ( seq_mode = = 1 )
{
2017-09-06 01:31:29 -05:00
send_info . seq + = raw_info . send_info . data_len ; //////////////////modify
2017-08-03 23:32:19 +08:00
} else if ( seq_mode = = 2 )
{
2018-08-29 20:00:37 -05:00
if ( get_true_random_number ( ) % 5 = = 3 )
2017-09-06 01:31:29 -05:00
send_info . seq + = raw_info . send_info . data_len ; //////////////////modify
2017-08-03 23:32:19 +08:00
}
2017-09-06 05:31:29 -05:00
else if ( seq_mode = = 3 | | seq_mode = = 4 )
{
send_info . seq + = raw_info . send_info . data_len ;
u32_t window_size ;
if ( seq_mode = = 3 )
{
window_size = ( u32_t ) ( ( u32_t ) receive_window_lower_bound < < ( u32_t ) wscale ) ;
}
else //seq_mode==4
{
window_size = ( u32_t ) ( ( u32_t ) receive_window_lower_bound ) ;
}
if ( larger_than_u32 ( send_info . seq + max_data_len , recv_info . ack_seq + window_size ) )
{
send_info . seq = raw_info . recv_info . ack_seq ;
}
2017-09-06 19:24:39 -05:00
if ( recv_info . ack_seq_counter > = 3 ) //simulate tcp fast re-transmit
2017-09-06 05:31:29 -05:00
{
recv_info . ack_seq_counter = 0 ;
send_info . seq = raw_info . recv_info . ack_seq ;
}
2017-09-06 19:24:39 -05:00
if ( larger_than_u32 ( raw_info . recv_info . ack_seq , send_info . seq ) ) //for further use,currently no effect.
2017-09-06 05:31:29 -05:00
{
send_info . seq = raw_info . recv_info . ack_seq ;
}
}
2017-08-03 23:32:19 +08:00
}
}
if ( raw_mode = = mode_icmp )
{
if ( program_mode = = client_mode )
{
2018-07-26 12:41:55 -05:00
send_info . my_icmp_seq + + ;
2017-08-03 23:32:19 +08:00
}
}
return 0 ;
}
int after_recv_raw0 ( raw_info_t & raw_info )
{
packet_info_t & send_info = raw_info . send_info ;
packet_info_t & recv_info = raw_info . recv_info ;
2017-07-29 20:32:26 +08:00
2017-08-03 23:32:19 +08:00
if ( raw_mode = = mode_faketcp )
{
if ( recv_info . has_ts )
send_info . ts_ack = recv_info . ts ;
2017-09-06 01:31:29 -05:00
if ( recv_info . syn = = 0 & & recv_info . ack = = 1 & & raw_info . recv_info . data_len ! = 0 ) //only modify send_info when the packet is not part of handshake
2017-08-03 23:32:19 +08:00
{
2017-09-06 01:31:29 -05:00
if ( seq_mode = = 0 | | seq_mode = = 1 | | seq_mode = = 2 )
{
if ( larger_than_u32 ( recv_info . seq + raw_info . recv_info . data_len , send_info . ack_seq ) )
send_info . ack_seq = recv_info . seq + raw_info . recv_info . data_len ; //TODO only update if its larger
}
2017-09-06 05:31:29 -05:00
else if ( seq_mode = = 3 | | seq_mode = = 4 )
2017-09-06 01:31:29 -05:00
{
2017-09-06 05:31:29 -05:00
if ( recv_info . seq = = send_info . ack_seq )
2017-09-06 01:37:14 -05:00
{
2017-09-06 05:31:29 -05:00
send_info . ack_seq = recv_info . seq + raw_info . recv_info . data_len ; //currently we dont remembr tcp segments,this is the simplest way
//TODO implement tcp segment remembering and SACK.
2017-09-06 01:37:14 -05:00
}
2017-09-06 01:31:29 -05:00
}
2017-08-03 23:32:19 +08:00
}
}
if ( raw_mode = = mode_icmp )
{
if ( program_mode = = server_mode )
{
2018-07-26 12:41:55 -05:00
if ( larger_than_u16 ( recv_info . my_icmp_seq , send_info . my_icmp_seq ) )
send_info . my_icmp_seq = recv_info . my_icmp_seq ; //TODO only update if its larger
2017-08-03 23:32:19 +08:00
}
}
return 0 ;
}
2017-07-29 20:32:26 +08:00
2017-08-04 03:29:53 +08:00
/*
2017-08-03 23:32:19 +08:00
int send_raw ( raw_info_t & raw_info , const char * payload , int payloadlen )
{
packet_info_t & send_info = raw_info . send_info ;
packet_info_t & recv_info = raw_info . recv_info ;
2017-08-04 03:29:53 +08:00
int ret = send_raw0 ( raw_info , payload , payloadlen ) ;
2017-08-03 23:32:19 +08:00
if ( ret < 0 ) return ret ;
else
{
after_send_raw0 ( raw_info ) ;
return ret ;
}
}
2017-07-30 23:18:37 +08:00
2017-08-03 23:32:19 +08:00
int recv_raw ( raw_info_t & raw_info , char * & payload , int & payloadlen )
{
packet_info_t & send_info = raw_info . send_info ;
packet_info_t & recv_info = raw_info . recv_info ;
2017-08-04 03:29:53 +08:00
int ret = recv_raw0 ( raw_info , payload , payloadlen ) ;
2017-08-03 23:32:19 +08:00
if ( ret < 0 ) return ret ;
else
{
after_recv_raw0 ( raw_info ) ;
return ret ;
}
2017-08-04 03:29:53 +08:00
} */
2017-07-30 23:18:37 +08:00
2018-07-23 13:52:36 -05:00
/*
2017-09-23 03:05:23 -05:00
int get_src_adress ( u32_t & ip , u32_t remote_ip_uint32 , int remote_port ) //a trick to get src adress for a dest adress,so that we can use the src address in raw socket as source ip
{
struct sockaddr_in remote_addr_in = { 0 } ;
socklen_t slen = sizeof ( sockaddr_in ) ;
//memset(&remote_addr_in, 0, sizeof(remote_addr_in));
remote_addr_in . sin_family = AF_INET ;
remote_addr_in . sin_port = htons ( remote_port ) ;
remote_addr_in . sin_addr . s_addr = remote_ip_uint32 ;
int new_udp_fd = socket ( AF_INET , SOCK_DGRAM , IPPROTO_UDP ) ;
if ( new_udp_fd < 0 )
{
mylog ( log_warn , " create udp_fd error \n " ) ;
return - 1 ;
}
//set_buf_size(new_udp_fd);
mylog ( log_debug , " created new udp_fd %d \n " , new_udp_fd ) ;
int ret = connect ( new_udp_fd , ( struct sockaddr * ) & remote_addr_in , slen ) ;
if ( ret ! = 0 )
{
mylog ( log_warn , " udp fd connect fail \n " ) ;
close ( new_udp_fd ) ;
return - 1 ;
}
struct sockaddr_in my_addr = { 0 } ;
socklen_t len = sizeof ( my_addr ) ;
2018-07-17 11:33:02 -05:00
if ( getsockname ( new_udp_fd , ( struct sockaddr * ) & my_addr , & len ) ! = 0 ) { close ( new_udp_fd ) ; return - 1 ; }
2017-09-23 03:05:23 -05:00
ip = my_addr . sin_addr . s_addr ;
close ( new_udp_fd ) ;
return 0 ;
2018-07-23 13:52:36 -05:00
} */
2017-09-23 03:05:23 -05:00
2018-07-17 11:33:02 -05:00
int get_src_adress2 ( address_t & output_addr , address_t remote_addr )
{
int new_udp_fd = remote_addr . new_connected_udp_fd ( ) ;
if ( new_udp_fd < 0 )
{
mylog ( log_warn , " create udp_fd error \n " ) ;
return - 1 ;
}
socklen_t len = sizeof ( output_addr . inner ) ;
if ( getsockname ( new_udp_fd , ( struct sockaddr * ) & output_addr . inner , & len ) ! = 0 ) { close ( new_udp_fd ) ; return - 1 ; }
assert ( output_addr . get_type ( ) = = remote_addr . get_type ( ) ) ;
close ( new_udp_fd ) ;
return 0 ;
}
2018-07-23 13:52:36 -05:00
/*
2017-11-01 08:25:50 -05:00
int try_to_list_and_bind ( int & fd , u32_t local_ip_uint32 , int port ) //try to bind to a port,may fail.
2017-09-23 03:05:23 -05:00
{
2017-11-01 08:25:50 -05:00
int old_bind_fd = fd ;
2017-09-23 03:05:23 -05:00
if ( raw_mode = = mode_faketcp )
{
2017-11-01 08:25:50 -05:00
fd = socket ( AF_INET , SOCK_STREAM , 0 ) ;
2017-09-23 03:05:23 -05:00
}
else if ( raw_mode = = mode_udp | | raw_mode = = mode_icmp )
{
2017-11-01 08:25:50 -05:00
fd = socket ( AF_INET , SOCK_DGRAM , 0 ) ;
2017-09-23 03:05:23 -05:00
}
if ( old_bind_fd ! = - 1 )
{
close ( old_bind_fd ) ;
}
struct sockaddr_in temp_bind_addr = { 0 } ;
//bzero(&temp_bind_addr, sizeof(temp_bind_addr));
temp_bind_addr . sin_family = AF_INET ;
temp_bind_addr . sin_port = htons ( port ) ;
temp_bind_addr . sin_addr . s_addr = local_ip_uint32 ;
2017-11-01 08:25:50 -05:00
if ( bind ( fd , ( struct sockaddr * ) & temp_bind_addr , sizeof ( temp_bind_addr ) ) ! = 0 )
2017-09-23 03:05:23 -05:00
{
mylog ( log_debug , " bind fail \n " ) ;
return - 1 ;
}
if ( raw_mode = = mode_faketcp )
{
2017-11-01 08:25:50 -05:00
if ( listen ( fd , SOMAXCONN ) ! = 0 ) {
2017-09-23 03:05:23 -05:00
mylog ( log_warn , " listen fail \n " ) ;
return - 1 ;
}
}
return 0 ;
2018-07-23 13:52:36 -05:00
} */
2018-07-17 11:33:02 -05:00
int try_to_list_and_bind2 ( int & fd , address_t address ) //try to bind to a port,may fail.
{
2018-07-25 01:52:18 -05:00
if ( fd ! = - 1 )
{
close ( fd ) ;
}
if ( raw_mode = = mode_faketcp )
{
fd = socket ( address . get_type ( ) , SOCK_STREAM , 0 ) ;
}
else if ( raw_mode = = mode_udp | | raw_mode = = mode_icmp )
{
fd = socket ( address . get_type ( ) , SOCK_DGRAM , 0 ) ;
}
2018-07-17 11:33:02 -05:00
2018-07-25 01:52:18 -05:00
if ( fd = = - 1 )
{
mylog ( log_debug , " create fd fail \n " ) ;
return - 1 ;
}
/*struct sockaddr_in temp_bind_addr={0};
//bzero(&temp_bind_addr, sizeof(temp_bind_addr));
2018-07-17 11:33:02 -05:00
2018-07-25 01:52:18 -05:00
temp_bind_addr . sin_family = AF_INET ;
temp_bind_addr . sin_port = htons ( port ) ;
temp_bind_addr . sin_addr . s_addr = local_ip_uint32 ; */
2018-07-17 11:33:02 -05:00
2018-07-25 01:52:18 -05:00
if ( bind ( fd , ( struct sockaddr * ) & address . inner , address . get_len ( ) ) ! = 0 )
{
mylog ( log_debug , " bind fail \n " ) ;
return - 1 ;
}
2018-08-29 12:19:09 -05:00
if ( raw_mode = = mode_faketcp & & ! use_tcp_dummy_socket )
2018-07-25 01:52:18 -05:00
{
2018-07-17 11:33:02 -05:00
if ( listen ( fd , SOMAXCONN ) ! = 0 ) {
mylog ( log_warn , " listen fail \n " ) ;
return - 1 ;
}
2018-07-25 01:52:18 -05:00
}
return 0 ;
2018-07-17 11:33:02 -05:00
}
2018-07-23 13:52:36 -05:00
/*
2017-11-01 08:25:50 -05:00
int client_bind_to_a_new_port ( int & fd , u32_t local_ip_uint32 ) //find a free port and bind to it.
2017-09-23 03:05:23 -05:00
{
int raw_send_port = 10000 + get_true_random_number ( ) % ( 65535 - 10000 ) ;
for ( int i = 0 ; i < 1000 ; i + + ) //try 1000 times at max,this should be enough
{
2017-11-01 08:25:50 -05:00
if ( try_to_list_and_bind ( fd , local_ip_uint32 , raw_send_port ) = = 0 )
2017-09-23 03:05:23 -05:00
{
return raw_send_port ;
}
}
mylog ( log_fatal , " bind port fail \n " ) ;
myexit ( - 1 ) ;
return - 1 ; ////for compiler check
2018-07-23 13:52:36 -05:00
} */
2018-07-17 11:33:02 -05:00
int client_bind_to_a_new_port2 ( int & fd , const address_t & address ) //find a free port and bind to it.
{
address_t tmp = address ;
for ( int i = 0 ; i < 1000 ; i + + ) //try 1000 times at max,this should be enough
{
2018-11-13 02:35:42 -06:00
int raw_send_port = 10000 + get_true_random_number ( ) % ( 65535 - 10000 ) ;
2018-07-17 11:33:02 -05:00
tmp . set_port ( raw_send_port ) ;
2018-08-29 12:19:09 -05:00
if ( try_to_list_and_bind2 ( fd , tmp ) = = 0 )
2018-07-17 11:33:02 -05:00
{
return raw_send_port ;
}
}
mylog ( log_fatal , " bind port fail \n " ) ;
myexit ( - 1 ) ;
return - 1 ; ////for compiler check
}