Skip to content

Commit

Permalink
Merge branch 'nonblock-pipe' into vita
Browse files Browse the repository at this point in the history
  • Loading branch information
d3m3vilurr committed Sep 30, 2023
2 parents a13cf8c + c8e99f0 commit a56aebe
Show file tree
Hide file tree
Showing 5 changed files with 95 additions and 5 deletions.
2 changes: 1 addition & 1 deletion newlib/libc/sys/vita/Makefile.am
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ AM_CCASFLAGS = $(INCLUDES)

noinst_LIBRARIES = lib.a

SOCKET_OBJS = accept.o bind.o connect.o getpeername.o getsockname.o getsockopt.o listen.o recv.o recvfrom.o recvmsg.o send.o sendto.o sendmsg.o setsockopt.o shutdown.o socket.o
SOCKET_OBJS = accept.o bind.o connect.o getpeername.o getsockname.o getsockopt.o listen.o recv.o recvfrom.o recvmsg.o send.o sendto.o sendmsg.o setsockopt.o shutdown.o socket.o socketpair.o
DIRENT_OBJS = dirfd.o closedir.o opendir.o readdir.o readdir_r.o rewinddir.o scandir.o seekdir.o telldir.o

NET_SOURCES = net/gethostbyaddr.c \
Expand Down
30 changes: 30 additions & 0 deletions newlib/libc/sys/vita/pipe.c
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,9 @@ DEALINGS IN THE SOFTWARE.
#include <stdio.h>
#include <stdlib.h>
#include <sys/syslimits.h>
#include <sys/socket.h>
#include <string.h>
#include <fcntl.h>

#include <psp2/types.h>
#include <psp2/kernel/threadmgr.h>
Expand Down Expand Up @@ -75,4 +77,32 @@ int pipe(int pipefd[2])
pipefd[1] = fd2;

return 0;
}


int pipe2(int pipefd[2], int flags) {
if (flags & O_NONBLOCK) {
// FIXME this action is simulated async io using the socket.
// this implemenation has limitation that this cannot use on every environment
// such as adhoc network mode
// implemenation have to be changed to async sce*MsgPiple
//
// https://github.com/vitasdk/newlib/pull/101
if (socketpair(AF_INET, SOCK_STREAM, 0, pipefd) == -1) {
return -1;
}

int val = 1;
if (setsockopt(pipefd[0], SOL_SOCKET, SO_NONBLOCK, &val, sizeof(val)) == -1 ||
setsockopt(pipefd[1], SOL_SOCKET, SO_NONBLOCK, &val, sizeof(val)) == -1) {

close(pipefd[0]);
close(pipefd[1]);
return -1;
}

return 0;
} else {
return pipe(pipefd);
}
}
57 changes: 57 additions & 0 deletions newlib/libc/sys/vita/socket.c
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,8 @@ DEALINGS IN THE SOFTWARE.
#include <sys/time.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <netinet/in.h>

#include <psp2/net/net.h>
#include <psp2/types.h>
Expand Down Expand Up @@ -491,3 +493,58 @@ int socket(int domain, int type, int protocol)
return s;
}
#endif

#ifdef F_socket
int socketpair(int domain, int type, int protocol, int sockfds[2])
{
// Usually socketpair is used with AF_UNIX, simulate that with INET.
int listener;
if ((listener = socket(AF_INET, type, protocol)) == -1) {
return -1;
}

struct sockaddr_in server_addr;
socklen_t addr_len = sizeof(struct sockaddr_in);

memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
server_addr.sin_port = 0;

if (bind(listener, (struct sockaddr*)&server_addr, addr_len) == -1) {
close(listener);
return -1;
}

if (listen(listener, 1) == -1) {
close(listener);
return -1;
}

if (getsockname(listener, (struct sockaddr *)&server_addr, &addr_len) == -1) {
close(listener);
return -1;
}

if ((sockfds[1] = socket(AF_INET, type, protocol)) == -1) {
close(listener);
return -1;
}

if (connect(sockfds[1], (struct sockaddr*)&server_addr, addr_len) == -1) {
close(sockfds[1]);
close(listener);
return -1;
}

if ((sockfds[0] = accept(listener, (struct sockaddr*)&server_addr, &addr_len)) == -1) {
close(sockfds[1]);
close(listener);
return -1;
}

close(listener);

return 0;
}
#endif
1 change: 1 addition & 0 deletions newlib/libc/sys/vita/sys/socket.h
Original file line number Diff line number Diff line change
Expand Up @@ -279,6 +279,7 @@ ssize_t sendmsg(int s, const struct msghdr *msg, int flags);
int setsockopt(int, int, int, const void *, socklen_t);
int shutdown(int, int);
int socket(int, int, int);
int socketpair(int, int, int, int *);

#ifdef __cplusplus
}
Expand Down
10 changes: 6 additions & 4 deletions newlib/libc/sys/vita/syscalls.c
Original file line number Diff line number Diff line change
Expand Up @@ -54,8 +54,9 @@ _write_r(struct _reent * reent, int fd, const void *buf, size_t nbytes)
{
size_t len = nbytes;
if (len > 4 * 4096) len = 4 * 4096;
ret = sceKernelSendMsgPipe(fdmap->sce_uid, buf, len, 1, NULL, NULL);
if (ret == 0) ret = len;
size_t p_res = 0;
ret = sceKernelSendMsgPipe(fdmap->sce_uid, buf, len, 1, &p_res, NULL);
if (ret == 0) ret = p_res;
break;
}
}
Expand Down Expand Up @@ -336,8 +337,9 @@ _read_r(struct _reent *reent, int fd, void *ptr, size_t len)
{
size_t rlen = len;
if (rlen > 4 * 4096) rlen = 4 * 4096;
ret = sceKernelReceiveMsgPipe(fdmap->sce_uid, ptr, rlen, 1, NULL, NULL);
if (ret == 0) ret = rlen;
size_t p_res = 0;
ret = sceKernelReceiveMsgPipe(fdmap->sce_uid, ptr, rlen, 1, &p_res, NULL);
if (ret == 0) ret = p_res;
break;
}
}
Expand Down

0 comments on commit a56aebe

Please sign in to comment.