@ -5,6 +5,7 @@
# include "params.h"
# include "helpers.h"
# include "hostlist.h"
# include "ipset.h"
# include "conntrack.h"
# include <string.h>
@ -145,20 +146,32 @@ enum dpi_desync_mode desync_mode_from_string(const char *s)
return DESYNC_INVALID ;
}
static bool dp_match_l3l4 ( struct desync_profile * dp , bool ipv6 , uint16_t tcp_port , uint16_t udp_por t)
static bool dp_match_l3l4 ( struct desync_profile * dp , uint8_t l3proto , const struct sockaddr * des t)
{
return \
( ( ! ipv6 & & dp - > filter_ipv4 ) | | ( ipv6 & & dp - > filter_ipv6 ) ) & &
( ! tcp_port | | pf_in_range ( tcp_port , & dp - > pf_tcp ) ) & &
( ! udp_port | | pf_in_range ( udp_port , & dp - > pf_udp ) ) ;
return ( ( dest - > sa_family = = AF_INET & & dp - > filter_ipv4 ) | | ( dest - > sa_family = = AF_INET6 & & dp - > filter_ipv6 ) ) & &
( l3proto = = IPPROTO_TCP & & pf_in_range ( saport ( dest ) , & dp - > pf_tcp ) | | l3proto = = IPPROTO_UDP & & pf_in_range ( saport ( dest ) , & dp - > pf_tcp ) ) & &
IpsetCheck ( dp , dest - > sa_family = = AF_INET ? & ( ( struct sockaddr_in * ) dest ) - > sin_addr : NULL , dest - > sa_family = = AF_INET6 ? & ( ( struct sockaddr_in6 * ) dest ) - > sin6_addr : NULL ) ;
}
static bool dp_impossible ( struct desync_profile * dp , const char * hostname , t_l7proto l7proto )
{
return ! PROFILE_IPSETS_EMPTY ( dp ) & &
( ( dp - > filter_l7 & & ! l7_proto_match ( l7proto , dp - > filter_l7 ) ) | | ( ! * dp - > hostlist_auto_filename & & ! hostname & & ( dp - > hostlist | | dp - > hostlist_exclude ) ) ) ;
}
static bool dp_match (
struct desync_profile * dp , bool ipv6 , uint16_t tcp_port , uint16_t udp_port , const char * hostname ,
struct desync_profile * dp ,
uint8_t l3proto , const struct sockaddr * dest , const char * hostname , t_l7proto l7proto ,
bool * bCheckDone , bool * bCheckResult , bool * bExcluded )
{
if ( bCheckDone ) * bCheckDone = false ;
if ( dp_match_l3l4 ( dp , ipv6 , tcp_port , udp_port ) )
// impossible case, hard filter
// impossible check avoids relatively slow ipset search
if ( ! dp_impossible ( dp , hostname , l7proto ) & & dp_match_l3l4 ( dp , l3proto , dest ) )
{
// soft filter
if ( dp - > filter_l7 & & ! l7_proto_match ( l7proto , dp - > filter_l7 ) )
return false ;
// autohostlist profile matching l3/l4 filter always win
if ( * dp - > hostlist_auto_filename ) return true ;
@ -181,15 +194,21 @@ static bool dp_match(
return false ;
}
static struct desync_profile * dp_find (
struct desync_profile_list_head * head , bool ipv6 , uint16_t tcp_port , uint16_t udp_port , const char * hostname ,
struct desync_profile_list_head * head ,
uint8_t l3proto , const struct sockaddr * dest , const char * hostname , t_l7proto l7proto ,
bool * bCheckDone , bool * bCheckResult , bool * bExcluded )
{
struct desync_profile_list * dpl ;
DLOG ( " desync profile search for hostname='%s' ipv6=%u tcp_port=%u udp_port=%u \n " , hostname ? hostname : " " , ipv6 , tcp_port , udp_port ) ;
if ( params . debug )
{
char ip_port [ 48 ] ;
ntop46_port ( dest , ip_port , sizeof ( ip_port ) ) ;
DLOG ( " desync profile search for %s target=%s l7proto=%s hostname='%s' \n " , proto_name ( l3proto ) , ip_port , l7proto_str ( l7proto ) , hostname ? hostname : " " ) ;
}
if ( bCheckDone ) * bCheckDone = false ;
LIST_FOREACH ( dpl , head , next )
{
if ( dp_match ( & dpl - > dp , ipv6 , tcp_port , udp_port , hostname , bCheckDone , bCheckResult , bExcluded ) )
if ( dp_match ( & dpl - > dp , l3proto , dest , hostname , l7proto , bCheckDone , bCheckResult , bExcluded ) )
{
DLOG ( " desync profile %d matches \n " , dpl - > dp . n ) ;
return & dpl - > dp ;
@ -326,7 +345,7 @@ static void auto_hostlist_failed(struct desync_profile *dp, const char *hostname
fail_counter = HostFailPoolAdd ( & dp - > hostlist_auto_fail_counters , hostname , dp - > hostlist_auto_fail_time ) ;
if ( ! fail_counter )
{
fprintf ( stderr , " HostFailPoolAdd: out of memory \n " ) ;
DLOG_ERR ( " HostFailPoolAdd: out of memory \n " ) ;
return ;
}
}
@ -346,7 +365,7 @@ static void auto_hostlist_failed(struct desync_profile *dp, const char *hostname
HOSTLIST_DEBUGLOG_APPEND ( " %s : profile %d : adding to %s " , hostname , dp - > n , dp - > hostlist_auto_filename ) ;
if ( ! StrPoolAddStr ( & dp - > hostlist , hostname ) )
{
fprintf ( stderr , " StrPoolAddStr out of memory \n " ) ;
DLOG_ERR ( " StrPoolAddStr out of memory \n " ) ;
return ;
}
if ( ! append_to_list_file ( dp - > hostlist_auto_filename , hostname ) )
@ -575,6 +594,21 @@ static size_t pos_normalize(size_t split_pos, size_t reasm_offset, size_t len_pa
return split_pos ;
}
static void autottl_discover ( t_ctrack * ctrack , bool bIpv6 )
{
if ( ctrack & & ctrack - > incoming_ttl )
{
autottl * attl = bIpv6 ? & ctrack - > dp - > desync_autottl6 : & ctrack - > dp - > desync_autottl ;
if ( AUTOTTL_ENABLED ( * attl ) )
{
ctrack - > autottl = autottl_guess ( ctrack - > incoming_ttl , attl ) ;
if ( ctrack - > autottl )
DLOG ( " autottl: guessed %u \n " , ctrack - > autottl ) ;
else
DLOG ( " autottl: could not guess \n " ) ;
}
}
}
static uint8_t dpi_desync_tcp_packet_play ( bool replay , size_t reasm_offset , uint32_t fwmark , const char * ifout , uint8_t * data_pkt , size_t * len_pkt , struct ip * ip , struct ip6_hdr * ip6hdr , struct tcphdr * tcphdr , size_t transport_len , uint8_t * data_payload , size_t len_payload )
{
@ -593,10 +627,10 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
size_t pkt1_len , pkt2_len ;
uint8_t ttl_orig , ttl_fake , flags_orig , scale_factor ;
uint32_t * timestamps ;
t_l7proto l7proto = UNKNOWN ;
ttl_orig = ip ? ip - > ip_ttl : ip6hdr - > ip6_ctlun . ip6_un1 . ip6_un1_hlim ;
uint32_t desync_fwmark = fwmark | params . desync_fwmark ;
extract_endpoints ( ip , ip6hdr , tcphdr , NULL , & src , & dst ) ;
if ( replay )
{
@ -611,10 +645,9 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
DLOG ( " using cached desync profile %d \n " , dp - > n ) ;
else if ( ! ctrack_replay - > dp_search_complete )
{
dp = ctrack_replay - > dp = dp_find ( & params . desync_profiles , ! ! ip6hdr , ntohs ( bReverse ? tcphdr - > th_sport : tcphdr - > th_dport ) , 0 , ctrack_replay - > hostname , NULL , NULL , NULL ) ;
dp = ctrack_replay - > dp = dp_find ( & params . desync_profiles , IPPROTO_TCP , ( struct sockaddr * ) & dst , ctrack_replay - > hostname , ctrack_replay - > l7proto , NULL , NULL , NULL ) ;
ctrack_replay - > dp_search_complete = true ;
}
if ( ! dp )
{
DLOG ( " matching desync profile not found \n " ) ;
@ -635,7 +668,7 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
DLOG ( " using cached desync profile %d \n " , dp - > n ) ;
else if ( ! ctrack | | ! ctrack - > dp_search_complete )
{
dp = dp_find ( & params . desync_profiles , ! ! ip6hdr , ntohs ( bReverse ? tcphdr - > th_sport : tcphdr - > th_dport ) , 0 , ctrack ? ctrack - > hostname : NULL , NULL , NULL , NULL ) ;
dp = dp_find ( & params . desync_profiles , IPPROTO_TCP , ( struct sockaddr * ) & dst , ctrack ? ctrack - > hostname : NULL , ctrack ? ctrack - > l7proto : UNKNOWN , NULL , NULL , NULL ) ;
if ( ctrack )
{
ctrack - > dp = dp ;
@ -661,17 +694,14 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
if ( bReverse )
{
if ( ctrack & & ! ctrack - > autottl & & ctrack - > pcounter_reply = = 1 )
if ( ctrack )
{
autottl * attl = ip ? & dp - > desync_autottl : & dp - > desync_autottl6 ;
if ( AUTOTTL_ENABLED ( * attl ) )
if ( ! ctrack - > incoming_ttl )
{
ctrack - > autottl = autottl_guess ( ttl_orig , attl ) ;
if ( ctrack - > autottl )
DLOG ( " autottl: guessed %u \n " , ctrack - > autottl ) ;
else
DLOG ( " autottl: could not guess \n " ) ;
DLOG ( " incoming TTL %u \n " , ttl_orig ) ;
ctrack - > incoming_ttl = ttl_orig ;
}
if ( ! ctrack - > autottl ) autottl_discover ( ctrack , ! ! ip6hdr ) ;
}
// process reply packets for auto hostlist mode
@ -744,10 +774,12 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
flags_orig = * ( ( uint8_t * ) tcphdr + 13 ) ;
scale_factor = tcp_find_scale_factor ( tcphdr ) ;
timestamps = tcp_find_timestamps ( tcphdr ) ;
extract_endpoints ( ip , ip6hdr , tcphdr , NULL , & src , & dst ) ;
if ( ! replay )
{
// start and cutoff limiters
if ( ! process_desync_interval ( dp , ctrack ) ) return verdict ;
if ( tcp_syn_segment ( tcphdr ) )
{
switch ( dp - > desync_mode0 )
@ -797,8 +829,6 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
return verdict ;
}
// start and cutoff limiters
if ( ! process_desync_interval ( dp , ctrack ) ) return verdict ;
} // !replay
if ( ! ( tcphdr - > th_flags & TH_SYN ) & & len_payload )
@ -811,6 +841,7 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
const uint8_t * rdata_payload = data_payload ;
size_t rlen_payload = len_payload ;
size_t split_pos ;
t_l7proto l7proto = UNKNOWN ;
if ( replay )
{
@ -829,7 +860,7 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
{
DLOG ( " packet contains HTTP request \n " ) ;
l7proto = HTTP ;
if ( ctrack & & ! ctrack - > l7proto ) ctrack - > l7proto = l7proto ;
if ( ctrack & & ctrack - > l7proto = = UNKNOWN ) ctrack - > l7proto = l7proto ;
// we do not reassemble http
reasm_orig_cancel ( ctrack ) ;
@ -874,7 +905,6 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
}
if ( ! ctrack - > req_seq_finalized )
{
@ -901,7 +931,7 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
}
else
{
fprintf ( stderr , " rawpacket_queue failed !' \n " ) ;
DLOG_ERR ( " rawpacket_queue failed ! \n " ) ;
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
@ -929,67 +959,94 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
if ( dseq > = 0x1000000 & & ! ( dseq & 0x80000000 ) ) ctrack - > req_seq_abandoned = true ;
}
if ( bHaveHost )
if ( bHaveHost ) DLOG ( " hostname: %s \n " , host ) ;
bool bDiscoveredL7 ;
if ( ctrack_replay )
{
bDiscoveredL7 = ! ctrack_replay - > l7proto_discovered & & ctrack_replay - > l7proto ! = UNKNOWN ;
ctrack_replay - > l7proto_discovered = true ;
}
else
bDiscoveredL7 = ! ctrack_replay & & l7proto ! = UNKNOWN ;
if ( bDiscoveredL7 ) DLOG ( " discovered l7 protocol \n " ) ;
bool bDiscoveredHostname = bHaveHost & & ! ( ctrack_replay & & ctrack_replay - > hostname ) ;
if ( bDiscoveredHostname )
{
bool bCheckDone = false , bCheckResult = false , bCheckExcluded = false ;
DLOG ( " hostname: %s \n " , host ) ;
DLOG ( " discovered hostname \n " ) ;
if ( ctrack_replay )
{
ctrack_replay - > hostname = strdup ( host ) ;
if ( ! ctrack_replay - > hostname )
{
ctrack_replay - > hostname = strdup ( host ) ;
if ( ! ctrack_replay - > hostname )
{
DLOG_ERR ( " hostname dup : out of memory " ) ;
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
DLOG ( " we have hostname now. searching desync profile again. \n " ) ;
struct desync_profile * dp_prev = dp ;
dp = ctrack_replay - > dp = dp_find ( & params . desync_profiles , ! ! ip6hdr , ntohs ( bReverse ? tcphdr - > th_sport : tcphdr - > th_dport ) , 0 , ctrack_replay - > hostname , & ctrack_replay - > bCheckDone , & ctrack_replay - > bCheckResult , & ctrack_replay - > bCheckExcluded ) ;
ctrack_replay - > dp_search_complete = true ;
if ( ! dp )
DLOG_ERR ( " hostname dup : out of memory " ) ;
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
}
}
bool bCheckDone = false , bCheckResult = false , bCheckExcluded = false ;
if ( bDiscoveredL7 | | bDiscoveredHostname )
{
struct desync_profile * dp_prev = dp ;
dp = dp_find ( & params . desync_profiles , IPPROTO_TCP , ( struct sockaddr * ) & dst , ctrack_replay ? ctrack_replay - > hostname : host , ctrack_replay ? ctrack_replay - > l7proto : l7proto , & bCheckDone , & bCheckResult , & bCheckExcluded ) ;
if ( ctrack_replay )
{
ctrack_replay - > dp = dp ;
ctrack_replay - > dp_search_complete = true ;
ctrack_replay - > bCheckDone = bCheckDone ;
ctrack_replay - > bCheckResult = bCheckResult ;
ctrack_replay - > bCheckExcluded = bCheckExcluded ;
}
if ( ! dp )
{
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
if ( dp ! = dp_prev )
{
DLOG ( " desync profile changed by revealed l7 protocol or hostname ! \n " ) ;
// rediscover autottl
autottl_discover ( ctrack_replay , ! ! ip6hdr ) ;
// re-evaluate start/cutoff limiters
if ( ! replay )
{
maybe_cutoff ( ctrack , IPPROTO_TCP ) ;
if ( ! process_desync_interval ( dp , ctrack ) )
{
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
if ( dp ! = dp_prev )
{
DLOG ( " desync profile changed by revealed hostname ! \n " ) ;
// re-evaluate start/cutoff limiters
if ( ! replay )
{
maybe_cutoff ( ctrack , IPPROTO_TCP ) ;
if ( ! process_desync_interval ( dp , ctrack ) )
{
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
}
}
}
bCheckDone = ctrack_replay - > bCheckDone ;
bCheckResult = ctrack_replay - > bCheckResult ;
bCheckExcluded = ctrack_replay - > bCheckExcluded ;
}
if ( dp - > hostlist | | dp - > hostlist_exclude )
}
else if ( ctrack_replay )
{
bCheckDone = ctrack_replay - > bCheckDone ;
bCheckResult = ctrack_replay - > bCheckResult ;
bCheckExcluded = ctrack_replay - > bCheckExcluded ;
}
if ( bHaveHost & & ( dp - > hostlist | | dp - > hostlist_exclude ) )
{
if ( ! bCheckDone )
bCheckResult = HostlistCheck ( dp , host , & bCheckExcluded ) ;
if ( bCheckResult )
ctrack_stop_retrans_counter ( ctrack_replay ) ;
else
{
if ( ! bCheckDone )
bCheckResult = HostlistCheck ( dp , host , & bCheckExcluded ) ;
if ( bCheckResult )
ctrack_stop_retrans_counter ( ctrack_replay ) ;
else
if ( ctrack_replay )
{
if ( ctrack_replay )
{
ctrack_replay - > hostname_ah_check = * dp - > hostlist_auto_filename & & ! bCheckExcluded ;
if ( ! ctrack_replay - > hostname_ah_check )
ctrack_stop_retrans_counter ( ctrack_replay ) ;
}
DLOG ( " not applying tampering to this request \n " ) ;
reasm_orig_cancel ( ctrack ) ;
return verdict ;
ctrack_replay - > hostname_ah_check = * dp - > hostlist_auto_filename & & ! bCheckExcluded ;
if ( ! ctrack_replay - > hostname_ah_check )
ctrack_stop_retrans_counter ( ctrack_replay ) ;
}
DLOG ( " not applying tampering to this request \n " ) ;
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
}
@ -1019,24 +1076,28 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
if ( l7proto = = UNKNOWN )
{
if ( ! dp - > desync_any_proto ) return verdict ;
if ( ! dp - > desync_any_proto )
{
DLOG ( " not applying tampering to unknown protocol \n " ) ;
return verdict ;
}
DLOG ( " applying tampering to unknown protocol \n " ) ;
}
ttl_fake = ( ctrack_replay & & ctrack_replay - > autottl ) ? ctrack_replay - > autottl : ( ip6hdr ? ( dp - > desync_ttl6 ? dp - > desync_ttl6 : ttl_orig ) : ( dp - > desync_ttl ? dp - > desync_ttl : ttl_orig ) ) ;
if ( ( l7proto = = HTTP ) & & ( dp - > hostcase | | dp - > hostnospace | | dp - > domcase ) & & ( phost = ( uint8_t * ) memmem ( data_payload , len_payload , " \r \n Host: " , 8 ) ) )
if ( ( l7proto = = HTTP ) & & ( dp - > hostcase | | dp - > hostnospace | | dp - > domcase ) & & HttpFindHost ( & phost , data_payload , len_payload ) )
{
if ( dp - > hostcase )
{
DLOG ( " modifying Host: => %c%c%c%c: \n " , dp - > hostspell [ 0 ] , dp - > hostspell [ 1 ] , dp - > hostspell [ 2 ] , dp - > hostspell [ 3 ] ) ;
memcpy ( phost + 2 , dp - > hostspell , 4 ) ;
memcpy ( phost , dp - > hostspell , 4 ) ;
verdict = VERDICT_MODIFY ;
}
if ( dp - > domcase )
{
DLOG ( " mixing domain case \n " ) ;
for ( p = phost + 7 ; p < ( data_payload + len_payload ) & & * p ! = ' \r ' & & * p ! = ' \n ' ; p + + )
for ( p = phost + 5 ; p < ( data_payload + len_payload ) & & * p ! = ' \r ' & & * p ! = ' \n ' ; p + + )
* p = ( ( ( size_t ) p ) & 1 ) ? tolower ( * p ) : toupper ( * p ) ;
verdict = VERDICT_MODIFY ;
}
@ -1048,12 +1109,12 @@ static uint8_t dpi_desync_tcp_packet_play(bool replay, size_t reasm_offset, uint
DLOG ( " removing space after Host: and adding it to User-Agent: \n " ) ;
if ( pua > phost )
{
memmove ( phost + 7 , phost + 8 , pua - phost - 8 ) ;
phost [ pua - phost - 1 ] = ' ' ;
memmove ( phost + 5 , phost + 6 , pua - phost - 6 ) ;
pua [ - 1 ] = ' ' ;
}
else
{
memmove ( pua + 1 , pua , phost - pua + 7 ) ;
memmove ( pua + 1 , pua , phost - pua + 5 ) ;
* pua = ' ' ;
}
verdict = VERDICT_MODIFY ;
@ -1404,6 +1465,9 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
uint8_t ttl_orig , ttl_fake ;
t_l7proto l7proto = UNKNOWN ;
ttl_orig = ip ? ip - > ip_ttl : ip6hdr - > ip6_ctlun . ip6_un1 . ip6_un1_hlim ;
extract_endpoints ( ip , ip6hdr , NULL , udphdr , & src , & dst ) ;
if ( replay )
{
// in replay mode conntrack_replay is not NULL and ctrack is NULL
@ -1417,7 +1481,7 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
DLOG ( " using cached desync profile %d \n " , dp - > n ) ;
else if ( ! ctrack_replay - > dp_search_complete )
{
dp = ctrack_replay - > dp = dp_find ( & params . desync_profiles , ! ! ip6hdr , 0 , ntohs ( bReverse ? udphdr - > uh_sport : udphdr - > uh_dport ) , ctrack_replay - > hostname , NULL , NULL , NULL ) ;
dp = ctrack_replay - > dp = dp_find ( & params . desync_profiles , IPPROTO_UDP , ( struct sockaddr * ) & dst , ctrack_replay - > hostname , ctrack_replay - > l7proto , NULL , NULL , NULL ) ;
ctrack_replay - > dp_search_complete = true ;
}
if ( ! dp )
@ -1440,7 +1504,7 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
DLOG ( " using cached desync profile %d \n " , dp - > n ) ;
else if ( ! ctrack | | ! ctrack - > dp_search_complete )
{
dp = dp_find ( & params . desync_profiles , ! ! ip6hdr , 0 , ntohs ( bReverse ? udphdr - > uh_sport : udphdr - > uh_dport ) , ctrack ? ctrack - > hostname : NULL , NULL , NULL , NULL ) ;
dp = dp_find ( & params . desync_profiles , IPPROTO_UDP , ( struct sockaddr * ) & dst , ctrack ? ctrack - > hostname : NULL , ctrack ? ctrack - > l7proto : UNKNOWN , NULL , NULL , NULL ) ;
if ( ctrack )
{
ctrack - > dp = dp ;
@ -1458,16 +1522,22 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
//ConntrackPoolDump(¶ms.conntrack);
}
if ( bReverse ) return verdict ; // nothing to do. do not waste cpu
if ( bReverse & & ctrack )
{
if ( ! ctrack - > incoming_ttl )
{
DLOG ( " incoming TTL %u \n " , ttl_orig ) ;
ctrack - > incoming_ttl = ttl_orig ;
}
if ( ! ctrack - > autottl ) autottl_discover ( ctrack , ! ! ip6hdr ) ;
return verdict ; // nothing to do. do not waste cpu
}
// start and cutoff limiters
if ( ! replay & & ! process_desync_interval ( dp , ctrack ) ) return verdict ;
uint32_t desync_fwmark = fwmark | params . desync_fwmark ;
ttl_orig = ip ? ip - > ip_ttl : ip6hdr - > ip6_ctlun . ip6_un1 . ip6_un1_hlim ;
ttl_fake = ip6hdr ? dp - > desync_ttl6 ? dp - > desync_ttl6 : ttl_orig : dp - > desync_ttl ? dp - > desync_ttl : ttl_orig ;
extract_endpoints ( ip , ip6hdr , NULL , udphdr , & src , & dst ) ;
if ( len_payload )
{
const uint8_t * fake ;
@ -1480,7 +1550,7 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
{
DLOG ( " packet contains QUIC initial \n " ) ;
l7proto = QUIC ;
if ( ctrack & & ! ctrack - > l7proto ) ctrack - > l7proto = l7proto ;
if ( ctrack & & ctrack - > l7proto = = UNKNOWN ) ctrack - > l7proto = l7proto ;
uint8_t clean [ 16384 ] , * pclean ;
size_t clean_len ;
@ -1542,7 +1612,7 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
}
else
{
fprintf ( stderr , " rawpacket_queue failed !' \n " ) ;
DLOG_ERR ( " rawpacket_queue failed ! \n " ) ;
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
@ -1593,77 +1663,113 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
{
DLOG ( " packet contains wireguard handshake initiation \n " ) ;
l7proto = WIREGUARD ;
if ( ctrack & & ! ctrack - > l7proto ) ctrack - > l7proto = l7proto ;
if ( ctrack & & ctrack - > l7proto = = UNKNOWN ) ctrack - > l7proto = l7proto ;
}
else if ( IsDhtD1 ( data_payload , len_payload ) )
{
DLOG ( " packet contains DHT d1...e \n " ) ;
l7proto = DHT ;
if ( ctrack & & ! ctrack - > l7proto ) ctrack - > l7proto = l7proto ;
if ( ctrack & & ctrack - > l7proto = = UNKNOWN ) ctrack - > l7proto = l7proto ;
}
else
{
if ( ! dp - > desync_any_proto ) return verdict ;
if ( ! dp - > desync_any_proto )
{
DLOG ( " not applying tampering to unknown protocol \n " ) ;
return verdict ;
}
DLOG ( " applying tampering to unknown protocol \n " ) ;
}
}
if ( bHaveHost )
if ( bHaveHost ) DLOG ( " hostname: %s \n " , host ) ;
bool bDiscoveredL7 ;
if ( ctrack_replay )
{
bDiscoveredL7 = ! ctrack_replay - > l7proto_discovered & & ctrack_replay - > l7proto ! = UNKNOWN ;
ctrack_replay - > l7proto_discovered = true ;
}
else
bDiscoveredL7 = ! ctrack_replay & & l7proto ! = UNKNOWN ;
if ( bDiscoveredL7 ) DLOG ( " discovered l7 protocol \n " ) ;
bool bDiscoveredHostname = bHaveHost & & ! ( ctrack_replay & & ctrack_replay - > hostname ) ;
if ( bDiscoveredHostname )
{
bool bCheckDone = false , bCheckResult = false , bCheckExcluded = false ;
DLOG ( " hostname: %s \n " , host ) ;
DLOG ( " discovered hostname \n " ) ;
if ( ctrack_replay )
{
ctrack_replay - > hostname = strdup ( host ) ;
if ( ! ctrack_replay - > hostname )
{
ctrack_replay - > hostname = strdup ( host ) ;
if ( ! ctrack_replay - > hostname )
{
DLOG_ERR ( " hostname dup : out of memory " ) ;
return verdict ;
}
DLOG ( " we have hostname now. searching desync profile again. \n " ) ;
struct desync_profile * dp_prev = dp ;
dp = ctrack_replay - > dp = dp_find ( & params . desync_profiles , ! ! ip6hdr , 0 , ntohs ( bReverse ? udphdr - > uh_sport : udphdr - > uh_dport ) , ctrack_replay - > hostname , & ctrack_replay - > bCheckDone , & ctrack_replay - > bCheckResult , & ctrack_replay - > bCheckExcluded ) ;
ctrack_replay - > dp_search_complete = true ;
if ( ! dp ) return verdict ;
if ( dp ! = dp_prev )
{
DLOG ( " desync profile changed by reavealed hostname ! \n " ) ;
// re-evaluate start/cutoff limiters
if ( ! replay )
{
maybe_cutoff ( ctrack , IPPROTO_UDP ) ;
if ( ! process_desync_interval ( dp , ctrack ) ) return verdict ;
}
}
DLOG_ERR ( " hostname dup : out of memory " ) ;
return verdict ;
}
bCheckDone = ctrack_replay - > bCheckDone ;
bCheckResult = ctrack_replay - > bCheckResult ;
bCheckExcluded = ctrack_replay - > bCheckExcluded ;
}
if ( dp - > hostlist | | dp - > hostlist_exclude )
}
bool bCheckDone = false , bCheckResult = false , bCheckExcluded = false ;
if ( bDiscoveredL7 | | bDiscoveredHostname )
{
struct desync_profile * dp_prev = dp ;
dp = dp_find ( & params . desync_profiles , IPPROTO_UDP , ( struct sockaddr * ) & dst , ctrack_replay ? ctrack_replay - > hostname : host , ctrack_replay ? ctrack_replay - > l7proto : l7proto , & bCheckDone , & bCheckResult , & bCheckExcluded ) ;
if ( ctrack_replay )
{
bool bCheckExcluded ;
if ( ! bCheckDone )
bCheckResult = HostlistCheck ( dp , host , & bCheckExcluded ) ;
if ( ! bCheckResult )
ctrack_replay - > dp = dp ;
ctrack_replay - > dp_search_complete = true ;
ctrack_replay - > bCheckDone = bCheckDone ;
ctrack_replay - > bCheckResult = bCheckResult ;
ctrack_replay - > bCheckExcluded = bCheckExcluded ;
}
if ( ! dp )
{
reasm_orig_cancel ( ctrack ) ;
return verdict ;
}
if ( dp ! = dp_prev )
{
DLOG ( " desync profile changed by revealed l7 protocol or hostname ! \n " ) ;
// rediscover autottl
autottl_discover ( ctrack_replay , ! ! ip6hdr ) ;
// re-evaluate start/cutoff limiters
if ( ! replay )
{
if ( ctrack_replay )
maybe_cutoff ( ctrack , IPPROTO_UDP ) ;
if ( ! process_desync_interval ( dp , ctrack ) ) return verdict ;
}
}
}
else if ( ctrack_replay )
{
bCheckDone = ctrack_replay - > bCheckDone ;
bCheckResult = ctrack_replay - > bCheckResult ;
bCheckExcluded = ctrack_replay - > bCheckExcluded ;
}
if ( bHaveHost & & ( dp - > hostlist | | dp - > hostlist_exclude ) )
{
if ( ! bCheckDone )
bCheckResult = HostlistCheck ( dp , host , & bCheckExcluded ) ;
if ( bCheckResult )
ctrack_stop_retrans_counter ( ctrack_replay ) ;
else
{
if ( ctrack_replay )
{
ctrack_replay - > hostname_ah_check = * dp - > hostlist_auto_filename & & ! bCheckExcluded ;
if ( ctrack_replay - > hostname_ah_check )
{
ctrack_replay - > hostname_ah_check = * dp - > hostlist_auto_filename & & ! bCheckExcluded ;
if ( ctrack_replay - > hostname_ah_check )
{
// first request is not retrans
if ( ctrack_replay - > hostname )
process_retrans_fail ( ctrack_replay , IPPROTO_UDP ) ;
else
ctrack_replay - > hostname = strdup ( host ) ;
}
// first request is not retrans
if ( ctrack_replay - > hostname )
process_retrans_fail ( ctrack_replay , IPPROTO_UDP ) ;
else
ctrack_replay - > hostname = strdup ( host ) ;
}
DLOG ( " not applying tampering to this request \n " ) ;
return verdict ;
}
DLOG ( " not applying tampering to this request \n " ) ;
return verdict ;
}
}
@ -1687,7 +1793,8 @@ static uint8_t dpi_desync_udp_packet_play(bool replay, size_t reasm_offset, uint
fake_size = dp - > fake_unknown_udp_size ;
break ;
}
ttl_fake = ip6hdr ? dp - > desync_ttl6 ? dp - > desync_ttl6 : ttl_orig : dp - > desync_ttl ? dp - > desync_ttl : ttl_orig ;
ttl_fake = ( ctrack_replay & & ctrack_replay - > autottl ) ? ctrack_replay - > autottl : ( ip6hdr ? ( dp - > desync_ttl6 ? dp - > desync_ttl6 : ttl_orig ) : ( dp - > desync_ttl ? dp - > desync_ttl : ttl_orig ) ) ;
enum dpi_desync_mode desync_mode = dp - > desync_mode ;
uint32_t fooling_orig = FOOL_NONE ;