Added support for using external slirp library instead of builtin one.
- Added configure option --enable-using-libslirp to turn on the search for the library. In config.h BX_HAVE_LIBSLIRP is set to 1 if found. The network Makefile is set up accordingly. If not enabled or not found, the builtin slirp code is used. - Moved samba server code and host forwarding code to eth_slirp.cc. - Modified builtin slirp API to reduce differences. - NOTE: IPv6 support present in libslirp is not yet enabled (needs more testing). - TODO: Update builtin slirp code based on libslirp 4.7.0 code.
This commit is contained in:
parent
e1d67d9ce0
commit
0ff2905998
@ -861,6 +861,8 @@ typedef Bit32u bx_phy_address;
|
||||
#define BX_NETMOD_SLIRP 0
|
||||
#define BX_NETMOD_SOCKET 0
|
||||
|
||||
#define BX_HAVE_LIBSLIRP 0
|
||||
|
||||
// Soundcard and gameport support
|
||||
#define BX_SUPPORT_SB16 0
|
||||
#define BX_SUPPORT_ES1370 0
|
||||
|
104
bochs/configure
vendored
104
bochs/configure
vendored
@ -911,6 +911,9 @@ CDROM_OBJS
|
||||
DISPLAY_DLL_TARGETS
|
||||
DISPLAY_EXTRA_OBJS
|
||||
DISPLAY_OBJS
|
||||
SLIRP_LINK_OPTS
|
||||
SLIRP_OBJS2
|
||||
SLIRP_OBJS
|
||||
NETDEV_DLL_TARGETS
|
||||
NETLOW_OBJS
|
||||
NETDEV_OBJS
|
||||
@ -1092,6 +1095,7 @@ enable_win32usbdbg
|
||||
enable_ne2000
|
||||
enable_pnic
|
||||
enable_e1000
|
||||
enable_using_libslirp
|
||||
enable_raw_serial
|
||||
enable_clgd54xx
|
||||
enable_voodoo
|
||||
@ -1836,6 +1840,7 @@ Optional Features:
|
||||
--enable-ne2000 enable NE2000 support (no)
|
||||
--enable-pnic enable PCI pseudo NIC support (no)
|
||||
--enable-e1000 enable Intel(R) Gigabit Ethernet support (no)
|
||||
--enable-using-libslirp enable using libslirp instead of builtin slirp (no)
|
||||
--enable-raw-serial use raw serial port access (no - incomplete)
|
||||
--enable-clgd54xx enable CLGD54XX emulation (no)
|
||||
--enable-voodoo enable 3dfx Voodoo Graphics emulation (no)
|
||||
@ -6170,7 +6175,7 @@ ia64-*-hpux*)
|
||||
;;
|
||||
*-*-irix6*)
|
||||
# Find out which ABI we are using.
|
||||
echo '#line 6173 "configure"' > conftest.$ac_ext
|
||||
echo '#line 6178 "configure"' > conftest.$ac_ext
|
||||
if { { eval echo "\"\$as_me\":${as_lineno-$LINENO}: \"$ac_compile\""; } >&5
|
||||
(eval $ac_compile) 2>&5
|
||||
ac_status=$?
|
||||
@ -7656,11 +7661,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:7659: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:7664: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>conftest.err)
|
||||
ac_status=$?
|
||||
cat conftest.err >&5
|
||||
echo "$as_me:7663: \$? = $ac_status" >&5
|
||||
echo "$as_me:7668: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s "$ac_outfile"; then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
# So say no if there are warnings
|
||||
@ -7890,11 +7895,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:7893: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:7898: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>conftest.err)
|
||||
ac_status=$?
|
||||
cat conftest.err >&5
|
||||
echo "$as_me:7897: \$? = $ac_status" >&5
|
||||
echo "$as_me:7902: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s "$ac_outfile"; then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
# So say no if there are warnings
|
||||
@ -7958,11 +7963,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:7961: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:7966: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>out/conftest.err)
|
||||
ac_status=$?
|
||||
cat out/conftest.err >&5
|
||||
echo "$as_me:7965: \$? = $ac_status" >&5
|
||||
echo "$as_me:7970: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s out/conftest2.$ac_objext
|
||||
then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
@ -9753,7 +9758,7 @@ else
|
||||
lt_dlunknown=0; lt_dlno_uscore=1; lt_dlneed_uscore=2
|
||||
lt_status=$lt_dlunknown
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 9756 "configure"
|
||||
#line 9761 "configure"
|
||||
#include "confdefs.h"
|
||||
|
||||
#if HAVE_DLFCN_H
|
||||
@ -9848,7 +9853,7 @@ else
|
||||
lt_dlunknown=0; lt_dlno_uscore=1; lt_dlneed_uscore=2
|
||||
lt_status=$lt_dlunknown
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 9851 "configure"
|
||||
#line 9856 "configure"
|
||||
#include "confdefs.h"
|
||||
|
||||
#if HAVE_DLFCN_H
|
||||
@ -11966,11 +11971,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:11969: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:11974: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>conftest.err)
|
||||
ac_status=$?
|
||||
cat conftest.err >&5
|
||||
echo "$as_me:11973: \$? = $ac_status" >&5
|
||||
echo "$as_me:11978: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s "$ac_outfile"; then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
# So say no if there are warnings
|
||||
@ -12034,11 +12039,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:12037: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:12042: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>out/conftest.err)
|
||||
ac_status=$?
|
||||
cat out/conftest.err >&5
|
||||
echo "$as_me:12041: \$? = $ac_status" >&5
|
||||
echo "$as_me:12046: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s out/conftest2.$ac_objext
|
||||
then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
@ -13057,7 +13062,7 @@ else
|
||||
lt_dlunknown=0; lt_dlno_uscore=1; lt_dlneed_uscore=2
|
||||
lt_status=$lt_dlunknown
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 13060 "configure"
|
||||
#line 13065 "configure"
|
||||
#include "confdefs.h"
|
||||
|
||||
#if HAVE_DLFCN_H
|
||||
@ -13152,7 +13157,7 @@ else
|
||||
lt_dlunknown=0; lt_dlno_uscore=1; lt_dlneed_uscore=2
|
||||
lt_status=$lt_dlunknown
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 13155 "configure"
|
||||
#line 13160 "configure"
|
||||
#include "confdefs.h"
|
||||
|
||||
#if HAVE_DLFCN_H
|
||||
@ -13972,11 +13977,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:13975: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:13980: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>conftest.err)
|
||||
ac_status=$?
|
||||
cat conftest.err >&5
|
||||
echo "$as_me:13979: \$? = $ac_status" >&5
|
||||
echo "$as_me:13984: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s "$ac_outfile"; then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
# So say no if there are warnings
|
||||
@ -14040,11 +14045,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:14043: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:14048: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>out/conftest.err)
|
||||
ac_status=$?
|
||||
cat out/conftest.err >&5
|
||||
echo "$as_me:14047: \$? = $ac_status" >&5
|
||||
echo "$as_me:14052: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s out/conftest2.$ac_objext
|
||||
then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
@ -16008,11 +16013,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:16011: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:16016: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>conftest.err)
|
||||
ac_status=$?
|
||||
cat conftest.err >&5
|
||||
echo "$as_me:16015: \$? = $ac_status" >&5
|
||||
echo "$as_me:16020: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s "$ac_outfile"; then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
# So say no if there are warnings
|
||||
@ -16242,11 +16247,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:16245: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:16250: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>conftest.err)
|
||||
ac_status=$?
|
||||
cat conftest.err >&5
|
||||
echo "$as_me:16249: \$? = $ac_status" >&5
|
||||
echo "$as_me:16254: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s "$ac_outfile"; then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
# So say no if there are warnings
|
||||
@ -16310,11 +16315,11 @@ else $as_nop
|
||||
-e 's:.*FLAGS}? :&$lt_compiler_flag :; t' \
|
||||
-e 's: [^ ]*conftest\.: $lt_compiler_flag&:; t' \
|
||||
-e 's:$: $lt_compiler_flag:'`
|
||||
(eval echo "\"\$as_me:16313: $lt_compile\"" >&5)
|
||||
(eval echo "\"\$as_me:16318: $lt_compile\"" >&5)
|
||||
(eval "$lt_compile" 2>out/conftest.err)
|
||||
ac_status=$?
|
||||
cat out/conftest.err >&5
|
||||
echo "$as_me:16317: \$? = $ac_status" >&5
|
||||
echo "$as_me:16322: \$? = $ac_status" >&5
|
||||
if (exit $ac_status) && test -s out/conftest2.$ac_objext
|
||||
then
|
||||
# The compiler can only warn and ignore the option if not recognized
|
||||
@ -18105,7 +18110,7 @@ else
|
||||
lt_dlunknown=0; lt_dlno_uscore=1; lt_dlneed_uscore=2
|
||||
lt_status=$lt_dlunknown
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 18108 "configure"
|
||||
#line 18113 "configure"
|
||||
#include "confdefs.h"
|
||||
|
||||
#if HAVE_DLFCN_H
|
||||
@ -18200,7 +18205,7 @@ else
|
||||
lt_dlunknown=0; lt_dlno_uscore=1; lt_dlneed_uscore=2
|
||||
lt_status=$lt_dlunknown
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 18203 "configure"
|
||||
#line 18208 "configure"
|
||||
#include "confdefs.h"
|
||||
|
||||
#if HAVE_DLFCN_H
|
||||
@ -19893,7 +19898,7 @@ else
|
||||
lt_dlunknown=0; lt_dlno_uscore=1; lt_dlneed_uscore=2
|
||||
lt_status=$lt_dlunknown
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 19896 "configure"
|
||||
#line 19901 "configure"
|
||||
#include "confdefs.h"
|
||||
|
||||
#if HAVE_DLFCN_H
|
||||
@ -24855,18 +24860,50 @@ fi
|
||||
|
||||
|
||||
NETLOW_OBJS=''
|
||||
SLIRP_OBJS=''
|
||||
SLIRP_OBJS2=''
|
||||
SLIRP_LINK_OPTS=''
|
||||
if test "$networking" = yes; then
|
||||
NETLOW_OBJS='eth_null.o eth_vnet.o'
|
||||
ethernet_modules='null vnet'
|
||||
can_compile_slirp=0
|
||||
{ printf "%s\n" "$as_me:${as_lineno-$LINENO}: checking for slirp networking support" >&5
|
||||
printf %s "checking for slirp networking support... " >&6; }
|
||||
# Check whether --enable-using-libslirp was given.
|
||||
if test ${enable_using_libslirp+y}
|
||||
then :
|
||||
enableval=$enable_using_libslirp; if test "$enableval" = yes; then
|
||||
if test "x$PKG_CONFIG" != x; then
|
||||
$PKG_CONFIG --exists slirp
|
||||
if test x$? = x0; then
|
||||
can_compile_slirp=1
|
||||
if test "$bx_plugins" = 1; then
|
||||
SLIRP_LINK_OPTS="`$PKG_CONFIG --libs slirp`"
|
||||
else
|
||||
DEVICE_LINK_OPTS="$DEVICE_LINK_OPTS `$PKG_CONFIG --libs slirp`"
|
||||
fi
|
||||
{ printf "%s\n" "$as_me:${as_lineno-$LINENO}: result: libslirp" >&5
|
||||
printf "%s\n" "libslirp" >&6; }
|
||||
printf "%s\n" "#define BX_HAVE_LIBSLIRP 1" >>confdefs.h
|
||||
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
|
||||
if test $can_compile_slirp = 0; then
|
||||
{ printf "%s\n" "$as_me:${as_lineno-$LINENO}: result: builtin" >&5
|
||||
printf "%s\n" "builtin" >&6; }
|
||||
case "$target" in
|
||||
*-cygwin* | *-mingw32* | *-msys)
|
||||
can_compile_slirp=1
|
||||
DEVICE_LINK_OPTS="$DEVICE_LINK_OPTS -liphlpapi"
|
||||
SLIRP_OBJS='$(SLIRP_OBJS)'
|
||||
;;
|
||||
*-pc-windows* | *-pc-winnt*)
|
||||
can_compile_slirp=1
|
||||
DEVICE_LINK_OPTS="$DEVICE_LINK_OPTS iphlpapi.lib"
|
||||
SLIRP_OBJS='$(SLIRP_OBJS)'
|
||||
;;
|
||||
*)
|
||||
ac_fn_c_check_header_compile "$LINENO" "netinet/in.h" "ac_cv_header_netinet_in_h" "$ac_includes_default"
|
||||
@ -24875,8 +24912,16 @@ then :
|
||||
can_compile_slirp=1
|
||||
fi
|
||||
|
||||
if test $can_compile_slirp = 1; then
|
||||
if test "$bx_plugins" = 1; then
|
||||
SLIRP_OBJS2='$(SLIRP_OBJS:.o=.lo)'
|
||||
else
|
||||
SLIRP_OBJS='$(SLIRP_OBJS)'
|
||||
fi
|
||||
fi
|
||||
;;
|
||||
esac
|
||||
fi
|
||||
if test $can_compile_slirp = 1; then
|
||||
NETLOW_OBJS="$NETLOW_OBJS eth_slirp.o"
|
||||
ethernet_modules="$ethernet_modules slirp"
|
||||
@ -25010,6 +25055,9 @@ fi
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
if test "$bx_plugins" = 0; then
|
||||
case $target in
|
||||
*-pc-windows*)
|
||||
|
@ -1759,23 +1759,56 @@ AC_ARG_ENABLE(e1000,
|
||||
)
|
||||
|
||||
NETLOW_OBJS=''
|
||||
SLIRP_OBJS=''
|
||||
SLIRP_OBJS2=''
|
||||
SLIRP_LINK_OPTS=''
|
||||
if test "$networking" = yes; then
|
||||
NETLOW_OBJS='eth_null.o eth_vnet.o'
|
||||
ethernet_modules='null vnet'
|
||||
can_compile_slirp=0
|
||||
AC_MSG_CHECKING(for slirp networking support)
|
||||
AC_ARG_ENABLE(using-libslirp,
|
||||
AS_HELP_STRING([--enable-using-libslirp], [enable using libslirp instead of builtin slirp (no)]),
|
||||
[if test "$enableval" = yes; then
|
||||
if test "x$PKG_CONFIG" != x; then
|
||||
$PKG_CONFIG --exists slirp
|
||||
if test x$? = x0; then
|
||||
can_compile_slirp=1
|
||||
if test "$bx_plugins" = 1; then
|
||||
SLIRP_LINK_OPTS="`$PKG_CONFIG --libs slirp`"
|
||||
else
|
||||
DEVICE_LINK_OPTS="$DEVICE_LINK_OPTS `$PKG_CONFIG --libs slirp`"
|
||||
fi
|
||||
AC_MSG_RESULT(libslirp)
|
||||
AC_DEFINE(BX_HAVE_LIBSLIRP, 1)
|
||||
fi
|
||||
fi
|
||||
fi], [])
|
||||
if test $can_compile_slirp = 0; then
|
||||
AC_MSG_RESULT(builtin)
|
||||
case "$target" in
|
||||
*-cygwin* | *-mingw32* | *-msys)
|
||||
can_compile_slirp=1
|
||||
DEVICE_LINK_OPTS="$DEVICE_LINK_OPTS -liphlpapi"
|
||||
SLIRP_OBJS='$(SLIRP_OBJS)'
|
||||
;;
|
||||
*-pc-windows* | *-pc-winnt*)
|
||||
can_compile_slirp=1
|
||||
DEVICE_LINK_OPTS="$DEVICE_LINK_OPTS iphlpapi.lib"
|
||||
SLIRP_OBJS='$(SLIRP_OBJS)'
|
||||
;;
|
||||
*)
|
||||
AC_CHECK_HEADER([netinet/in.h], [ can_compile_slirp=1 ], [])
|
||||
AC_CHECK_HEADER([netinet/in.h],[can_compile_slirp=1])
|
||||
if test $can_compile_slirp = 1; then
|
||||
if test "$bx_plugins" = 1; then
|
||||
SLIRP_OBJS2='$(SLIRP_OBJS:.o=.lo)'
|
||||
else
|
||||
SLIRP_OBJS='$(SLIRP_OBJS)'
|
||||
fi
|
||||
fi
|
||||
;;
|
||||
esac
|
||||
fi
|
||||
if test $can_compile_slirp = 1; then
|
||||
NETLOW_OBJS="$NETLOW_OBJS eth_slirp.o"
|
||||
ethernet_modules="$ethernet_modules slirp"
|
||||
@ -1864,6 +1897,9 @@ fi
|
||||
AC_SUBST(NETDEV_OBJS)
|
||||
AC_SUBST(NETLOW_OBJS)
|
||||
AC_SUBST(NETDEV_DLL_TARGETS)
|
||||
AC_SUBST(SLIRP_OBJS)
|
||||
AC_SUBST(SLIRP_OBJS2)
|
||||
AC_SUBST(SLIRP_LINK_OPTS)
|
||||
|
||||
|
||||
dnl // serial mode 'socket' needs ws2_32.dll in non-plugin mode
|
||||
|
@ -1,4 +1,4 @@
|
||||
# Copyright (C) 2012-2021 The Bochs Project
|
||||
# Copyright (C) 2012-2024 The Bochs Project
|
||||
#
|
||||
# This library is free software; you can redistribute it and/or
|
||||
# modify it under the terms of the GNU Lesser General Public
|
||||
@ -87,8 +87,8 @@ OBJS_THAT_CAN_BE_PLUGINS = \
|
||||
$(NETLOW_OBJS)
|
||||
|
||||
OBJS_THAT_SUPPORT_OTHER_PLUGINS = \
|
||||
netutil.o \
|
||||
$(SLIRP_OBJS)
|
||||
@SLIRP_OBJS@ \
|
||||
netutil.o
|
||||
|
||||
NONPLUGIN_OBJS = @IODEV_EXT_NON_PLUGIN_OBJS@
|
||||
PLUGIN_OBJS = @IODEV_EXT_PLUGIN_OBJS@
|
||||
@ -120,8 +120,8 @@ libbx_%.la: %.lo
|
||||
$(LIBTOOL) --mode=link --tag CXX $(CXX) $(LDFLAGS) -module $< -o $@ -rpath $(PLUGIN_PATH)
|
||||
|
||||
# special link rules for plugins that require more than one object file
|
||||
libbx_eth_slirp.la: eth_slirp.lo $(SLIRP_OBJS:.o=.lo)
|
||||
$(LIBTOOL) --mode=link --tag CXX $(CXX) $(LDFLAGS) -module eth_slirp.lo $(SLIRP_OBJS:.o=.lo) -o libbx_eth_slirp.la -rpath $(PLUGIN_PATH)
|
||||
libbx_eth_slirp.la: eth_slirp.lo @SLIRP_OBJS2@
|
||||
$(LIBTOOL) --mode=link --tag CXX $(CXX) $(LDFLAGS) -module eth_slirp.lo @SLIRP_OBJS2@ -o libbx_eth_slirp.la -rpath $(PLUGIN_PATH) @SLIRP_LINK_OPTS@
|
||||
|
||||
libbx_eth_vnet.la: eth_vnet.lo netutil.lo
|
||||
$(LIBTOOL) --mode=link --tag CXX $(CXX) $(LDFLAGS) -module eth_vnet.lo netutil.lo -o libbx_eth_vnet.la -rpath $(PLUGIN_PATH)
|
||||
@ -134,8 +134,8 @@ bx_%.dll: %.o
|
||||
bx_eth_null.dll: eth_null.o
|
||||
@LINK_DLL@ eth_null.o $(WIN32_DLL_IMPORT_LIBRARY)
|
||||
|
||||
bx_eth_slirp.dll: eth_slirp.o $(SLIRP_OBJS)
|
||||
@LINK_DLL@ eth_slirp.o $(SLIRP_OBJS) $(WIN32_DLL_IMPORT_LIBRARY) $(NETMOD_LINK_OPTS@LINK_VAR@)
|
||||
bx_eth_slirp.dll: eth_slirp.o @SLIRP_OBJS@
|
||||
@LINK_DLL@ eth_slirp.o @SLIRP_OBJS@ $(WIN32_DLL_IMPORT_LIBRARY) $(NETMOD_LINK_OPTS@LINK_VAR@)
|
||||
|
||||
bx_eth_socket.dll: eth_socket.o
|
||||
@LINK_DLL@ eth_socket.o $(WIN32_DLL_IMPORT_LIBRARY) $(NETMOD_LINK_OPTS@LINK_VAR@)
|
||||
|
@ -1,8 +1,6 @@
|
||||
/////////////////////////////////////////////////////////////////////////
|
||||
// $Id$
|
||||
/////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// Copyright (C) 2014-2021 The Bochs Project
|
||||
// Copyright (C) 2014-2024 The Bochs Project
|
||||
//
|
||||
// This library is free software; you can redistribute it and/or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
@ -35,8 +33,13 @@
|
||||
|
||||
#if BX_NETWORKING && BX_NETMOD_SLIRP
|
||||
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
#include <slirp/libslirp.h>
|
||||
#include <signal.h>
|
||||
#else
|
||||
#include "slirp/slirp.h"
|
||||
#include "slirp/libslirp.h"
|
||||
#endif
|
||||
|
||||
static unsigned int bx_slirp_instances = 0;
|
||||
|
||||
@ -60,13 +63,6 @@ static int rx_timer_index = BX_NULL_TIMER_HANDLE;
|
||||
fd_set rfds, wfds, xfds;
|
||||
int nfds;
|
||||
|
||||
extern int slirp_hostfwd(Slirp *s, const char *redir_str, int legacy_format);
|
||||
#ifndef WIN32
|
||||
extern int slirp_smb(Slirp *s, char *smb_tmpdir, const char *exported_dir,
|
||||
struct in_addr vserver_addr);
|
||||
void slirp_smb_cleanup(Slirp *s, char *smb_tmpdir);
|
||||
#endif
|
||||
|
||||
class bx_slirp_pktmover_c : public eth_pktmover_c {
|
||||
public:
|
||||
bx_slirp_pktmover_c(const char *netif, const char *macaddr,
|
||||
@ -80,9 +76,7 @@ private:
|
||||
Slirp *slirp;
|
||||
unsigned netdev_speed;
|
||||
|
||||
int restricted;
|
||||
struct in_addr net, mask, host, dhcp, dns;
|
||||
char *bootfile, *hostname, **dnssearch;
|
||||
SlirpConfig config;
|
||||
char *hostfwd[MAX_HOSTFWD];
|
||||
int n_hostfwd;
|
||||
#ifndef WIN32
|
||||
@ -95,6 +89,14 @@ private:
|
||||
|
||||
bool parse_slirp_conf(const char *conf);
|
||||
static void rx_timer_handler(void *);
|
||||
void rx_timer(void);
|
||||
|
||||
#ifndef WIN32
|
||||
int slirp_smb(Slirp *s, char *smb_tmpdir, const char *exported_dir,
|
||||
struct in_addr vserver_addr);
|
||||
void slirp_smb_cleanup(Slirp *s, char *smb_tmpdir);
|
||||
#endif
|
||||
int slirp_hostfwd(Slirp *s, const char *redir_str, int legacy_format);
|
||||
};
|
||||
|
||||
class bx_slirp_locator_c : public eth_locator_c {
|
||||
@ -109,6 +111,158 @@ protected:
|
||||
} bx_slirp_match;
|
||||
|
||||
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
static slirp_ssize_t send_packet(const void *buf, size_t len, void *opaque);
|
||||
|
||||
static void guest_error(const char *msg, void *opaque)
|
||||
{
|
||||
fprintf(stderr, "guest_error\n");
|
||||
}
|
||||
|
||||
static int64_t clock_get_ns(void *opaque)
|
||||
{
|
||||
return bx_pc_system.time_usec() * 1000;
|
||||
}
|
||||
|
||||
static int npoll;
|
||||
|
||||
static void register_poll_fd(int fd, void *opaque)
|
||||
{
|
||||
npoll++;
|
||||
}
|
||||
|
||||
static void unregister_poll_fd(int fd, void *opaque)
|
||||
{
|
||||
npoll--;
|
||||
}
|
||||
|
||||
static void notify(void *opaque)
|
||||
{
|
||||
// Nothing here yet
|
||||
}
|
||||
|
||||
static struct SlirpCb callbacks = {
|
||||
.send_packet = send_packet,
|
||||
.guest_error = guest_error,
|
||||
.clock_get_ns = clock_get_ns,
|
||||
.register_poll_fd = register_poll_fd,
|
||||
.unregister_poll_fd = unregister_poll_fd,
|
||||
.notify = notify,
|
||||
};
|
||||
#endif
|
||||
|
||||
bx_slirp_pktmover_c::bx_slirp_pktmover_c(const char *netif,
|
||||
const char *macaddr,
|
||||
eth_rx_handler_t rxh,
|
||||
eth_rx_status_t rxstat,
|
||||
logfunctions *netdev,
|
||||
const char *script)
|
||||
{
|
||||
#if !BX_HAVE_LIBSLIRP
|
||||
logfunctions *slirplog;
|
||||
char prefix[10];
|
||||
#endif
|
||||
|
||||
slirp = NULL;
|
||||
pktlog_fn = NULL;
|
||||
n_hostfwd = 0;
|
||||
/* default settings according to historic slirp */
|
||||
memset(&config, 0, sizeof(config));
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
config.version = 4;
|
||||
config.in_enabled = true;
|
||||
config.disable_host_loopback = false;
|
||||
config.enable_emu = false;
|
||||
config.disable_dns = false;
|
||||
config.tftp_server_name = "tftp";
|
||||
#endif
|
||||
config.restricted = false;
|
||||
config.vnetwork.s_addr = htonl(0x0a000200); /* 10.0.2.0 */
|
||||
config.vnetmask.s_addr = htonl(0xffffff00); /* 255.255.255.0 */
|
||||
config.vhost.s_addr = htonl(0x0a000202); /* 10.0.2.2 */
|
||||
config.vdhcp_start.s_addr = htonl(0x0a00020f); /* 10.0.2.15 */
|
||||
config.vnameserver.s_addr = htonl(0x0a000203); /* 10.0.2.3 */
|
||||
config.tftp_path = netif;
|
||||
#ifndef WIN32
|
||||
smb_export = NULL;
|
||||
smb_tmpdir = NULL;
|
||||
smb_srv.s_addr = 0;
|
||||
#endif
|
||||
|
||||
this->netdev = netdev;
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
BX_INFO(("slirp network driver (libslirp version %s)", slirp_version_string()));
|
||||
#else
|
||||
if (sizeof(struct arphdr) != 28) {
|
||||
BX_FATAL(("system error: invalid ARP header structure size"));
|
||||
}
|
||||
BX_INFO(("slirp network driver"));
|
||||
#endif
|
||||
|
||||
this->rxh = rxh;
|
||||
this->rxstat = rxstat;
|
||||
Bit32u status = this->rxstat(this->netdev) & BX_NETDEV_SPEED;
|
||||
this->netdev_speed = (status == BX_NETDEV_1GBIT) ? 1000 :
|
||||
(status == BX_NETDEV_100MBIT) ? 100 : 10;
|
||||
if (bx_slirp_instances == 0) {
|
||||
rx_timer_index =
|
||||
DEV_register_timer(this, this->rx_timer_handler, 1000, 1, 1,
|
||||
"eth_slirp");
|
||||
#ifndef WIN32
|
||||
signal(SIGPIPE, SIG_IGN);
|
||||
#endif
|
||||
}
|
||||
|
||||
if ((strlen(script) > 0) && (strcmp(script, "none"))) {
|
||||
if (!parse_slirp_conf(script)) {
|
||||
BX_ERROR(("reading slirp config failed"));
|
||||
}
|
||||
}
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
slirp = slirp_new(&config, &callbacks, this);
|
||||
#else
|
||||
slirplog = new logfunctions();
|
||||
sprintf(prefix, "SLIRP%d", bx_slirp_instances);
|
||||
slirplog->put(prefix);
|
||||
slirp = slirp_new(&config, this, slirplog);
|
||||
#endif
|
||||
if (n_hostfwd > 0) {
|
||||
for (int i = 0; i < n_hostfwd; i++) {
|
||||
slirp_hostfwd(slirp, hostfwd[i], 0);
|
||||
}
|
||||
}
|
||||
#ifndef WIN32
|
||||
if (smb_export != NULL) {
|
||||
smb_tmpdir = (char*)malloc(128);
|
||||
if (slirp_smb(slirp, smb_tmpdir, smb_export, smb_srv) < 0) {
|
||||
BX_ERROR(("failed to initialize SMB support"));
|
||||
}
|
||||
}
|
||||
#endif
|
||||
if (pktlog_fn != NULL) {
|
||||
pktlog_txt = fopen(pktlog_fn, "wb");
|
||||
slirp_logging = (pktlog_txt != NULL);
|
||||
if (slirp_logging) {
|
||||
fprintf(pktlog_txt, "slirp packetmover readable log file\n");
|
||||
if (strlen(netif) > 0) {
|
||||
fprintf(pktlog_txt, "TFTP root = %s\n", netif);
|
||||
} else {
|
||||
fprintf(pktlog_txt, "TFTP service disabled\n");
|
||||
}
|
||||
fprintf(pktlog_txt, "guest MAC address = ");
|
||||
int i;
|
||||
for (i=0; i<6; i++)
|
||||
fprintf(pktlog_txt, "%02x%s", 0xff & macaddr[i], i<5?":" : "\n");
|
||||
fprintf(pktlog_txt, "--\n");
|
||||
fflush(pktlog_txt);
|
||||
}
|
||||
free(pktlog_fn);
|
||||
} else {
|
||||
slirp_logging = 0;
|
||||
}
|
||||
bx_slirp_instances++;
|
||||
}
|
||||
|
||||
bx_slirp_pktmover_c::~bx_slirp_pktmover_c()
|
||||
{
|
||||
if (slirp != NULL) {
|
||||
@ -120,14 +274,14 @@ bx_slirp_pktmover_c::~bx_slirp_pktmover_c()
|
||||
free(smb_export);
|
||||
}
|
||||
#endif
|
||||
if (bootfile != NULL) free(bootfile);
|
||||
if (hostname != NULL) free(hostname);
|
||||
if (dnssearch != NULL) {
|
||||
if (config.bootfile != NULL) free((void*)config.bootfile);
|
||||
if (config.vhostname != NULL) free((void*)config.vhostname);
|
||||
if (config.vdnssearch != NULL) {
|
||||
size_t i = 0;
|
||||
while (dnssearch[i] != NULL) {
|
||||
free(dnssearch[i++]);
|
||||
while (config.vdnssearch[i] != NULL) {
|
||||
free((void*)config.vdnssearch[i++]);
|
||||
}
|
||||
free(dnssearch);
|
||||
free((void*)config.vdnssearch);
|
||||
}
|
||||
while (n_hostfwd > 0) {
|
||||
free(hostfwd[--n_hostfwd]);
|
||||
@ -187,18 +341,18 @@ bool bx_slirp_pktmover_c::parse_slirp_conf(const char *conf)
|
||||
len2 = strip_whitespace(val);
|
||||
if ((len1 == 0) || (len2 == 0)) continue;
|
||||
if (!stricmp(param, "restricted")) {
|
||||
restricted = atoi(val);
|
||||
config.restricted = atoi(val);
|
||||
} else if (!stricmp(param, "hostname")) {
|
||||
if (len2 < 33) {
|
||||
hostname = (char*)malloc(len2+1);
|
||||
strcpy(hostname, val);
|
||||
config.vhostname = (char*)malloc(len2+1);
|
||||
strcpy((char*)config.vhostname, val);
|
||||
} else {
|
||||
BX_ERROR(("slirp: wrong format for 'hostname'"));
|
||||
}
|
||||
} else if (!stricmp(param, "bootfile")) {
|
||||
if (len2 < 128) {
|
||||
bootfile = (char*)malloc(len2+1);
|
||||
strcpy(bootfile, val);
|
||||
config.bootfile = (char*)malloc(len2+1);
|
||||
strcpy((char*)config.bootfile, val);
|
||||
} else {
|
||||
BX_ERROR(("slirp: wrong format for 'bootfile'"));
|
||||
}
|
||||
@ -208,38 +362,38 @@ bool bx_slirp_pktmover_c::parse_slirp_conf(const char *conf)
|
||||
for (i = 0; i < len2; i++) {
|
||||
if (val[i] == ',') count++;
|
||||
}
|
||||
dnssearch = (char**)malloc((count + 1) * sizeof(char*));
|
||||
config.vdnssearch = (const char**)malloc((count + 1) * sizeof(char*));
|
||||
i = 0;
|
||||
tmp = strtok(val, ",");
|
||||
while (tmp != NULL) {
|
||||
len2 = strip_whitespace(tmp);
|
||||
dnssearch[i] = (char*)malloc(len2+1);
|
||||
strcpy(dnssearch[i], tmp);
|
||||
config.vdnssearch[i] = (char*)malloc(len2+1);
|
||||
strcpy((char*)config.vdnssearch[i], tmp);
|
||||
i++;
|
||||
tmp = strtok(NULL, ",");
|
||||
}
|
||||
dnssearch[i] = NULL;
|
||||
config.vdnssearch[i] = NULL;
|
||||
} else {
|
||||
BX_ERROR(("slirp: wrong format for 'dnssearch'"));
|
||||
}
|
||||
} else if (!stricmp(param, "net")) {
|
||||
if (!inet_aton(val, &net)) {
|
||||
if (!inet_aton(val, &config.vnetwork)) {
|
||||
BX_ERROR(("slirp: wrong format for 'net'"));
|
||||
}
|
||||
} else if (!stricmp(param, "mask")) {
|
||||
if (!inet_aton(val, &mask)) {
|
||||
if (!inet_aton(val, &config.vnetmask)) {
|
||||
BX_ERROR(("slirp: wrong format for 'mask'"));
|
||||
}
|
||||
} else if (!stricmp(param, "host")) {
|
||||
if (!inet_aton(val, &host)) {
|
||||
if (!inet_aton(val, &config.vhost)) {
|
||||
BX_ERROR(("slirp: wrong format for 'host'"));
|
||||
}
|
||||
} else if (!stricmp(param, "dhcpstart")) {
|
||||
if (!inet_aton(val, &dhcp)) {
|
||||
if (!inet_aton(val, &config.vdhcp_start)) {
|
||||
BX_ERROR(("slirp: wrong format for 'dhcpstart'"));
|
||||
}
|
||||
} else if (!stricmp(param, "dns")) {
|
||||
if (!inet_aton(val, &dns)) {
|
||||
if (!inet_aton(val, &config.vnameserver)) {
|
||||
BX_ERROR(("slirp: wrong format for 'dns'"));
|
||||
}
|
||||
#ifndef WIN32
|
||||
@ -284,102 +438,6 @@ bool bx_slirp_pktmover_c::parse_slirp_conf(const char *conf)
|
||||
return 1;
|
||||
}
|
||||
|
||||
bx_slirp_pktmover_c::bx_slirp_pktmover_c(const char *netif,
|
||||
const char *macaddr,
|
||||
eth_rx_handler_t rxh,
|
||||
eth_rx_status_t rxstat,
|
||||
logfunctions *netdev,
|
||||
const char *script)
|
||||
{
|
||||
logfunctions *slirplog;
|
||||
char prefix[10];
|
||||
|
||||
restricted = 0;
|
||||
slirp = NULL;
|
||||
hostname = NULL;
|
||||
bootfile = NULL;
|
||||
dnssearch = NULL;
|
||||
pktlog_fn = NULL;
|
||||
n_hostfwd = 0;
|
||||
/* default settings according to historic slirp */
|
||||
net.s_addr = htonl(0x0a000200); /* 10.0.2.0 */
|
||||
mask.s_addr = htonl(0xffffff00); /* 255.255.255.0 */
|
||||
host.s_addr = htonl(0x0a000202); /* 10.0.2.2 */
|
||||
dhcp.s_addr = htonl(0x0a00020f); /* 10.0.2.15 */
|
||||
dns.s_addr = htonl(0x0a000203); /* 10.0.2.3 */
|
||||
#ifndef WIN32
|
||||
smb_export = NULL;
|
||||
smb_tmpdir = NULL;
|
||||
smb_srv.s_addr = 0;
|
||||
#endif
|
||||
|
||||
this->netdev = netdev;
|
||||
if (sizeof(struct arphdr) != 28) {
|
||||
BX_FATAL(("system error: invalid ARP header structure size"));
|
||||
}
|
||||
BX_INFO(("slirp network driver"));
|
||||
|
||||
this->rxh = rxh;
|
||||
this->rxstat = rxstat;
|
||||
Bit32u status = this->rxstat(this->netdev) & BX_NETDEV_SPEED;
|
||||
this->netdev_speed = (status == BX_NETDEV_1GBIT) ? 1000 :
|
||||
(status == BX_NETDEV_100MBIT) ? 100 : 10;
|
||||
if (bx_slirp_instances == 0) {
|
||||
rx_timer_index =
|
||||
DEV_register_timer(this, this->rx_timer_handler, 1000, 1, 1,
|
||||
"eth_slirp");
|
||||
#ifndef WIN32
|
||||
signal(SIGPIPE, SIG_IGN);
|
||||
#endif
|
||||
}
|
||||
|
||||
if ((strlen(script) > 0) && (strcmp(script, "none"))) {
|
||||
if (!parse_slirp_conf(script)) {
|
||||
BX_ERROR(("reading slirp config failed"));
|
||||
}
|
||||
}
|
||||
slirplog = new logfunctions();
|
||||
sprintf(prefix, "SLIRP%d", bx_slirp_instances);
|
||||
slirplog->put(prefix);
|
||||
slirp = slirp_init(restricted, net, mask, host, hostname, netif, bootfile, dhcp, dns,
|
||||
(const char**)dnssearch, this, slirplog);
|
||||
if (n_hostfwd > 0) {
|
||||
for (int i = 0; i < n_hostfwd; i++) {
|
||||
slirp_hostfwd(slirp, hostfwd[i], 0);
|
||||
}
|
||||
}
|
||||
#ifndef WIN32
|
||||
if (smb_export != NULL) {
|
||||
smb_tmpdir = (char*)malloc(128);
|
||||
if (slirp_smb(slirp, smb_tmpdir, smb_export, smb_srv) < 0) {
|
||||
BX_ERROR(("failed to initialize SMB support"));
|
||||
}
|
||||
}
|
||||
#endif
|
||||
if (pktlog_fn != NULL) {
|
||||
pktlog_txt = fopen(pktlog_fn, "wb");
|
||||
slirp_logging = (pktlog_txt != NULL);
|
||||
if (slirp_logging) {
|
||||
fprintf(pktlog_txt, "slirp packetmover readable log file\n");
|
||||
if (strlen(netif) > 0) {
|
||||
fprintf(pktlog_txt, "TFTP root = %s\n", netif);
|
||||
} else {
|
||||
fprintf(pktlog_txt, "TFTP service disabled\n");
|
||||
}
|
||||
fprintf(pktlog_txt, "guest MAC address = ");
|
||||
int i;
|
||||
for (i=0; i<6; i++)
|
||||
fprintf(pktlog_txt, "%02x%s", 0xff & macaddr[i], i<5?":" : "\n");
|
||||
fprintf(pktlog_txt, "--\n");
|
||||
fflush(pktlog_txt);
|
||||
}
|
||||
free(pktlog_fn);
|
||||
} else {
|
||||
slirp_logging = 0;
|
||||
}
|
||||
bx_slirp_instances++;
|
||||
}
|
||||
|
||||
void bx_slirp_pktmover_c::sendpkt(void *buf, unsigned io_len)
|
||||
{
|
||||
if (slirp_logging) {
|
||||
@ -389,6 +447,39 @@ void bx_slirp_pktmover_c::sendpkt(void *buf, unsigned io_len)
|
||||
}
|
||||
|
||||
void bx_slirp_pktmover_c::rx_timer_handler(void *this_ptr)
|
||||
{
|
||||
((bx_slirp_pktmover_c*)this_ptr)->rx_timer();
|
||||
}
|
||||
|
||||
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
static int add_poll_cb(int fd, int events, void *opaque)
|
||||
{
|
||||
if (events & SLIRP_POLL_IN)
|
||||
FD_SET(fd, &rfds);
|
||||
if (events & SLIRP_POLL_OUT)
|
||||
FD_SET(fd, &wfds);
|
||||
if (events & SLIRP_POLL_PRI)
|
||||
FD_SET(fd, &xfds);
|
||||
if (nfds < fd)
|
||||
nfds = fd;
|
||||
return fd;
|
||||
}
|
||||
|
||||
static int get_revents_cb(int idx, void *opaque)
|
||||
{
|
||||
int event = 0;
|
||||
if (FD_ISSET(idx, &rfds))
|
||||
event |= SLIRP_POLL_IN;
|
||||
if (FD_ISSET(idx, &wfds))
|
||||
event |= SLIRP_POLL_OUT;
|
||||
if (FD_ISSET(idx, &xfds))
|
||||
event |= SLIRP_POLL_PRI;
|
||||
return event;
|
||||
}
|
||||
#endif
|
||||
|
||||
void bx_slirp_pktmover_c::rx_timer(void)
|
||||
{
|
||||
Bit32u timeout = 0;
|
||||
int ret;
|
||||
@ -402,11 +493,19 @@ void bx_slirp_pktmover_c::rx_timer_handler(void *this_ptr)
|
||||
FD_ZERO(&rfds);
|
||||
FD_ZERO(&wfds);
|
||||
FD_ZERO(&xfds);
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
slirp_pollfds_fill(slirp, &timeout, add_poll_cb, this);
|
||||
#else
|
||||
slirp_select_fill(&nfds, &rfds, &wfds, &xfds, &timeout);
|
||||
#endif
|
||||
tv.tv_sec = 0;
|
||||
tv.tv_usec = 0;
|
||||
ret = select(nfds + 1, &rfds, &wfds, &xfds, &tv);
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
slirp_pollfds_poll(slirp, (ret < 0), get_revents_cb, this);
|
||||
#else
|
||||
slirp_select_poll(&rfds, &wfds, &xfds, (ret < 0));
|
||||
#endif
|
||||
}
|
||||
|
||||
int slirp_can_output(void *this_ptr)
|
||||
@ -439,4 +538,223 @@ void bx_slirp_pktmover_c::receive(void *pkt, unsigned pkt_len)
|
||||
}
|
||||
}
|
||||
|
||||
#if BX_HAVE_LIBSLIRP
|
||||
static slirp_ssize_t send_packet(const void *buf, size_t len, void *opaque)
|
||||
{
|
||||
bx_slirp_pktmover_c *class_ptr = (bx_slirp_pktmover_c *)opaque;
|
||||
class_ptr->receive((void*)buf, len);
|
||||
return len;
|
||||
}
|
||||
#endif
|
||||
|
||||
#if !defined(_WIN32) && !defined(__CYGWIN__)
|
||||
|
||||
#define CONFIG_SMBD_COMMAND "/usr/sbin/smbd"
|
||||
|
||||
#include <pwd.h>
|
||||
|
||||
/* automatic user mode samba server configuration */
|
||||
int bx_slirp_pktmover_c::slirp_smb(Slirp *s, char *smb_tmpdir, const char *exported_dir,
|
||||
struct in_addr vserver_addr)
|
||||
{
|
||||
static int instance;
|
||||
int i;
|
||||
char smb_conf[128], smb_cmdline[150];
|
||||
char share[64], error_msg[256];
|
||||
struct passwd *passwd;
|
||||
FILE *f;
|
||||
|
||||
passwd = getpwuid(geteuid());
|
||||
if (!passwd) {
|
||||
BX_ERROR(("failed to retrieve user name"));
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (access(CONFIG_SMBD_COMMAND, F_OK)) {
|
||||
BX_ERROR(("could not find '%s', please install it", CONFIG_SMBD_COMMAND));
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (access(exported_dir, R_OK | X_OK)) {
|
||||
snprintf(error_msg, sizeof(error_msg), "error accessing shared directory '%s': %s",
|
||||
exported_dir, strerror(errno));
|
||||
BX_ERROR(("%s", error_msg));
|
||||
return -1;
|
||||
}
|
||||
|
||||
i = strlen(exported_dir) - 2;
|
||||
while ((i > 0) && (exported_dir[i] != '/')) i--;
|
||||
snprintf(share, sizeof(share), "%s", &exported_dir[i+1]);
|
||||
if (share[strlen(share)-1] == '/') share[strlen(share)-1] = '\0';
|
||||
|
||||
snprintf(smb_tmpdir, 128, "/tmp/bochs-smb.%ld-%d",
|
||||
(long)getpid(), instance++);
|
||||
if (mkdir(smb_tmpdir, 0700) < 0) {
|
||||
snprintf(error_msg, sizeof(error_msg), "could not create samba server dir '%s'",
|
||||
smb_tmpdir);
|
||||
BX_ERROR(("%s", error_msg));
|
||||
return -1;
|
||||
}
|
||||
snprintf(smb_conf, sizeof(smb_conf), "%s/%s", smb_tmpdir, "smb.conf");
|
||||
|
||||
f = fopen(smb_conf, "w");
|
||||
if (!f) {
|
||||
slirp_smb_cleanup(s, smb_tmpdir);
|
||||
snprintf(error_msg, sizeof(error_msg), "could not create samba server configuration file '%s'",
|
||||
smb_conf);
|
||||
BX_ERROR(("%s", error_msg));
|
||||
return -1;
|
||||
}
|
||||
fprintf(f,
|
||||
"[global]\n"
|
||||
"private dir=%s\n"
|
||||
"interfaces=127.0.0.1\n"
|
||||
"bind interfaces only=yes\n"
|
||||
"pid directory=%s\n"
|
||||
"lock directory=%s\n"
|
||||
"state directory=%s\n"
|
||||
"cache directory=%s\n"
|
||||
"ncalrpc dir=%s/ncalrpc\n"
|
||||
"log file=%s/log.smbd\n"
|
||||
"smb passwd file=%s/smbpasswd\n"
|
||||
"security = user\n"
|
||||
"map to guest = Bad User\n"
|
||||
"[%s]\n"
|
||||
"path=%s\n"
|
||||
"read only=no\n"
|
||||
"guest ok=yes\n"
|
||||
"force user=%s\n",
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
share,
|
||||
exported_dir,
|
||||
passwd->pw_name
|
||||
);
|
||||
fclose(f);
|
||||
|
||||
snprintf(smb_cmdline, sizeof(smb_cmdline), "%s -s %s",
|
||||
CONFIG_SMBD_COMMAND, smb_conf);
|
||||
|
||||
if ((slirp_add_exec(s, smb_cmdline, &vserver_addr, 139) < 0) ||
|
||||
(slirp_add_exec(s, smb_cmdline, &vserver_addr, 445) < 0)) {
|
||||
slirp_smb_cleanup(s, smb_tmpdir);
|
||||
BX_ERROR(("conflicting/invalid smbserver address"));
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void bx_slirp_pktmover_c::slirp_smb_cleanup(Slirp *s, char *smb_tmpdir)
|
||||
{
|
||||
char cmd[128];
|
||||
char error_msg[256];
|
||||
int ret;
|
||||
|
||||
if (smb_tmpdir[0] != '\0') {
|
||||
snprintf(cmd, sizeof(cmd), "rm -rf %s", smb_tmpdir);
|
||||
ret = system(cmd);
|
||||
if (ret == -1 || !WIFEXITED(ret)) {
|
||||
snprintf(error_msg, sizeof(error_msg), "'%s' failed.", cmd);
|
||||
BX_ERROR(("%s", error_msg));
|
||||
} else if (WEXITSTATUS(ret)) {
|
||||
snprintf(error_msg, sizeof(error_msg), "'%s' failed. Error code: %d",
|
||||
cmd, WEXITSTATUS(ret));
|
||||
BX_ERROR(("%s", error_msg));
|
||||
}
|
||||
smb_tmpdir[0] = '\0';
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
static int get_str_sep(char *buf, int buf_size, const char **pp, int sep)
|
||||
{
|
||||
const char *p, *p1;
|
||||
int len;
|
||||
p = *pp;
|
||||
p1 = strchr(p, sep);
|
||||
if (!p1)
|
||||
return -1;
|
||||
len = p1 - p;
|
||||
p1++;
|
||||
if (buf_size > 0) {
|
||||
if (len > buf_size - 1)
|
||||
len = buf_size - 1;
|
||||
memcpy(buf, p, len);
|
||||
buf[len] = '\0';
|
||||
}
|
||||
*pp = p1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int bx_slirp_pktmover_c::slirp_hostfwd(Slirp *s, const char *redir_str, int legacy_format)
|
||||
{
|
||||
struct in_addr host_addr;
|
||||
struct in_addr guest_addr;
|
||||
int host_port, guest_port;
|
||||
const char *p;
|
||||
char buf[256];
|
||||
int is_udp;
|
||||
char *end;
|
||||
|
||||
host_addr.s_addr = INADDR_ANY;
|
||||
guest_addr.s_addr = 0;
|
||||
p = redir_str;
|
||||
if (!p || get_str_sep(buf, sizeof(buf), &p, ':') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
if (!strcmp(buf, "tcp") || buf[0] == '\0') {
|
||||
is_udp = 0;
|
||||
} else if (!strcmp(buf, "udp")) {
|
||||
is_udp = 1;
|
||||
} else {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
if (!legacy_format) {
|
||||
if (get_str_sep(buf, sizeof(buf), &p, ':') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
if (buf[0] != '\0' && !inet_aton(buf, &host_addr)) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
}
|
||||
|
||||
if (get_str_sep(buf, sizeof(buf), &p, legacy_format ? ':' : '-') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
host_port = strtol(buf, &end, 0);
|
||||
if (*end != '\0' || host_port < 1 || host_port > 65535) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
if (get_str_sep(buf, sizeof(buf), &p, ':') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
if (buf[0] != '\0' && !inet_aton(buf, &guest_addr)) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
guest_port = strtol(p, &end, 0);
|
||||
if (*end != '\0' || guest_port < 1 || guest_port > 65535) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
if (slirp_add_hostfwd(s, is_udp, host_addr, host_port, guest_addr,
|
||||
guest_port) < 0) {
|
||||
BX_ERROR(("could not set up host forwarding rule '%s'", redir_str));
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
|
||||
fail_syntax:
|
||||
BX_ERROR(("invalid host forwarding rule '%s'", redir_str));
|
||||
return -1;
|
||||
}
|
||||
|
||||
#endif /* if BX_NETWORKING && BX_NETMOD_SLIRP */
|
||||
|
@ -138,220 +138,4 @@ int qemu_socket(int domain, int type, int protocol)
|
||||
return ret;
|
||||
}
|
||||
|
||||
#if !defined(_WIN32) && !defined(__CYGWIN__)
|
||||
|
||||
#define CONFIG_SMBD_COMMAND "/usr/sbin/smbd"
|
||||
|
||||
#include <pwd.h>
|
||||
|
||||
/* automatic user mode samba server configuration */
|
||||
void slirp_smb_cleanup(Slirp *s, char *smb_tmpdir)
|
||||
{
|
||||
char cmd[128];
|
||||
char error_msg[256];
|
||||
int ret;
|
||||
|
||||
if (smb_tmpdir[0] != '\0') {
|
||||
snprintf(cmd, sizeof(cmd), "rm -rf %s", smb_tmpdir);
|
||||
ret = system(cmd);
|
||||
if (ret == -1 || !WIFEXITED(ret)) {
|
||||
snprintf(error_msg, sizeof(error_msg), "'%s' failed.", cmd);
|
||||
slirp_warning(s, error_msg);
|
||||
} else if (WEXITSTATUS(ret)) {
|
||||
snprintf(error_msg, sizeof(error_msg), "'%s' failed. Error code: %d",
|
||||
cmd, WEXITSTATUS(ret));
|
||||
slirp_warning(s, error_msg);
|
||||
}
|
||||
smb_tmpdir[0] = '\0';
|
||||
}
|
||||
}
|
||||
|
||||
int slirp_smb(Slirp *s, char *smb_tmpdir, const char *exported_dir,
|
||||
struct in_addr vserver_addr)
|
||||
{
|
||||
static int instance;
|
||||
int i;
|
||||
char smb_conf[128], smb_cmdline[150];
|
||||
char share[64], error_msg[256];
|
||||
struct passwd *passwd;
|
||||
FILE *f;
|
||||
|
||||
passwd = getpwuid(geteuid());
|
||||
if (!passwd) {
|
||||
sprintf(error_msg, "failed to retrieve user name");
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (access(CONFIG_SMBD_COMMAND, F_OK)) {
|
||||
sprintf(error_msg, "could not find '%s', please install it",
|
||||
CONFIG_SMBD_COMMAND);
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (access(exported_dir, R_OK | X_OK)) {
|
||||
snprintf(error_msg, sizeof(error_msg), "error accessing shared directory '%s': %s",
|
||||
exported_dir, strerror(errno));
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
|
||||
i = strlen(exported_dir) - 2;
|
||||
while ((i > 0) && (exported_dir[i] != '/')) i--;
|
||||
snprintf(share, sizeof(share), "%s", &exported_dir[i+1]);
|
||||
if (share[strlen(share)-1] == '/') share[strlen(share)-1] = '\0';
|
||||
|
||||
snprintf(smb_tmpdir, 128, "/tmp/bochs-smb.%ld-%d",
|
||||
(long)getpid(), instance++);
|
||||
if (mkdir(smb_tmpdir, 0700) < 0) {
|
||||
snprintf(error_msg, sizeof(error_msg), "could not create samba server dir '%s'",
|
||||
smb_tmpdir);
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
snprintf(smb_conf, sizeof(smb_conf), "%s/%s", smb_tmpdir, "smb.conf");
|
||||
|
||||
f = fopen(smb_conf, "w");
|
||||
if (!f) {
|
||||
slirp_smb_cleanup(s, smb_tmpdir);
|
||||
snprintf(error_msg, sizeof(error_msg), "could not create samba server configuration file '%s'",
|
||||
smb_conf);
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
fprintf(f,
|
||||
"[global]\n"
|
||||
"private dir=%s\n"
|
||||
"interfaces=127.0.0.1\n"
|
||||
"bind interfaces only=yes\n"
|
||||
"pid directory=%s\n"
|
||||
"lock directory=%s\n"
|
||||
"state directory=%s\n"
|
||||
"cache directory=%s\n"
|
||||
"ncalrpc dir=%s/ncalrpc\n"
|
||||
"log file=%s/log.smbd\n"
|
||||
"smb passwd file=%s/smbpasswd\n"
|
||||
"security = user\n"
|
||||
"map to guest = Bad User\n"
|
||||
"[%s]\n"
|
||||
"path=%s\n"
|
||||
"read only=no\n"
|
||||
"guest ok=yes\n"
|
||||
"force user=%s\n",
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
smb_tmpdir,
|
||||
share,
|
||||
exported_dir,
|
||||
passwd->pw_name
|
||||
);
|
||||
fclose(f);
|
||||
|
||||
snprintf(smb_cmdline, sizeof(smb_cmdline), "%s -s %s",
|
||||
CONFIG_SMBD_COMMAND, smb_conf);
|
||||
|
||||
if (slirp_add_exec(s, 0, smb_cmdline, &vserver_addr, 139) < 0 ||
|
||||
slirp_add_exec(s, 0, smb_cmdline, &vserver_addr, 445) < 0) {
|
||||
slirp_smb_cleanup(s, smb_tmpdir);
|
||||
sprintf(error_msg, "conflicting/invalid smbserver address");
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
static int get_str_sep(char *buf, int buf_size, const char **pp, int sep)
|
||||
{
|
||||
const char *p, *p1;
|
||||
int len;
|
||||
p = *pp;
|
||||
p1 = strchr(p, sep);
|
||||
if (!p1)
|
||||
return -1;
|
||||
len = p1 - p;
|
||||
p1++;
|
||||
if (buf_size > 0) {
|
||||
if (len > buf_size - 1)
|
||||
len = buf_size - 1;
|
||||
memcpy(buf, p, len);
|
||||
buf[len] = '\0';
|
||||
}
|
||||
*pp = p1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int slirp_hostfwd(Slirp *s, const char *redir_str, int legacy_format)
|
||||
{
|
||||
struct in_addr host_addr;
|
||||
struct in_addr guest_addr;
|
||||
int host_port, guest_port;
|
||||
const char *p;
|
||||
char buf[256], error_msg[256];
|
||||
int is_udp;
|
||||
char *end;
|
||||
|
||||
host_addr.s_addr = INADDR_ANY;
|
||||
guest_addr.s_addr = 0;
|
||||
p = redir_str;
|
||||
if (!p || get_str_sep(buf, sizeof(buf), &p, ':') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
if (!strcmp(buf, "tcp") || buf[0] == '\0') {
|
||||
is_udp = 0;
|
||||
} else if (!strcmp(buf, "udp")) {
|
||||
is_udp = 1;
|
||||
} else {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
if (!legacy_format) {
|
||||
if (get_str_sep(buf, sizeof(buf), &p, ':') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
if (buf[0] != '\0' && !inet_aton(buf, &host_addr)) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
}
|
||||
|
||||
if (get_str_sep(buf, sizeof(buf), &p, legacy_format ? ':' : '-') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
host_port = strtol(buf, &end, 0);
|
||||
if (*end != '\0' || host_port < 1 || host_port > 65535) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
if (get_str_sep(buf, sizeof(buf), &p, ':') < 0) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
if (buf[0] != '\0' && !inet_aton(buf, &guest_addr)) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
guest_port = strtol(p, &end, 0);
|
||||
if (*end != '\0' || guest_port < 1 || guest_port > 65535) {
|
||||
goto fail_syntax;
|
||||
}
|
||||
|
||||
if (slirp_add_hostfwd(s, is_udp, host_addr, host_port, guest_addr,
|
||||
guest_port) < 0) {
|
||||
sprintf(error_msg, "could not set up host forwarding rule '%s'", redir_str);
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
|
||||
fail_syntax:
|
||||
sprintf(error_msg, "invalid host forwarding rule '%s'", redir_str);
|
||||
slirp_warning(s, error_msg);
|
||||
return -1;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
@ -10,6 +10,12 @@
|
||||
struct Slirp;
|
||||
typedef struct Slirp Slirp;
|
||||
|
||||
typedef struct SlirpConfig {
|
||||
int restricted;
|
||||
struct in_addr vnetwork, vnetmask, vhost, vdhcp_start, vnameserver;
|
||||
const char *bootfile, *vhostname, **vdnssearch, *tftp_path;
|
||||
} SlirpConfig;
|
||||
|
||||
int get_dns_addr(struct in_addr *pdns_addr);
|
||||
|
||||
Slirp *slirp_init(int restricted, struct in_addr vnetwork,
|
||||
@ -18,6 +24,7 @@ Slirp *slirp_init(int restricted, struct in_addr vnetwork,
|
||||
const char *bootfile, struct in_addr vdhcp_start,
|
||||
struct in_addr vnameserver, const char **vdnssearch,
|
||||
void *opaque, void *logfn);
|
||||
Slirp *slirp_new(SlirpConfig *cfg, void *opaque, void *logfn);
|
||||
void slirp_cleanup(Slirp *slirp);
|
||||
|
||||
void slirp_select_fill(int *pnfds, fd_set *readfds, fd_set *writefds,
|
||||
@ -37,7 +44,7 @@ int slirp_add_hostfwd(Slirp *slirp, int is_udp,
|
||||
struct in_addr guest_addr, int guest_port);
|
||||
int slirp_remove_hostfwd(Slirp *slirp, int is_udp,
|
||||
struct in_addr host_addr, int host_port);
|
||||
int slirp_add_exec(Slirp *slirp, int do_pty, const void *args,
|
||||
int slirp_add_exec(Slirp *slirp, const void *args,
|
||||
struct in_addr *guest_addr, int guest_port);
|
||||
|
||||
void slirp_socket_recv(Slirp *slirp, struct in_addr guest_addr,
|
||||
|
@ -210,19 +210,14 @@ static void slirp_init_once(void)
|
||||
loopback_mask = htonl(IN_CLASSA_NET);
|
||||
}
|
||||
|
||||
Slirp *slirp_init(int restricted, struct in_addr vnetwork,
|
||||
struct in_addr vnetmask, struct in_addr vhost,
|
||||
const char *vhostname, const char *tftp_path,
|
||||
const char *bootfile, struct in_addr vdhcp_start,
|
||||
struct in_addr vnameserver, const char **vdnssearch,
|
||||
void *opaque, void *logfn)
|
||||
Slirp *slirp_new(SlirpConfig *cfg, void *opaque, void *logfn)
|
||||
{
|
||||
Slirp *slirp = (Slirp*)malloc(sizeof(Slirp));
|
||||
memset(slirp, 0, sizeof(Slirp));
|
||||
|
||||
slirp_init_once();
|
||||
|
||||
slirp->restricted = restricted;
|
||||
slirp->restricted = cfg->restricted;
|
||||
|
||||
if_init(slirp);
|
||||
ip_init(slirp);
|
||||
@ -230,24 +225,24 @@ Slirp *slirp_init(int restricted, struct in_addr vnetwork,
|
||||
/* Initialise mbufs *after* setting the MTU */
|
||||
m_init(slirp);
|
||||
|
||||
slirp->vnetwork_addr = vnetwork;
|
||||
slirp->vnetwork_mask = vnetmask;
|
||||
slirp->vhost_addr = vhost;
|
||||
if (vhostname) {
|
||||
slirp->vnetwork_addr = cfg->vnetwork;
|
||||
slirp->vnetwork_mask = cfg->vnetmask;
|
||||
slirp->vhost_addr = cfg->vhost;
|
||||
if (cfg->vhostname) {
|
||||
pstrcpy(slirp->client_hostname, sizeof(slirp->client_hostname),
|
||||
vhostname);
|
||||
cfg->vhostname);
|
||||
}
|
||||
if (tftp_path) {
|
||||
slirp->tftp_prefix = strdup(tftp_path);
|
||||
if (cfg->tftp_path) {
|
||||
slirp->tftp_prefix = strdup(cfg->tftp_path);
|
||||
}
|
||||
if (bootfile) {
|
||||
slirp->bootp_filename = strdup(bootfile);
|
||||
if (cfg->bootfile) {
|
||||
slirp->bootp_filename = strdup(cfg->bootfile);
|
||||
}
|
||||
slirp->vdhcp_startaddr = vdhcp_start;
|
||||
slirp->vnameserver_addr = vnameserver;
|
||||
slirp->vdhcp_startaddr = cfg->vdhcp_start;
|
||||
slirp->vnameserver_addr = cfg->vnameserver;
|
||||
|
||||
if (vdnssearch) {
|
||||
translate_dnssearch(slirp, vdnssearch);
|
||||
if (cfg->vdnssearch) {
|
||||
translate_dnssearch(slirp, cfg->vdnssearch);
|
||||
}
|
||||
|
||||
slirp->opaque = opaque;
|
||||
@ -258,6 +253,28 @@ Slirp *slirp_init(int restricted, struct in_addr vnetwork,
|
||||
return slirp;
|
||||
}
|
||||
|
||||
Slirp *slirp_init(int restricted, struct in_addr vnetwork,
|
||||
struct in_addr vnetmask, struct in_addr vhost,
|
||||
const char *vhostname, const char *tftp_path,
|
||||
const char *bootfile, struct in_addr vdhcp_start,
|
||||
struct in_addr vnameserver, const char **vdnssearch,
|
||||
void *opaque, void *logfn)
|
||||
{
|
||||
SlirpConfig cfg;
|
||||
memset(&cfg, 0, sizeof(cfg));
|
||||
cfg.restricted = restricted;
|
||||
cfg.vnetwork = vnetwork;
|
||||
cfg.vnetmask = vnetmask;
|
||||
cfg.vhost = vhost;
|
||||
cfg.vhostname = vhostname;
|
||||
cfg.tftp_path = tftp_path;
|
||||
cfg.bootfile = bootfile;
|
||||
cfg.vdhcp_start = vdhcp_start;
|
||||
cfg.vnameserver = vnameserver;
|
||||
cfg.vdnssearch = vdnssearch;
|
||||
return slirp_new(&cfg, opaque, logfn);
|
||||
}
|
||||
|
||||
void slirp_cleanup(Slirp *slirp)
|
||||
{
|
||||
struct ex_list *ex_ptr;
|
||||
@ -867,7 +884,7 @@ int slirp_add_hostfwd(Slirp *slirp, int is_udp, struct in_addr host_addr,
|
||||
return 0;
|
||||
}
|
||||
|
||||
int slirp_add_exec(Slirp *slirp, int do_pty, const void *args,
|
||||
int slirp_add_exec(Slirp *slirp, const void *args,
|
||||
struct in_addr *guest_addr, int guest_port)
|
||||
{
|
||||
if (!guest_addr->s_addr) {
|
||||
@ -880,7 +897,7 @@ int slirp_add_exec(Slirp *slirp, int do_pty, const void *args,
|
||||
guest_addr->s_addr == slirp->vnameserver_addr.s_addr) {
|
||||
return -1;
|
||||
}
|
||||
return add_exec(&slirp->exec_list, do_pty, (char *)args, *guest_addr,
|
||||
return add_exec(&slirp->exec_list, 0, (char *)args, *guest_addr,
|
||||
htons(guest_port));
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user