Получение многоадресной рассылки UDP в macOS Big Sur

Я знаю, что это может быть 10.000-й вопрос о получении многоадресных сообщений UDP. Однако у меня уже полгода назад он заработал. Следующий фрагмент я получил от http://www.cs.tau.ac.il/%7Eeddiea/samples/Multicast/multicast-listen.c.html (с небольшими изменениями: bind-ip был изменен на IPADDR_ANY) работал должным образом, прежде чем я обновился до macOS Big Sur.

/**************************************************************/
/* Multicast listener (server)                                */
/*                                                            */
/* Activation using: {program name} {Multicast IP} {port}     */
/*   {program name} - This program name                       */
/*   {Multicast IP} - The IP address to listen to (Class D)   */
/*   {port} - The port hnumber to listen on                   */
/*                                                            */
/* This is free software released under the GPL license.      */
/* See the GNU GPL for details.                               */
/*                                                            */
/* (c) Juan-Mariano de Goyeneche. 1998, 1999.                 */
/**************************************************************/


#include <stdio.h>          /* printf(), snprintf() */
#include <stdlib.h>         /* strtol(), exit() */
#include <sys/types.h>
#include <sys/socket.h>     /* socket(), setsockopt(), bind(), recvfrom(), sendto() */
#include <errno.h>          /* perror() */
#include <netinet/in.h>     /* IPPROTO_IP, sockaddr_in, htons(), htonl() */
#include <arpa/inet.h>      /* inet_addr() */
#include <unistd.h>         /* fork(), sleep() */
#include <sys/utsname.h>    /* uname() */
#include <string.h>         /* memset() */

#define MAXLEN 1024


int main(int argc, char* argv[])
{
  u_char no = 0;
  u_int yes = 1;      /* Used with SO_REUSEADDR.
                             In Linux both u_int */
  /* and u_char are valid. */
  int send_s, recv_s;     /* Sockets for sending and receiving. */
  u_char ttl;
  struct sockaddr_in mcast_group;
  struct ip_mreq mreq;
  struct utsname name;
  int n;
  socklen_t socklen;
  struct sockaddr_in from;
  char message [MAXLEN+1];

  if (argc != 3) {
    fprintf(stderr, "Usage: %s mcast_group port\n", argv[0]);
    exit(1);
  }

  memset(&mcast_group, 0, sizeof(mcast_group));
  mcast_group.sin_family = AF_INET;
  mcast_group.sin_port = htons((unsigned short int)strtol(argv[2], NULL, 0));
  mcast_group.sin_addr.s_addr = htonl(INADDR_ANY);

  if ( (recv_s=socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
    perror ("recv socket");
    exit(1);
  }

  if (setsockopt(recv_s, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)) < 0) {
    perror("reuseaddr setsockopt");
    exit(1);
  }

  if (bind(recv_s, (struct sockaddr*)&mcast_group, sizeof(mcast_group)) < 0) {
    perror ("bind");
    exit(1);
  }

  struct in_addr sin;
  inet_pton(AF_INET, argv[1], &sin);
  /* Preparatios for using Multicast */
  mreq.imr_multiaddr = sin;
  mreq.imr_interface.s_addr = 0;

  /* Tell the kernel we want to join that multicast group. */
  if (setsockopt(recv_s, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0) {
    perror ("add_membership setsockopt");
    exit(1);
  }

  for (;;) {
    socklen=sizeof(from);
    if ( (n=recvfrom(recv_s, message, MAXLEN, 0,
                     (struct sockaddr*)&from, &socklen)) < 0) {
      perror ("recv");
      exit(1);
    }
    message[n] = '\0'; /* null-terminate string */
    printf("%s: Received message from %s, size=%d !!\n",
           name.nodename,
           inet_ntoa(from.sin_addr), n);
    printf("\t%s \n", message);
  }
}

Теперь я пытаюсь запустить его снова и не могу найти причину, по которой это больше не работает. Изменились ли какие-то правила в Big Sur, и это ошибка разрешения? Я уже запустил приложение как root с sudo ./multicast-listen 239.255.255.250 1900, но это тоже не сработало. Я пытаюсь получить пакеты SSDP.


person bam    schedule 01.01.2021    source источник
comment
Ссылка в википедии: en.wikipedia.org/wiki/Simple_Service_Discovery_Protocol описывает атаку ddos ​​для этого. С обновлением macOS, возможно, они отключили его по умолчанию. Это [все еще] включено в вашем брандмауэре (ах)? Поскольку это локальный адрес сайта, у вас должна быть возможность создать программу-отправитель для тестирования. У вас есть доступ к другой системе в той же сети, в которой вы можете запускать программу (ы) (например, linux)? Поскольку это локальный сайт, уверены ли вы, что программа-отправитель [в другой системе] все еще работает?   -  person Craig Estey    schedule 02.01.2021
comment
Привет, @CraigEstey, спасибо за ваши предложения. Я вижу все интересующие меня пакеты хотя бы в Wireshark. IIRC Wireshark видит пакеты до брандмауэра. Ваш комментарий навел меня на мысли, которые решили мою проблему, спасибо!   -  person bam    schedule 02.01.2021


Ответы (1)


Итак, комментарий @CraigEstey привел меня к выводу, что он должен что-то делать с брандмауэром. На самом деле это была простая проблема с брандмауэром. В Settings - ›Security -› Firewall - ›Firewall Options... вы должны отключить Block all incoming connections и отключить скрытый режим. Теперь я получаю все пакеты SSDP, как и предполагалось.

person bam    schedule 02.01.2021
comment
У меня проблемы с получением многоадресных пакетов в приложении Java на Big Sur. Это не решило мою проблему, так как брандмауэр был полностью отключен. По-прежнему нет многоадресных пакетов. Кажется, macOS действительно ужасен с многоадресной рассылкой. Многоадресная рассылка работает безупречно без дополнительной настройки в Windows. - person moonlightcheese; 15.01.2021
comment
@moonlightcheese Я не очень хорошо разбираюсь в сетевых технологиях с Java. Но в Linux, а также в macOS вы должны зарегистрировать свое приложение в группе многоадресной рассылки в дополнение к открытию сокета. См. Часть с IP_ADD_MEMBERSHIP и setsocktop. Я почти уверен, что вам нужно сделать что-то подобное в java. - person bam; 16.01.2021