hio/mio/lib/nwif.c

652 lines
13 KiB
C

/*
* $Id$
*
Copyright (c) 2016-2020 Chung, Hyung-Hwan. All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "mio-nwif.h"
#include "mio-prv.h"
#if defined(_WIN32)
/* TODO: */
#elif defined(__OS2__)
/* TODO: */
#elif defined(__DOS__)
/* TODO: */
#else
# include <unistd.h>
# include <errno.h>
# include <sys/socket.h>
# if defined(HAVE_SYS_IOCTL_H)
# include <sys/ioctl.h>
# endif
# if defined(HAVE_NET_IF_H)
# include <net/if.h>
# endif
# if defined(HAVE_SYS_SOCKIO_H)
# include <sys/sockio.h>
# endif
# if !defined(IF_NAMESIZE)
# define IF_NAMESIZE 63
# endif
#endif
#if defined(SIOCGIFCONF) && (defined(SIOCGIFANUM) || defined(SIOCGIFNUM))
static int get_sco_ifconf (mio_t* mio, struct ifconf* ifc)
{
/* SCO doesn't have have any IFINDEX thing.
* i emultate it using IFCONF */
int h, num;
struct ifreq* ifr;
h = socket(AF_INET, SOCK_DGRAM, 0);
if (h <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
ifc->ifc_len = 0;
ifc->ifc_buf = MIO_NULL;
#if defined(SIOCGIFANUM)
if (ioctl(h, SIOCGIFANUM, &num) <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
goto oops;
}
#else
if (ioctl(h, SIOCGIFNUM, &num) <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
goto oops;
}
#endif
/* sco needs reboot when you add an network interface.
* it should be safe not to consider the case when the interface
* is added after SIOCGIFANUM above.
* another thing to note is that SIOCGIFCONF ends with segfault
* if the buffer is not large enough unlike some other OSes
* like opensolaris which truncates the configuration. */
ifc->ifc_len = num * MIO_SIZEOF(*ifr);
ifc->ifc_buf = mio_allocmem(mio, ifc->ifc_len);
if (ifc->ifc_buf == MIO_NULL) goto oops;
if (ioctl(h, SIOCGIFCONF, ifc) <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
goto oops;
}
close (h); h = -1;
return 0;
oops:
if (ifc->ifc_buf) mio_freemem (mio, ifc->ifc_buf);
if (h >= 0) close (h);
return -1;
}
static MIO_INLINE void free_sco_ifconf (mio_t* mio, struct ifconf* ifc)
{
mio_freemem (mio, ifc->ifc_buf);
}
#endif
int mio_bcstrtoifindex (mio_t* mio, const mio_bch_t* ptr, unsigned int* index)
{
#if defined(_WIN32)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__OS2__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__DOS__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(SIOCGIFINDEX)
int h, x;
mio_oow_t len;
struct ifreq ifr;
h = socket(AF_INET, SOCK_DGRAM, 0);
if (h <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
MIO_MEMSET (&ifr, 0, MIO_SIZEOF(ifr));
len = mio_copy_bcstr(ifr.ifr_name, MIO_COUNTOF(ifr.ifr_name), ptr);
if (ptr[len] != '\0') return -1; /* name too long */
x = ioctl(h, SIOCGIFINDEX, &ifr);
close (h);
if (x >= 0)
{
#if defined(HAVE_STRUCT_IFREQ_IFR_IFINDEX)
*index = ifr.ifr_ifindex;
#else
*index = ifr.ifr_index;
#endif
}
return x;
#elif defined(HAVE_IF_NAMETOINDEX)
mio_bch_t tmp[IF_NAMESIZE + 1];
mio_oow_t len;
unsigned int tmpidx;
len = mio_copy_bcstr(tmp, MIO_COUNTOF(tmp), ptr);
if (ptr[len] != '\0') return -1; /* name too long */
tmpidx = if_nametoindex(tmp);
if (tmpidx == 0)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
*index = tmpidx;
return 0;
#elif defined(SIOCGIFCONF) && (defined(SIOCGIFANUM) || defined(SIOCGIFNUM))
struct ifconf ifc;
int num, i;
if (get_sco_ifconf(mio, &ifc) <= -1) return -1;
num = ifc.ifc_len / MIO_SIZEOF(struct ifreq);
for (i = 0; i < num; i++)
{
if (mio_comp_bcstr(ptr, ifc.ifc_req[i].ifr_name, 0) == 0)
{
free_sco_ifconf (mio, &ifc);
*index = i + 1;
return 0;
}
}
free_sco_ifconf (mio, &ifc);
return -1;
#else
return -1;
#endif
}
int mio_bcharstoifindex (mio_t* mio, const mio_bch_t* ptr, mio_oow_t len, unsigned int* index)
{
#if defined(_WIN32)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__OS2__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__DOS__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(SIOCGIFINDEX)
int h, x;
struct ifreq ifr;
h = socket(AF_INET, SOCK_DGRAM, 0);
if (h <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
MIO_MEMSET (&ifr, 0, MIO_SIZEOF(ifr));
if (mio_copy_bchars_to_bcstr(ifr.ifr_name, MIO_COUNTOF(ifr.ifr_name), ptr, len) < len) return -1; /* name too long */
x = ioctl(h, SIOCGIFINDEX, &ifr);
close (h);
if (x >= 0)
{
#if defined(HAVE_STRUCT_IFREQ_IFR_IFINDEX)
*index = ifr.ifr_ifindex;
#else
*index = ifr.ifr_index;
#endif
}
return x;
#elif defined(HAVE_IF_NAMETOINDEX)
mio_bch_t tmp[IF_NAMESIZE + 1];
unsigned int tmpidx;
if (mio_copy_bchars_to_bcstr(tmp, MIO_COUNTOF(tmp), ptr, len) < len) return -1;
tmpidx = if_nametoindex(tmp);
if (tmpidx == 0)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
*index = tmpidx;
return 0;
#elif defined(SIOCGIFCONF) && (defined(SIOCGIFANUM) || defined(SIOCGIFNUM))
struct ifconf ifc;
int num, i;
if (get_sco_ifconf(mio, &ifc) <= -1) return -1;
num = ifc.ifc_len / MIO_SIZEOF(struct ifreq);
for (i = 0; i < num; i++)
{
if (mio_comp_bchars_bcstr(ptr, len, ifc.ifc_req[i].ifr_name) == 0)
{
free_sco_ifconf (mio, &ifc);
*index = i + 1;
return 0;
}
}
free_sco_ifconf (mio, &ifc);
return -1;
#else
return -1;
#endif
}
int mio_ucstrtoifindex (mio_t* mio, const mio_uch_t* ptr, unsigned int* index)
{
#if defined(_WIN32)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__OS2__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__DOS__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(SIOCGIFINDEX)
int h, x;
struct ifreq ifr;
mio_oow_t wl, ml;
h = socket(AF_INET, SOCK_DGRAM, 0);
if (h <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
ml = MIO_COUNTOF(ifr.ifr_name);
if (mio_convutobcstr(mio, ptr, &wl, ifr.ifr_name, &ml) <= -1) return -1;
x = ioctl(h, SIOCGIFINDEX, &ifr);
close (h);
if (x >= 0)
{
#if defined(HAVE_STRUCT_IFREQ_IFR_IFINDEX)
*index = ifr.ifr_ifindex;
#else
*index = ifr.ifr_index;
#endif
}
return x;
#elif defined(HAVE_IF_NAMETOINDEX)
mio_bch_t tmp[IF_NAMESIZE + 1];
mio_oow_t wl, ml;
unsigned int tmpidx;
ml = MIO_COUNTOF(tmp);
if (mio_convutobcstr(mio, ptr, &wl, tmp, &ml) <= -1) return -1;
tmpidx = if_nametoindex(tmp);
if (tmpidx == 0)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
*index = tmpidx;
return 0;
#elif defined(SIOCGIFCONF) && (defined(SIOCGIFANUM) || defined(SIOCGIFNUM))
struct ifconf ifc;
int num, i;
mio_bch_t tmp[IF_NAMESIZE + 1];
mio_oow_t wl, ml;
ml = MIO_COUNTOF(tmp);
if (mio_convutobcstr(mio, ptr, &wl, tmp, &ml) <= -1) return -1;
if (get_sco_ifconf(mio, &ifc) <= -1) return -1;
num = ifc.ifc_len / MIO_SIZEOF(struct ifreq);
for (i = 0; i < num; i++)
{
if (mio_comp_bcstr(tmp, ifc.ifc_req[i].ifr_name, 0) == 0)
{
free_sco_ifconf (mio, &ifc);
*index = i + 1;
return 0;
}
}
free_sco_ifconf (mio, &ifc);
return -1;
#else
return -1;
#endif
}
int mio_ucharstoifindex (mio_t* mio, const mio_uch_t* ptr, mio_oow_t len, unsigned int* index)
{
#if defined(_WIN32)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__OS2__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__DOS__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(SIOCGIFINDEX)
int h, x;
struct ifreq ifr;
mio_oow_t wl, ml;
h = socket(AF_INET, SOCK_DGRAM, 0);
if (h <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
wl = len; ml = MIO_COUNTOF(ifr.ifr_name) - 1;
if (mio_convutobchars(mio, ptr, &wl, ifr.ifr_name, &ml) <= -1) return -1;
ifr.ifr_name[ml] = '\0';
x = ioctl(h, SIOCGIFINDEX, &ifr);
close (h);
if (x >= 0)
{
#if defined(HAVE_STRUCT_IFREQ_IFR_IFINDEX)
*index = ifr.ifr_ifindex;
#else
*index = ifr.ifr_index;
#endif
}
return x;
#elif defined(HAVE_IF_NAMETOINDEX)
mio_bch_t tmp[IF_NAMESIZE + 1];
mio_oow_t wl, ml;
unsigned int tmpidx;
wl = len; ml = MIO_COUNTOF(tmp) - 1;
if (mio_convutobchars(mio, ptr, &wl, tmp, &ml) <= -1) return -1;
tmp[ml] = '\0';
tmpidx = if_nametoindex(tmp);
if (tmpidx == 0)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
*index = tmpidx;
return 0;
#elif defined(SIOCGIFCONF) && (defined(SIOCGIFANUM) || defined(SIOCGIFNUM))
struct ifconf ifc;
int num, i;
mio_bch_t tmp[IF_NAMESIZE + 1];
mio_oow_t wl, ml;
wl = len; ml = MIO_COUNTOF(tmp) - 1;
if (mio_convutobchars(ptr, &wl, tmp, &ml) <= -1) return -1;
tmp[ml] = '\0';
if (get_sco_ifconf(mio, &ifc) <= -1) return -1;
num = ifc.ifc_len / MIO_SIZEOF(struct ifreq);
for (i = 0; i < num; i++)
{
if (mio_comp_bcstr(tmp, ifc.ifc_req[i].ifr_name, 0) == 0)
{
free_sco_ifconf (mio, &ifc);
*index = i + 1;
return 0;
}
}
free_sco_ifconf (mio, &ifc);
return -1;
#else
return -1;
#endif
}
/* ---------------------------------------------------------- */
int mio_ifindextobcstr (mio_t* mio, unsigned int index, mio_bch_t* buf, mio_oow_t len)
{
#if defined(_WIN32)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__OS2__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__DOS__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(SIOCGIFNAME)
int h, x;
struct ifreq ifr;
h = socket(AF_INET, SOCK_DGRAM, 0);
if (h <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
MIO_MEMSET (&ifr, 0, MIO_SIZEOF(ifr));
#if defined(HAVE_STRUCT_IFREQ_IFR_IFINDEX)
ifr.ifr_ifindex = index;
#else
ifr.ifr_index = index;
#endif
x = ioctl(h, SIOCGIFNAME, &ifr);
close (h);
if (x <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
return mio_copy_bcstr(buf, len, ifr.ifr_name);
#elif defined(HAVE_IF_INDEXTONAME)
mio_bch_t tmp[IF_NAMESIZE + 1];
if (if_indextoname (index, tmp) == MIO_NULL)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
return mio_copy_bcstr(buf, len, tmp);
#elif defined(SIOCGIFCONF) && (defined(SIOCGIFANUM) || defined(SIOCGIFNUM))
struct ifconf ifc;
mio_oow_t ml;
int num;
if (index <= 0) return -1;
if (get_sco_ifconf(mio, &ifc) <= -1) return -1;
num = ifc.ifc_len / MIO_SIZEOF(struct ifreq);
if (index > num)
{
mio_seterrnum (mio, MIO_ENOENT);
free_sco_ifconf (mio, &ifc);
return -1;
}
ml = mio_copy_bcstr(buf, len, ifc.ifc_req[index - 1].ifr_name);
free_sco_ifconf (mio, &ifc);
return ml;
#else
return -1;
#endif
}
int mio_ifindextoucstr (mio_t* mio, unsigned int index, mio_uch_t* buf, mio_oow_t len)
{
#if defined(_WIN32)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__OS2__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(__DOS__)
/* TODO: */
mio_seterrnum (mio, MIO_ENOIMPL);
return -1;
#elif defined(SIOCGIFNAME)
int h, x;
struct ifreq ifr;
mio_oow_t wl, ml;
h = socket(AF_INET, SOCK_DGRAM, 0);
if (h <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
MIO_MEMSET (&ifr, 0, MIO_SIZEOF(ifr));
#if defined(HAVE_STRUCT_IFREQ_IFR_IFINDEX)
ifr.ifr_ifindex = index;
#else
ifr.ifr_index = index;
#endif
x = ioctl(h, SIOCGIFNAME, &ifr);
close (h);
if (x <= -1)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
wl = len;
x = mio_convbtoucstr(mio, ifr.ifr_name, &ml, buf, &wl, 0);
if (x == -2 && wl > 1) buf[wl - 1] = '\0';
else if (x != 0) return -1;
return wl;
#elif defined(HAVE_IF_INDEXTONAME)
mio_bch_t tmp[IF_NAMESIZE + 1];
mio_oow_t ml, wl;
int x;
if (if_indextoname(index, tmp) == MIO_NULL)
{
mio_seterrwithsyserr (mio, 0, errno);
return -1;
}
wl = len;
x = mio_convbtoucstr(mio, tmp, &ml, buf, &wl, 0);
if (x == -2 && wl > 1) buf[wl - 1] = '\0';
else if (x != 0) return -1;
return wl;
#elif defined(SIOCGIFCONF) && (defined(SIOCGIFANUM) || defined(SIOCGIFNUM))
struct ifconf ifc;
mio_oow_t wl, ml;
int num, x;
if (index <= 0) return -1;
if (get_sco_ifconf(mio, &ifc) <= -1) return -1;
num = ifc.ifc_len / MIO_SIZEOF(struct ifreq);
if (index > num)
{
free_sco_ifconf (mio, &ifc);
return -1;
}
wl = len;
x = mio_convbtoucstr(ifc.ifc_req[index - 1].ifr_name, &ml, buf, &wl, 0);
free_sco_ifconf (mio, &ifc);
if (x == -2 && wl > 1) buf[wl - 1] = '\0';
else if (x != 0) return -1;
return wl;
#else
return -1;
#endif
}