#endif
struct ip_mreq mreq;
- mreq.imr_multiaddr.s_addr = ::inet_addr("225.0.0.37");
+ mreq.imr_multiaddr.s_addr = ::inet_addr("225.0.0.37"); /* ipMIDI group multicast address */
mreq.imr_interface.s_addr = if_addr_in.s_addr;
if(::setsockopt (sockin, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char *) &mreq, sizeof(mreq)) < 0) {
::perror("setsockopt(IP_ADD_MEMBERSHIP)");
}
// Will Hall, Oct 2007
+#ifndef PLATFORM_WINDOWS
if (!ifname.empty()) {
struct in_addr if_addr_out;
if (!get_address(sockout, &if_addr_out, ifname)) {
return false;
}
}
+#endif
::memset(&addrout, 0, sizeof(struct sockaddr_in));
addrout.sin_family = AF_INET;
// Turn off loopback...
int loop = 0;
- if (::setsockopt(sockout, IPPROTO_IP, IP_MULTICAST_LOOP, (char *) &loop, sizeof (loop)) < 0) {
+
+#ifdef PLATFORM_WINDOWS
+
+ /* https://msdn.microsoft.com/en-us/library/windows/desktop/ms739161%28v=vs.85%29.aspx
+ *
+ * ------------------------------------------------------------------------------
+ * Note The Winsock version of the IP_MULTICAST_LOOP option is
+ * semantically different than the UNIX version of the
+ * IP_MULTICAST_LOOP option:
+ *
+ * In Winsock, the IP_MULTICAST_LOOP option applies only to the receive path.
+ * In the UNIX version, the IP_MULTICAST_LOOP option applies to the send path.
+ *
+ * For example, applications ON and OFF (which are easier to track than
+ * X and Y) join the same group on the same interface; application ON
+ * sets the IP_MULTICAST_LOOP option on, application OFF sets the
+ * IP_MULTICAST_LOOP option off. If ON and OFF are Winsock
+ * applications, OFF can send to ON, but ON cannot sent to OFF. In
+ * contrast, if ON and OFF are UNIX applications, ON can send to OFF,
+ * but OFF cannot send to ON.
+ * ------------------------------------------------------------------------------
+ *
+ * Alles klar? Gut!
+ */
+
+ const int target_sock = sockin;
+#else
+ const int target_sock = sockout;
+#endif
+
+ if (::setsockopt (target_sock, IPPROTO_IP, IP_MULTICAST_LOOP, (char *) &loop, sizeof (loop)) < 0) {
::perror("setsockopt(IP_MULTICAST_LOOP)");
return false;
}
}
#else
- // If iMode!=0, non-blocking mode is enabled.
+ // If imode !=0, non-blocking mode is enabled.
u_long mode=1;
- if (ioctlsocket(sockin,FIONBIO,&imode)) {
+ if (ioctlsocket(sockin,FIONBIO,&mode)) {
error << "cannot set non-blocking mode for IP MIDI input socket (" << ::strerror (errno) << ')' << endmsg;
return false;
}
- imode = 1;
- if (ioctlsocket(sockout,FIONBIO,&iMode)) {
+ mode = 1; /* just in case it was modified in the previous call */
+ if (ioctlsocket(sockout,FIONBIO,&mode)) {
error << "cannot set non-blocking mode for IP MIDI output socket (" << ::strerror (errno) << ')' << endmsg;
return false;
}
}
void
-IPMIDIPort::parse (framecnt_t timestamp)
+IPMIDIPort::parse (samplecnt_t timestamp)
{
/* input was detected on the socket, so go get it and hand it to the
* parser. This will emit appropriate signals that will be handled