- fixes thread safety issues
- fixes problems with non-IP (e.g. IPX) addresses
- updates comments to reflect winsock, netapi32 changes
diff --git a/dlls/iphlpapi/ifenum.c b/dlls/iphlpapi/ifenum.c
index 052fa49..e5749d9 100644
--- a/dlls/iphlpapi/ifenum.c
+++ b/dlls/iphlpapi/ifenum.c
@@ -27,27 +27,13 @@
* an index is a key, and that an interface has 0 or 1 IP addresses.
* If that behavior were consistent across UNIXen (I don't know), it could
* help me implement multiple IP addresses more in the Windows way.
- * But here's a problem with doing so:
- * - Netbios() has a concept of an LAN adapter number (LANA), which is an 8-bit
- * number in the range 0-254, inclusive. The MSDN pages for Netbios() says
- * the original Netbios() spec allowed only 0 or 1 to be used, though "new"
- * applications should enumerate available adapters rather than assuming 0
- * is the default adapter.
- * I'm concerned that some old application might depend on being able to get
- * "the" MAC address of a machine by opening LANA 0 and getting its MAC
- * address. This also implies LANA 0 should correspond to a non-loopback
- * interface.
- * On Linux, the if_nametoindex index is 1-based, and "lo" typically has
- * index 1.
- * I could make netapi32 do its own LANA map, independent of my index
- * assignment, but it seems simpler just to assign 0-based indexes and put
- * non-loopback adapters first, so the first 255 indexes (!) on a system will
- * automatically map to LANA numbers without difficulty.
- * - One more argument for doing it this way, if you don't buy the Netbios()
- * argument: WsControl() (in wsock32) uses the same index to refer to an IP
- * address and an interface. If I assigned multiple IP addresses to an
- * interface, wsock32 would have to maintain a table of IP addresses with its
- * own indexing scheme. No thanks.
+ * I used to assert I could not use UNIX interface indexes as my iphlpapi
+ * indexes due to restrictions in netapi32 and wsock32, but I have removed
+ * those restrictions, so using if_nametoindex and if_indextoname rather
+ * than my current mess would probably be better.
+ * FIXME:
+ * - I don't support IPv6 addresses here, since SIOCGIFCONF can't return them
+ * - the memory interface uses malloc/free; it should be using HeapAlloc instead
*
* There are three implemened methods for determining the MAC address of an
* interface:
@@ -115,9 +101,6 @@
#include <net/if_types.h>
#endif
-#include "windef.h"
-#include "winbase.h"
-#include "iprtrmib.h"
#include "ifenum.h"
#ifdef HAVE_STRUCT_SOCKADDR_SA_LEN
@@ -156,17 +139,32 @@
/* Global variables */
+static CRITICAL_SECTION mapCS;
static InterfaceNameMap *gNonLoopbackInterfaceMap = NULL;
static InterfaceNameMap *gLoopbackInterfaceMap = NULL;
/* Functions */
+void interfaceMapInit(void)
+{
+ InitializeCriticalSection(&mapCS);
+}
+
+void interfaceMapFree(void)
+{
+ DeleteCriticalSection(&mapCS);
+ if (gNonLoopbackInterfaceMap)
+ free(gNonLoopbackInterfaceMap);
+ if (gLoopbackInterfaceMap)
+ free(gLoopbackInterfaceMap);
+}
+
/* Sizes the passed-in map to have enough space for numInterfaces interfaces.
* If map is NULL, allocates a new map. If it is not, may reallocate the
* existing map and return a map of increased size. Returns the allocated map,
* or NULL if it could not allocate a map of the requested size.
*/
-InterfaceNameMap *sizeMap(InterfaceNameMap *map, DWORD numInterfaces)
+static InterfaceNameMap *sizeMap(InterfaceNameMap *map, DWORD numInterfaces)
{
if (!map) {
numInterfaces = max(numInterfaces, INITIAL_INTERFACES_ASSUMED);
@@ -286,10 +284,12 @@
while (ifPtr && ifPtr < buf + len) {
struct ifreq *ifr = (struct ifreq *)ifPtr;
- if (isLoopbackInterface(fd, ifr->ifr_name))
- storeInterfaceInMap(gLoopbackInterfaceMap, ifr->ifr_name);
- else
- storeInterfaceInMap(gNonLoopbackInterfaceMap, ifr->ifr_name);
+ if (ifr->ifr_addr.sa_family == AF_INET) {
+ if (isLoopbackInterface(fd, ifr->ifr_name))
+ storeInterfaceInMap(gLoopbackInterfaceMap, ifr->ifr_name);
+ else
+ storeInterfaceInMap(gNonLoopbackInterfaceMap, ifr->ifr_name);
+ }
ifPtr += ifreq_len(ifr);
}
}
@@ -329,8 +329,10 @@
ifc.ifc_len == (sizeof(struct ifreq) * guessedNumInterfaces));
if (ret == 0) {
+ EnterCriticalSection(&mapCS);
countInterfaces(fd, ifc.ifc_buf, ifc.ifc_len);
classifyInterfaces(fd, ifc.ifc_buf, ifc.ifc_len);
+ LeaveCriticalSection(&mapCS);
}
if (ifc.ifc_buf)
@@ -359,6 +361,7 @@
InterfaceNameMap *map;
const char *ret = NULL;
+ EnterCriticalSection(&mapCS);
if (index & INDEX_IS_LOOPBACK) {
realIndex = index ^ INDEX_IS_LOOPBACK;
map = gLoopbackInterfaceMap;
@@ -369,12 +372,13 @@
}
if (map && realIndex < map->nextAvailable)
ret = map->table[realIndex].name;
+ LeaveCriticalSection(&mapCS);
return ret;
}
DWORD getInterfaceIndexByName(const char *name, PDWORD index)
{
- DWORD ndx;
+ DWORD ndx, ret;
BOOL found = FALSE;
if (!name)
@@ -382,6 +386,7 @@
if (!index)
return ERROR_INVALID_PARAMETER;
+ EnterCriticalSection(&mapCS);
for (ndx = 0; !found && gNonLoopbackInterfaceMap &&
ndx < gNonLoopbackInterfaceMap->nextAvailable; ndx++)
if (!strncmp(gNonLoopbackInterfaceMap->table[ndx].name, name, IFNAMSIZ)) {
@@ -394,10 +399,12 @@
found = TRUE;
*index = ndx | INDEX_IS_LOOPBACK;
}
+ LeaveCriticalSection(&mapCS);
if (found)
- return NO_ERROR;
+ ret = NO_ERROR;
else
- return ERROR_INVALID_DATA;
+ ret = ERROR_INVALID_DATA;
+ return ret;
}
static void addMapEntriesToIndexTable(InterfaceIndexTable *table,
@@ -420,28 +427,36 @@
InterfaceIndexTable *getInterfaceIndexTable(void)
{
- DWORD numInterfaces = getNumInterfaces();
- InterfaceIndexTable *ret = (InterfaceIndexTable *)calloc(1,
+ DWORD numInterfaces;
+ InterfaceIndexTable *ret;
+
+ EnterCriticalSection(&mapCS);
+ numInterfaces = getNumInterfaces();
+ ret = (InterfaceIndexTable *)calloc(1,
sizeof(InterfaceIndexTable) + (numInterfaces - 1) * sizeof(DWORD));
-
if (ret) {
ret->numAllocated = numInterfaces;
addMapEntriesToIndexTable(ret, gNonLoopbackInterfaceMap);
addMapEntriesToIndexTable(ret, gLoopbackInterfaceMap);
}
+ LeaveCriticalSection(&mapCS);
return ret;
}
InterfaceIndexTable *getNonLoopbackInterfaceIndexTable(void)
{
- DWORD numInterfaces = getNumNonLoopbackInterfaces();
- InterfaceIndexTable *ret = (InterfaceIndexTable *)calloc(1,
- sizeof(InterfaceIndexTable) + (numInterfaces - 1) * sizeof(DWORD));
+ DWORD numInterfaces;
+ InterfaceIndexTable *ret;
+ EnterCriticalSection(&mapCS);
+ numInterfaces = getNumNonLoopbackInterfaces();
+ ret = (InterfaceIndexTable *)calloc(1,
+ sizeof(InterfaceIndexTable) + (numInterfaces - 1) * sizeof(DWORD));
if (ret) {
ret->numAllocated = numInterfaces;
addMapEntriesToIndexTable(ret, gNonLoopbackInterfaceMap);
}
+ LeaveCriticalSection(&mapCS);
return ret;
}