mirror of
https://github.com/bol-van/zapret.git
synced 2025-05-29 05:57:50 +05:00
Compare commits
No commits in common. "master" and "v71" have entirely different histories.
@ -1062,7 +1062,7 @@ tpws_curl_test()
|
|||||||
# $1 - test function
|
# $1 - test function
|
||||||
# $2 - domain
|
# $2 - domain
|
||||||
# $3,$4,$5, ... - tpws params
|
# $3,$4,$5, ... - tpws params
|
||||||
echo - $1 ipv$IPV $2 : tpws $3 $4 $5 $6 $7 $8 $9${TPWS_EXTRA:+ $TPWS_EXTRA}${TPWS_EXTRA_1:+ "$TPWS_EXTRA_1"}${TPWS_EXTRA_2:+ "$TPWS_EXTRA_2"}${TPWS_EXTRA_3:+ "$TPWS_EXTRA_3"}${TPWS_EXTRA_4:+ "$TPWS_EXTRA_4"}${TPWS_EXTRA_5:+ "$TPWS_EXTRA_5"}${TPWS_EXTRA_6:+ "$TPWS_EXTRA_6"}${TPWS_EXTRA_7:+ "$TPWS_EXTRA_7"}${TPWS_EXTRA_8:+ "$TPWS_EXTRA_8"}${TPWS_EXTRA_9:+ "$TPWS_EXTRA_9"}
|
echo - $1 $2 : tpws $3 $4 $5 $6 $7 $8 $9${TPWS_EXTRA:+ $TPWS_EXTRA}${TPWS_EXTRA_1:+ "$TPWS_EXTRA_1"}${TPWS_EXTRA_2:+ "$TPWS_EXTRA_2"}${TPWS_EXTRA_3:+ "$TPWS_EXTRA_3"}${TPWS_EXTRA_4:+ "$TPWS_EXTRA_4"}${TPWS_EXTRA_5:+ "$TPWS_EXTRA_5"}${TPWS_EXTRA_6:+ "$TPWS_EXTRA_6"}${TPWS_EXTRA_7:+ "$TPWS_EXTRA_7"}${TPWS_EXTRA_8:+ "$TPWS_EXTRA_8"}${TPWS_EXTRA_9:+ "$TPWS_EXTRA_9"}
|
||||||
local ALL_PROXY="socks5://127.0.0.1:$SOCKS_PORT"
|
local ALL_PROXY="socks5://127.0.0.1:$SOCKS_PORT"
|
||||||
ws_curl_test tpws_start "$@"${TPWS_EXTRA:+ $TPWS_EXTRA}${TPWS_EXTRA_1:+ "$TPWS_EXTRA_1"}${TPWS_EXTRA_2:+ "$TPWS_EXTRA_2"}${TPWS_EXTRA_3:+ "$TPWS_EXTRA_3"}${TPWS_EXTRA_4:+ "$TPWS_EXTRA_4"}${TPWS_EXTRA_5:+ "$TPWS_EXTRA_5"}${TPWS_EXTRA_6:+ "$TPWS_EXTRA_6"}${TPWS_EXTRA_7:+ "$TPWS_EXTRA_7"}${TPWS_EXTRA_8:+ "$TPWS_EXTRA_8"}${TPWS_EXTRA_9:+ "$TPWS_EXTRA_9"}
|
ws_curl_test tpws_start "$@"${TPWS_EXTRA:+ $TPWS_EXTRA}${TPWS_EXTRA_1:+ "$TPWS_EXTRA_1"}${TPWS_EXTRA_2:+ "$TPWS_EXTRA_2"}${TPWS_EXTRA_3:+ "$TPWS_EXTRA_3"}${TPWS_EXTRA_4:+ "$TPWS_EXTRA_4"}${TPWS_EXTRA_5:+ "$TPWS_EXTRA_5"}${TPWS_EXTRA_6:+ "$TPWS_EXTRA_6"}${TPWS_EXTRA_7:+ "$TPWS_EXTRA_7"}${TPWS_EXTRA_8:+ "$TPWS_EXTRA_8"}${TPWS_EXTRA_9:+ "$TPWS_EXTRA_9"}
|
||||||
local testf=$1 dom=$2 strategy code=$?
|
local testf=$1 dom=$2 strategy code=$?
|
||||||
@ -1082,7 +1082,7 @@ pktws_curl_test()
|
|||||||
local testf=$1 dom=$2 strategy code
|
local testf=$1 dom=$2 strategy code
|
||||||
|
|
||||||
shift; shift;
|
shift; shift;
|
||||||
echo - $testf ipv$IPV $dom : $PKTWSD ${WF:+$WF }${PKTWS_EXTRA_PRE:+$PKTWS_EXTRA_PRE }${PKTWS_EXTRA_PRE_1:+"$PKTWS_EXTRA_PRE_1" }${PKTWS_EXTRA_PRE_2:+"$PKTWS_EXTRA_PRE_2" }${PKTWS_EXTRA_PRE_3:+"$PKTWS_EXTRA_PRE_3" }${PKTWS_EXTRA_PRE_4:+"$PKTWS_EXTRA_PRE_4" }${PKTWS_EXTRA_PRE_5:+"$PKTWS_EXTRA_PRE_5" }${PKTWS_EXTRA_PRE_6:+"$PKTWS_EXTRA_PRE_6" }${PKTWS_EXTRA_PRE_7:+"$PKTWS_EXTRA_PRE_7" }${PKTWS_EXTRA_PRE_8:+"$PKTWS_EXTRA_PRE_8" }${PKTWS_EXTRA_PRE_9:+"$PKTWS_EXTRA_PRE_9" }$@${PKTWS_EXTRA:+ $PKTWS_EXTRA}${PKTWS_EXTRA_1:+ "$PKTWS_EXTRA_1"}${PKTWS_EXTRA_2:+ "$PKTWS_EXTRA_2"}${PKTWS_EXTRA_3:+ "$PKTWS_EXTRA_3"}${PKTWS_EXTRA_4:+ "$PKTWS_EXTRA_4"}${PKTWS_EXTRA_5:+ "$PKTWS_EXTRA_5"}${PKTWS_EXTRA_6:+ "$PKTWS_EXTRA_6"}${PKTWS_EXTRA_7:+ "$PKTWS_EXTRA_7"}${PKTWS_EXTRA_8:+ "$PKTWS_EXTRA_8"}${PKTWS_EXTRA_9:+ "$PKTWS_EXTRA_9"}
|
echo - $testf $dom : $PKTWSD ${WF:+$WF }${PKTWS_EXTRA_PRE:+$PKTWS_EXTRA_PRE }${PKTWS_EXTRA_PRE_1:+"$PKTWS_EXTRA_PRE_1" }${PKTWS_EXTRA_PRE_2:+"$PKTWS_EXTRA_PRE_2" }${PKTWS_EXTRA_PRE_3:+"$PKTWS_EXTRA_PRE_3" }${PKTWS_EXTRA_PRE_4:+"$PKTWS_EXTRA_PRE_4" }${PKTWS_EXTRA_PRE_5:+"$PKTWS_EXTRA_PRE_5" }${PKTWS_EXTRA_PRE_6:+"$PKTWS_EXTRA_PRE_6" }${PKTWS_EXTRA_PRE_7:+"$PKTWS_EXTRA_PRE_7" }${PKTWS_EXTRA_PRE_8:+"$PKTWS_EXTRA_PRE_8" }${PKTWS_EXTRA_PRE_9:+"$PKTWS_EXTRA_PRE_9" }$@${PKTWS_EXTRA:+ $PKTWS_EXTRA}${PKTWS_EXTRA_1:+ "$PKTWS_EXTRA_1"}${PKTWS_EXTRA_2:+ "$PKTWS_EXTRA_2"}${PKTWS_EXTRA_3:+ "$PKTWS_EXTRA_3"}${PKTWS_EXTRA_4:+ "$PKTWS_EXTRA_4"}${PKTWS_EXTRA_5:+ "$PKTWS_EXTRA_5"}${PKTWS_EXTRA_6:+ "$PKTWS_EXTRA_6"}${PKTWS_EXTRA_7:+ "$PKTWS_EXTRA_7"}${PKTWS_EXTRA_8:+ "$PKTWS_EXTRA_8"}${PKTWS_EXTRA_9:+ "$PKTWS_EXTRA_9"}
|
||||||
ws_curl_test pktws_start $testf $dom ${PKTWS_EXTRA_PRE:+$PKTWS_EXTRA_PRE }${PKTWS_EXTRA_PRE_1:+"$PKTWS_EXTRA_PRE_1" }${PKTWS_EXTRA_PRE_2:+"$PKTWS_EXTRA_PRE_2" }${PKTWS_EXTRA_PRE_3:+"$PKTWS_EXTRA_PRE_3" }${PKTWS_EXTRA_PRE_4:+"$PKTWS_EXTRA_PRE_4" }${PKTWS_EXTRA_PRE_5:+"$PKTWS_EXTRA_PRE_5" }${PKTWS_EXTRA_PRE_6:+"$PKTWS_EXTRA_PRE_6" }${PKTWS_EXTRA_PRE_7:+"$PKTWS_EXTRA_PRE_7" }${PKTWS_EXTRA_PRE_8:+"$PKTWS_EXTRA_PRE_8" }${PKTWS_EXTRA_PRE_9:+"$PKTWS_EXTRA_PRE_9" }"$@"${PKTWS_EXTRA:+ $PKTWS_EXTRA}${PKTWS_EXTRA_1:+ "$PKTWS_EXTRA_1"}${PKTWS_EXTRA_2:+ "$PKTWS_EXTRA_2"}${PKTWS_EXTRA_3:+ "$PKTWS_EXTRA_3"}${PKTWS_EXTRA_4:+ "$PKTWS_EXTRA_4"}${PKTWS_EXTRA_5:+ "$PKTWS_EXTRA_5"}${PKTWS_EXTRA_6:+ "$PKTWS_EXTRA_6"}${PKTWS_EXTRA_7:+ "$PKTWS_EXTRA_7"}${PKTWS_EXTRA_8:+ "$PKTWS_EXTRA_8"}${PKTWS_EXTRA_9:+ "$PKTWS_EXTRA_9"}
|
ws_curl_test pktws_start $testf $dom ${PKTWS_EXTRA_PRE:+$PKTWS_EXTRA_PRE }${PKTWS_EXTRA_PRE_1:+"$PKTWS_EXTRA_PRE_1" }${PKTWS_EXTRA_PRE_2:+"$PKTWS_EXTRA_PRE_2" }${PKTWS_EXTRA_PRE_3:+"$PKTWS_EXTRA_PRE_3" }${PKTWS_EXTRA_PRE_4:+"$PKTWS_EXTRA_PRE_4" }${PKTWS_EXTRA_PRE_5:+"$PKTWS_EXTRA_PRE_5" }${PKTWS_EXTRA_PRE_6:+"$PKTWS_EXTRA_PRE_6" }${PKTWS_EXTRA_PRE_7:+"$PKTWS_EXTRA_PRE_7" }${PKTWS_EXTRA_PRE_8:+"$PKTWS_EXTRA_PRE_8" }${PKTWS_EXTRA_PRE_9:+"$PKTWS_EXTRA_PRE_9" }"$@"${PKTWS_EXTRA:+ $PKTWS_EXTRA}${PKTWS_EXTRA_1:+ "$PKTWS_EXTRA_1"}${PKTWS_EXTRA_2:+ "$PKTWS_EXTRA_2"}${PKTWS_EXTRA_3:+ "$PKTWS_EXTRA_3"}${PKTWS_EXTRA_4:+ "$PKTWS_EXTRA_4"}${PKTWS_EXTRA_5:+ "$PKTWS_EXTRA_5"}${PKTWS_EXTRA_6:+ "$PKTWS_EXTRA_6"}${PKTWS_EXTRA_7:+ "$PKTWS_EXTRA_7"}${PKTWS_EXTRA_8:+ "$PKTWS_EXTRA_8"}${PKTWS_EXTRA_9:+ "$PKTWS_EXTRA_9"}
|
||||||
|
|
||||||
code=$?
|
code=$?
|
||||||
|
@ -502,9 +502,3 @@ init.d, blockcheck: drop time exceeded icmp for nfqws-related connections
|
|||||||
blockcheck: some dup and orig-ttl mods
|
blockcheck: some dup and orig-ttl mods
|
||||||
blockcheck: PKTWS_EXTRA_PRE
|
blockcheck: PKTWS_EXTRA_PRE
|
||||||
blockcheck: report test function and domain every test
|
blockcheck: report test function and domain every test
|
||||||
|
|
||||||
v71.1
|
|
||||||
|
|
||||||
nfqws,tpws: much faster ipset implementation. move from hash to avl tree
|
|
||||||
install_easy: stop if running embedded release on traditional linux system (some files missing)
|
|
||||||
install_bin: add "read elf" arch detection method
|
|
||||||
|
@ -226,7 +226,6 @@ dvtws, собираемый из тех же исходников (см. [док
|
|||||||
--dpi-desync-fake-unknown=<filename>|0xHEX ; файл, содержащий фейковый пейлоад неизвестного протокола для dpi-desync=fake, на замену стандартным нулям 256 байт
|
--dpi-desync-fake-unknown=<filename>|0xHEX ; файл, содержащий фейковый пейлоад неизвестного протокола для dpi-desync=fake, на замену стандартным нулям 256 байт
|
||||||
--dpi-desync-fake-syndata=<filename>|0xHEX ; файл, содержащий фейковый пейлоад пакета SYN для режима десинхронизации syndata
|
--dpi-desync-fake-syndata=<filename>|0xHEX ; файл, содержащий фейковый пейлоад пакета SYN для режима десинхронизации syndata
|
||||||
--dpi-desync-fake-quic=<filename>|0xHEX ; файл, содержащий фейковый QUIC Initial
|
--dpi-desync-fake-quic=<filename>|0xHEX ; файл, содержащий фейковый QUIC Initial
|
||||||
--dpi-desync-fake-wireguard=<filename>|0xHEX ; файл, содержащий фейковый wireguard handshake initiation
|
|
||||||
--dpi-desync-fake-dht=<filename>|0xHEX ; файл, содержащий фейковый пейлоад DHT протокола для dpi-desync=fake, на замену стандартным нулям 64 байт
|
--dpi-desync-fake-dht=<filename>|0xHEX ; файл, содержащий фейковый пейлоад DHT протокола для dpi-desync=fake, на замену стандартным нулям 64 байт
|
||||||
--dpi-desync-fake-discord=<filename>|0xHEX ; файл, содержащий фейковый пейлоад Discord протокола нахождения IP адреса для голосовых чатов для dpi-desync=fake, на замену стандартным нулям 64 байт
|
--dpi-desync-fake-discord=<filename>|0xHEX ; файл, содержащий фейковый пейлоад Discord протокола нахождения IP адреса для голосовых чатов для dpi-desync=fake, на замену стандартным нулям 64 байт
|
||||||
--dpi-desync-fake-stun=<filename>|0xHEX ; файл, содержащий фейковый пейлоад STUN протокола для dpi-desync=fake, на замену стандартным нулям 64 байт
|
--dpi-desync-fake-stun=<filename>|0xHEX ; файл, содержащий фейковый пейлоад STUN протокола для dpi-desync=fake, на замену стандартным нулям 64 байт
|
||||||
|
@ -24,15 +24,17 @@ Tor поддерживает "из коробки" режим transparent proxy.
|
|||||||
Устанавливать их нужно с опцией opkg --force-overwrite, поскольку они перепишут ssh клиент от dropbear.
|
Устанавливать их нужно с опцией opkg --force-overwrite, поскольку они перепишут ssh клиент от dropbear.
|
||||||
После установки пакетов расслабим неоправданно жестокие права : chmod 755 /etc/ssh.
|
После установки пакетов расслабим неоправданно жестокие права : chmod 755 /etc/ssh.
|
||||||
Следует создать пользователя, под которым будем крутить ssh client. Допустим, это будет 'proxy'.
|
Следует создать пользователя, под которым будем крутить ssh client. Допустим, это будет 'proxy'.
|
||||||
Сначала установить пакеты shadow-useradd и shadow-su.
|
Сначала установить пакет shadow-useradd.
|
||||||
------------------
|
------------------
|
||||||
useradd -s /bin/false -d /home/proxy proxy
|
useradd -d /home/proxy proxy
|
||||||
mkdir -p /home/proxy
|
mkdir -p /home/proxy
|
||||||
chown proxy:proxy /home/proxy
|
chown proxy:proxy /home/proxy
|
||||||
------------------
|
------------------
|
||||||
Сгенерируем ключ RSA для доступа к ssh серверу.
|
Openssh ловит разные глюки, если у него нет доступа к /dev/tty.
|
||||||
|
Добавим в /etc/rc.local строчку : "chmod 666 /dev/tty"
|
||||||
|
Сгенерируем для него ключ RSA для доступа к ssh серверу.
|
||||||
------------------
|
------------------
|
||||||
su -s /bin/ash proxy
|
su proxy
|
||||||
cd
|
cd
|
||||||
mkdir -m 700 .ssh
|
mkdir -m 700 .ssh
|
||||||
cd .ssh
|
cd .ssh
|
||||||
|
@ -56,6 +56,9 @@ RemoveIPC=true
|
|||||||
RestrictNamespaces=true
|
RestrictNamespaces=true
|
||||||
RestrictRealtime=true
|
RestrictRealtime=true
|
||||||
RestrictSUIDSGID=true
|
RestrictSUIDSGID=true
|
||||||
|
SystemCallArchitectures=native
|
||||||
|
SystemCallFilter=@system-service
|
||||||
|
SystemCallFilter=~@resources
|
||||||
UMask=0077
|
UMask=0077
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
|
@ -55,6 +55,8 @@ RemoveIPC=true
|
|||||||
RestrictNamespaces=true
|
RestrictNamespaces=true
|
||||||
RestrictRealtime=true
|
RestrictRealtime=true
|
||||||
RestrictSUIDSGID=true
|
RestrictSUIDSGID=true
|
||||||
|
SystemCallArchitectures=native
|
||||||
|
SystemCallFilter=@system-service
|
||||||
UMask=0077
|
UMask=0077
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
|
@ -8,62 +8,6 @@ BINDIR="$EXEDIR/$BINS"
|
|||||||
ZAPRET_BASE=${ZAPRET_BASE:-"$EXEDIR"}
|
ZAPRET_BASE=${ZAPRET_BASE:-"$EXEDIR"}
|
||||||
. "$ZAPRET_BASE/common/base.sh"
|
. "$ZAPRET_BASE/common/base.sh"
|
||||||
|
|
||||||
|
|
||||||
read_elf_arch()
|
|
||||||
{
|
|
||||||
# $1 - elf file
|
|
||||||
|
|
||||||
local arch=$(dd if="$1" count=2 bs=1 skip=18 2>/dev/null | hexdump -e '2/1 "%02x"')
|
|
||||||
local bit=$(dd if="$1" count=1 bs=1 skip=4 2>/dev/null | hexdump -e '1/1 "%02x"')
|
|
||||||
echo $bit$arch
|
|
||||||
}
|
|
||||||
|
|
||||||
select_test_method()
|
|
||||||
{
|
|
||||||
local f ELF
|
|
||||||
|
|
||||||
TEST=run
|
|
||||||
|
|
||||||
# ash and dash try to execute invalid executables as a script. they interpret binary garbage with possible negative consequences
|
|
||||||
# bash and zsh do not do this
|
|
||||||
if exists bash; then
|
|
||||||
TEST=bash
|
|
||||||
elif exists zsh && [ "$UNAME" != CYGWIN ] ; then
|
|
||||||
TEST=zsh
|
|
||||||
elif [ "$UNAME" != Darwin -a "$UNAME" != CYGWIN ]; then
|
|
||||||
if exists hexdump and exists dd; then
|
|
||||||
# macos does not use ELF
|
|
||||||
TEST=elf
|
|
||||||
ELF=
|
|
||||||
ELF_ARCH=
|
|
||||||
for f in /bin/sh /system/bin/sh; do
|
|
||||||
[ -x "$f" ] && {
|
|
||||||
ELF=$f
|
|
||||||
break
|
|
||||||
}
|
|
||||||
done
|
|
||||||
[ -n "$ELF" ] && ELF_ARCH=$(read_elf_arch "$ELF")
|
|
||||||
[ -n "$ELF_ARCH" ] && return
|
|
||||||
fi
|
|
||||||
|
|
||||||
# find does not use its own shell exec
|
|
||||||
# it uses execvp(). in musl libc it does not call shell, in glibc it DOES call /bin/sh
|
|
||||||
# that's why prefer bash or zsh if present. otherwise it's our last chance
|
|
||||||
if exists find; then
|
|
||||||
TEST=find
|
|
||||||
FIND=find
|
|
||||||
elif exists busybox; then
|
|
||||||
busybox find /jGHUa3fh1A 2>/dev/null
|
|
||||||
# 127 - command not found
|
|
||||||
[ "$?" = 127 ] || {
|
|
||||||
TEST=find
|
|
||||||
FIND="busybox find"
|
|
||||||
}
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
check_dir()
|
check_dir()
|
||||||
{
|
{
|
||||||
local dir="$BINDIR/$1"
|
local dir="$BINDIR/$1"
|
||||||
@ -71,35 +15,23 @@ check_dir()
|
|||||||
local out
|
local out
|
||||||
if [ -f "$exe" ]; then
|
if [ -f "$exe" ]; then
|
||||||
if [ -x "$exe" ]; then
|
if [ -x "$exe" ]; then
|
||||||
case $TEST in
|
# ash and dash try to execute invalid executables as a script. they interpret binary garbage with possible negative consequences
|
||||||
bash)
|
# bash and zsh do not do this
|
||||||
|
if exists bash; then
|
||||||
out=$(echo 0.0.0.0 | bash -c "\"$exe"\" 2>/dev/null)
|
out=$(echo 0.0.0.0 | bash -c "\"$exe"\" 2>/dev/null)
|
||||||
[ -n "$out" ]
|
elif exists zsh; then
|
||||||
;;
|
|
||||||
zsh)
|
|
||||||
out=$(echo 0.0.0.0 | zsh -c "\"$exe\"" 2>/dev/null)
|
out=$(echo 0.0.0.0 | zsh -c "\"$exe\"" 2>/dev/null)
|
||||||
[ -n "$out" ]
|
else
|
||||||
;;
|
# find does not use its own shell exec
|
||||||
elf)
|
# it uses execvp(). in musl libc it does not call shell, in glibc it DOES call /bin/sh
|
||||||
out=$(read_elf_arch "$exe")
|
# that's why prefer bash or zsh if present. otherwise it's our last chance
|
||||||
[ "$ELF_ARCH" = "$out" ] && {
|
local FIND=find
|
||||||
# exec test to verify it actually works. no illegal instruction or crash.
|
if ! exists find && exists busybox; then
|
||||||
out=$(echo 0.0.0.0 | "$exe" 2>/dev/null)
|
FIND="busybox find"
|
||||||
[ -n "$out" ]
|
fi
|
||||||
}
|
|
||||||
;;
|
|
||||||
find)
|
|
||||||
out=$(echo 0.0.0.0 | $FIND "$dir" -maxdepth 1 -name ip2net -exec {} \; 2>/dev/null)
|
out=$(echo 0.0.0.0 | $FIND "$dir" -maxdepth 1 -name ip2net -exec {} \; 2>/dev/null)
|
||||||
|
fi
|
||||||
[ -n "$out" ]
|
[ -n "$out" ]
|
||||||
;;
|
|
||||||
run)
|
|
||||||
out=$(echo 0.0.0.0 | "$exe" 2>/dev/null)
|
|
||||||
[ -n "$out" ]
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
false
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
else
|
else
|
||||||
echo >&2 "$exe is not executable. set proper chmod."
|
echo >&2 "$exe is not executable. set proper chmod."
|
||||||
return 1
|
return 1
|
||||||
@ -121,7 +53,6 @@ ccp()
|
|||||||
}
|
}
|
||||||
|
|
||||||
UNAME=$(uname)
|
UNAME=$(uname)
|
||||||
|
|
||||||
unset PKTWS
|
unset PKTWS
|
||||||
case $UNAME in
|
case $UNAME in
|
||||||
Linux)
|
Linux)
|
||||||
@ -144,8 +75,6 @@ case $UNAME in
|
|||||||
ARCHLIST="my"
|
ARCHLIST="my"
|
||||||
esac
|
esac
|
||||||
|
|
||||||
select_test_method
|
|
||||||
|
|
||||||
if [ "$1" = "getarch" ]; then
|
if [ "$1" = "getarch" ]; then
|
||||||
for arch in $ARCHLIST
|
for arch in $ARCHLIST
|
||||||
do
|
do
|
||||||
@ -156,8 +85,6 @@ if [ "$1" = "getarch" ]; then
|
|||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
else
|
else
|
||||||
echo "using arch detect method : $TEST${ELF_ARCH:+ $ELF_ARCH}"
|
|
||||||
|
|
||||||
for arch in $ARCHLIST
|
for arch in $ARCHLIST
|
||||||
do
|
do
|
||||||
[ -d "$BINDIR/$arch" ] || continue
|
[ -d "$BINDIR/$arch" ] || continue
|
||||||
|
@ -50,26 +50,6 @@ check_readonly_system()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
check_source()
|
|
||||||
{
|
|
||||||
local bad=0
|
|
||||||
echo \* checking source files
|
|
||||||
case $SYSTEM in
|
|
||||||
systemd)
|
|
||||||
[ -f "$EXEDIR/init.d/systemd/zapret.service" ] || bad=1
|
|
||||||
;;
|
|
||||||
openrc)
|
|
||||||
[ -f "$EXEDIR/init.d/openrc/zapret" ] || bad=1
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
esac
|
|
||||||
[ "$bad" = 1 ] && {
|
|
||||||
echo 'some critical files are missing'
|
|
||||||
echo 'are you sure you are not using embedded release ? you need full version for traditional linux'
|
|
||||||
exitp 5
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
check_bins()
|
check_bins()
|
||||||
{
|
{
|
||||||
echo \* checking executables
|
echo \* checking executables
|
||||||
@ -903,7 +883,6 @@ umask 0022
|
|||||||
fix_sbin_path
|
fix_sbin_path
|
||||||
fsleep_setup
|
fsleep_setup
|
||||||
check_system
|
check_system
|
||||||
check_source
|
|
||||||
|
|
||||||
[ "$SYSTEM" = "macos" ] && . "$EXEDIR/init.d/macos/functions"
|
[ "$SYSTEM" = "macos" ] && . "$EXEDIR/init.d/macos/functions"
|
||||||
|
|
||||||
|
@ -113,16 +113,6 @@ bool append_to_list_file(const char *filename, const char *s)
|
|||||||
return bOK;
|
return bOK;
|
||||||
}
|
}
|
||||||
|
|
||||||
void expand_bits(void *target, const void *source, unsigned int source_bitlen, unsigned int target_bytelen)
|
|
||||||
{
|
|
||||||
unsigned int target_bitlen = target_bytelen<<3;
|
|
||||||
unsigned int bitlen = target_bitlen<source_bitlen ? target_bitlen : source_bitlen;
|
|
||||||
unsigned int bytelen = bitlen>>3;
|
|
||||||
|
|
||||||
if ((target_bytelen-bytelen)>=1) memset(target+bytelen,0,target_bytelen-bytelen);
|
|
||||||
memcpy(target,source,bytelen);
|
|
||||||
if ((bitlen &= 7)) ((uint8_t*)target)[bytelen] = ((uint8_t*)source)[bytelen] & (~((1 << (8-bitlen)) - 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
void ntop46(const struct sockaddr *sa, char *str, size_t len)
|
void ntop46(const struct sockaddr *sa, char *str, size_t len)
|
||||||
{
|
{
|
||||||
|
@ -31,8 +31,6 @@ bool load_file_nonempty(const char *filename,void *buffer,size_t *buffer_size);
|
|||||||
bool save_file(const char *filename, const void *buffer, size_t buffer_size);
|
bool save_file(const char *filename, const void *buffer, size_t buffer_size);
|
||||||
bool append_to_list_file(const char *filename, const char *s);
|
bool append_to_list_file(const char *filename, const char *s);
|
||||||
|
|
||||||
void expand_bits(void *target, const void *source, unsigned int source_bitlen, unsigned int target_bytelen);
|
|
||||||
|
|
||||||
void print_sockaddr(const struct sockaddr *sa);
|
void print_sockaddr(const struct sockaddr *sa);
|
||||||
void ntop46(const struct sockaddr *sa, char *str, size_t len);
|
void ntop46(const struct sockaddr *sa, char *str, size_t len);
|
||||||
void ntop46_port(const struct sockaddr *sa, char *str, size_t len);
|
void ntop46_port(const struct sockaddr *sa, char *str, size_t len);
|
||||||
|
400
nfq/kavl.h
400
nfq/kavl.h
@ -1,400 +0,0 @@
|
|||||||
/* The MIT License
|
|
||||||
|
|
||||||
Copyright (c) 2018 by Attractive Chaos <attractor@live.co.uk>
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining
|
|
||||||
a copy of this software and associated documentation files (the
|
|
||||||
"Software"), to deal in the Software without restriction, including
|
|
||||||
without limitation the rights to use, copy, modify, merge, publish,
|
|
||||||
distribute, sublicense, and/or sell copies of the Software, and to
|
|
||||||
permit persons to whom the Software is furnished to do so, subject to
|
|
||||||
the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be
|
|
||||||
included in all copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
||||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
||||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
||||||
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
||||||
BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
||||||
ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
||||||
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* An example:
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include "kavl.h"
|
|
||||||
|
|
||||||
struct my_node {
|
|
||||||
char key;
|
|
||||||
KAVL_HEAD(struct my_node) head;
|
|
||||||
};
|
|
||||||
#define my_cmp(p, q) (((q)->key < (p)->key) - ((p)->key < (q)->key))
|
|
||||||
KAVL_INIT(my, struct my_node, head, my_cmp)
|
|
||||||
|
|
||||||
int main(void) {
|
|
||||||
const char *str = "MNOLKQOPHIA"; // from wiki, except a duplicate
|
|
||||||
struct my_node *root = 0;
|
|
||||||
int i, l = strlen(str);
|
|
||||||
for (i = 0; i < l; ++i) { // insert in the input order
|
|
||||||
struct my_node *q, *p = malloc(sizeof(*p));
|
|
||||||
p->key = str[i];
|
|
||||||
q = kavl_insert(my, &root, p, 0);
|
|
||||||
if (p != q) free(p); // if already present, free
|
|
||||||
}
|
|
||||||
kavl_itr_t(my) itr;
|
|
||||||
kavl_itr_first(my, root, &itr); // place at first
|
|
||||||
do { // traverse
|
|
||||||
const struct my_node *p = kavl_at(&itr);
|
|
||||||
putchar(p->key);
|
|
||||||
free((void*)p); // free node
|
|
||||||
} while (kavl_itr_next(my, &itr));
|
|
||||||
putchar('\n');
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef KAVL_H
|
|
||||||
#define KAVL_H
|
|
||||||
|
|
||||||
#ifdef __STRICT_ANSI__
|
|
||||||
#define inline __inline__
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define KAVL_MAX_DEPTH 64
|
|
||||||
|
|
||||||
#define kavl_size(head, p) ((p)? (p)->head.size : 0)
|
|
||||||
#define kavl_size_child(head, q, i) ((q)->head.p[(i)]? (q)->head.p[(i)]->head.size : 0)
|
|
||||||
|
|
||||||
#define KAVL_HEAD(__type) \
|
|
||||||
struct { \
|
|
||||||
__type *p[2]; \
|
|
||||||
signed char balance; /* balance factor */ \
|
|
||||||
unsigned size; /* #elements in subtree */ \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_FIND(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__scope __type *kavl_find_##suf(const __type *root, const __type *x, unsigned *cnt_) { \
|
|
||||||
const __type *p = root; \
|
|
||||||
unsigned cnt = 0; \
|
|
||||||
while (p != 0) { \
|
|
||||||
int cmp; \
|
|
||||||
cmp = __cmp(x, p); \
|
|
||||||
if (cmp >= 0) cnt += kavl_size_child(__head, p, 0) + 1; \
|
|
||||||
if (cmp < 0) p = p->__head.p[0]; \
|
|
||||||
else if (cmp > 0) p = p->__head.p[1]; \
|
|
||||||
else break; \
|
|
||||||
} \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
return (__type*)p; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_ROTATE(suf, __type, __head) \
|
|
||||||
/* one rotation: (a,(b,c)q)p => ((a,b)p,c)q */ \
|
|
||||||
static inline __type *kavl_rotate1_##suf(__type *p, int dir) { /* dir=0 to left; dir=1 to right */ \
|
|
||||||
int opp = 1 - dir; /* opposite direction */ \
|
|
||||||
__type *q = p->__head.p[opp]; \
|
|
||||||
unsigned size_p = p->__head.size; \
|
|
||||||
p->__head.size -= q->__head.size - kavl_size_child(__head, q, dir); \
|
|
||||||
q->__head.size = size_p; \
|
|
||||||
p->__head.p[opp] = q->__head.p[dir]; \
|
|
||||||
q->__head.p[dir] = p; \
|
|
||||||
return q; \
|
|
||||||
} \
|
|
||||||
/* two consecutive rotations: (a,((b,c)r,d)q)p => ((a,b)p,(c,d)q)r */ \
|
|
||||||
static inline __type *kavl_rotate2_##suf(__type *p, int dir) { \
|
|
||||||
int b1, opp = 1 - dir; \
|
|
||||||
__type *q = p->__head.p[opp], *r = q->__head.p[dir]; \
|
|
||||||
unsigned size_x_dir = kavl_size_child(__head, r, dir); \
|
|
||||||
r->__head.size = p->__head.size; \
|
|
||||||
p->__head.size -= q->__head.size - size_x_dir; \
|
|
||||||
q->__head.size -= size_x_dir + 1; \
|
|
||||||
p->__head.p[opp] = r->__head.p[dir]; \
|
|
||||||
r->__head.p[dir] = p; \
|
|
||||||
q->__head.p[dir] = r->__head.p[opp]; \
|
|
||||||
r->__head.p[opp] = q; \
|
|
||||||
b1 = dir == 0? +1 : -1; \
|
|
||||||
if (r->__head.balance == b1) q->__head.balance = 0, p->__head.balance = -b1; \
|
|
||||||
else if (r->__head.balance == 0) q->__head.balance = p->__head.balance = 0; \
|
|
||||||
else q->__head.balance = b1, p->__head.balance = 0; \
|
|
||||||
r->__head.balance = 0; \
|
|
||||||
return r; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_INSERT(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__scope __type *kavl_insert_##suf(__type **root_, __type *x, unsigned *cnt_) { \
|
|
||||||
unsigned char stack[KAVL_MAX_DEPTH]; \
|
|
||||||
__type *path[KAVL_MAX_DEPTH]; \
|
|
||||||
__type *bp, *bq; \
|
|
||||||
__type *p, *q, *r = 0; /* _r_ is potentially the new root */ \
|
|
||||||
int i, which = 0, top, b1, path_len; \
|
|
||||||
unsigned cnt = 0; \
|
|
||||||
bp = *root_, bq = 0; \
|
|
||||||
/* find the insertion location */ \
|
|
||||||
for (p = bp, q = bq, top = path_len = 0; p; q = p, p = p->__head.p[which]) { \
|
|
||||||
int cmp; \
|
|
||||||
cmp = __cmp(x, p); \
|
|
||||||
if (cmp >= 0) cnt += kavl_size_child(__head, p, 0) + 1; \
|
|
||||||
if (cmp == 0) { \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
return p; \
|
|
||||||
} \
|
|
||||||
if (p->__head.balance != 0) \
|
|
||||||
bq = q, bp = p, top = 0; \
|
|
||||||
stack[top++] = which = (cmp > 0); \
|
|
||||||
path[path_len++] = p; \
|
|
||||||
} \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
x->__head.balance = 0, x->__head.size = 1, x->__head.p[0] = x->__head.p[1] = 0; \
|
|
||||||
if (q == 0) *root_ = x; \
|
|
||||||
else q->__head.p[which] = x; \
|
|
||||||
if (bp == 0) return x; \
|
|
||||||
for (i = 0; i < path_len; ++i) ++path[i]->__head.size; \
|
|
||||||
for (p = bp, top = 0; p != x; p = p->__head.p[stack[top]], ++top) /* update balance factors */ \
|
|
||||||
if (stack[top] == 0) --p->__head.balance; \
|
|
||||||
else ++p->__head.balance; \
|
|
||||||
if (bp->__head.balance > -2 && bp->__head.balance < 2) return x; /* no re-balance needed */ \
|
|
||||||
/* re-balance */ \
|
|
||||||
which = (bp->__head.balance < 0); \
|
|
||||||
b1 = which == 0? +1 : -1; \
|
|
||||||
q = bp->__head.p[1 - which]; \
|
|
||||||
if (q->__head.balance == b1) { \
|
|
||||||
r = kavl_rotate1_##suf(bp, which); \
|
|
||||||
q->__head.balance = bp->__head.balance = 0; \
|
|
||||||
} else r = kavl_rotate2_##suf(bp, which); \
|
|
||||||
if (bq == 0) *root_ = r; \
|
|
||||||
else bq->__head.p[bp != bq->__head.p[0]] = r; \
|
|
||||||
return x; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_ERASE(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__scope __type *kavl_erase_##suf(__type **root_, const __type *x, unsigned *cnt_) { \
|
|
||||||
__type *p, *path[KAVL_MAX_DEPTH], fake; \
|
|
||||||
unsigned char dir[KAVL_MAX_DEPTH]; \
|
|
||||||
int i, d = 0, cmp; \
|
|
||||||
unsigned cnt = 0; \
|
|
||||||
fake.__head.p[0] = *root_, fake.__head.p[1] = 0; \
|
|
||||||
if (cnt_) *cnt_ = 0; \
|
|
||||||
if (x) { \
|
|
||||||
for (cmp = -1, p = &fake; cmp; cmp = __cmp(x, p)) { \
|
|
||||||
int which = (cmp > 0); \
|
|
||||||
if (cmp > 0) cnt += kavl_size_child(__head, p, 0) + 1; \
|
|
||||||
dir[d] = which; \
|
|
||||||
path[d++] = p; \
|
|
||||||
p = p->__head.p[which]; \
|
|
||||||
if (p == 0) { \
|
|
||||||
if (cnt_) *cnt_ = 0; \
|
|
||||||
return 0; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
cnt += kavl_size_child(__head, p, 0) + 1; /* because p==x is not counted */ \
|
|
||||||
} else { \
|
|
||||||
for (p = &fake, cnt = 1; p; p = p->__head.p[0]) \
|
|
||||||
dir[d] = 0, path[d++] = p; \
|
|
||||||
p = path[--d]; \
|
|
||||||
} \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
for (i = 1; i < d; ++i) --path[i]->__head.size; \
|
|
||||||
if (p->__head.p[1] == 0) { /* ((1,.)2,3)4 => (1,3)4; p=2 */ \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = p->__head.p[0]; \
|
|
||||||
} else { \
|
|
||||||
__type *q = p->__head.p[1]; \
|
|
||||||
if (q->__head.p[0] == 0) { /* ((1,2)3,4)5 => ((1)2,4)5; p=3 */ \
|
|
||||||
q->__head.p[0] = p->__head.p[0]; \
|
|
||||||
q->__head.balance = p->__head.balance; \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = q; \
|
|
||||||
path[d] = q, dir[d++] = 1; \
|
|
||||||
q->__head.size = p->__head.size - 1; \
|
|
||||||
} else { /* ((1,((.,2)3,4)5)6,7)8 => ((1,(2,4)5)3,7)8; p=6 */ \
|
|
||||||
__type *r; \
|
|
||||||
int e = d++; /* backup _d_ */\
|
|
||||||
for (;;) { \
|
|
||||||
dir[d] = 0; \
|
|
||||||
path[d++] = q; \
|
|
||||||
r = q->__head.p[0]; \
|
|
||||||
if (r->__head.p[0] == 0) break; \
|
|
||||||
q = r; \
|
|
||||||
} \
|
|
||||||
r->__head.p[0] = p->__head.p[0]; \
|
|
||||||
q->__head.p[0] = r->__head.p[1]; \
|
|
||||||
r->__head.p[1] = p->__head.p[1]; \
|
|
||||||
r->__head.balance = p->__head.balance; \
|
|
||||||
path[e-1]->__head.p[dir[e-1]] = r; \
|
|
||||||
path[e] = r, dir[e] = 1; \
|
|
||||||
for (i = e + 1; i < d; ++i) --path[i]->__head.size; \
|
|
||||||
r->__head.size = p->__head.size - 1; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
while (--d > 0) { \
|
|
||||||
__type *q = path[d]; \
|
|
||||||
int which, other, b1 = 1, b2 = 2; \
|
|
||||||
which = dir[d], other = 1 - which; \
|
|
||||||
if (which) b1 = -b1, b2 = -b2; \
|
|
||||||
q->__head.balance += b1; \
|
|
||||||
if (q->__head.balance == b1) break; \
|
|
||||||
else if (q->__head.balance == b2) { \
|
|
||||||
__type *r = q->__head.p[other]; \
|
|
||||||
if (r->__head.balance == -b1) { \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = kavl_rotate2_##suf(q, which); \
|
|
||||||
} else { \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = kavl_rotate1_##suf(q, which); \
|
|
||||||
if (r->__head.balance == 0) { \
|
|
||||||
r->__head.balance = -b1; \
|
|
||||||
q->__head.balance = b1; \
|
|
||||||
break; \
|
|
||||||
} else r->__head.balance = q->__head.balance = 0; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
*root_ = fake.__head.p[0]; \
|
|
||||||
return p; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define kavl_free(__type, __head, __root, __free) do { \
|
|
||||||
__type *_p, *_q; \
|
|
||||||
for (_p = __root; _p; _p = _q) { \
|
|
||||||
if (_p->__head.p[0] == 0) { \
|
|
||||||
_q = _p->__head.p[1]; \
|
|
||||||
__free(_p); \
|
|
||||||
} else { \
|
|
||||||
_q = _p->__head.p[0]; \
|
|
||||||
_p->__head.p[0] = _q->__head.p[1]; \
|
|
||||||
_q->__head.p[1] = _p; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define __KAVL_ITR(suf, __scope, __type, __head, __cmp) \
|
|
||||||
struct kavl_itr_##suf { \
|
|
||||||
const __type *stack[KAVL_MAX_DEPTH], **top, *right; /* _right_ points to the right child of *top */ \
|
|
||||||
}; \
|
|
||||||
__scope void kavl_itr_first_##suf(const __type *root, struct kavl_itr_##suf *itr) { \
|
|
||||||
const __type *p; \
|
|
||||||
for (itr->top = itr->stack - 1, p = root; p; p = p->__head.p[0]) \
|
|
||||||
*++itr->top = p; \
|
|
||||||
itr->right = (*itr->top)->__head.p[1]; \
|
|
||||||
} \
|
|
||||||
__scope int kavl_itr_find_##suf(const __type *root, const __type *x, struct kavl_itr_##suf *itr) { \
|
|
||||||
const __type *p = root; \
|
|
||||||
itr->top = itr->stack - 1; \
|
|
||||||
while (p != 0) { \
|
|
||||||
int cmp; \
|
|
||||||
cmp = __cmp(x, p); \
|
|
||||||
if (cmp < 0) *++itr->top = p, p = p->__head.p[0]; \
|
|
||||||
else if (cmp > 0) p = p->__head.p[1]; \
|
|
||||||
else break; \
|
|
||||||
} \
|
|
||||||
if (p) { \
|
|
||||||
*++itr->top = p; \
|
|
||||||
itr->right = p->__head.p[1]; \
|
|
||||||
return 1; \
|
|
||||||
} else if (itr->top >= itr->stack) { \
|
|
||||||
itr->right = (*itr->top)->__head.p[1]; \
|
|
||||||
return 0; \
|
|
||||||
} else return 0; \
|
|
||||||
} \
|
|
||||||
__scope int kavl_itr_next_##suf(struct kavl_itr_##suf *itr) { \
|
|
||||||
for (;;) { \
|
|
||||||
const __type *p; \
|
|
||||||
for (p = itr->right, --itr->top; p; p = p->__head.p[0]) \
|
|
||||||
*++itr->top = p; \
|
|
||||||
if (itr->top < itr->stack) return 0; \
|
|
||||||
itr->right = (*itr->top)->__head.p[1]; \
|
|
||||||
return 1; \
|
|
||||||
} \
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Insert a node to the tree
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param proot pointer to the root of the tree (in/out: root may change)
|
|
||||||
* @param x node to insert (in)
|
|
||||||
* @param cnt number of nodes smaller than or equal to _x_; can be NULL (out)
|
|
||||||
*
|
|
||||||
* @return _x_ if not present in the tree, or the node equal to x.
|
|
||||||
*/
|
|
||||||
#define kavl_insert(suf, proot, x, cnt) kavl_insert_##suf(proot, x, cnt)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Find a node in the tree
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param root root of the tree
|
|
||||||
* @param x node value to find (in)
|
|
||||||
* @param cnt number of nodes smaller than or equal to _x_; can be NULL (out)
|
|
||||||
*
|
|
||||||
* @return node equal to _x_ if present, or NULL if absent
|
|
||||||
*/
|
|
||||||
#define kavl_find(suf, root, x, cnt) kavl_find_##suf(root, x, cnt)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Delete a node from the tree
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param proot pointer to the root of the tree (in/out: root may change)
|
|
||||||
* @param x node value to delete; if NULL, delete the first node (in)
|
|
||||||
*
|
|
||||||
* @return node removed from the tree if present, or NULL if absent
|
|
||||||
*/
|
|
||||||
#define kavl_erase(suf, proot, x, cnt) kavl_erase_##suf(proot, x, cnt)
|
|
||||||
#define kavl_erase_first(suf, proot) kavl_erase_##suf(proot, 0, 0)
|
|
||||||
|
|
||||||
#define kavl_itr_t(suf) struct kavl_itr_##suf
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Place the iterator at the smallest object
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param root root of the tree
|
|
||||||
* @param itr iterator
|
|
||||||
*/
|
|
||||||
#define kavl_itr_first(suf, root, itr) kavl_itr_first_##suf(root, itr)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Place the iterator at the object equal to or greater than the query
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param root root of the tree
|
|
||||||
* @param x query (in)
|
|
||||||
* @param itr iterator (out)
|
|
||||||
*
|
|
||||||
* @return 1 if find; 0 otherwise. kavl_at(itr) is NULL if and only if query is
|
|
||||||
* larger than all objects in the tree
|
|
||||||
*/
|
|
||||||
#define kavl_itr_find(suf, root, x, itr) kavl_itr_find_##suf(root, x, itr)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Move to the next object in order
|
|
||||||
*
|
|
||||||
* @param itr iterator (modified)
|
|
||||||
*
|
|
||||||
* @return 1 if there is a next object; 0 otherwise
|
|
||||||
*/
|
|
||||||
#define kavl_itr_next(suf, itr) kavl_itr_next_##suf(itr)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the pointer at the iterator
|
|
||||||
*
|
|
||||||
* @param itr iterator
|
|
||||||
*
|
|
||||||
* @return pointer if present; NULL otherwise
|
|
||||||
*/
|
|
||||||
#define kavl_at(itr) ((itr)->top < (itr)->stack? 0 : *(itr)->top)
|
|
||||||
|
|
||||||
#define KAVL_INIT2(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_FIND(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_ROTATE(suf, __type, __head) \
|
|
||||||
__KAVL_INSERT(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_ERASE(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_ITR(suf, __scope, __type, __head, __cmp)
|
|
||||||
|
|
||||||
#define KAVL_INIT(suf, __type, __head, __cmp) \
|
|
||||||
KAVL_INIT2(suf,, __type, __head, __cmp)
|
|
||||||
|
|
||||||
#endif
|
|
@ -1675,8 +1675,7 @@ void check_dp(const struct desync_profile *dp)
|
|||||||
DLOG_CONDUP("WARNING !!! it's completely ok if connbytes or payload based ip/nf tables limiter is applied. Make sure it exists.\n");
|
DLOG_CONDUP("WARNING !!! it's completely ok if connbytes or payload based ip/nf tables limiter is applied. Make sure it exists.\n");
|
||||||
#else
|
#else
|
||||||
DLOG_CONDUP("WARNING !!! possible TRASH FLOOD configuration detected in profile %d\n", dp->n);
|
DLOG_CONDUP("WARNING !!! possible TRASH FLOOD configuration detected in profile %d\n", dp->n);
|
||||||
DLOG_CONDUP("WARNING !!! in profile %d you are using --dpi-desync-any-protocol without --dpi-desync-cutoff or --dup without --dup-cutoff\n", dp->n);
|
DLOG_CONDUP("WARNING !!! it's highly recommended to use --dpi-desync-cutoff limiter or fakes will be sent on every processed packet\n");
|
||||||
DLOG_CONDUP("WARNING !!! fakes or dups will be sent on every processed packet\n");
|
|
||||||
DLOG_CONDUP("WARNING !!! make sure it's really what you want\n");
|
DLOG_CONDUP("WARNING !!! make sure it's really what you want\n");
|
||||||
#ifdef __CYGWIN__
|
#ifdef __CYGWIN__
|
||||||
DLOG_CONDUP("WARNING !!! in most cases this is acceptable only with custom payload based windivert filter (--wf-raw)\n");
|
DLOG_CONDUP("WARNING !!! in most cases this is acceptable only with custom payload based windivert filter (--wf-raw)\n");
|
||||||
|
205
nfq/pools.c
205
nfq/pools.c
@ -260,146 +260,121 @@ bool hostlist_collection_is_empty(const struct hostlist_collection_head *head)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int kavl_bit_cmp(const struct kavl_bit_elem *p, const struct kavl_bit_elem *q)
|
void ipset4Destroy(ipset4 **ipset)
|
||||||
{
|
{
|
||||||
unsigned int bitlen = q->bitlen < p->bitlen ? q->bitlen : p->bitlen;
|
ipset4 *elem, *tmp;
|
||||||
unsigned int df = bitlen & 7, bytes = bitlen >> 3;
|
HASH_ITER(hh, *ipset, elem, tmp)
|
||||||
int cmp = memcmp(p->data, q->data, bytes);
|
{
|
||||||
|
HASH_DEL(*ipset, elem);
|
||||||
|
free(elem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
bool ipset4Check(ipset4 *ipset, const struct in_addr *a, uint8_t preflen)
|
||||||
|
{
|
||||||
|
uint32_t ip = ntohl(a->s_addr);
|
||||||
|
struct cidr4 cidr;
|
||||||
|
ipset4 *ips_found;
|
||||||
|
|
||||||
if (cmp || !df) return cmp;
|
// zero alignment bytes
|
||||||
|
memset(&cidr,0,sizeof(cidr));
|
||||||
|
cidr.preflen = preflen+1;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
cidr.preflen--;
|
||||||
|
cidr.addr.s_addr = htonl(ip & mask_from_preflen(cidr.preflen));
|
||||||
|
HASH_FIND(hh, ipset, &cidr, sizeof(cidr), ips_found);
|
||||||
|
if (ips_found) return true;
|
||||||
|
} while(cidr.preflen);
|
||||||
|
|
||||||
uint8_t c1 = p->data[bytes] >> (8 - df);
|
|
||||||
uint8_t c2 = q->data[bytes] >> (8 - df);
|
|
||||||
return c1<c2 ? -1 : c1==c2 ? 0 : 1;
|
|
||||||
}
|
|
||||||
KAVL_INIT(kavl_bit, struct kavl_bit_elem, head, kavl_bit_cmp)
|
|
||||||
static void kavl_bit_destroy_elem(struct kavl_bit_elem *e)
|
|
||||||
{
|
|
||||||
if (e)
|
|
||||||
{
|
|
||||||
free(e->data);
|
|
||||||
free(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void kavl_bit_delete(struct kavl_bit_elem **hdr, const void *data, unsigned int bitlen)
|
|
||||||
{
|
|
||||||
struct kavl_bit_elem temp = {
|
|
||||||
.bitlen = bitlen, .data = (uint8_t*)data
|
|
||||||
};
|
|
||||||
kavl_bit_destroy_elem(kavl_erase(kavl_bit, hdr, &temp, 0));
|
|
||||||
}
|
|
||||||
void kavl_bit_destroy(struct kavl_bit_elem **hdr)
|
|
||||||
{
|
|
||||||
while (*hdr)
|
|
||||||
{
|
|
||||||
struct kavl_bit_elem *e = kavl_erase_first(kavl_bit, hdr);
|
|
||||||
if (!e) break;
|
|
||||||
kavl_bit_destroy_elem(e);
|
|
||||||
}
|
|
||||||
free(*hdr);
|
|
||||||
}
|
|
||||||
struct kavl_bit_elem *kavl_bit_add(struct kavl_bit_elem **hdr, void *data, unsigned int bitlen, size_t struct_size)
|
|
||||||
{
|
|
||||||
if (!struct_size) struct_size=sizeof(struct kavl_bit_elem);
|
|
||||||
|
|
||||||
struct kavl_bit_elem *v, *e = calloc(1, struct_size);
|
|
||||||
if (!e) return 0;
|
|
||||||
|
|
||||||
e->bitlen = bitlen;
|
|
||||||
e->data = data;
|
|
||||||
|
|
||||||
v = kavl_insert(kavl_bit, hdr, e, 0);
|
|
||||||
while (e != v && e->bitlen < v->bitlen)
|
|
||||||
{
|
|
||||||
kavl_bit_delete(hdr, v->data, v->bitlen);
|
|
||||||
v = kavl_insert(kavl_bit, hdr, e, 0);
|
|
||||||
}
|
|
||||||
if (e != v) kavl_bit_destroy_elem(e);
|
|
||||||
return v;
|
|
||||||
}
|
|
||||||
struct kavl_bit_elem *kavl_bit_get(const struct kavl_bit_elem *hdr, const void *data, unsigned int bitlen)
|
|
||||||
{
|
|
||||||
struct kavl_bit_elem temp = {
|
|
||||||
.bitlen = bitlen, .data = (uint8_t*)data
|
|
||||||
};
|
|
||||||
return kavl_find(kavl_bit, hdr, &temp, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool ipset_kavl_add(struct kavl_bit_elem **ipset, const void *a, uint8_t preflen)
|
|
||||||
{
|
|
||||||
uint8_t bytelen = (preflen+7)>>3;
|
|
||||||
uint8_t *abuf = malloc(bytelen);
|
|
||||||
if (!abuf) return false;
|
|
||||||
memcpy(abuf,a,bytelen);
|
|
||||||
if (!kavl_bit_add(ipset,abuf,preflen,0))
|
|
||||||
{
|
|
||||||
free(abuf);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
bool ipset4Add(ipset4 **ipset, const struct in_addr *a, uint8_t preflen)
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool ipset4Check(const struct kavl_bit_elem *ipset, const struct in_addr *a, uint8_t preflen)
|
|
||||||
{
|
|
||||||
return !!kavl_bit_get(ipset,a,preflen);
|
|
||||||
}
|
|
||||||
bool ipset4Add(struct kavl_bit_elem **ipset, const struct in_addr *a, uint8_t preflen)
|
|
||||||
{
|
{
|
||||||
if (preflen>32) return false;
|
if (preflen>32) return false;
|
||||||
return ipset_kavl_add(ipset,a,preflen);
|
|
||||||
}
|
|
||||||
void ipset4Print(struct kavl_bit_elem *ipset)
|
|
||||||
{
|
|
||||||
if (!ipset) return;
|
|
||||||
|
|
||||||
struct cidr4 c;
|
// avoid dups
|
||||||
const struct kavl_bit_elem *elem;
|
if (ipset4Check(*ipset, a, preflen)) return true; // already included
|
||||||
kavl_itr_t(kavl_bit) itr;
|
|
||||||
kavl_itr_first(kavl_bit, ipset, &itr);
|
struct ipset4 *entry = calloc(1,sizeof(ipset4));
|
||||||
do
|
if (!entry) return false;
|
||||||
|
|
||||||
|
entry->cidr.addr.s_addr = htonl(ntohl(a->s_addr) & mask_from_preflen(preflen));
|
||||||
|
entry->cidr.preflen = preflen;
|
||||||
|
oom = false;
|
||||||
|
HASH_ADD(hh, *ipset, cidr, sizeof(entry->cidr), entry);
|
||||||
|
if (oom) { free(entry); return false; }
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
void ipset4Print(ipset4 *ipset)
|
||||||
{
|
{
|
||||||
elem = kavl_at(&itr);
|
ipset4 *ips, *tmp;
|
||||||
c.preflen = elem->bitlen;
|
HASH_ITER(hh, ipset , ips, tmp)
|
||||||
expand_bits(&c.addr, elem->data, elem->bitlen, sizeof(c.addr));
|
{
|
||||||
print_cidr4(&c);
|
print_cidr4(&ips->cidr);
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
while (kavl_itr_next(kavl_bit, &itr));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ipset6Check(const struct kavl_bit_elem *ipset, const struct in6_addr *a, uint8_t preflen)
|
void ipset6Destroy(ipset6 **ipset)
|
||||||
{
|
{
|
||||||
return !!kavl_bit_get(ipset,a,preflen);
|
ipset6 *elem, *tmp;
|
||||||
|
HASH_ITER(hh, *ipset, elem, tmp)
|
||||||
|
{
|
||||||
|
HASH_DEL(*ipset, elem);
|
||||||
|
free(elem);
|
||||||
}
|
}
|
||||||
bool ipset6Add(struct kavl_bit_elem **ipset, const struct in6_addr *a, uint8_t preflen)
|
}
|
||||||
|
bool ipset6Check(ipset6 *ipset, const struct in6_addr *a, uint8_t preflen)
|
||||||
|
{
|
||||||
|
struct cidr6 cidr;
|
||||||
|
ipset6 *ips_found;
|
||||||
|
|
||||||
|
// zero alignment bytes
|
||||||
|
memset(&cidr,0,sizeof(cidr));
|
||||||
|
cidr.preflen = preflen+1;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
cidr.preflen--;
|
||||||
|
ip6_and(a, mask_from_preflen6(cidr.preflen), &cidr.addr);
|
||||||
|
HASH_FIND(hh, ipset, &cidr, sizeof(cidr), ips_found);
|
||||||
|
if (ips_found) return true;
|
||||||
|
} while(cidr.preflen);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
bool ipset6Add(ipset6 **ipset, const struct in6_addr *a, uint8_t preflen)
|
||||||
{
|
{
|
||||||
if (preflen>128) return false;
|
if (preflen>128) return false;
|
||||||
return ipset_kavl_add(ipset,a,preflen);
|
|
||||||
}
|
|
||||||
void ipset6Print(struct kavl_bit_elem *ipset)
|
|
||||||
{
|
|
||||||
if (!ipset) return;
|
|
||||||
|
|
||||||
struct cidr6 c;
|
// avoid dups
|
||||||
const struct kavl_bit_elem *elem;
|
if (ipset6Check(*ipset, a, preflen)) return true; // already included
|
||||||
kavl_itr_t(kavl_bit) itr;
|
|
||||||
kavl_itr_first(kavl_bit, ipset, &itr);
|
struct ipset6 *entry = calloc(1,sizeof(ipset6));
|
||||||
do
|
if (!entry) return false;
|
||||||
|
|
||||||
|
ip6_and(a, mask_from_preflen6(preflen), &entry->cidr.addr);
|
||||||
|
entry->cidr.preflen = preflen;
|
||||||
|
oom = false;
|
||||||
|
HASH_ADD(hh, *ipset, cidr, sizeof(entry->cidr), entry);
|
||||||
|
if (oom) { free(entry); return false; }
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
void ipset6Print(ipset6 *ipset)
|
||||||
{
|
{
|
||||||
elem = kavl_at(&itr);
|
ipset6 *ips, *tmp;
|
||||||
c.preflen = elem->bitlen;
|
HASH_ITER(hh, ipset , ips, tmp)
|
||||||
expand_bits(&c.addr, elem->data, elem->bitlen, sizeof(c.addr));
|
{
|
||||||
print_cidr6(&c);
|
print_cidr6(&ips->cidr);
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
while (kavl_itr_next(kavl_bit, &itr));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ipsetDestroy(ipset *ipset)
|
void ipsetDestroy(ipset *ipset)
|
||||||
{
|
{
|
||||||
kavl_bit_destroy(&ipset->ips4);
|
ipset4Destroy(&ipset->ips4);
|
||||||
kavl_bit_destroy(&ipset->ips6);
|
ipset6Destroy(&ipset->ips6);
|
||||||
}
|
}
|
||||||
void ipsetPrint(ipset *ipset)
|
void ipsetPrint(ipset *ipset)
|
||||||
{
|
{
|
||||||
|
43
nfq/pools.h
43
nfq/pools.h
@ -13,8 +13,6 @@
|
|||||||
#define HASH_FUNCTION HASH_BER
|
#define HASH_FUNCTION HASH_BER
|
||||||
#include "uthash.h"
|
#include "uthash.h"
|
||||||
|
|
||||||
#include "kavl.h"
|
|
||||||
|
|
||||||
#define HOSTLIST_POOL_FLAG_STRICT_MATCH 1
|
#define HOSTLIST_POOL_FLAG_STRICT_MATCH 1
|
||||||
|
|
||||||
typedef struct hostlist_pool {
|
typedef struct hostlist_pool {
|
||||||
@ -78,40 +76,39 @@ struct hostlist_item *hostlist_collection_search(struct hostlist_collection_head
|
|||||||
bool hostlist_collection_is_empty(const struct hostlist_collection_head *head);
|
bool hostlist_collection_is_empty(const struct hostlist_collection_head *head);
|
||||||
|
|
||||||
|
|
||||||
struct kavl_bit_elem
|
typedef struct ipset4 {
|
||||||
{
|
struct cidr4 cidr; /* key */
|
||||||
unsigned int bitlen;
|
UT_hash_handle hh; /* makes this structure hashable */
|
||||||
uint8_t *data;
|
} ipset4;
|
||||||
KAVL_HEAD(struct kavl_bit_elem) head;
|
typedef struct ipset6 {
|
||||||
};
|
struct cidr6 cidr; /* key */
|
||||||
|
UT_hash_handle hh; /* makes this structure hashable */
|
||||||
struct kavl_bit_elem *kavl_bit_get(const struct kavl_bit_elem *hdr, const void *data, unsigned int bitlen);
|
} ipset6;
|
||||||
struct kavl_bit_elem *kavl_bit_add(struct kavl_bit_elem **hdr, void *data, unsigned int bitlen, size_t struct_size);
|
|
||||||
void kavl_bit_delete(struct kavl_bit_elem **hdr, const void *data, unsigned int bitlen);
|
|
||||||
void kavl_bit_destroy(struct kavl_bit_elem **hdr);
|
|
||||||
|
|
||||||
// combined ipset ipv4 and ipv6
|
// combined ipset ipv4 and ipv6
|
||||||
typedef struct ipset {
|
typedef struct ipset {
|
||||||
struct kavl_bit_elem *ips4,*ips6;
|
ipset4 *ips4;
|
||||||
|
ipset6 *ips6;
|
||||||
} ipset;
|
} ipset;
|
||||||
|
|
||||||
#define IPSET_EMPTY(ips) (!(ips)->ips4 && !(ips)->ips6)
|
#define IPSET_EMPTY(ips) (!(ips)->ips4 && !(ips)->ips6)
|
||||||
|
|
||||||
bool ipset4Add(struct kavl_bit_elem **ipset, const struct in_addr *a, uint8_t preflen);
|
void ipset4Destroy(ipset4 **ipset);
|
||||||
static inline bool ipset4AddCidr(struct kavl_bit_elem **ipset, const struct cidr4 *cidr)
|
bool ipset4Add(ipset4 **ipset, const struct in_addr *a, uint8_t preflen);
|
||||||
|
static inline bool ipset4AddCidr(ipset4 **ipset, const struct cidr4 *cidr)
|
||||||
{
|
{
|
||||||
return ipset4Add(ipset,&cidr->addr,cidr->preflen);
|
return ipset4Add(ipset,&cidr->addr,cidr->preflen);
|
||||||
}
|
}
|
||||||
bool ipset4Check(const struct kavl_bit_elem *ipset, const struct in_addr *a, uint8_t preflen);
|
bool ipset4Check(ipset4 *ipset, const struct in_addr *a, uint8_t preflen);
|
||||||
void ipset4Print(struct kavl_bit_elem *ipset);
|
void ipset4Print(ipset4 *ipset);
|
||||||
|
|
||||||
bool ipset6Add(struct kavl_bit_elem **ipset, const struct in6_addr *a, uint8_t preflen);
|
void ipset6Destroy(ipset6 **ipset);
|
||||||
static inline bool ipset6AddCidr(struct kavl_bit_elem **ipset, const struct cidr6 *cidr)
|
bool ipset6Add(ipset6 **ipset, const struct in6_addr *a, uint8_t preflen);
|
||||||
|
static inline bool ipset6AddCidr(ipset6 **ipset, const struct cidr6 *cidr)
|
||||||
{
|
{
|
||||||
return ipset6Add(ipset,&cidr->addr,cidr->preflen);
|
return ipset6Add(ipset,&cidr->addr,cidr->preflen);
|
||||||
}
|
}
|
||||||
bool ipset6Check(const struct kavl_bit_elem *ipset, const struct in6_addr *a, uint8_t preflen);
|
bool ipset6Check(ipset6 *ipset, const struct in6_addr *a, uint8_t preflen);
|
||||||
void ipset6Print(struct kavl_bit_elem *ipset);
|
void ipset6Print(ipset6 *ipset);
|
||||||
|
|
||||||
void ipsetDestroy(ipset *ipset);
|
void ipsetDestroy(ipset *ipset);
|
||||||
void ipsetPrint(ipset *ipset);
|
void ipsetPrint(ipset *ipset);
|
||||||
|
@ -112,17 +112,6 @@ bool append_to_list_file(const char *filename, const char *s)
|
|||||||
return bOK;
|
return bOK;
|
||||||
}
|
}
|
||||||
|
|
||||||
void expand_bits(void *target, const void *source, unsigned int source_bitlen, unsigned int target_bytelen)
|
|
||||||
{
|
|
||||||
unsigned int target_bitlen = target_bytelen<<3;
|
|
||||||
unsigned int bitlen = target_bitlen<source_bitlen ? target_bitlen : source_bitlen;
|
|
||||||
unsigned int bytelen = bitlen>>3;
|
|
||||||
|
|
||||||
if ((target_bytelen-bytelen)>=1) memset(target+bytelen,0,target_bytelen-bytelen);
|
|
||||||
memcpy(target,source,bytelen);
|
|
||||||
if ((bitlen &= 7)) ((uint8_t*)target)[bytelen] = ((uint8_t*)source)[bytelen] & (~((1 << (8-bitlen)) - 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
void ntop46(const struct sockaddr *sa, char *str, size_t len)
|
void ntop46(const struct sockaddr *sa, char *str, size_t len)
|
||||||
{
|
{
|
||||||
if (!len) return;
|
if (!len) return;
|
||||||
|
@ -29,8 +29,6 @@ bool str_ends_with(const char *s, const char *suffix);
|
|||||||
bool load_file(const char *filename,void *buffer,size_t *buffer_size);
|
bool load_file(const char *filename,void *buffer,size_t *buffer_size);
|
||||||
bool append_to_list_file(const char *filename, const char *s);
|
bool append_to_list_file(const char *filename, const char *s);
|
||||||
|
|
||||||
void expand_bits(void *target, const void *source, unsigned int source_bitlen, unsigned int target_bytelen);
|
|
||||||
|
|
||||||
void ntop46(const struct sockaddr *sa, char *str, size_t len);
|
void ntop46(const struct sockaddr *sa, char *str, size_t len);
|
||||||
void ntop46_port(const struct sockaddr *sa, char *str, size_t len);
|
void ntop46_port(const struct sockaddr *sa, char *str, size_t len);
|
||||||
void print_sockaddr(const struct sockaddr *sa);
|
void print_sockaddr(const struct sockaddr *sa);
|
||||||
|
400
tpws/kavl.h
400
tpws/kavl.h
@ -1,400 +0,0 @@
|
|||||||
/* The MIT License
|
|
||||||
|
|
||||||
Copyright (c) 2018 by Attractive Chaos <attractor@live.co.uk>
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining
|
|
||||||
a copy of this software and associated documentation files (the
|
|
||||||
"Software"), to deal in the Software without restriction, including
|
|
||||||
without limitation the rights to use, copy, modify, merge, publish,
|
|
||||||
distribute, sublicense, and/or sell copies of the Software, and to
|
|
||||||
permit persons to whom the Software is furnished to do so, subject to
|
|
||||||
the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be
|
|
||||||
included in all copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
||||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
||||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
||||||
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
||||||
BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
||||||
ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
||||||
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* An example:
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include "kavl.h"
|
|
||||||
|
|
||||||
struct my_node {
|
|
||||||
char key;
|
|
||||||
KAVL_HEAD(struct my_node) head;
|
|
||||||
};
|
|
||||||
#define my_cmp(p, q) (((q)->key < (p)->key) - ((p)->key < (q)->key))
|
|
||||||
KAVL_INIT(my, struct my_node, head, my_cmp)
|
|
||||||
|
|
||||||
int main(void) {
|
|
||||||
const char *str = "MNOLKQOPHIA"; // from wiki, except a duplicate
|
|
||||||
struct my_node *root = 0;
|
|
||||||
int i, l = strlen(str);
|
|
||||||
for (i = 0; i < l; ++i) { // insert in the input order
|
|
||||||
struct my_node *q, *p = malloc(sizeof(*p));
|
|
||||||
p->key = str[i];
|
|
||||||
q = kavl_insert(my, &root, p, 0);
|
|
||||||
if (p != q) free(p); // if already present, free
|
|
||||||
}
|
|
||||||
kavl_itr_t(my) itr;
|
|
||||||
kavl_itr_first(my, root, &itr); // place at first
|
|
||||||
do { // traverse
|
|
||||||
const struct my_node *p = kavl_at(&itr);
|
|
||||||
putchar(p->key);
|
|
||||||
free((void*)p); // free node
|
|
||||||
} while (kavl_itr_next(my, &itr));
|
|
||||||
putchar('\n');
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef KAVL_H
|
|
||||||
#define KAVL_H
|
|
||||||
|
|
||||||
#ifdef __STRICT_ANSI__
|
|
||||||
#define inline __inline__
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define KAVL_MAX_DEPTH 64
|
|
||||||
|
|
||||||
#define kavl_size(head, p) ((p)? (p)->head.size : 0)
|
|
||||||
#define kavl_size_child(head, q, i) ((q)->head.p[(i)]? (q)->head.p[(i)]->head.size : 0)
|
|
||||||
|
|
||||||
#define KAVL_HEAD(__type) \
|
|
||||||
struct { \
|
|
||||||
__type *p[2]; \
|
|
||||||
signed char balance; /* balance factor */ \
|
|
||||||
unsigned size; /* #elements in subtree */ \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_FIND(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__scope __type *kavl_find_##suf(const __type *root, const __type *x, unsigned *cnt_) { \
|
|
||||||
const __type *p = root; \
|
|
||||||
unsigned cnt = 0; \
|
|
||||||
while (p != 0) { \
|
|
||||||
int cmp; \
|
|
||||||
cmp = __cmp(x, p); \
|
|
||||||
if (cmp >= 0) cnt += kavl_size_child(__head, p, 0) + 1; \
|
|
||||||
if (cmp < 0) p = p->__head.p[0]; \
|
|
||||||
else if (cmp > 0) p = p->__head.p[1]; \
|
|
||||||
else break; \
|
|
||||||
} \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
return (__type*)p; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_ROTATE(suf, __type, __head) \
|
|
||||||
/* one rotation: (a,(b,c)q)p => ((a,b)p,c)q */ \
|
|
||||||
static inline __type *kavl_rotate1_##suf(__type *p, int dir) { /* dir=0 to left; dir=1 to right */ \
|
|
||||||
int opp = 1 - dir; /* opposite direction */ \
|
|
||||||
__type *q = p->__head.p[opp]; \
|
|
||||||
unsigned size_p = p->__head.size; \
|
|
||||||
p->__head.size -= q->__head.size - kavl_size_child(__head, q, dir); \
|
|
||||||
q->__head.size = size_p; \
|
|
||||||
p->__head.p[opp] = q->__head.p[dir]; \
|
|
||||||
q->__head.p[dir] = p; \
|
|
||||||
return q; \
|
|
||||||
} \
|
|
||||||
/* two consecutive rotations: (a,((b,c)r,d)q)p => ((a,b)p,(c,d)q)r */ \
|
|
||||||
static inline __type *kavl_rotate2_##suf(__type *p, int dir) { \
|
|
||||||
int b1, opp = 1 - dir; \
|
|
||||||
__type *q = p->__head.p[opp], *r = q->__head.p[dir]; \
|
|
||||||
unsigned size_x_dir = kavl_size_child(__head, r, dir); \
|
|
||||||
r->__head.size = p->__head.size; \
|
|
||||||
p->__head.size -= q->__head.size - size_x_dir; \
|
|
||||||
q->__head.size -= size_x_dir + 1; \
|
|
||||||
p->__head.p[opp] = r->__head.p[dir]; \
|
|
||||||
r->__head.p[dir] = p; \
|
|
||||||
q->__head.p[dir] = r->__head.p[opp]; \
|
|
||||||
r->__head.p[opp] = q; \
|
|
||||||
b1 = dir == 0? +1 : -1; \
|
|
||||||
if (r->__head.balance == b1) q->__head.balance = 0, p->__head.balance = -b1; \
|
|
||||||
else if (r->__head.balance == 0) q->__head.balance = p->__head.balance = 0; \
|
|
||||||
else q->__head.balance = b1, p->__head.balance = 0; \
|
|
||||||
r->__head.balance = 0; \
|
|
||||||
return r; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_INSERT(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__scope __type *kavl_insert_##suf(__type **root_, __type *x, unsigned *cnt_) { \
|
|
||||||
unsigned char stack[KAVL_MAX_DEPTH]; \
|
|
||||||
__type *path[KAVL_MAX_DEPTH]; \
|
|
||||||
__type *bp, *bq; \
|
|
||||||
__type *p, *q, *r = 0; /* _r_ is potentially the new root */ \
|
|
||||||
int i, which = 0, top, b1, path_len; \
|
|
||||||
unsigned cnt = 0; \
|
|
||||||
bp = *root_, bq = 0; \
|
|
||||||
/* find the insertion location */ \
|
|
||||||
for (p = bp, q = bq, top = path_len = 0; p; q = p, p = p->__head.p[which]) { \
|
|
||||||
int cmp; \
|
|
||||||
cmp = __cmp(x, p); \
|
|
||||||
if (cmp >= 0) cnt += kavl_size_child(__head, p, 0) + 1; \
|
|
||||||
if (cmp == 0) { \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
return p; \
|
|
||||||
} \
|
|
||||||
if (p->__head.balance != 0) \
|
|
||||||
bq = q, bp = p, top = 0; \
|
|
||||||
stack[top++] = which = (cmp > 0); \
|
|
||||||
path[path_len++] = p; \
|
|
||||||
} \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
x->__head.balance = 0, x->__head.size = 1, x->__head.p[0] = x->__head.p[1] = 0; \
|
|
||||||
if (q == 0) *root_ = x; \
|
|
||||||
else q->__head.p[which] = x; \
|
|
||||||
if (bp == 0) return x; \
|
|
||||||
for (i = 0; i < path_len; ++i) ++path[i]->__head.size; \
|
|
||||||
for (p = bp, top = 0; p != x; p = p->__head.p[stack[top]], ++top) /* update balance factors */ \
|
|
||||||
if (stack[top] == 0) --p->__head.balance; \
|
|
||||||
else ++p->__head.balance; \
|
|
||||||
if (bp->__head.balance > -2 && bp->__head.balance < 2) return x; /* no re-balance needed */ \
|
|
||||||
/* re-balance */ \
|
|
||||||
which = (bp->__head.balance < 0); \
|
|
||||||
b1 = which == 0? +1 : -1; \
|
|
||||||
q = bp->__head.p[1 - which]; \
|
|
||||||
if (q->__head.balance == b1) { \
|
|
||||||
r = kavl_rotate1_##suf(bp, which); \
|
|
||||||
q->__head.balance = bp->__head.balance = 0; \
|
|
||||||
} else r = kavl_rotate2_##suf(bp, which); \
|
|
||||||
if (bq == 0) *root_ = r; \
|
|
||||||
else bq->__head.p[bp != bq->__head.p[0]] = r; \
|
|
||||||
return x; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define __KAVL_ERASE(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__scope __type *kavl_erase_##suf(__type **root_, const __type *x, unsigned *cnt_) { \
|
|
||||||
__type *p, *path[KAVL_MAX_DEPTH], fake; \
|
|
||||||
unsigned char dir[KAVL_MAX_DEPTH]; \
|
|
||||||
int i, d = 0, cmp; \
|
|
||||||
unsigned cnt = 0; \
|
|
||||||
fake.__head.p[0] = *root_, fake.__head.p[1] = 0; \
|
|
||||||
if (cnt_) *cnt_ = 0; \
|
|
||||||
if (x) { \
|
|
||||||
for (cmp = -1, p = &fake; cmp; cmp = __cmp(x, p)) { \
|
|
||||||
int which = (cmp > 0); \
|
|
||||||
if (cmp > 0) cnt += kavl_size_child(__head, p, 0) + 1; \
|
|
||||||
dir[d] = which; \
|
|
||||||
path[d++] = p; \
|
|
||||||
p = p->__head.p[which]; \
|
|
||||||
if (p == 0) { \
|
|
||||||
if (cnt_) *cnt_ = 0; \
|
|
||||||
return 0; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
cnt += kavl_size_child(__head, p, 0) + 1; /* because p==x is not counted */ \
|
|
||||||
} else { \
|
|
||||||
for (p = &fake, cnt = 1; p; p = p->__head.p[0]) \
|
|
||||||
dir[d] = 0, path[d++] = p; \
|
|
||||||
p = path[--d]; \
|
|
||||||
} \
|
|
||||||
if (cnt_) *cnt_ = cnt; \
|
|
||||||
for (i = 1; i < d; ++i) --path[i]->__head.size; \
|
|
||||||
if (p->__head.p[1] == 0) { /* ((1,.)2,3)4 => (1,3)4; p=2 */ \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = p->__head.p[0]; \
|
|
||||||
} else { \
|
|
||||||
__type *q = p->__head.p[1]; \
|
|
||||||
if (q->__head.p[0] == 0) { /* ((1,2)3,4)5 => ((1)2,4)5; p=3 */ \
|
|
||||||
q->__head.p[0] = p->__head.p[0]; \
|
|
||||||
q->__head.balance = p->__head.balance; \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = q; \
|
|
||||||
path[d] = q, dir[d++] = 1; \
|
|
||||||
q->__head.size = p->__head.size - 1; \
|
|
||||||
} else { /* ((1,((.,2)3,4)5)6,7)8 => ((1,(2,4)5)3,7)8; p=6 */ \
|
|
||||||
__type *r; \
|
|
||||||
int e = d++; /* backup _d_ */\
|
|
||||||
for (;;) { \
|
|
||||||
dir[d] = 0; \
|
|
||||||
path[d++] = q; \
|
|
||||||
r = q->__head.p[0]; \
|
|
||||||
if (r->__head.p[0] == 0) break; \
|
|
||||||
q = r; \
|
|
||||||
} \
|
|
||||||
r->__head.p[0] = p->__head.p[0]; \
|
|
||||||
q->__head.p[0] = r->__head.p[1]; \
|
|
||||||
r->__head.p[1] = p->__head.p[1]; \
|
|
||||||
r->__head.balance = p->__head.balance; \
|
|
||||||
path[e-1]->__head.p[dir[e-1]] = r; \
|
|
||||||
path[e] = r, dir[e] = 1; \
|
|
||||||
for (i = e + 1; i < d; ++i) --path[i]->__head.size; \
|
|
||||||
r->__head.size = p->__head.size - 1; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
while (--d > 0) { \
|
|
||||||
__type *q = path[d]; \
|
|
||||||
int which, other, b1 = 1, b2 = 2; \
|
|
||||||
which = dir[d], other = 1 - which; \
|
|
||||||
if (which) b1 = -b1, b2 = -b2; \
|
|
||||||
q->__head.balance += b1; \
|
|
||||||
if (q->__head.balance == b1) break; \
|
|
||||||
else if (q->__head.balance == b2) { \
|
|
||||||
__type *r = q->__head.p[other]; \
|
|
||||||
if (r->__head.balance == -b1) { \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = kavl_rotate2_##suf(q, which); \
|
|
||||||
} else { \
|
|
||||||
path[d-1]->__head.p[dir[d-1]] = kavl_rotate1_##suf(q, which); \
|
|
||||||
if (r->__head.balance == 0) { \
|
|
||||||
r->__head.balance = -b1; \
|
|
||||||
q->__head.balance = b1; \
|
|
||||||
break; \
|
|
||||||
} else r->__head.balance = q->__head.balance = 0; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
*root_ = fake.__head.p[0]; \
|
|
||||||
return p; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define kavl_free(__type, __head, __root, __free) do { \
|
|
||||||
__type *_p, *_q; \
|
|
||||||
for (_p = __root; _p; _p = _q) { \
|
|
||||||
if (_p->__head.p[0] == 0) { \
|
|
||||||
_q = _p->__head.p[1]; \
|
|
||||||
__free(_p); \
|
|
||||||
} else { \
|
|
||||||
_q = _p->__head.p[0]; \
|
|
||||||
_p->__head.p[0] = _q->__head.p[1]; \
|
|
||||||
_q->__head.p[1] = _p; \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define __KAVL_ITR(suf, __scope, __type, __head, __cmp) \
|
|
||||||
struct kavl_itr_##suf { \
|
|
||||||
const __type *stack[KAVL_MAX_DEPTH], **top, *right; /* _right_ points to the right child of *top */ \
|
|
||||||
}; \
|
|
||||||
__scope void kavl_itr_first_##suf(const __type *root, struct kavl_itr_##suf *itr) { \
|
|
||||||
const __type *p; \
|
|
||||||
for (itr->top = itr->stack - 1, p = root; p; p = p->__head.p[0]) \
|
|
||||||
*++itr->top = p; \
|
|
||||||
itr->right = (*itr->top)->__head.p[1]; \
|
|
||||||
} \
|
|
||||||
__scope int kavl_itr_find_##suf(const __type *root, const __type *x, struct kavl_itr_##suf *itr) { \
|
|
||||||
const __type *p = root; \
|
|
||||||
itr->top = itr->stack - 1; \
|
|
||||||
while (p != 0) { \
|
|
||||||
int cmp; \
|
|
||||||
cmp = __cmp(x, p); \
|
|
||||||
if (cmp < 0) *++itr->top = p, p = p->__head.p[0]; \
|
|
||||||
else if (cmp > 0) p = p->__head.p[1]; \
|
|
||||||
else break; \
|
|
||||||
} \
|
|
||||||
if (p) { \
|
|
||||||
*++itr->top = p; \
|
|
||||||
itr->right = p->__head.p[1]; \
|
|
||||||
return 1; \
|
|
||||||
} else if (itr->top >= itr->stack) { \
|
|
||||||
itr->right = (*itr->top)->__head.p[1]; \
|
|
||||||
return 0; \
|
|
||||||
} else return 0; \
|
|
||||||
} \
|
|
||||||
__scope int kavl_itr_next_##suf(struct kavl_itr_##suf *itr) { \
|
|
||||||
for (;;) { \
|
|
||||||
const __type *p; \
|
|
||||||
for (p = itr->right, --itr->top; p; p = p->__head.p[0]) \
|
|
||||||
*++itr->top = p; \
|
|
||||||
if (itr->top < itr->stack) return 0; \
|
|
||||||
itr->right = (*itr->top)->__head.p[1]; \
|
|
||||||
return 1; \
|
|
||||||
} \
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Insert a node to the tree
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param proot pointer to the root of the tree (in/out: root may change)
|
|
||||||
* @param x node to insert (in)
|
|
||||||
* @param cnt number of nodes smaller than or equal to _x_; can be NULL (out)
|
|
||||||
*
|
|
||||||
* @return _x_ if not present in the tree, or the node equal to x.
|
|
||||||
*/
|
|
||||||
#define kavl_insert(suf, proot, x, cnt) kavl_insert_##suf(proot, x, cnt)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Find a node in the tree
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param root root of the tree
|
|
||||||
* @param x node value to find (in)
|
|
||||||
* @param cnt number of nodes smaller than or equal to _x_; can be NULL (out)
|
|
||||||
*
|
|
||||||
* @return node equal to _x_ if present, or NULL if absent
|
|
||||||
*/
|
|
||||||
#define kavl_find(suf, root, x, cnt) kavl_find_##suf(root, x, cnt)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Delete a node from the tree
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param proot pointer to the root of the tree (in/out: root may change)
|
|
||||||
* @param x node value to delete; if NULL, delete the first node (in)
|
|
||||||
*
|
|
||||||
* @return node removed from the tree if present, or NULL if absent
|
|
||||||
*/
|
|
||||||
#define kavl_erase(suf, proot, x, cnt) kavl_erase_##suf(proot, x, cnt)
|
|
||||||
#define kavl_erase_first(suf, proot) kavl_erase_##suf(proot, 0, 0)
|
|
||||||
|
|
||||||
#define kavl_itr_t(suf) struct kavl_itr_##suf
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Place the iterator at the smallest object
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param root root of the tree
|
|
||||||
* @param itr iterator
|
|
||||||
*/
|
|
||||||
#define kavl_itr_first(suf, root, itr) kavl_itr_first_##suf(root, itr)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Place the iterator at the object equal to or greater than the query
|
|
||||||
*
|
|
||||||
* @param suf name suffix used in KAVL_INIT()
|
|
||||||
* @param root root of the tree
|
|
||||||
* @param x query (in)
|
|
||||||
* @param itr iterator (out)
|
|
||||||
*
|
|
||||||
* @return 1 if find; 0 otherwise. kavl_at(itr) is NULL if and only if query is
|
|
||||||
* larger than all objects in the tree
|
|
||||||
*/
|
|
||||||
#define kavl_itr_find(suf, root, x, itr) kavl_itr_find_##suf(root, x, itr)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Move to the next object in order
|
|
||||||
*
|
|
||||||
* @param itr iterator (modified)
|
|
||||||
*
|
|
||||||
* @return 1 if there is a next object; 0 otherwise
|
|
||||||
*/
|
|
||||||
#define kavl_itr_next(suf, itr) kavl_itr_next_##suf(itr)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the pointer at the iterator
|
|
||||||
*
|
|
||||||
* @param itr iterator
|
|
||||||
*
|
|
||||||
* @return pointer if present; NULL otherwise
|
|
||||||
*/
|
|
||||||
#define kavl_at(itr) ((itr)->top < (itr)->stack? 0 : *(itr)->top)
|
|
||||||
|
|
||||||
#define KAVL_INIT2(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_FIND(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_ROTATE(suf, __type, __head) \
|
|
||||||
__KAVL_INSERT(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_ERASE(suf, __scope, __type, __head, __cmp) \
|
|
||||||
__KAVL_ITR(suf, __scope, __type, __head, __cmp)
|
|
||||||
|
|
||||||
#define KAVL_INIT(suf, __type, __head, __cmp) \
|
|
||||||
KAVL_INIT2(suf,, __type, __head, __cmp)
|
|
||||||
|
|
||||||
#endif
|
|
205
tpws/pools.c
205
tpws/pools.c
@ -260,146 +260,121 @@ bool hostlist_collection_is_empty(const struct hostlist_collection_head *head)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int kavl_bit_cmp(const struct kavl_bit_elem *p, const struct kavl_bit_elem *q)
|
void ipset4Destroy(ipset4 **ipset)
|
||||||
{
|
{
|
||||||
unsigned int bitlen = q->bitlen < p->bitlen ? q->bitlen : p->bitlen;
|
ipset4 *elem, *tmp;
|
||||||
unsigned int df = bitlen & 7, bytes = bitlen >> 3;
|
HASH_ITER(hh, *ipset, elem, tmp)
|
||||||
int cmp = memcmp(p->data, q->data, bytes);
|
{
|
||||||
|
HASH_DEL(*ipset, elem);
|
||||||
|
free(elem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
bool ipset4Check(ipset4 *ipset, const struct in_addr *a, uint8_t preflen)
|
||||||
|
{
|
||||||
|
uint32_t ip = ntohl(a->s_addr);
|
||||||
|
struct cidr4 cidr;
|
||||||
|
ipset4 *ips_found;
|
||||||
|
|
||||||
if (cmp || !df) return cmp;
|
// zero alignment bytes
|
||||||
|
memset(&cidr,0,sizeof(cidr));
|
||||||
|
cidr.preflen = preflen+1;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
cidr.preflen--;
|
||||||
|
cidr.addr.s_addr = htonl(ip & mask_from_preflen(cidr.preflen));
|
||||||
|
HASH_FIND(hh, ipset, &cidr, sizeof(cidr), ips_found);
|
||||||
|
if (ips_found) return true;
|
||||||
|
} while(cidr.preflen);
|
||||||
|
|
||||||
uint8_t c1 = p->data[bytes] >> (8 - df);
|
|
||||||
uint8_t c2 = q->data[bytes] >> (8 - df);
|
|
||||||
return c1<c2 ? -1 : c1==c2 ? 0 : 1;
|
|
||||||
}
|
|
||||||
KAVL_INIT(kavl_bit, struct kavl_bit_elem, head, kavl_bit_cmp)
|
|
||||||
static void kavl_bit_destroy_elem(struct kavl_bit_elem *e)
|
|
||||||
{
|
|
||||||
if (e)
|
|
||||||
{
|
|
||||||
free(e->data);
|
|
||||||
free(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void kavl_bit_delete(struct kavl_bit_elem **hdr, const void *data, unsigned int bitlen)
|
|
||||||
{
|
|
||||||
struct kavl_bit_elem temp = {
|
|
||||||
.bitlen = bitlen, .data = (uint8_t*)data
|
|
||||||
};
|
|
||||||
kavl_bit_destroy_elem(kavl_erase(kavl_bit, hdr, &temp, 0));
|
|
||||||
}
|
|
||||||
void kavl_bit_destroy(struct kavl_bit_elem **hdr)
|
|
||||||
{
|
|
||||||
while (*hdr)
|
|
||||||
{
|
|
||||||
struct kavl_bit_elem *e = kavl_erase_first(kavl_bit, hdr);
|
|
||||||
if (!e) break;
|
|
||||||
kavl_bit_destroy_elem(e);
|
|
||||||
}
|
|
||||||
free(*hdr);
|
|
||||||
}
|
|
||||||
struct kavl_bit_elem *kavl_bit_add(struct kavl_bit_elem **hdr, void *data, unsigned int bitlen, size_t struct_size)
|
|
||||||
{
|
|
||||||
if (!struct_size) struct_size=sizeof(struct kavl_bit_elem);
|
|
||||||
|
|
||||||
struct kavl_bit_elem *v, *e = calloc(1, struct_size);
|
|
||||||
if (!e) return 0;
|
|
||||||
|
|
||||||
e->bitlen = bitlen;
|
|
||||||
e->data = data;
|
|
||||||
|
|
||||||
v = kavl_insert(kavl_bit, hdr, e, 0);
|
|
||||||
while (e != v && e->bitlen < v->bitlen)
|
|
||||||
{
|
|
||||||
kavl_bit_delete(hdr, v->data, v->bitlen);
|
|
||||||
v = kavl_insert(kavl_bit, hdr, e, 0);
|
|
||||||
}
|
|
||||||
if (e != v) kavl_bit_destroy_elem(e);
|
|
||||||
return v;
|
|
||||||
}
|
|
||||||
struct kavl_bit_elem *kavl_bit_get(const struct kavl_bit_elem *hdr, const void *data, unsigned int bitlen)
|
|
||||||
{
|
|
||||||
struct kavl_bit_elem temp = {
|
|
||||||
.bitlen = bitlen, .data = (uint8_t*)data
|
|
||||||
};
|
|
||||||
return kavl_find(kavl_bit, hdr, &temp, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool ipset_kavl_add(struct kavl_bit_elem **ipset, const void *a, uint8_t preflen)
|
|
||||||
{
|
|
||||||
uint8_t bytelen = (preflen+7)>>3;
|
|
||||||
uint8_t *abuf = malloc(bytelen);
|
|
||||||
if (!abuf) return false;
|
|
||||||
memcpy(abuf,a,bytelen);
|
|
||||||
if (!kavl_bit_add(ipset,abuf,preflen,0))
|
|
||||||
{
|
|
||||||
free(abuf);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
bool ipset4Add(ipset4 **ipset, const struct in_addr *a, uint8_t preflen)
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool ipset4Check(const struct kavl_bit_elem *ipset, const struct in_addr *a, uint8_t preflen)
|
|
||||||
{
|
|
||||||
return !!kavl_bit_get(ipset,a,preflen);
|
|
||||||
}
|
|
||||||
bool ipset4Add(struct kavl_bit_elem **ipset, const struct in_addr *a, uint8_t preflen)
|
|
||||||
{
|
{
|
||||||
if (preflen>32) return false;
|
if (preflen>32) return false;
|
||||||
return ipset_kavl_add(ipset,a,preflen);
|
|
||||||
}
|
|
||||||
void ipset4Print(struct kavl_bit_elem *ipset)
|
|
||||||
{
|
|
||||||
if (!ipset) return;
|
|
||||||
|
|
||||||
struct cidr4 c;
|
// avoid dups
|
||||||
const struct kavl_bit_elem *elem;
|
if (ipset4Check(*ipset, a, preflen)) return true; // already included
|
||||||
kavl_itr_t(kavl_bit) itr;
|
|
||||||
kavl_itr_first(kavl_bit, ipset, &itr);
|
struct ipset4 *entry = calloc(1,sizeof(ipset4));
|
||||||
do
|
if (!entry) return false;
|
||||||
|
|
||||||
|
entry->cidr.addr.s_addr = htonl(ntohl(a->s_addr) & mask_from_preflen(preflen));
|
||||||
|
entry->cidr.preflen = preflen;
|
||||||
|
oom = false;
|
||||||
|
HASH_ADD(hh, *ipset, cidr, sizeof(entry->cidr), entry);
|
||||||
|
if (oom) { free(entry); return false; }
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
void ipset4Print(ipset4 *ipset)
|
||||||
{
|
{
|
||||||
elem = kavl_at(&itr);
|
ipset4 *ips, *tmp;
|
||||||
c.preflen = elem->bitlen;
|
HASH_ITER(hh, ipset , ips, tmp)
|
||||||
expand_bits(&c.addr, elem->data, elem->bitlen, sizeof(c.addr));
|
{
|
||||||
print_cidr4(&c);
|
print_cidr4(&ips->cidr);
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
while (kavl_itr_next(kavl_bit, &itr));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ipset6Check(const struct kavl_bit_elem *ipset, const struct in6_addr *a, uint8_t preflen)
|
void ipset6Destroy(ipset6 **ipset)
|
||||||
{
|
{
|
||||||
return !!kavl_bit_get(ipset,a,preflen);
|
ipset6 *elem, *tmp;
|
||||||
|
HASH_ITER(hh, *ipset, elem, tmp)
|
||||||
|
{
|
||||||
|
HASH_DEL(*ipset, elem);
|
||||||
|
free(elem);
|
||||||
}
|
}
|
||||||
bool ipset6Add(struct kavl_bit_elem **ipset, const struct in6_addr *a, uint8_t preflen)
|
}
|
||||||
|
bool ipset6Check(ipset6 *ipset, const struct in6_addr *a, uint8_t preflen)
|
||||||
|
{
|
||||||
|
struct cidr6 cidr;
|
||||||
|
ipset6 *ips_found;
|
||||||
|
|
||||||
|
// zero alignment bytes
|
||||||
|
memset(&cidr,0,sizeof(cidr));
|
||||||
|
cidr.preflen = preflen+1;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
cidr.preflen--;
|
||||||
|
ip6_and(a, mask_from_preflen6(cidr.preflen), &cidr.addr);
|
||||||
|
HASH_FIND(hh, ipset, &cidr, sizeof(cidr), ips_found);
|
||||||
|
if (ips_found) return true;
|
||||||
|
} while(cidr.preflen);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
bool ipset6Add(ipset6 **ipset, const struct in6_addr *a, uint8_t preflen)
|
||||||
{
|
{
|
||||||
if (preflen>128) return false;
|
if (preflen>128) return false;
|
||||||
return ipset_kavl_add(ipset,a,preflen);
|
|
||||||
}
|
|
||||||
void ipset6Print(struct kavl_bit_elem *ipset)
|
|
||||||
{
|
|
||||||
if (!ipset) return;
|
|
||||||
|
|
||||||
struct cidr6 c;
|
// avoid dups
|
||||||
const struct kavl_bit_elem *elem;
|
if (ipset6Check(*ipset, a, preflen)) return true; // already included
|
||||||
kavl_itr_t(kavl_bit) itr;
|
|
||||||
kavl_itr_first(kavl_bit, ipset, &itr);
|
struct ipset6 *entry = calloc(1,sizeof(ipset6));
|
||||||
do
|
if (!entry) return false;
|
||||||
|
|
||||||
|
ip6_and(a, mask_from_preflen6(preflen), &entry->cidr.addr);
|
||||||
|
entry->cidr.preflen = preflen;
|
||||||
|
oom = false;
|
||||||
|
HASH_ADD(hh, *ipset, cidr, sizeof(entry->cidr), entry);
|
||||||
|
if (oom) { free(entry); return false; }
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
void ipset6Print(ipset6 *ipset)
|
||||||
{
|
{
|
||||||
elem = kavl_at(&itr);
|
ipset6 *ips, *tmp;
|
||||||
c.preflen = elem->bitlen;
|
HASH_ITER(hh, ipset , ips, tmp)
|
||||||
expand_bits(&c.addr, elem->data, elem->bitlen, sizeof(c.addr));
|
{
|
||||||
print_cidr6(&c);
|
print_cidr6(&ips->cidr);
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
while (kavl_itr_next(kavl_bit, &itr));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ipsetDestroy(ipset *ipset)
|
void ipsetDestroy(ipset *ipset)
|
||||||
{
|
{
|
||||||
kavl_bit_destroy(&ipset->ips4);
|
ipset4Destroy(&ipset->ips4);
|
||||||
kavl_bit_destroy(&ipset->ips6);
|
ipset6Destroy(&ipset->ips6);
|
||||||
}
|
}
|
||||||
void ipsetPrint(ipset *ipset)
|
void ipsetPrint(ipset *ipset)
|
||||||
{
|
{
|
||||||
|
43
tpws/pools.h
43
tpws/pools.h
@ -13,8 +13,6 @@
|
|||||||
#define HASH_FUNCTION HASH_BER
|
#define HASH_FUNCTION HASH_BER
|
||||||
#include "uthash.h"
|
#include "uthash.h"
|
||||||
|
|
||||||
#include "kavl.h"
|
|
||||||
|
|
||||||
#define HOSTLIST_POOL_FLAG_STRICT_MATCH 1
|
#define HOSTLIST_POOL_FLAG_STRICT_MATCH 1
|
||||||
|
|
||||||
typedef struct hostlist_pool {
|
typedef struct hostlist_pool {
|
||||||
@ -78,40 +76,39 @@ struct hostlist_item *hostlist_collection_search(struct hostlist_collection_head
|
|||||||
bool hostlist_collection_is_empty(const struct hostlist_collection_head *head);
|
bool hostlist_collection_is_empty(const struct hostlist_collection_head *head);
|
||||||
|
|
||||||
|
|
||||||
struct kavl_bit_elem
|
typedef struct ipset4 {
|
||||||
{
|
struct cidr4 cidr; /* key */
|
||||||
unsigned int bitlen;
|
UT_hash_handle hh; /* makes this structure hashable */
|
||||||
uint8_t *data;
|
} ipset4;
|
||||||
KAVL_HEAD(struct kavl_bit_elem) head;
|
typedef struct ipset6 {
|
||||||
};
|
struct cidr6 cidr; /* key */
|
||||||
|
UT_hash_handle hh; /* makes this structure hashable */
|
||||||
struct kavl_bit_elem *kavl_bit_get(const struct kavl_bit_elem *hdr, const void *data, unsigned int bitlen);
|
} ipset6;
|
||||||
struct kavl_bit_elem *kavl_bit_add(struct kavl_bit_elem **hdr, void *data, unsigned int bitlen, size_t struct_size);
|
|
||||||
void kavl_bit_delete(struct kavl_bit_elem **hdr, const void *data, unsigned int bitlen);
|
|
||||||
void kavl_bit_destroy(struct kavl_bit_elem **hdr);
|
|
||||||
|
|
||||||
// combined ipset ipv4 and ipv6
|
// combined ipset ipv4 and ipv6
|
||||||
typedef struct ipset {
|
typedef struct ipset {
|
||||||
struct kavl_bit_elem *ips4,*ips6;
|
ipset4 *ips4;
|
||||||
|
ipset6 *ips6;
|
||||||
} ipset;
|
} ipset;
|
||||||
|
|
||||||
#define IPSET_EMPTY(ips) (!(ips)->ips4 && !(ips)->ips6)
|
#define IPSET_EMPTY(ips) (!(ips)->ips4 && !(ips)->ips6)
|
||||||
|
|
||||||
bool ipset4Add(struct kavl_bit_elem **ipset, const struct in_addr *a, uint8_t preflen);
|
void ipset4Destroy(ipset4 **ipset);
|
||||||
static inline bool ipset4AddCidr(struct kavl_bit_elem **ipset, const struct cidr4 *cidr)
|
bool ipset4Add(ipset4 **ipset, const struct in_addr *a, uint8_t preflen);
|
||||||
|
static inline bool ipset4AddCidr(ipset4 **ipset, const struct cidr4 *cidr)
|
||||||
{
|
{
|
||||||
return ipset4Add(ipset,&cidr->addr,cidr->preflen);
|
return ipset4Add(ipset,&cidr->addr,cidr->preflen);
|
||||||
}
|
}
|
||||||
bool ipset4Check(const struct kavl_bit_elem *ipset, const struct in_addr *a, uint8_t preflen);
|
bool ipset4Check(ipset4 *ipset, const struct in_addr *a, uint8_t preflen);
|
||||||
void ipset4Print(struct kavl_bit_elem *ipset);
|
void ipset4Print(ipset4 *ipset);
|
||||||
|
|
||||||
bool ipset6Add(struct kavl_bit_elem **ipset, const struct in6_addr *a, uint8_t preflen);
|
void ipset6Destroy(ipset6 **ipset);
|
||||||
static inline bool ipset6AddCidr(struct kavl_bit_elem **ipset, const struct cidr6 *cidr)
|
bool ipset6Add(ipset6 **ipset, const struct in6_addr *a, uint8_t preflen);
|
||||||
|
static inline bool ipset6AddCidr(ipset6 **ipset, const struct cidr6 *cidr)
|
||||||
{
|
{
|
||||||
return ipset6Add(ipset,&cidr->addr,cidr->preflen);
|
return ipset6Add(ipset,&cidr->addr,cidr->preflen);
|
||||||
}
|
}
|
||||||
bool ipset6Check(const struct kavl_bit_elem *ipset, const struct in6_addr *a, uint8_t preflen);
|
bool ipset6Check(ipset6 *ipset, const struct in6_addr *a, uint8_t preflen);
|
||||||
void ipset6Print(struct kavl_bit_elem *ipset);
|
void ipset6Print(ipset6 *ipset);
|
||||||
|
|
||||||
void ipsetDestroy(ipset *ipset);
|
void ipsetDestroy(ipset *ipset);
|
||||||
void ipsetPrint(ipset *ipset);
|
void ipsetPrint(ipset *ipset);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user