⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 qnetworkinterface_unix.cpp

📁 奇趣公司比较新的qt/emd版本
💻 CPP
字号:
/******************************************************************************** Copyright (C) 1992-2007 Trolltech ASA. All rights reserved.**** This file is part of the QtNetwork module of the Qt Toolkit.**** This file may be used under the terms of the GNU General Public** License version 2.0 as published by the Free Software Foundation** and appearing in the file LICENSE.GPL included in the packaging of** this file.  Please review the following information to ensure GNU** General Public Licensing requirements will be met:** http://trolltech.com/products/qt/licenses/licensing/opensource/**** If you are unsure which license is appropriate for your use, please** review the following information:** http://trolltech.com/products/qt/licenses/licensing/licensingoverview** or contact the sales department at sales@trolltech.com.**** In addition, as a special exception, Trolltech gives you certain** additional rights. These rights are described in the Trolltech GPL** Exception version 1.0, which can be found at** http://www.trolltech.com/products/qt/gplexception/ and in the file** GPL_EXCEPTION.txt in this package.**** In addition, as a special exception, Trolltech, as the sole copyright** holder for Qt Designer, grants users of the Qt/Eclipse Integration** plug-in the right for the Qt/Eclipse Integration to link to** functionality provided by Qt Designer and its related libraries.**** Trolltech reserves all rights not expressly granted herein.**** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.******************************************************************************/#include "qset.h"#include "qnetworkinterface.h"#include "qnetworkinterface_p.h"#include "qalgorithms.h"#define IP_MULTICAST    // make AIX happy and define IFF_MULTICAST#include <sys/types.h>#include <sys/socket.h>#ifdef Q_OS_SOLARIS# include <sys/sockio.h>#endif#include <net/if.h>#ifndef QT_NO_GETIFADDRS# include <ifaddrs.h>#endif#ifdef QT_LSB#  include <arpa/inet.h>#  ifndef SIOCGIFBRDADDR#    define SIOCGIFBRDADDR 0x8919#  endif#endif // QT_LSB#include <qplatformdefs.h>static QHostAddress addressFromSockaddr(sockaddr *sa){    QHostAddress address;    if (!sa)        return address;    if (sa->sa_family == AF_INET)        address.setAddress(htonl(((sockaddr_in *)sa)->sin_addr.s_addr));#ifndef QT_NO_IPV6    else if (sa->sa_family == AF_INET6)        address.setAddress(((sockaddr_in6 *)sa)->sin6_addr.s6_addr);#endif    return address;}static QNetworkInterface::InterfaceFlags convertFlags(uint rawFlags){    QNetworkInterface::InterfaceFlags flags = 0;    flags |= (rawFlags & IFF_UP) ? QNetworkInterface::IsUp : QNetworkInterface::InterfaceFlag(0);    flags |= (rawFlags & IFF_RUNNING) ? QNetworkInterface::IsRunning : QNetworkInterface::InterfaceFlag(0);    flags |= (rawFlags & IFF_BROADCAST) ? QNetworkInterface::CanBroadcast : QNetworkInterface::InterfaceFlag(0);    flags |= (rawFlags & IFF_LOOPBACK) ? QNetworkInterface::IsLoopBack : QNetworkInterface::InterfaceFlag(0);    flags |= (rawFlags & IFF_POINTOPOINT) ? QNetworkInterface::IsPointToPoint : QNetworkInterface::InterfaceFlag(0);#ifdef IFF_MULTICAST    flags |= (rawFlags & IFF_MULTICAST) ? QNetworkInterface::CanMulticast : QNetworkInterface::InterfaceFlag(0);#endif    return flags;}#ifdef QT_NO_GETIFADDRS// getifaddrs not availablestatic const int STORAGEBUFFER_GROWTH = 256;static QSet<QByteArray> interfaceNames(int socket){    QSet<QByteArray> result;#ifdef QT_NO_IPV6IFNAME    QByteArray storageBuffer;    struct ifconf interfaceList;    forever {        // grow the storage buffer        storageBuffer.resize(storageBuffer.size() + STORAGEBUFFER_GROWTH);        interfaceList.ifc_buf = storageBuffer.data();        interfaceList.ifc_len = storageBuffer.size();        // get the interface list        if (::ioctl(socket, SIOCGIFCONF, &interfaceList) >= 0) {            if (int(interfaceList.ifc_len + sizeof(ifreq) + 64) < storageBuffer.size()) {                // if the buffer was big enough, break                storageBuffer.resize(interfaceList.ifc_len);                break;            }        } else {            // internal error            return result;        }        if (storageBuffer.size() > 100000) {            // out of space            return result;        }    }    int interfaceCount = interfaceList.ifc_len / sizeof(ifreq);    for (int i = 0; i < interfaceCount; ++i) {        QByteArray name = QByteArray(interfaceList.ifc_req[i].ifr_name);        if (!name.isEmpty())            result << name;    }    return result;#else    Q_UNUSED(socket);    // use if_nameindex    struct if_nameindex *interfaceList = ::if_nameindex();    for (struct if_nameindex *ptr = interfaceList; ptr && ptr->if_name; ++ptr)        result << ptr->if_name;    if_freenameindex(interfaceList);    return result;#endif}static QNetworkInterfacePrivate *findInterface(int socket, QList<QNetworkInterfacePrivate *> &interfaces,                                               struct ifreq &req){    QNetworkInterfacePrivate *iface = 0;    int ifindex = 0;#ifndef QT_NO_IPV6IFNAME    // Get the interface index    ifindex = if_nametoindex(req.ifr_name);    // find the interface data    QList<QNetworkInterfacePrivate *>::Iterator if_it = interfaces.begin();    for ( ; if_it != interfaces.end(); ++if_it)        if ((*if_it)->index == ifindex) {            // existing interface            iface = *if_it;            break;        }#else    // Search by name    QList<QNetworkInterfacePrivate *>::Iterator if_it = interfaces.begin();    for ( ; if_it != interfaces.end(); ++if_it)        if ((*if_it)->name == QLatin1String(req.ifr_name)) {            // existing interface            iface = *if_it;            break;        }#endif    if (!iface) {        // new interface, create data:        iface = new QNetworkInterfacePrivate;        iface->index = ifindex;        interfaces << iface;#ifdef SIOCGIFNAME        // Get the canonical name        QByteArray oldName = req.ifr_name;        if (::ioctl(socket, SIOCGIFNAME, &req) >= 0) {            iface->name = QString::fromLatin1(req.ifr_name);            // reset the name:            memcpy(req.ifr_name, oldName, qMin<int>(oldName.length() + 1, sizeof(req.ifr_name) - 1));        } else#endif	{            // use this name anyways            iface->name = QString::fromLatin1(req.ifr_name);        }        // Get interface flags        if (::ioctl(socket, SIOCGIFFLAGS, &req) >= 0) {            iface->flags = convertFlags(req.ifr_flags);        }#ifdef SIOCGIFHWADDR        // Get the HW address        if (::ioctl(socket, SIOCGIFHWADDR, &req) >= 0) {            uchar *addr = (uchar *)&req.ifr_addr;            iface->hardwareAddress = iface->makeHwAddress(6, addr);        }#endif    }    return iface;}static QList<QNetworkInterfacePrivate *> interfaceListing(){    QList<QNetworkInterfacePrivate *> interfaces;    int socket;    if ((socket = ::socket(AF_INET, SOCK_STREAM, IPPROTO_IP)) == -1)        return interfaces;      // error    QSet<QByteArray> names = interfaceNames(socket);    QSet<QByteArray>::ConstIterator it = names.constBegin();    for ( ; it != names.constEnd(); ++it) {        ifreq req;        memset(&req, 0, sizeof(ifreq));        memcpy(req.ifr_name, *it, qMin<int>(it->length() + 1, sizeof(req.ifr_name) - 1));        QNetworkInterfacePrivate *iface = findInterface(socket, interfaces, req);        // Get the interface broadcast address        QNetworkAddressEntry entry;        if (iface->flags & QNetworkInterface::CanBroadcast) {            if (::ioctl(socket, SIOCGIFBRDADDR, &req) >= 0) {                sockaddr *sa = &req.ifr_addr;                if (sa->sa_family == AF_INET)                    entry.setBroadcast(addressFromSockaddr(sa));            }        }        // Get the interface netmask        if (::ioctl(socket, SIOCGIFNETMASK, &req) >= 0) {            sockaddr *sa = &req.ifr_addr;	    entry.setNetmask(addressFromSockaddr(sa));        }        // Get the address of the interface        if (::ioctl(socket, SIOCGIFADDR, &req) >= 0) {            sockaddr *sa = &req.ifr_addr;            entry.setIp(addressFromSockaddr(sa));        }        iface->addressEntries << entry;    }    ::close(socket);    return interfaces;}#else// use getifaddrs// platform-specific defs:# ifdef Q_OS_LINUX#  include <features.h># endif# if defined(Q_OS_LINUX) &&  __GLIBC__ - 0 >= 2 && __GLIBC_MINOR__ - 0 >= 1#  include <netpacket/packet.h>static QList<QNetworkInterfacePrivate *> createInterfaces(ifaddrs *rawList){    QList<QNetworkInterfacePrivate *> interfaces;    for (ifaddrs *ptr = rawList; ptr; ptr = ptr->ifa_next) {        if ( !ptr->ifa_addr )            continue;        // Get the interface index        int ifindex = if_nametoindex(ptr->ifa_name);        // on Linux we use AF_PACKET and sockaddr_ll to obtain hHwAddress        QList<QNetworkInterfacePrivate *>::Iterator if_it = interfaces.begin();        for ( ; if_it != interfaces.end(); ++if_it)            if ((*if_it)->index == ifindex) {                // this one has been added already                if ( ptr->ifa_addr->sa_family == AF_PACKET                        && (*if_it)->hardwareAddress.isEmpty()) {                    sockaddr_ll *sll = (sockaddr_ll *)ptr->ifa_addr;                    (*if_it)->hardwareAddress = (*if_it)->makeHwAddress(sll->sll_halen, (uchar*)sll->sll_addr);                }                break;            }        if ( if_it != interfaces.end() )             continue;                QNetworkInterfacePrivate *iface = new QNetworkInterfacePrivate;        interfaces << iface;        iface->index = ifindex;        iface->name = QString::fromLatin1(ptr->ifa_name);        iface->flags = convertFlags(ptr->ifa_flags);        if ( ptr->ifa_addr->sa_family == AF_PACKET ) {            sockaddr_ll *sll = (sockaddr_ll *)ptr->ifa_addr;            iface->hardwareAddress = iface->makeHwAddress(sll->sll_halen, (uchar*)sll->sll_addr);        }    }    return interfaces;}# elif defined(Q_OS_BSD4)#  include <net/if_dl.h>static QList<QNetworkInterfacePrivate *> createInterfaces(ifaddrs *rawList){    QList<QNetworkInterfacePrivate *> interfaces;    // on NetBSD we use AF_LINK and sockaddr_dl    // scan the list for that family    for (ifaddrs *ptr = rawList; ptr; ptr = ptr->ifa_next)        if (ptr->ifa_addr && ptr->ifa_addr->sa_family == AF_LINK) {            QNetworkInterfacePrivate *iface = new QNetworkInterfacePrivate;            interfaces << iface;            sockaddr_dl *sdl = (sockaddr_dl *)ptr->ifa_addr;            iface->index = sdl->sdl_index;            iface->name = QString::fromLatin1(ptr->ifa_name);            iface->flags = convertFlags(ptr->ifa_flags);            iface->hardwareAddress = iface->makeHwAddress(sdl->sdl_alen, (uchar*)LLADDR(sdl));        }    return interfaces;}# else  // Generic versionstatic QList<QNetworkInterfacePrivate *> createInterfaces(ifaddrs *rawList){    QList<QNetworkInterfacePrivate *> interfaces;    // make sure there's one entry for each interface    for (ifaddrs *ptr = rawList; ptr; ptr = ptr->ifa_next) {        // Get the interface index        int ifindex = if_nametoindex(ptr->ifa_name);        QList<QNetworkInterfacePrivate *>::Iterator if_it = interfaces.begin();        for ( ; if_it != interfaces.end(); ++if_it)            if ((*if_it)->index == ifindex)                // this one has been added already                break;        if (if_it == interfaces.end()) {            // none found, create            QNetworkInterfacePrivate *iface = new QNetworkInterfacePrivate;            interfaces << iface;            iface->index = ifindex;            iface->name = QString::fromLatin1(ptr->ifa_name);            iface->flags = convertFlags(ptr->ifa_flags);        }    }    return interfaces;}# endifstatic QList<QNetworkInterfacePrivate *> interfaceListing(){    QList<QNetworkInterfacePrivate *> interfaces;    int socket;    if ((socket = ::socket(AF_INET, SOCK_STREAM, IPPROTO_IP)) == -1)        return interfaces;      // error    ifaddrs *interfaceListing;    if (getifaddrs(&interfaceListing) == -1) {        // error        ::close(socket);        return interfaces;    }    interfaces = createInterfaces(interfaceListing);    for (ifaddrs *ptr = interfaceListing; ptr; ptr = ptr->ifa_next) {        // Get the interface index        int ifindex = if_nametoindex(ptr->ifa_name);        QNetworkInterfacePrivate *iface = 0;        QList<QNetworkInterfacePrivate *>::Iterator if_it = interfaces.begin();        for ( ; if_it != interfaces.end(); ++if_it)            if ((*if_it)->index == ifindex) {                // found this interface already                iface = *if_it;                break;            }        if (!iface) {            // skip all non-IP interfaces            continue;        }        QNetworkAddressEntry entry;        entry.setIp(addressFromSockaddr(ptr->ifa_addr));        if (entry.ip().isNull())            // could not parse the address            continue;        entry.setNetmask(addressFromSockaddr(ptr->ifa_netmask));        if (iface->flags & QNetworkInterface::CanBroadcast)            entry.setBroadcast(addressFromSockaddr(ptr->ifa_broadaddr));        iface->addressEntries << entry;    }    freeifaddrs(interfaceListing);    ::close(socket);    return interfaces;}#endifQList<QNetworkInterfacePrivate *> QNetworkInterfaceManager::scan(){    return interfaceListing();}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -