|
@@ -3,7 +3,7 @@
|
|
|
* Copyright (c) 2000-2003 Intel Corporation
|
|
|
* Copyright (c) 2006-2007 Sony Corporation
|
|
|
* Copyright (c) 2008-2009 Atheros Communications
|
|
|
- * Copyright (c) 2009, Jouni Malinen <j@w1.fi>
|
|
|
+ * Copyright (c) 2009-2013, Jouni Malinen <j@w1.fi>
|
|
|
*
|
|
|
* See wps_upnp.c for more details on licensing and code history.
|
|
|
*/
|
|
@@ -13,6 +13,9 @@
|
|
|
#include <fcntl.h>
|
|
|
#include <sys/ioctl.h>
|
|
|
#include <net/route.h>
|
|
|
+#ifdef __linux__
|
|
|
+#include <net/if.h>
|
|
|
+#endif /* __linux__ */
|
|
|
|
|
|
#include "common.h"
|
|
|
#include "uuid.h"
|
|
@@ -854,7 +857,7 @@ fail:
|
|
|
}
|
|
|
|
|
|
|
|
|
-int ssdp_open_multicast_sock(u32 ip_addr)
|
|
|
+int ssdp_open_multicast_sock(u32 ip_addr, const char *forced_ifname)
|
|
|
{
|
|
|
int sd;
|
|
|
/* per UPnP-arch-DeviceArchitecture, 1. Discovery, keep IP packet
|
|
@@ -865,6 +868,22 @@ int ssdp_open_multicast_sock(u32 ip_addr)
|
|
|
if (sd < 0)
|
|
|
return -1;
|
|
|
|
|
|
+ if (forced_ifname) {
|
|
|
+#ifdef __linux__
|
|
|
+ struct ifreq req;
|
|
|
+ os_memset(&req, 0, sizeof(req));
|
|
|
+ os_strlcpy(req.ifr_name, forced_ifname, sizeof(req.ifr_name));
|
|
|
+ if (setsockopt(sd, SOL_SOCKET, SO_BINDTODEVICE, &req,
|
|
|
+ sizeof(req)) < 0) {
|
|
|
+ wpa_printf(MSG_INFO, "WPS UPnP: Failed to bind "
|
|
|
+ "multicast socket to ifname %s: %s",
|
|
|
+ forced_ifname, strerror(errno));
|
|
|
+ close(sd);
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
+#endif /* __linux__ */
|
|
|
+ }
|
|
|
+
|
|
|
#if 0 /* maybe ok if we sometimes block on writes */
|
|
|
if (fcntl(sd, F_SETFL, O_NONBLOCK) != 0) {
|
|
|
close(sd);
|
|
@@ -924,7 +943,7 @@ int ssdp_open_multicast_sock(u32 ip_addr)
|
|
|
*/
|
|
|
int ssdp_open_multicast(struct upnp_wps_device_sm *sm)
|
|
|
{
|
|
|
- sm->multicast_sd = ssdp_open_multicast_sock(sm->ip_addr);
|
|
|
+ sm->multicast_sd = ssdp_open_multicast_sock(sm->ip_addr, NULL);
|
|
|
if (sm->multicast_sd < 0)
|
|
|
return -1;
|
|
|
return 0;
|