Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
#include "net.h"
|
|
|
|
|
2018-02-16 21:55:33 +08:00
|
|
|
#include <stdio.h>
|
2019-12-16 04:55:43 +08:00
|
|
|
#include <SDL2/SDL_platform.h>
|
2018-02-16 21:55:33 +08:00
|
|
|
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
#include "log.h"
|
|
|
|
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
typedef int socklen_t;
|
|
|
|
#else
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <sys/socket.h>
|
|
|
|
# include <netinet/in.h>
|
|
|
|
# include <arpa/inet.h>
|
|
|
|
# include <unistd.h>
|
|
|
|
# define SOCKET_ERROR -1
|
|
|
|
typedef struct sockaddr_in SOCKADDR_IN;
|
|
|
|
typedef struct sockaddr SOCKADDR;
|
|
|
|
typedef struct in_addr IN_ADDR;
|
|
|
|
#endif
|
|
|
|
|
2021-09-10 00:07:38 +08:00
|
|
|
static void
|
|
|
|
net_perror(const char *s) {
|
|
|
|
#ifdef _WIN32
|
|
|
|
int error = WSAGetLastError();
|
|
|
|
char *wsa_message;
|
|
|
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
|
|
|
|
NULL, error, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
|
|
(char *) &wsa_message, 0, NULL);
|
|
|
|
// no explicit '\n', wsa_message already contains a trailing '\n'
|
|
|
|
fprintf(stderr, "%s: [%d] %s", s, error, wsa_message);
|
|
|
|
LocalFree(wsa_message);
|
|
|
|
#else
|
|
|
|
perror(s);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2019-03-03 03:09:56 +08:00
|
|
|
socket_t
|
2019-03-03 06:52:22 +08:00
|
|
|
net_connect(uint32_t addr, uint16_t port) {
|
2018-03-12 15:35:51 +08:00
|
|
|
socket_t sock = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
if (sock == INVALID_SOCKET) {
|
2021-09-10 00:07:38 +08:00
|
|
|
net_perror("socket");
|
2018-03-12 15:35:51 +08:00
|
|
|
return INVALID_SOCKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
SOCKADDR_IN sin;
|
|
|
|
sin.sin_family = AF_INET;
|
|
|
|
sin.sin_addr.s_addr = htonl(addr);
|
|
|
|
sin.sin_port = htons(port);
|
|
|
|
|
|
|
|
if (connect(sock, (SOCKADDR *) &sin, sizeof(sin)) == SOCKET_ERROR) {
|
2021-09-10 00:07:38 +08:00
|
|
|
net_perror("connect");
|
2019-06-01 05:32:53 +08:00
|
|
|
net_close(sock);
|
2018-03-12 15:35:51 +08:00
|
|
|
return INVALID_SOCKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
return sock;
|
|
|
|
}
|
|
|
|
|
2019-03-03 03:09:56 +08:00
|
|
|
socket_t
|
2019-03-03 06:52:22 +08:00
|
|
|
net_listen(uint32_t addr, uint16_t port, int backlog) {
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
socket_t sock = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
if (sock == INVALID_SOCKET) {
|
2021-09-10 00:07:38 +08:00
|
|
|
net_perror("socket");
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
return INVALID_SOCKET;
|
|
|
|
}
|
|
|
|
|
2018-02-16 22:05:43 +08:00
|
|
|
int reuse = 1;
|
2019-03-03 03:09:56 +08:00
|
|
|
if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, (const void *) &reuse,
|
|
|
|
sizeof(reuse)) == -1) {
|
2021-09-10 00:07:38 +08:00
|
|
|
net_perror("setsockopt(SO_REUSEADDR)");
|
2018-02-16 22:05:43 +08:00
|
|
|
}
|
|
|
|
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
SOCKADDR_IN sin;
|
|
|
|
sin.sin_family = AF_INET;
|
|
|
|
sin.sin_addr.s_addr = htonl(addr); // htonl() harmless on INADDR_ANY
|
|
|
|
sin.sin_port = htons(port);
|
|
|
|
|
|
|
|
if (bind(sock, (SOCKADDR *) &sin, sizeof(sin)) == SOCKET_ERROR) {
|
2021-09-10 00:07:38 +08:00
|
|
|
net_perror("bind");
|
2019-06-01 05:32:53 +08:00
|
|
|
net_close(sock);
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
return INVALID_SOCKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (listen(sock, backlog) == SOCKET_ERROR) {
|
2021-09-10 00:07:38 +08:00
|
|
|
net_perror("listen");
|
2019-06-01 05:32:53 +08:00
|
|
|
net_close(sock);
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
return INVALID_SOCKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
return sock;
|
|
|
|
}
|
|
|
|
|
2019-03-03 03:09:56 +08:00
|
|
|
socket_t
|
|
|
|
net_accept(socket_t server_socket) {
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
SOCKADDR_IN csin;
|
|
|
|
socklen_t sinsize = sizeof(csin);
|
|
|
|
return accept(server_socket, (SOCKADDR *) &csin, &sinsize);
|
|
|
|
}
|
|
|
|
|
2019-03-03 03:09:56 +08:00
|
|
|
ssize_t
|
|
|
|
net_recv(socket_t socket, void *buf, size_t len) {
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
return recv(socket, buf, len, 0);
|
|
|
|
}
|
|
|
|
|
2019-03-03 03:09:56 +08:00
|
|
|
ssize_t
|
|
|
|
net_recv_all(socket_t socket, void *buf, size_t len) {
|
2018-02-16 06:55:52 +08:00
|
|
|
return recv(socket, buf, len, MSG_WAITALL);
|
|
|
|
}
|
|
|
|
|
2019-03-03 03:09:56 +08:00
|
|
|
ssize_t
|
|
|
|
net_send(socket_t socket, const void *buf, size_t len) {
|
Replace SDL_net by custom implementation
SDL_net is not very suitable for scrcpy.
For example, SDLNet_TCP_Accept() is non-blocking, so we have to wrap it
by calling many SDL_Net-specific functions to make it blocking.
But above all, SDLNet_TCP_Open() is a server socket only when no IP is
provided; otherwise, it's a client socket. Therefore, it is not possible
to create a server socket bound to localhost, so it accepts connections
from anywhere.
This is a problem for scrcpy, because on start, the application listens
for nearly 1 second until it accepts the first connection, supposedly
from the device. If someone on the local network manages to connect to
the server socket first, then they can stream arbitrary H.264 video.
This may be troublesome, for example during a public presentation ;-)
Provide our own simplified API (net.h) instead, implemented for the
different platforms.
2018-02-16 05:59:21 +08:00
|
|
|
return send(socket, buf, len, 0);
|
|
|
|
}
|
2018-02-16 06:55:52 +08:00
|
|
|
|
2019-03-03 03:09:56 +08:00
|
|
|
ssize_t
|
|
|
|
net_send_all(socket_t socket, const void *buf, size_t len) {
|
2021-07-16 00:16:26 +08:00
|
|
|
size_t copied = 0;
|
2018-05-27 21:49:20 +08:00
|
|
|
ssize_t w = 0;
|
2018-02-16 06:55:52 +08:00
|
|
|
while (len > 0) {
|
|
|
|
w = send(socket, buf, len, 0);
|
|
|
|
if (w == -1) {
|
2021-07-16 00:16:26 +08:00
|
|
|
return copied ? (ssize_t) copied : -1;
|
2018-02-16 06:55:52 +08:00
|
|
|
}
|
|
|
|
len -= w;
|
2018-04-04 16:48:14 +08:00
|
|
|
buf = (char *) buf + w;
|
2021-07-16 00:16:26 +08:00
|
|
|
copied += w;
|
2018-02-16 06:55:52 +08:00
|
|
|
}
|
2021-07-16 00:16:26 +08:00
|
|
|
return copied;
|
2018-02-16 06:55:52 +08:00
|
|
|
}
|
2018-02-16 18:11:07 +08:00
|
|
|
|
2019-03-03 06:52:22 +08:00
|
|
|
bool
|
2019-03-03 03:09:56 +08:00
|
|
|
net_shutdown(socket_t socket, int how) {
|
2018-02-16 18:11:07 +08:00
|
|
|
return !shutdown(socket, how);
|
|
|
|
}
|
2019-12-16 04:55:43 +08:00
|
|
|
|
|
|
|
bool
|
|
|
|
net_init(void) {
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
WSADATA wsa;
|
|
|
|
int res = WSAStartup(MAKEWORD(2, 2), &wsa) < 0;
|
|
|
|
if (res < 0) {
|
|
|
|
LOGC("WSAStartup failed with error %d", res);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
net_cleanup(void) {
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
WSACleanup();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
net_close(socket_t socket) {
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
return !closesocket(socket);
|
|
|
|
#else
|
|
|
|
return !close(socket);
|
|
|
|
#endif
|
|
|
|
}
|