multicast.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Gustavo Lopes <cataphract@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* $Id$ */
  19. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #include "php_network.h"
  24. #ifdef PHP_WIN32
  25. # include "windows_common.h"
  26. #else
  27. #include <sys/socket.h>
  28. #include <sys/ioctl.h>
  29. #include <net/if.h>
  30. #ifdef HAVE_SYS_SOCKIO_H
  31. #include <sys/sockio.h>
  32. #endif
  33. #include <netinet/in.h>
  34. #include <arpa/inet.h>
  35. #endif
  36. #include "php_sockets.h"
  37. #include "multicast.h"
  38. #include "sockaddr_conv.h"
  39. #include "main/php_network.h"
  40. enum source_op {
  41. JOIN_SOURCE,
  42. LEAVE_SOURCE,
  43. BLOCK_SOURCE,
  44. UNBLOCK_SOURCE
  45. };
  46. static int _php_mcast_join_leave(php_socket *sock, int level, struct sockaddr *group, socklen_t group_len, unsigned int if_index, int join TSRMLS_DC);
  47. #ifdef HAS_MCAST_EXT
  48. static int _php_mcast_source_op(php_socket *sock, int level, struct sockaddr *group, socklen_t group_len, struct sockaddr *source, socklen_t source_len, unsigned int if_index, enum source_op sop TSRMLS_DC);
  49. #endif
  50. #ifdef RFC3678_API
  51. static int _php_source_op_to_rfc3678_op(enum source_op sop);
  52. #elif HAS_MCAST_EXT
  53. static const char *_php_source_op_to_string(enum source_op sop);
  54. static int _php_source_op_to_ipv4_op(enum source_op sop);
  55. #endif
  56. int php_string_to_if_index(const char *val, unsigned *out TSRMLS_DC)
  57. {
  58. #if HAVE_IF_NAMETOINDEX
  59. unsigned int ind;
  60. ind = if_nametoindex(val);
  61. if (ind == 0) {
  62. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  63. "no interface with name \"%s\" could be found", val);
  64. return FAILURE;
  65. } else {
  66. *out = ind;
  67. return SUCCESS;
  68. }
  69. #else
  70. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  71. "this platform does not support looking up an interface by "
  72. "name, an integer interface index must be supplied instead");
  73. return FAILURE;
  74. #endif
  75. }
  76. static int php_get_if_index_from_zval(zval *val, unsigned *out TSRMLS_DC)
  77. {
  78. int ret;
  79. if (Z_TYPE_P(val) == IS_LONG) {
  80. if (Z_LVAL_P(val) < 0 || Z_LVAL_P(val) > UINT_MAX) {
  81. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  82. "the interface index cannot be negative or larger than %u;"
  83. " given %ld", UINT_MAX, Z_LVAL_P(val));
  84. ret = FAILURE;
  85. } else {
  86. *out = Z_LVAL_P(val);
  87. ret = SUCCESS;
  88. }
  89. } else {
  90. zval_add_ref(&val);
  91. convert_to_string_ex(&val);
  92. ret = php_string_to_if_index(Z_STRVAL_P(val), out TSRMLS_CC);
  93. zval_ptr_dtor(&val);
  94. }
  95. return ret;
  96. }
  97. static int php_get_if_index_from_array(const HashTable *ht, const char *key,
  98. php_socket *sock, unsigned int *if_index TSRMLS_DC)
  99. {
  100. zval **val;
  101. if (zend_hash_find(ht, key, strlen(key) + 1, (void **)&val) == FAILURE) {
  102. *if_index = 0; /* default: 0 */
  103. return SUCCESS;
  104. }
  105. return php_get_if_index_from_zval(*val, if_index TSRMLS_CC);
  106. }
  107. static int php_get_address_from_array(const HashTable *ht, const char *key,
  108. php_socket *sock, php_sockaddr_storage *ss, socklen_t *ss_len TSRMLS_DC)
  109. {
  110. zval **val,
  111. *valcp;
  112. if (zend_hash_find(ht, key, strlen(key) + 1, (void **)&val) == FAILURE) {
  113. php_error_docref(NULL TSRMLS_CC, E_WARNING, "no key \"%s\" passed in optval", key);
  114. return FAILURE;
  115. }
  116. valcp = *val;
  117. zval_add_ref(&valcp);
  118. convert_to_string_ex(val);
  119. if (!php_set_inet46_addr(ss, ss_len, Z_STRVAL_P(valcp), sock TSRMLS_CC)) {
  120. zval_ptr_dtor(&valcp);
  121. return FAILURE;
  122. }
  123. zval_ptr_dtor(&valcp);
  124. return SUCCESS;
  125. }
  126. static int php_do_mcast_opt(php_socket *php_sock, int level, int optname, zval **arg4 TSRMLS_DC)
  127. {
  128. HashTable *opt_ht;
  129. unsigned int if_index;
  130. int retval;
  131. int (*mcast_req_fun)(php_socket *, int, struct sockaddr *, socklen_t,
  132. unsigned TSRMLS_DC);
  133. #ifdef HAS_MCAST_EXT
  134. int (*mcast_sreq_fun)(php_socket *, int, struct sockaddr *, socklen_t,
  135. struct sockaddr *, socklen_t, unsigned TSRMLS_DC);
  136. #endif
  137. switch (optname) {
  138. case PHP_MCAST_JOIN_GROUP:
  139. mcast_req_fun = &php_mcast_join;
  140. goto mcast_req_fun;
  141. case PHP_MCAST_LEAVE_GROUP:
  142. {
  143. php_sockaddr_storage group = {0};
  144. socklen_t glen;
  145. mcast_req_fun = &php_mcast_leave;
  146. mcast_req_fun:
  147. convert_to_array_ex(arg4);
  148. opt_ht = HASH_OF(*arg4);
  149. if (php_get_address_from_array(opt_ht, "group", php_sock, &group,
  150. &glen TSRMLS_CC) == FAILURE) {
  151. return FAILURE;
  152. }
  153. if (php_get_if_index_from_array(opt_ht, "interface", php_sock,
  154. &if_index TSRMLS_CC) == FAILURE) {
  155. return FAILURE;
  156. }
  157. retval = mcast_req_fun(php_sock, level, (struct sockaddr*)&group,
  158. glen, if_index TSRMLS_CC);
  159. break;
  160. }
  161. #ifdef HAS_MCAST_EXT
  162. case PHP_MCAST_BLOCK_SOURCE:
  163. mcast_sreq_fun = &php_mcast_block_source;
  164. goto mcast_sreq_fun;
  165. case PHP_MCAST_UNBLOCK_SOURCE:
  166. mcast_sreq_fun = &php_mcast_unblock_source;
  167. goto mcast_sreq_fun;
  168. case PHP_MCAST_JOIN_SOURCE_GROUP:
  169. mcast_sreq_fun = &php_mcast_join_source;
  170. goto mcast_sreq_fun;
  171. case PHP_MCAST_LEAVE_SOURCE_GROUP:
  172. {
  173. php_sockaddr_storage group = {0},
  174. source = {0};
  175. socklen_t glen,
  176. slen;
  177. mcast_sreq_fun = &php_mcast_leave_source;
  178. mcast_sreq_fun:
  179. convert_to_array_ex(arg4);
  180. opt_ht = HASH_OF(*arg4);
  181. if (php_get_address_from_array(opt_ht, "group", php_sock, &group,
  182. &glen TSRMLS_CC) == FAILURE) {
  183. return FAILURE;
  184. }
  185. if (php_get_address_from_array(opt_ht, "source", php_sock, &source,
  186. &slen TSRMLS_CC) == FAILURE) {
  187. return FAILURE;
  188. }
  189. if (php_get_if_index_from_array(opt_ht, "interface", php_sock,
  190. &if_index TSRMLS_CC) == FAILURE) {
  191. return FAILURE;
  192. }
  193. retval = mcast_sreq_fun(php_sock, level, (struct sockaddr*)&group,
  194. glen, (struct sockaddr*)&source, slen, if_index TSRMLS_CC);
  195. break;
  196. }
  197. #endif
  198. default:
  199. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  200. "unexpected option in php_do_mcast_opt (level %d, option %d). "
  201. "This is a bug.", level, optname);
  202. return FAILURE;
  203. }
  204. if (retval != 0) {
  205. if (retval != -2) { /* error, but message already emitted */
  206. PHP_SOCKET_ERROR(php_sock, "unable to set socket option", errno);
  207. }
  208. return FAILURE;
  209. }
  210. return SUCCESS;
  211. }
  212. int php_do_setsockopt_ip_mcast(php_socket *php_sock,
  213. int level,
  214. int optname,
  215. zval **arg4 TSRMLS_DC)
  216. {
  217. unsigned int if_index;
  218. struct in_addr if_addr;
  219. void *opt_ptr;
  220. socklen_t optlen;
  221. unsigned char ipv4_mcast_ttl_lback;
  222. int retval;
  223. switch (optname) {
  224. case PHP_MCAST_JOIN_GROUP:
  225. case PHP_MCAST_LEAVE_GROUP:
  226. #ifdef HAS_MCAST_EXT
  227. case PHP_MCAST_BLOCK_SOURCE:
  228. case PHP_MCAST_UNBLOCK_SOURCE:
  229. case PHP_MCAST_JOIN_SOURCE_GROUP:
  230. case PHP_MCAST_LEAVE_SOURCE_GROUP:
  231. #endif
  232. if (php_do_mcast_opt(php_sock, level, optname, arg4 TSRMLS_CC) == FAILURE) {
  233. return FAILURE;
  234. } else {
  235. return SUCCESS;
  236. }
  237. case IP_MULTICAST_IF:
  238. if (php_get_if_index_from_zval(*arg4, &if_index TSRMLS_CC) == FAILURE) {
  239. return FAILURE;
  240. }
  241. if (php_if_index_to_addr4(if_index, php_sock, &if_addr TSRMLS_CC) == FAILURE) {
  242. return FAILURE;
  243. }
  244. opt_ptr = &if_addr;
  245. optlen = sizeof(if_addr);
  246. goto dosockopt;
  247. case IP_MULTICAST_LOOP:
  248. convert_to_boolean_ex(arg4);
  249. goto ipv4_loop_ttl;
  250. case IP_MULTICAST_TTL:
  251. convert_to_long_ex(arg4);
  252. if (Z_LVAL_PP(arg4) < 0L || Z_LVAL_PP(arg4) > 255L) {
  253. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  254. "Expected a value between 0 and 255");
  255. return FAILURE;
  256. }
  257. ipv4_loop_ttl:
  258. ipv4_mcast_ttl_lback = (unsigned char) Z_LVAL_PP(arg4);
  259. opt_ptr = &ipv4_mcast_ttl_lback;
  260. optlen = sizeof(ipv4_mcast_ttl_lback);
  261. goto dosockopt;
  262. }
  263. return 1;
  264. dosockopt:
  265. retval = setsockopt(php_sock->bsd_socket, level, optname, opt_ptr, optlen);
  266. if (retval != 0) {
  267. PHP_SOCKET_ERROR(php_sock, "unable to set socket option", errno);
  268. return FAILURE;
  269. }
  270. return SUCCESS;
  271. }
  272. int php_do_setsockopt_ipv6_mcast(php_socket *php_sock,
  273. int level,
  274. int optname,
  275. zval **arg4 TSRMLS_DC)
  276. {
  277. unsigned int if_index;
  278. void *opt_ptr;
  279. socklen_t optlen;
  280. int ov;
  281. int retval;
  282. switch (optname) {
  283. case PHP_MCAST_JOIN_GROUP:
  284. case PHP_MCAST_LEAVE_GROUP:
  285. #ifdef HAS_MCAST_EXT
  286. case PHP_MCAST_BLOCK_SOURCE:
  287. case PHP_MCAST_UNBLOCK_SOURCE:
  288. case PHP_MCAST_JOIN_SOURCE_GROUP:
  289. case PHP_MCAST_LEAVE_SOURCE_GROUP:
  290. #endif
  291. if (php_do_mcast_opt(php_sock, level, optname, arg4 TSRMLS_CC) == FAILURE) {
  292. return FAILURE;
  293. } else {
  294. return SUCCESS;
  295. }
  296. case IPV6_MULTICAST_IF:
  297. if (php_get_if_index_from_zval(*arg4, &if_index TSRMLS_CC) == FAILURE) {
  298. return FAILURE;
  299. }
  300. opt_ptr = &if_index;
  301. optlen = sizeof(if_index);
  302. goto dosockopt;
  303. case IPV6_MULTICAST_LOOP:
  304. convert_to_boolean_ex(arg4);
  305. goto ipv6_loop_hops;
  306. case IPV6_MULTICAST_HOPS:
  307. convert_to_long_ex(arg4);
  308. if (Z_LVAL_PP(arg4) < -1L || Z_LVAL_PP(arg4) > 255L) {
  309. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  310. "Expected a value between -1 and 255");
  311. return FAILURE;
  312. }
  313. ipv6_loop_hops:
  314. ov = (int) Z_LVAL_PP(arg4);
  315. opt_ptr = &ov;
  316. optlen = sizeof(ov);
  317. goto dosockopt;
  318. }
  319. return 1; /* not handled */
  320. dosockopt:
  321. retval = setsockopt(php_sock->bsd_socket, level, optname, opt_ptr, optlen);
  322. if (retval != 0) {
  323. PHP_SOCKET_ERROR(php_sock, "unable to set socket option", errno);
  324. return FAILURE;
  325. }
  326. return SUCCESS;
  327. }
  328. int php_mcast_join(
  329. php_socket *sock,
  330. int level,
  331. struct sockaddr *group,
  332. socklen_t group_len,
  333. unsigned int if_index TSRMLS_DC)
  334. {
  335. return _php_mcast_join_leave(sock, level, group, group_len, if_index, 1 TSRMLS_CC);
  336. }
  337. int php_mcast_leave(
  338. php_socket *sock,
  339. int level,
  340. struct sockaddr *group,
  341. socklen_t group_len,
  342. unsigned int if_index TSRMLS_DC)
  343. {
  344. return _php_mcast_join_leave(sock, level, group, group_len, if_index, 0 TSRMLS_CC);
  345. }
  346. #ifdef HAS_MCAST_EXT
  347. int php_mcast_join_source(
  348. php_socket *sock,
  349. int level,
  350. struct sockaddr *group,
  351. socklen_t group_len,
  352. struct sockaddr *source,
  353. socklen_t source_len,
  354. unsigned int if_index TSRMLS_DC)
  355. {
  356. return _php_mcast_source_op(sock, level, group, group_len, source, source_len, if_index, JOIN_SOURCE TSRMLS_CC);
  357. }
  358. int php_mcast_leave_source(
  359. php_socket *sock,
  360. int level,
  361. struct sockaddr *group,
  362. socklen_t group_len,
  363. struct sockaddr *source,
  364. socklen_t source_len,
  365. unsigned int if_index TSRMLS_DC)
  366. {
  367. return _php_mcast_source_op(sock, level, group, group_len, source, source_len, if_index, LEAVE_SOURCE TSRMLS_CC);
  368. }
  369. int php_mcast_block_source(
  370. php_socket *sock,
  371. int level,
  372. struct sockaddr *group,
  373. socklen_t group_len,
  374. struct sockaddr *source,
  375. socklen_t source_len,
  376. unsigned int if_index TSRMLS_DC)
  377. {
  378. return _php_mcast_source_op(sock, level, group, group_len, source, source_len, if_index, BLOCK_SOURCE TSRMLS_CC);
  379. }
  380. int php_mcast_unblock_source(
  381. php_socket *sock,
  382. int level,
  383. struct sockaddr *group,
  384. socklen_t group_len,
  385. struct sockaddr *source,
  386. socklen_t source_len,
  387. unsigned int if_index TSRMLS_DC)
  388. {
  389. return _php_mcast_source_op(sock, level, group, group_len, source, source_len, if_index, UNBLOCK_SOURCE TSRMLS_CC);
  390. }
  391. #endif /* HAS_MCAST_EXT */
  392. static int _php_mcast_join_leave(
  393. php_socket *sock,
  394. int level,
  395. struct sockaddr *group, /* struct sockaddr_in/sockaddr_in6 */
  396. socklen_t group_len,
  397. unsigned int if_index,
  398. int join TSRMLS_DC)
  399. {
  400. #ifdef RFC3678_API
  401. struct group_req greq = {0};
  402. memcpy(&greq.gr_group, group, group_len);
  403. assert(greq.gr_group.ss_family != 0); /* the caller has set this */
  404. greq.gr_interface = if_index;
  405. return setsockopt(sock->bsd_socket, level,
  406. join ? MCAST_JOIN_GROUP : MCAST_LEAVE_GROUP, (char*)&greq,
  407. sizeof(greq));
  408. #else
  409. if (sock->type == AF_INET) {
  410. struct ip_mreq mreq = {0};
  411. struct in_addr addr;
  412. assert(group_len == sizeof(struct sockaddr_in));
  413. if (if_index != 0) {
  414. if (php_if_index_to_addr4(if_index, sock, &addr TSRMLS_CC) ==
  415. FAILURE)
  416. return -2; /* failure, but notice already emitted */
  417. mreq.imr_interface = addr;
  418. } else {
  419. mreq.imr_interface.s_addr = htonl(INADDR_ANY);
  420. }
  421. mreq.imr_multiaddr = ((struct sockaddr_in*)group)->sin_addr;
  422. return setsockopt(sock->bsd_socket, level,
  423. join ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP, (char*)&mreq,
  424. sizeof(mreq));
  425. }
  426. #if HAVE_IPV6
  427. else if (sock->type == AF_INET6) {
  428. struct ipv6_mreq mreq = {0};
  429. assert(group_len == sizeof(struct sockaddr_in6));
  430. mreq.ipv6mr_multiaddr = ((struct sockaddr_in6*)group)->sin6_addr;
  431. mreq.ipv6mr_interface = if_index;
  432. return setsockopt(sock->bsd_socket, level,
  433. join ? IPV6_JOIN_GROUP : IPV6_LEAVE_GROUP, (char*)&mreq,
  434. sizeof(mreq));
  435. }
  436. #endif
  437. else {
  438. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  439. "Option %s is inapplicable to this socket type",
  440. join ? "MCAST_JOIN_GROUP" : "MCAST_LEAVE_GROUP");
  441. return -2;
  442. }
  443. #endif
  444. }
  445. #ifdef HAS_MCAST_EXT
  446. static int _php_mcast_source_op(
  447. php_socket *sock,
  448. int level,
  449. struct sockaddr *group,
  450. socklen_t group_len,
  451. struct sockaddr *source,
  452. socklen_t source_len,
  453. unsigned int if_index,
  454. enum source_op sop TSRMLS_DC)
  455. {
  456. #ifdef RFC3678_API
  457. struct group_source_req gsreq = {0};
  458. memcpy(&gsreq.gsr_group, group, group_len);
  459. assert(gsreq.gsr_group.ss_family != 0);
  460. memcpy(&gsreq.gsr_source, source, source_len);
  461. assert(gsreq.gsr_source.ss_family != 0);
  462. gsreq.gsr_interface = if_index;
  463. return setsockopt(sock->bsd_socket, level,
  464. _php_source_op_to_rfc3678_op(sop), (char*)&gsreq, sizeof(gsreq));
  465. #else
  466. if (sock->type == AF_INET) {
  467. struct ip_mreq_source mreqs = {0};
  468. struct in_addr addr;
  469. mreqs.imr_multiaddr = ((struct sockaddr_in*)group)->sin_addr;
  470. mreqs.imr_sourceaddr = ((struct sockaddr_in*)source)->sin_addr;
  471. assert(group_len == sizeof(struct sockaddr_in));
  472. assert(source_len == sizeof(struct sockaddr_in));
  473. if (if_index != 0) {
  474. if (php_if_index_to_addr4(if_index, sock, &addr TSRMLS_CC) ==
  475. FAILURE)
  476. return -2; /* failure, but notice already emitted */
  477. mreqs.imr_interface = addr;
  478. } else {
  479. mreqs.imr_interface.s_addr = htonl(INADDR_ANY);
  480. }
  481. return setsockopt(sock->bsd_socket, level,
  482. _php_source_op_to_ipv4_op(sop), (char*)&mreqs, sizeof(mreqs));
  483. }
  484. #if HAVE_IPV6
  485. else if (sock->type == AF_INET6) {
  486. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  487. "This platform does not support %s for IPv6 sockets",
  488. _php_source_op_to_string(sop));
  489. return -2;
  490. }
  491. #endif
  492. else {
  493. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  494. "Option %s is inapplicable to this socket type",
  495. _php_source_op_to_string(sop));
  496. return -2;
  497. }
  498. #endif
  499. }
  500. #if RFC3678_API
  501. static int _php_source_op_to_rfc3678_op(enum source_op sop)
  502. {
  503. switch (sop) {
  504. case JOIN_SOURCE:
  505. return MCAST_JOIN_SOURCE_GROUP;
  506. case LEAVE_SOURCE:
  507. return MCAST_LEAVE_SOURCE_GROUP;
  508. case BLOCK_SOURCE:
  509. return MCAST_BLOCK_SOURCE;
  510. case UNBLOCK_SOURCE:
  511. return MCAST_UNBLOCK_SOURCE;
  512. }
  513. assert(0);
  514. return 0;
  515. }
  516. #else
  517. static const char *_php_source_op_to_string(enum source_op sop)
  518. {
  519. switch (sop) {
  520. case JOIN_SOURCE:
  521. return "MCAST_JOIN_SOURCE_GROUP";
  522. case LEAVE_SOURCE:
  523. return "MCAST_LEAVE_SOURCE_GROUP";
  524. case BLOCK_SOURCE:
  525. return "MCAST_BLOCK_SOURCE";
  526. case UNBLOCK_SOURCE:
  527. return "MCAST_UNBLOCK_SOURCE";
  528. }
  529. assert(0);
  530. return "";
  531. }
  532. static int _php_source_op_to_ipv4_op(enum source_op sop)
  533. {
  534. switch (sop) {
  535. case JOIN_SOURCE:
  536. return IP_ADD_SOURCE_MEMBERSHIP;
  537. case LEAVE_SOURCE:
  538. return IP_DROP_SOURCE_MEMBERSHIP;
  539. case BLOCK_SOURCE:
  540. return IP_BLOCK_SOURCE;
  541. case UNBLOCK_SOURCE:
  542. return IP_UNBLOCK_SOURCE;
  543. }
  544. assert(0);
  545. return 0;
  546. }
  547. #endif
  548. #endif /* HAS_MCAST_EXT */
  549. #if PHP_WIN32
  550. int php_if_index_to_addr4(unsigned if_index, php_socket *php_sock, struct in_addr *out_addr TSRMLS_DC)
  551. {
  552. MIB_IPADDRTABLE *addr_table;
  553. ULONG size;
  554. DWORD retval;
  555. DWORD i;
  556. (void) php_sock; /* not necessary */
  557. if (if_index == 0) {
  558. out_addr->s_addr = INADDR_ANY;
  559. return SUCCESS;
  560. }
  561. size = 4 * (sizeof *addr_table);
  562. addr_table = emalloc(size);
  563. retry:
  564. retval = GetIpAddrTable(addr_table, &size, 0);
  565. if (retval == ERROR_INSUFFICIENT_BUFFER) {
  566. efree(addr_table);
  567. addr_table = emalloc(size);
  568. goto retry;
  569. }
  570. if (retval != NO_ERROR) {
  571. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  572. "GetIpAddrTable failed with error %lu", retval);
  573. return FAILURE;
  574. }
  575. for (i = 0; i < addr_table->dwNumEntries; i++) {
  576. MIB_IPADDRROW r = addr_table->table[i];
  577. if (r.dwIndex == if_index) {
  578. out_addr->s_addr = r.dwAddr;
  579. return SUCCESS;
  580. }
  581. }
  582. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  583. "No interface with index %u was found", if_index);
  584. return FAILURE;
  585. }
  586. int php_add4_to_if_index(struct in_addr *addr, php_socket *php_sock, unsigned *if_index TSRMLS_DC)
  587. {
  588. MIB_IPADDRTABLE *addr_table;
  589. ULONG size;
  590. DWORD retval;
  591. DWORD i;
  592. (void) php_sock; /* not necessary */
  593. if (addr->s_addr == INADDR_ANY) {
  594. *if_index = 0;
  595. return SUCCESS;
  596. }
  597. size = 4 * (sizeof *addr_table);
  598. addr_table = emalloc(size);
  599. retry:
  600. retval = GetIpAddrTable(addr_table, &size, 0);
  601. if (retval == ERROR_INSUFFICIENT_BUFFER) {
  602. efree(addr_table);
  603. addr_table = emalloc(size);
  604. goto retry;
  605. }
  606. if (retval != NO_ERROR) {
  607. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  608. "GetIpAddrTable failed with error %lu", retval);
  609. return FAILURE;
  610. }
  611. for (i = 0; i < addr_table->dwNumEntries; i++) {
  612. MIB_IPADDRROW r = addr_table->table[i];
  613. if (r.dwAddr == addr->s_addr) {
  614. *if_index = r.dwIndex;
  615. return SUCCESS;
  616. }
  617. }
  618. {
  619. char addr_str[17] = {0};
  620. inet_ntop(AF_INET, addr, addr_str, sizeof(addr_str));
  621. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  622. "The interface with IP address %s was not found", addr_str);
  623. }
  624. return FAILURE;
  625. }
  626. #else
  627. int php_if_index_to_addr4(unsigned if_index, php_socket *php_sock, struct in_addr *out_addr TSRMLS_DC)
  628. {
  629. struct ifreq if_req;
  630. if (if_index == 0) {
  631. out_addr->s_addr = INADDR_ANY;
  632. return SUCCESS;
  633. }
  634. #if !defined(ifr_ifindex) && defined(ifr_index)
  635. #define ifr_ifindex ifr_index
  636. #endif
  637. #if defined(SIOCGIFNAME)
  638. if_req.ifr_ifindex = if_index;
  639. if (ioctl(php_sock->bsd_socket, SIOCGIFNAME, &if_req) == -1) {
  640. #elif defined(HAVE_IF_INDEXTONAME)
  641. if (if_indextoname(if_index, if_req.ifr_name) == NULL) {
  642. #else
  643. #error Neither SIOCGIFNAME nor if_indextoname are available
  644. #endif
  645. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  646. "Failed obtaining address for interface %u: error %d", if_index, errno);
  647. return FAILURE;
  648. }
  649. if (ioctl(php_sock->bsd_socket, SIOCGIFADDR, &if_req) == -1) {
  650. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  651. "Failed obtaining address for interface %u: error %d", if_index, errno);
  652. return FAILURE;
  653. }
  654. memcpy(out_addr, &((struct sockaddr_in *) &if_req.ifr_addr)->sin_addr,
  655. sizeof *out_addr);
  656. return SUCCESS;
  657. }
  658. int php_add4_to_if_index(struct in_addr *addr, php_socket *php_sock, unsigned *if_index TSRMLS_DC)
  659. {
  660. struct ifconf if_conf = {0};
  661. char *buf = NULL,
  662. *p;
  663. int size = 0,
  664. lastsize = 0;
  665. size_t entry_len;
  666. if (addr->s_addr == INADDR_ANY) {
  667. *if_index = 0;
  668. return SUCCESS;
  669. }
  670. for(;;) {
  671. size += 5 * sizeof(struct ifreq);
  672. buf = ecalloc(size, 1);
  673. if_conf.ifc_len = size;
  674. if_conf.ifc_buf = buf;
  675. if (ioctl(php_sock->bsd_socket, SIOCGIFCONF, (char*)&if_conf) == -1 &&
  676. (errno != EINVAL || lastsize != 0)) {
  677. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  678. "Failed obtaining interfaces list: error %d", errno);
  679. goto err;
  680. }
  681. if (if_conf.ifc_len == lastsize)
  682. /* not increasing anymore */
  683. break;
  684. else {
  685. lastsize = if_conf.ifc_len;
  686. efree(buf);
  687. buf = NULL;
  688. }
  689. }
  690. for (p = if_conf.ifc_buf;
  691. p < if_conf.ifc_buf + if_conf.ifc_len;
  692. p += entry_len) {
  693. struct ifreq *cur_req;
  694. /* let's hope the pointer is aligned */
  695. cur_req = (struct ifreq*) p;
  696. #ifdef HAVE_SOCKADDR_SA_LEN
  697. entry_len = cur_req->ifr_addr.sa_len + sizeof(cur_req->ifr_name);
  698. #else
  699. /* if there's no sa_len, assume the ifr_addr field is a sockaddr */
  700. entry_len = sizeof(struct sockaddr) + sizeof(cur_req->ifr_name);
  701. #endif
  702. entry_len = MAX(entry_len, sizeof(*cur_req));
  703. if ((((struct sockaddr*)&cur_req->ifr_addr)->sa_family == AF_INET) &&
  704. (((struct sockaddr_in*)&cur_req->ifr_addr)->sin_addr.s_addr ==
  705. addr->s_addr)) {
  706. #if defined(SIOCGIFINDEX)
  707. if (ioctl(php_sock->bsd_socket, SIOCGIFINDEX, (char*)cur_req)
  708. == -1) {
  709. #elif defined(HAVE_IF_NAMETOINDEX)
  710. unsigned index_tmp;
  711. if ((index_tmp = if_nametoindex(cur_req->ifr_name)) == 0) {
  712. #else
  713. #error Neither SIOCGIFINDEX nor if_nametoindex are available
  714. #endif
  715. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  716. "Error converting interface name to index: error %d",
  717. errno);
  718. goto err;
  719. } else {
  720. #if defined(SIOCGIFINDEX)
  721. *if_index = cur_req->ifr_ifindex;
  722. #else
  723. *if_index = index_tmp;
  724. #endif
  725. efree(buf);
  726. return SUCCESS;
  727. }
  728. }
  729. }
  730. {
  731. char addr_str[17] = {0};
  732. inet_ntop(AF_INET, addr, addr_str, sizeof(addr_str));
  733. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  734. "The interface with IP address %s was not found", addr_str);
  735. }
  736. err:
  737. if (buf != NULL)
  738. efree(buf);
  739. return FAILURE;
  740. }
  741. #endif