minissdpc.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875
  1. /* $Id: minissdpc.c,v 1.33 2016/12/16 08:57:20 nanard Exp $ */
  2. /* vim: tabstop=4 shiftwidth=4 noexpandtab
  3. * Project : miniupnp
  4. * Web : http://miniupnp.free.fr/
  5. * Author : Thomas BERNARD
  6. * copyright (c) 2005-2016 Thomas Bernard
  7. * This software is subjet to the conditions detailed in the
  8. * provided LICENCE file. */
  9. /*#include <syslog.h>*/
  10. #include <stdio.h>
  11. #include <string.h>
  12. #include <stdlib.h>
  13. #include <sys/types.h>
  14. #if defined (__NetBSD__)
  15. #include <net/if.h>
  16. #endif
  17. #if defined(_WIN32) || defined(__amigaos__) || defined(__amigaos4__)
  18. #ifdef _WIN32
  19. #include <winsock2.h>
  20. #include <ws2tcpip.h>
  21. #include <io.h>
  22. #include <iphlpapi.h>
  23. #include <winsock.h>
  24. #define snprintf _snprintf
  25. #if !defined(_MSC_VER)
  26. #include <stdint.h>
  27. #else /* !defined(_MSC_VER) */
  28. typedef unsigned short uint16_t;
  29. #endif /* !defined(_MSC_VER) */
  30. #ifndef strncasecmp
  31. #if defined(_MSC_VER) && (_MSC_VER >= 1400)
  32. #define strncasecmp _memicmp
  33. #else /* defined(_MSC_VER) && (_MSC_VER >= 1400) */
  34. #define strncasecmp memicmp
  35. #endif /* defined(_MSC_VER) && (_MSC_VER >= 1400) */
  36. #endif /* #ifndef strncasecmp */
  37. #endif /* _WIN32 */
  38. #if defined(__amigaos__) || defined(__amigaos4__)
  39. #include <sys/socket.h>
  40. #endif /* defined(__amigaos__) || defined(__amigaos4__) */
  41. #if defined(__amigaos__)
  42. #define uint16_t unsigned short
  43. #endif /* defined(__amigaos__) */
  44. /* Hack */
  45. #define UNIX_PATH_LEN 108
  46. struct sockaddr_un {
  47. uint16_t sun_family;
  48. char sun_path[UNIX_PATH_LEN];
  49. };
  50. #else /* defined(_WIN32) || defined(__amigaos__) || defined(__amigaos4__) */
  51. #include <strings.h>
  52. #include <unistd.h>
  53. #include <sys/socket.h>
  54. #include <sys/param.h>
  55. #include <sys/time.h>
  56. #include <sys/un.h>
  57. #include <netinet/in.h>
  58. #include <arpa/inet.h>
  59. #include <netdb.h>
  60. #include <net/if.h>
  61. #define closesocket close
  62. #endif
  63. #ifdef _WIN32
  64. #define PRINT_SOCKET_ERROR(x) printf("Socket error: %s, %d\n", x, WSAGetLastError());
  65. #else
  66. #define PRINT_SOCKET_ERROR(x) perror(x)
  67. #endif
  68. #if !defined(__DragonFly__) && !defined(__OpenBSD__) && !defined(__NetBSD__) && !defined(__APPLE__) && !defined(_WIN32) && !defined(__CYGWIN__) && !defined(__sun) && !defined(__GNU__) && !defined(__FreeBSD_kernel__)
  69. #define HAS_IP_MREQN
  70. #endif
  71. #if !defined(HAS_IP_MREQN) && !defined(_WIN32)
  72. #include <sys/ioctl.h>
  73. #if defined(__sun)
  74. #include <sys/sockio.h>
  75. #endif
  76. #endif
  77. #if defined(HAS_IP_MREQN) && defined(NEED_STRUCT_IP_MREQN)
  78. /* Several versions of glibc don't define this structure,
  79. * define it here and compile with CFLAGS NEED_STRUCT_IP_MREQN */
  80. struct ip_mreqn
  81. {
  82. struct in_addr imr_multiaddr; /* IP multicast address of group */
  83. struct in_addr imr_address; /* local IP address of interface */
  84. int imr_ifindex; /* Interface index */
  85. };
  86. #endif
  87. #if defined(__amigaos__) || defined(__amigaos4__)
  88. /* Amiga OS specific stuff */
  89. #define TIMEVAL struct timeval
  90. #endif
  91. #include "minissdpc.h"
  92. #include "miniupnpc.h"
  93. #include "receivedata.h"
  94. #if !(defined(_WIN32) || defined(__amigaos__) || defined(__amigaos4__))
  95. #include "codelength.h"
  96. struct UPNPDev *
  97. getDevicesFromMiniSSDPD(const char * devtype, const char * socketpath, int * error)
  98. {
  99. struct UPNPDev * devlist = NULL;
  100. int s;
  101. int res;
  102. s = connectToMiniSSDPD(socketpath);
  103. if (s < 0) {
  104. if (error)
  105. *error = s;
  106. return NULL;
  107. }
  108. res = requestDevicesFromMiniSSDPD(s, devtype);
  109. if (res < 0) {
  110. if (error)
  111. *error = res;
  112. } else {
  113. devlist = receiveDevicesFromMiniSSDPD(s, error);
  114. }
  115. disconnectFromMiniSSDPD(s);
  116. return devlist;
  117. }
  118. /* macros used to read from unix socket */
  119. #define READ_BYTE_BUFFER(c) \
  120. if((int)bufferindex >= n) { \
  121. n = read(s, buffer, sizeof(buffer)); \
  122. if(n<=0) break; \
  123. bufferindex = 0; \
  124. } \
  125. c = buffer[bufferindex++];
  126. #ifndef MIN
  127. #define MIN(a, b) (((a) < (b)) ? (a) : (b))
  128. #endif /* MIN */
  129. #define READ_COPY_BUFFER(dst, len) \
  130. for(l = len, p = (unsigned char *)dst; l > 0; ) { \
  131. unsigned int lcopy; \
  132. if((int)bufferindex >= n) { \
  133. n = read(s, buffer, sizeof(buffer)); \
  134. if(n<=0) break; \
  135. bufferindex = 0; \
  136. } \
  137. lcopy = MIN(l, (n - bufferindex)); \
  138. memcpy(p, buffer + bufferindex, lcopy); \
  139. l -= lcopy; \
  140. p += lcopy; \
  141. bufferindex += lcopy; \
  142. }
  143. #define READ_DISCARD_BUFFER(len) \
  144. for(l = len; l > 0; ) { \
  145. unsigned int lcopy; \
  146. if(bufferindex >= n) { \
  147. n = read(s, buffer, sizeof(buffer)); \
  148. if(n<=0) break; \
  149. bufferindex = 0; \
  150. } \
  151. lcopy = MIN(l, (n - bufferindex)); \
  152. l -= lcopy; \
  153. bufferindex += lcopy; \
  154. }
  155. int
  156. connectToMiniSSDPD(const char * socketpath)
  157. {
  158. int s;
  159. struct sockaddr_un addr;
  160. #if defined(MINIUPNPC_SET_SOCKET_TIMEOUT) && !defined(__sun)
  161. struct timeval timeout;
  162. #endif /* #ifdef MINIUPNPC_SET_SOCKET_TIMEOUT */
  163. s = socket(AF_UNIX, SOCK_STREAM, 0);
  164. if(s < 0)
  165. {
  166. /*syslog(LOG_ERR, "socket(unix): %m");*/
  167. perror("socket(unix)");
  168. return MINISSDPC_SOCKET_ERROR;
  169. }
  170. #if defined(MINIUPNPC_SET_SOCKET_TIMEOUT) && !defined(__sun)
  171. /* setting a 3 seconds timeout */
  172. /* not supported for AF_UNIX sockets under Solaris */
  173. timeout.tv_sec = 3;
  174. timeout.tv_usec = 0;
  175. if(setsockopt(s, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(struct timeval)) < 0)
  176. {
  177. perror("setsockopt SO_RCVTIMEO unix");
  178. }
  179. timeout.tv_sec = 3;
  180. timeout.tv_usec = 0;
  181. if(setsockopt(s, SOL_SOCKET, SO_SNDTIMEO, &timeout, sizeof(struct timeval)) < 0)
  182. {
  183. perror("setsockopt SO_SNDTIMEO unix");
  184. }
  185. #endif /* #ifdef MINIUPNPC_SET_SOCKET_TIMEOUT */
  186. if(!socketpath)
  187. socketpath = "/var/run/minissdpd.sock";
  188. addr.sun_family = AF_UNIX;
  189. strncpy(addr.sun_path, socketpath, sizeof(addr.sun_path));
  190. /* TODO : check if we need to handle the EINTR */
  191. if(connect(s, (struct sockaddr *)&addr, sizeof(struct sockaddr_un)) < 0)
  192. {
  193. /*syslog(LOG_WARNING, "connect(\"%s\"): %m", socketpath);*/
  194. close(s);
  195. return MINISSDPC_SOCKET_ERROR;
  196. }
  197. return s;
  198. }
  199. int
  200. disconnectFromMiniSSDPD(int s)
  201. {
  202. if (close(s) < 0)
  203. return MINISSDPC_SOCKET_ERROR;
  204. return MINISSDPC_SUCCESS;
  205. }
  206. int
  207. requestDevicesFromMiniSSDPD(int s, const char * devtype)
  208. {
  209. unsigned char buffer[256];
  210. unsigned char * p;
  211. unsigned int stsize, l;
  212. stsize = strlen(devtype);
  213. if(stsize == 8 && 0 == memcmp(devtype, "ssdp:all", 8))
  214. {
  215. buffer[0] = 3; /* request type 3 : everything */
  216. }
  217. else
  218. {
  219. buffer[0] = 1; /* request type 1 : request devices/services by type */
  220. }
  221. p = buffer + 1;
  222. l = stsize; CODELENGTH(l, p);
  223. if(p + stsize > buffer + sizeof(buffer))
  224. {
  225. /* devtype is too long ! */
  226. #ifdef DEBUG
  227. fprintf(stderr, "devtype is too long ! stsize=%u sizeof(buffer)=%u\n",
  228. stsize, (unsigned)sizeof(buffer));
  229. #endif /* DEBUG */
  230. return MINISSDPC_INVALID_INPUT;
  231. }
  232. memcpy(p, devtype, stsize);
  233. p += stsize;
  234. if(write(s, buffer, p - buffer) < 0)
  235. {
  236. /*syslog(LOG_ERR, "write(): %m");*/
  237. perror("minissdpc.c: write()");
  238. return MINISSDPC_SOCKET_ERROR;
  239. }
  240. return MINISSDPC_SUCCESS;
  241. }
  242. struct UPNPDev *
  243. receiveDevicesFromMiniSSDPD(int s, int * error)
  244. {
  245. struct UPNPDev * tmp;
  246. struct UPNPDev * devlist = NULL;
  247. unsigned char buffer[256];
  248. ssize_t n;
  249. unsigned char * p;
  250. unsigned char * url;
  251. unsigned char * st;
  252. unsigned int bufferindex;
  253. unsigned int i, ndev;
  254. unsigned int urlsize, stsize, usnsize, l;
  255. n = read(s, buffer, sizeof(buffer));
  256. if(n<=0)
  257. {
  258. perror("minissdpc.c: read()");
  259. if (error)
  260. *error = MINISSDPC_SOCKET_ERROR;
  261. return NULL;
  262. }
  263. ndev = buffer[0];
  264. bufferindex = 1;
  265. for(i = 0; i < ndev; i++)
  266. {
  267. DECODELENGTH_READ(urlsize, READ_BYTE_BUFFER);
  268. if(n<=0) {
  269. if (error)
  270. *error = MINISSDPC_INVALID_SERVER_REPLY;
  271. return devlist;
  272. }
  273. #ifdef DEBUG
  274. printf(" urlsize=%u", urlsize);
  275. #endif /* DEBUG */
  276. url = malloc(urlsize);
  277. if(url == NULL) {
  278. if (error)
  279. *error = MINISSDPC_MEMORY_ERROR;
  280. return devlist;
  281. }
  282. READ_COPY_BUFFER(url, urlsize);
  283. if(n<=0) {
  284. if (error)
  285. *error = MINISSDPC_INVALID_SERVER_REPLY;
  286. goto free_url_and_return;
  287. }
  288. DECODELENGTH_READ(stsize, READ_BYTE_BUFFER);
  289. if(n<=0) {
  290. if (error)
  291. *error = MINISSDPC_INVALID_SERVER_REPLY;
  292. goto free_url_and_return;
  293. }
  294. #ifdef DEBUG
  295. printf(" stsize=%u", stsize);
  296. #endif /* DEBUG */
  297. st = malloc(stsize);
  298. if (st == NULL) {
  299. if (error)
  300. *error = MINISSDPC_MEMORY_ERROR;
  301. goto free_url_and_return;
  302. }
  303. READ_COPY_BUFFER(st, stsize);
  304. if(n<=0) {
  305. if (error)
  306. *error = MINISSDPC_INVALID_SERVER_REPLY;
  307. goto free_url_and_st_and_return;
  308. }
  309. DECODELENGTH_READ(usnsize, READ_BYTE_BUFFER);
  310. if(n<=0) {
  311. if (error)
  312. *error = MINISSDPC_INVALID_SERVER_REPLY;
  313. goto free_url_and_st_and_return;
  314. }
  315. #ifdef DEBUG
  316. printf(" usnsize=%u\n", usnsize);
  317. #endif /* DEBUG */
  318. tmp = (struct UPNPDev *)malloc(sizeof(struct UPNPDev)+urlsize+stsize+usnsize);
  319. if(tmp == NULL) {
  320. if (error)
  321. *error = MINISSDPC_MEMORY_ERROR;
  322. goto free_url_and_st_and_return;
  323. }
  324. tmp->pNext = devlist;
  325. tmp->descURL = tmp->buffer;
  326. tmp->st = tmp->buffer + 1 + urlsize;
  327. memcpy(tmp->buffer, url, urlsize);
  328. tmp->buffer[urlsize] = '\0';
  329. memcpy(tmp->st, st, stsize);
  330. tmp->buffer[urlsize+1+stsize] = '\0';
  331. free(url);
  332. free(st);
  333. url = NULL;
  334. st = NULL;
  335. tmp->usn = tmp->buffer + 1 + urlsize + 1 + stsize;
  336. READ_COPY_BUFFER(tmp->usn, usnsize);
  337. if(n<=0) {
  338. if (error)
  339. *error = MINISSDPC_INVALID_SERVER_REPLY;
  340. goto free_tmp_and_return;
  341. }
  342. tmp->buffer[urlsize+1+stsize+1+usnsize] = '\0';
  343. tmp->scope_id = 0; /* default value. scope_id is not available with MiniSSDPd */
  344. devlist = tmp;
  345. }
  346. if (error)
  347. *error = MINISSDPC_SUCCESS;
  348. return devlist;
  349. free_url_and_st_and_return:
  350. free(st);
  351. free_url_and_return:
  352. free(url);
  353. return devlist;
  354. free_tmp_and_return:
  355. free(tmp);
  356. return devlist;
  357. }
  358. #endif /* !(defined(_WIN32) || defined(__amigaos__) || defined(__amigaos4__)) */
  359. /* parseMSEARCHReply()
  360. * the last 4 arguments are filled during the parsing :
  361. * - location/locationsize : "location:" field of the SSDP reply packet
  362. * - st/stsize : "st:" field of the SSDP reply packet.
  363. * The strings are NOT null terminated */
  364. static void
  365. parseMSEARCHReply(const char * reply, int size,
  366. const char * * location, int * locationsize,
  367. const char * * st, int * stsize,
  368. const char * * usn, int * usnsize)
  369. {
  370. int a, b, i;
  371. i = 0;
  372. a = i; /* start of the line */
  373. b = 0; /* end of the "header" (position of the colon) */
  374. while(i<size)
  375. {
  376. switch(reply[i])
  377. {
  378. case ':':
  379. if(b==0)
  380. {
  381. b = i; /* end of the "header" */
  382. /*for(j=a; j<b; j++)
  383. {
  384. putchar(reply[j]);
  385. }
  386. */
  387. }
  388. break;
  389. case '\x0a':
  390. case '\x0d':
  391. if(b!=0)
  392. {
  393. /*for(j=b+1; j<i; j++)
  394. {
  395. putchar(reply[j]);
  396. }
  397. putchar('\n');*/
  398. /* skip the colon and white spaces */
  399. do { b++; } while(reply[b]==' ');
  400. if(0==strncasecmp(reply+a, "location", 8))
  401. {
  402. *location = reply+b;
  403. *locationsize = i-b;
  404. }
  405. else if(0==strncasecmp(reply+a, "st", 2))
  406. {
  407. *st = reply+b;
  408. *stsize = i-b;
  409. }
  410. else if(0==strncasecmp(reply+a, "usn", 3))
  411. {
  412. *usn = reply+b;
  413. *usnsize = i-b;
  414. }
  415. b = 0;
  416. }
  417. a = i+1;
  418. break;
  419. default:
  420. break;
  421. }
  422. i++;
  423. }
  424. }
  425. /* port upnp discover : SSDP protocol */
  426. #define SSDP_PORT 1900
  427. #define XSTR(s) STR(s)
  428. #define STR(s) #s
  429. #define UPNP_MCAST_ADDR "239.255.255.250"
  430. /* for IPv6 */
  431. #define UPNP_MCAST_LL_ADDR "FF02::C" /* link-local */
  432. #define UPNP_MCAST_SL_ADDR "FF05::C" /* site-local */
  433. /* direct discovery if minissdpd responses are not sufficient */
  434. /* ssdpDiscoverDevices() :
  435. * return a chained list of all devices found or NULL if
  436. * no devices was found.
  437. * It is up to the caller to free the chained list
  438. * delay is in millisecond (poll).
  439. * UDA v1.1 says :
  440. * The TTL for the IP packet SHOULD default to 2 and
  441. * SHOULD be configurable. */
  442. struct UPNPDev *
  443. ssdpDiscoverDevices(const char * const deviceTypes[],
  444. int delay, const char * multicastif,
  445. int localport,
  446. int ipv6, unsigned char ttl,
  447. int * error,
  448. int searchalltypes)
  449. {
  450. struct UPNPDev * tmp;
  451. struct UPNPDev * devlist = 0;
  452. unsigned int scope_id = 0;
  453. int opt = 1;
  454. static const char MSearchMsgFmt[] =
  455. "M-SEARCH * HTTP/1.1\r\n"
  456. "HOST: %s:" XSTR(SSDP_PORT) "\r\n"
  457. "ST: %s\r\n"
  458. "MAN: \"ssdp:discover\"\r\n"
  459. "MX: %u\r\n"
  460. "\r\n";
  461. int deviceIndex;
  462. char bufr[1536]; /* reception and emission buffer */
  463. int sudp;
  464. int n;
  465. struct sockaddr_storage sockudp_r;
  466. unsigned int mx;
  467. #ifdef NO_GETADDRINFO
  468. struct sockaddr_storage sockudp_w;
  469. #else
  470. int rv;
  471. struct addrinfo hints, *servinfo, *p;
  472. #endif
  473. #ifdef _WIN32
  474. MIB_IPFORWARDROW ip_forward;
  475. unsigned long _ttl = (unsigned long)ttl;
  476. #endif
  477. int linklocal = 1;
  478. if(error)
  479. *error = MINISSDPC_UNKNOWN_ERROR;
  480. if(localport==UPNP_LOCAL_PORT_SAME)
  481. localport = SSDP_PORT;
  482. #ifdef _WIN32
  483. sudp = socket(ipv6 ? PF_INET6 : PF_INET, SOCK_DGRAM, IPPROTO_UDP);
  484. #else
  485. sudp = socket(ipv6 ? PF_INET6 : PF_INET, SOCK_DGRAM, 0);
  486. #endif
  487. if(sudp < 0)
  488. {
  489. if(error)
  490. *error = MINISSDPC_SOCKET_ERROR;
  491. PRINT_SOCKET_ERROR("socket");
  492. return NULL;
  493. }
  494. /* reception */
  495. memset(&sockudp_r, 0, sizeof(struct sockaddr_storage));
  496. if(ipv6) {
  497. struct sockaddr_in6 * p = (struct sockaddr_in6 *)&sockudp_r;
  498. p->sin6_family = AF_INET6;
  499. if(localport > 0 && localport < 65536)
  500. p->sin6_port = htons((unsigned short)localport);
  501. p->sin6_addr = in6addr_any; /* in6addr_any is not available with MinGW32 3.4.2 */
  502. } else {
  503. struct sockaddr_in * p = (struct sockaddr_in *)&sockudp_r;
  504. p->sin_family = AF_INET;
  505. if(localport > 0 && localport < 65536)
  506. p->sin_port = htons((unsigned short)localport);
  507. p->sin_addr.s_addr = INADDR_ANY;
  508. }
  509. #ifdef _WIN32
  510. /* This code could help us to use the right Network interface for
  511. * SSDP multicast traffic */
  512. /* Get IP associated with the index given in the ip_forward struct
  513. * in order to give this ip to setsockopt(sudp, IPPROTO_IP, IP_MULTICAST_IF) */
  514. if(!ipv6
  515. && (GetBestRoute(inet_addr("223.255.255.255"), 0, &ip_forward) == NO_ERROR)) {
  516. DWORD dwRetVal = 0;
  517. PMIB_IPADDRTABLE pIPAddrTable;
  518. DWORD dwSize = 0;
  519. #ifdef DEBUG
  520. IN_ADDR IPAddr;
  521. #endif
  522. int i;
  523. #ifdef DEBUG
  524. printf("ifIndex=%lu nextHop=%lx \n", ip_forward.dwForwardIfIndex, ip_forward.dwForwardNextHop);
  525. #endif
  526. pIPAddrTable = (MIB_IPADDRTABLE *) malloc(sizeof (MIB_IPADDRTABLE));
  527. if(pIPAddrTable) {
  528. if (GetIpAddrTable(pIPAddrTable, &dwSize, 0) == ERROR_INSUFFICIENT_BUFFER) {
  529. free(pIPAddrTable);
  530. pIPAddrTable = (MIB_IPADDRTABLE *) malloc(dwSize);
  531. }
  532. }
  533. if(pIPAddrTable) {
  534. dwRetVal = GetIpAddrTable( pIPAddrTable, &dwSize, 0 );
  535. if (dwRetVal == NO_ERROR) {
  536. #ifdef DEBUG
  537. printf("\tNum Entries: %ld\n", pIPAddrTable->dwNumEntries);
  538. #endif
  539. for (i=0; i < (int) pIPAddrTable->dwNumEntries; i++) {
  540. #ifdef DEBUG
  541. printf("\n\tInterface Index[%d]:\t%ld\n", i, pIPAddrTable->table[i].dwIndex);
  542. IPAddr.S_un.S_addr = (u_long) pIPAddrTable->table[i].dwAddr;
  543. printf("\tIP Address[%d]: \t%s\n", i, inet_ntoa(IPAddr) );
  544. IPAddr.S_un.S_addr = (u_long) pIPAddrTable->table[i].dwMask;
  545. printf("\tSubnet Mask[%d]: \t%s\n", i, inet_ntoa(IPAddr) );
  546. IPAddr.S_un.S_addr = (u_long) pIPAddrTable->table[i].dwBCastAddr;
  547. printf("\tBroadCast[%d]: \t%s (%ld)\n", i, inet_ntoa(IPAddr), pIPAddrTable->table[i].dwBCastAddr);
  548. printf("\tReassembly size[%d]:\t%ld\n", i, pIPAddrTable->table[i].dwReasmSize);
  549. printf("\tType and State[%d]:", i);
  550. printf("\n");
  551. #endif
  552. if (pIPAddrTable->table[i].dwIndex == ip_forward.dwForwardIfIndex) {
  553. /* Set the address of this interface to be used */
  554. struct in_addr mc_if;
  555. memset(&mc_if, 0, sizeof(mc_if));
  556. mc_if.s_addr = pIPAddrTable->table[i].dwAddr;
  557. if(setsockopt(sudp, IPPROTO_IP, IP_MULTICAST_IF, (const char *)&mc_if, sizeof(mc_if)) < 0) {
  558. PRINT_SOCKET_ERROR("setsockopt");
  559. }
  560. ((struct sockaddr_in *)&sockudp_r)->sin_addr.s_addr = pIPAddrTable->table[i].dwAddr;
  561. #ifndef DEBUG
  562. break;
  563. #endif
  564. }
  565. }
  566. }
  567. free(pIPAddrTable);
  568. pIPAddrTable = NULL;
  569. }
  570. }
  571. #endif /* _WIN32 */
  572. #ifdef _WIN32
  573. if (setsockopt(sudp, SOL_SOCKET, SO_REUSEADDR, (const char *)&opt, sizeof (opt)) < 0)
  574. #else
  575. if (setsockopt(sudp, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof (opt)) < 0)
  576. #endif
  577. {
  578. if(error)
  579. *error = MINISSDPC_SOCKET_ERROR;
  580. PRINT_SOCKET_ERROR("setsockopt(SO_REUSEADDR,...)");
  581. return NULL;
  582. }
  583. #ifdef _WIN32
  584. if(setsockopt(sudp, IPPROTO_IP, IP_MULTICAST_TTL, (const char *)&_ttl, sizeof(_ttl)) < 0)
  585. #else /* _WIN32 */
  586. if(setsockopt(sudp, IPPROTO_IP, IP_MULTICAST_TTL, &ttl, sizeof(ttl)) < 0)
  587. #endif /* _WIN32 */
  588. {
  589. /* not a fatal error */
  590. PRINT_SOCKET_ERROR("setsockopt(IP_MULTICAST_TTL,...)");
  591. }
  592. if(multicastif)
  593. {
  594. if(ipv6) {
  595. #if !defined(_WIN32)
  596. /* according to MSDN, if_nametoindex() is supported since
  597. * MS Windows Vista and MS Windows Server 2008.
  598. * http://msdn.microsoft.com/en-us/library/bb408409%28v=vs.85%29.aspx */
  599. unsigned int ifindex = if_nametoindex(multicastif); /* eth0, etc. */
  600. if(setsockopt(sudp, IPPROTO_IPV6, IPV6_MULTICAST_IF, &ifindex, sizeof(ifindex)) < 0)
  601. {
  602. PRINT_SOCKET_ERROR("setsockopt IPV6_MULTICAST_IF");
  603. }
  604. #else
  605. #ifdef DEBUG
  606. printf("Setting of multicast interface not supported in IPv6 under Windows.\n");
  607. #endif
  608. #endif
  609. } else {
  610. struct in_addr mc_if;
  611. mc_if.s_addr = inet_addr(multicastif); /* ex: 192.168.x.x */
  612. if(mc_if.s_addr != INADDR_NONE)
  613. {
  614. ((struct sockaddr_in *)&sockudp_r)->sin_addr.s_addr = mc_if.s_addr;
  615. if(setsockopt(sudp, IPPROTO_IP, IP_MULTICAST_IF, (const char *)&mc_if, sizeof(mc_if)) < 0)
  616. {
  617. PRINT_SOCKET_ERROR("setsockopt IP_MULTICAST_IF");
  618. }
  619. } else {
  620. #ifdef HAS_IP_MREQN
  621. /* was not an ip address, try with an interface name */
  622. struct ip_mreqn reqn; /* only defined with -D_BSD_SOURCE or -D_GNU_SOURCE */
  623. memset(&reqn, 0, sizeof(struct ip_mreqn));
  624. reqn.imr_ifindex = if_nametoindex(multicastif);
  625. if(setsockopt(sudp, IPPROTO_IP, IP_MULTICAST_IF, (const char *)&reqn, sizeof(reqn)) < 0)
  626. {
  627. PRINT_SOCKET_ERROR("setsockopt IP_MULTICAST_IF");
  628. }
  629. #elif !defined(_WIN32)
  630. struct ifreq ifr;
  631. int ifrlen = sizeof(ifr);
  632. strncpy(ifr.ifr_name, multicastif, IFNAMSIZ);
  633. ifr.ifr_name[IFNAMSIZ-1] = '\0';
  634. if(ioctl(sudp, SIOCGIFADDR, &ifr, &ifrlen) < 0)
  635. {
  636. PRINT_SOCKET_ERROR("ioctl(...SIOCGIFADDR...)");
  637. }
  638. mc_if.s_addr = ((struct sockaddr_in *)&ifr.ifr_addr)->sin_addr.s_addr;
  639. if(setsockopt(sudp, IPPROTO_IP, IP_MULTICAST_IF, (const char *)&mc_if, sizeof(mc_if)) < 0)
  640. {
  641. PRINT_SOCKET_ERROR("setsockopt IP_MULTICAST_IF");
  642. }
  643. #else /* _WIN32 */
  644. #ifdef DEBUG
  645. printf("Setting of multicast interface not supported with interface name.\n");
  646. #endif
  647. #endif /* #ifdef HAS_IP_MREQN / !defined(_WIN32) */
  648. }
  649. }
  650. }
  651. /* Before sending the packed, we first "bind" in order to be able
  652. * to receive the response */
  653. if (bind(sudp, (const struct sockaddr *)&sockudp_r,
  654. ipv6 ? sizeof(struct sockaddr_in6) : sizeof(struct sockaddr_in)) != 0)
  655. {
  656. if(error)
  657. *error = MINISSDPC_SOCKET_ERROR;
  658. PRINT_SOCKET_ERROR("bind");
  659. closesocket(sudp);
  660. return NULL;
  661. }
  662. if(error)
  663. *error = MINISSDPC_SUCCESS;
  664. /* Calculating maximum response time in seconds */
  665. mx = ((unsigned int)delay) / 1000u;
  666. if(mx == 0) {
  667. mx = 1;
  668. delay = 1000;
  669. }
  670. /* receiving SSDP response packet */
  671. for(deviceIndex = 0; deviceTypes[deviceIndex]; deviceIndex++) {
  672. /* sending the SSDP M-SEARCH packet */
  673. n = snprintf(bufr, sizeof(bufr),
  674. MSearchMsgFmt,
  675. ipv6 ?
  676. (linklocal ? "[" UPNP_MCAST_LL_ADDR "]" : "[" UPNP_MCAST_SL_ADDR "]")
  677. : UPNP_MCAST_ADDR,
  678. deviceTypes[deviceIndex], mx);
  679. if ((unsigned int)n >= sizeof(bufr)) {
  680. if(error)
  681. *error = MINISSDPC_MEMORY_ERROR;
  682. goto error;
  683. }
  684. #ifdef DEBUG
  685. /*printf("Sending %s", bufr);*/
  686. printf("Sending M-SEARCH request to %s with ST: %s\n",
  687. ipv6 ?
  688. (linklocal ? "[" UPNP_MCAST_LL_ADDR "]" : "[" UPNP_MCAST_SL_ADDR "]")
  689. : UPNP_MCAST_ADDR,
  690. deviceTypes[deviceIndex]);
  691. #endif
  692. #ifdef NO_GETADDRINFO
  693. /* the following code is not using getaddrinfo */
  694. /* emission */
  695. memset(&sockudp_w, 0, sizeof(struct sockaddr_storage));
  696. if(ipv6) {
  697. struct sockaddr_in6 * p = (struct sockaddr_in6 *)&sockudp_w;
  698. p->sin6_family = AF_INET6;
  699. p->sin6_port = htons(SSDP_PORT);
  700. inet_pton(AF_INET6,
  701. linklocal ? UPNP_MCAST_LL_ADDR : UPNP_MCAST_SL_ADDR,
  702. &(p->sin6_addr));
  703. } else {
  704. struct sockaddr_in * p = (struct sockaddr_in *)&sockudp_w;
  705. p->sin_family = AF_INET;
  706. p->sin_port = htons(SSDP_PORT);
  707. p->sin_addr.s_addr = inet_addr(UPNP_MCAST_ADDR);
  708. }
  709. n = sendto(sudp, bufr, n, 0, &sockudp_w,
  710. ipv6 ? sizeof(struct sockaddr_in6) : sizeof(struct sockaddr_in));
  711. if (n < 0) {
  712. if(error)
  713. *error = MINISSDPC_SOCKET_ERROR;
  714. PRINT_SOCKET_ERROR("sendto");
  715. break;
  716. }
  717. #else /* #ifdef NO_GETADDRINFO */
  718. memset(&hints, 0, sizeof(hints));
  719. hints.ai_family = AF_UNSPEC; /* AF_INET6 or AF_INET */
  720. hints.ai_socktype = SOCK_DGRAM;
  721. /*hints.ai_flags = */
  722. if ((rv = getaddrinfo(ipv6
  723. ? (linklocal ? UPNP_MCAST_LL_ADDR : UPNP_MCAST_SL_ADDR)
  724. : UPNP_MCAST_ADDR,
  725. XSTR(SSDP_PORT), &hints, &servinfo)) != 0) {
  726. if(error)
  727. *error = MINISSDPC_SOCKET_ERROR;
  728. #ifdef _WIN32
  729. fprintf(stderr, "getaddrinfo() failed: %d\n", rv);
  730. #else
  731. fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv));
  732. #endif
  733. break;
  734. }
  735. for(p = servinfo; p; p = p->ai_next) {
  736. n = sendto(sudp, bufr, n, 0, p->ai_addr, p->ai_addrlen);
  737. if (n < 0) {
  738. #ifdef DEBUG
  739. char hbuf[NI_MAXHOST], sbuf[NI_MAXSERV];
  740. if (getnameinfo(p->ai_addr, p->ai_addrlen, hbuf, sizeof(hbuf), sbuf,
  741. sizeof(sbuf), NI_NUMERICHOST | NI_NUMERICSERV) == 0) {
  742. fprintf(stderr, "host:%s port:%s\n", hbuf, sbuf);
  743. }
  744. #endif
  745. PRINT_SOCKET_ERROR("sendto");
  746. continue;
  747. }
  748. }
  749. freeaddrinfo(servinfo);
  750. if(n < 0) {
  751. if(error)
  752. *error = MINISSDPC_SOCKET_ERROR;
  753. break;
  754. }
  755. #endif /* #ifdef NO_GETADDRINFO */
  756. /* Waiting for SSDP REPLY packet to M-SEARCH
  757. * if searchalltypes is set, enter the loop only
  758. * when the last deviceType is reached */
  759. if(!searchalltypes || !deviceTypes[deviceIndex + 1]) do {
  760. n = receivedata(sudp, bufr, sizeof(bufr), delay, &scope_id);
  761. if (n < 0) {
  762. /* error */
  763. if(error)
  764. *error = MINISSDPC_SOCKET_ERROR;
  765. goto error;
  766. } else if (n == 0) {
  767. /* no data or Time Out */
  768. #ifdef DEBUG
  769. printf("NODATA or TIMEOUT\n");
  770. #endif /* DEBUG */
  771. if (devlist && !searchalltypes) {
  772. /* found some devices, stop now*/
  773. if(error)
  774. *error = MINISSDPC_SUCCESS;
  775. goto error;
  776. }
  777. } else {
  778. const char * descURL=NULL;
  779. int urlsize=0;
  780. const char * st=NULL;
  781. int stsize=0;
  782. const char * usn=NULL;
  783. int usnsize=0;
  784. parseMSEARCHReply(bufr, n, &descURL, &urlsize, &st, &stsize, &usn, &usnsize);
  785. if(st&&descURL) {
  786. #ifdef DEBUG
  787. printf("M-SEARCH Reply:\n ST: %.*s\n USN: %.*s\n Location: %.*s\n",
  788. stsize, st, usnsize, (usn?usn:""), urlsize, descURL);
  789. #endif /* DEBUG */
  790. for(tmp=devlist; tmp; tmp = tmp->pNext) {
  791. if(memcmp(tmp->descURL, descURL, urlsize) == 0 &&
  792. tmp->descURL[urlsize] == '\0' &&
  793. memcmp(tmp->st, st, stsize) == 0 &&
  794. tmp->st[stsize] == '\0' &&
  795. (usnsize == 0 || memcmp(tmp->usn, usn, usnsize) == 0) &&
  796. tmp->usn[usnsize] == '\0')
  797. break;
  798. }
  799. /* at the exit of the loop above, tmp is null if
  800. * no duplicate device was found */
  801. if(tmp)
  802. continue;
  803. tmp = (struct UPNPDev *)malloc(sizeof(struct UPNPDev)+urlsize+stsize+usnsize);
  804. if(!tmp) {
  805. /* memory allocation error */
  806. if(error)
  807. *error = MINISSDPC_MEMORY_ERROR;
  808. goto error;
  809. }
  810. tmp->pNext = devlist;
  811. tmp->descURL = tmp->buffer;
  812. tmp->st = tmp->buffer + 1 + urlsize;
  813. tmp->usn = tmp->st + 1 + stsize;
  814. memcpy(tmp->buffer, descURL, urlsize);
  815. tmp->buffer[urlsize] = '\0';
  816. memcpy(tmp->st, st, stsize);
  817. tmp->buffer[urlsize+1+stsize] = '\0';
  818. if(usn != NULL)
  819. memcpy(tmp->usn, usn, usnsize);
  820. tmp->buffer[urlsize+1+stsize+1+usnsize] = '\0';
  821. tmp->scope_id = scope_id;
  822. devlist = tmp;
  823. }
  824. }
  825. } while(n > 0);
  826. if(ipv6) {
  827. /* switch linklocal flag */
  828. if(linklocal) {
  829. linklocal = 0;
  830. --deviceIndex;
  831. } else {
  832. linklocal = 1;
  833. }
  834. }
  835. }
  836. error:
  837. closesocket(sudp);
  838. return devlist;
  839. }