1
0
mirror of https://github.com/aap/pdp6.git synced 2026-05-01 22:16:01 +00:00
Files
aap.pdp6/newemu/common.c

397 lines
7.1 KiB
C

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#include <ctype.h>
#include <signal.h>
#include <pthread.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <netdb.h>
#include "common.h"
void
panic(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
vfprintf(stderr, fmt, ap);
fprintf(stderr, "\n");
va_end(ap);
exit(1);
}
int
hasinput(int fd)
{
fd_set fds;
struct timeval timeout;
if(fd < 0) return 0;
timeout.tv_sec = 0;
timeout.tv_usec = 0;
FD_ZERO(&fds);
FD_SET(fd, &fds);
return select(fd+1, &fds, NULL, NULL, &timeout) > 0;
}
int
readn(int fd, void *data, int n)
{
int m;
while(n > 0) {
m = read(fd, data, n);
if(m <= 0)
return -1;
data += m;
n -= m;
}
return 0;
}
int
dial(const char *host, int port)
{
char portstr[32];
int sockfd;
struct addrinfo *result, *rp, hints;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
snprintf(portstr, 32, "%d", port);
if(getaddrinfo(host, portstr, &hints, &result)){
perror("error: getaddrinfo");
return -1;
}
for(rp = result; rp; rp = rp->ai_next){
sockfd = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol);
if(sockfd < 0)
continue;
if(connect(sockfd, rp->ai_addr, rp->ai_addrlen) >= 0)
goto win;
close(sockfd);
}
freeaddrinfo(result);
perror("error");
return -1;
win:
freeaddrinfo(result);
return sockfd;
}
int
serve1(int port)
{
int sockfd, confd;
socklen_t len;
struct sockaddr_in server, client;
int x;
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if(sockfd < 0){
perror("error: socket");
return -1;
}
x = 1;
setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (void *)&x, sizeof x);
memset(&server, 0, sizeof(server));
server.sin_family = AF_INET;
server.sin_addr.s_addr = INADDR_ANY;
server.sin_port = htons(port);
if(bind(sockfd, (struct sockaddr*)&server, sizeof(server)) < 0){
perror("error: bind");
return -1;
}
listen(sockfd, 5);
len = sizeof(client);
confd = accept(sockfd, (struct sockaddr*)&client, &len),
close(sockfd);
if(confd >= 0)
return confd;
perror("error: accept");
return -1;
}
void
nodelay(int fd)
{
int flag = 1;
setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &flag, sizeof(flag));
}
void*
createseg(const char *name, size_t sz)
{
int fd;
void *p;
mode_t mask = umask(0);
fd = open(name, O_RDWR|O_CREAT, 0666);
umask(mask);
// if we try to open a /tmp file owned by another user
// with O_CREAT, the above will fail (even for root).
// so try again without O_CREAT
if(fd == -1)
fd = open(name, O_RDWR);
if(fd == -1) {
fprintf(stderr, "couldn't open file %s\n", name);
return nil;
}
ftruncate(fd, sz);
p = mmap(nil, sz, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
if(p == MAP_FAILED) {
fprintf(stderr, "couldn't mmap file\n");
return nil;
}
return p;
}
void*
attachseg(const char *name, size_t sz)
{
int fd;
void *p;
fd = open(name, O_RDWR);
if(fd == -1) {
fprintf(stderr, "couldn't open file %s\n", name);
return nil;
}
p = mmap(NULL, sz, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
if(p == MAP_FAILED) {
fprintf(stderr, "couldn't mmap file\n");
return nil;
}
return p;
}
static struct timespec starttime;
void
inittime(void)
{
clock_gettime(CLOCK_MONOTONIC, &starttime);
}
u64
gettime(void)
{
struct timespec tm;
u64 t;
clock_gettime(CLOCK_MONOTONIC, &tm);
tm.tv_sec -= starttime.tv_sec;
t = tm.tv_nsec;
t += (u64)tm.tv_sec * 1000 * 1000 * 1000;
return t;
}
void
nsleep(u64 ns)
{
struct timespec tm;
tm.tv_sec = ns / (1000 * 1000 * 1000);
tm.tv_nsec = ns % (1000 * 1000 * 1000);
nanosleep(&tm, nil);
}
static int
isdelim(char c)
{
return c == '\0' || strchr(" \t\n;'\"", c) != nil;
}
/* have to free argv[0] and argv to clean up */
char**
split(char *line, int *pargc)
{
int argc, n;
char **argv, *lp, delim;
n = strlen(line)+1;
// just allocate enough
lp = (char*)malloc(2*n);
argv = (char**)malloc(sizeof(char*)*n);
argc = 0;
for(; *line; line++) {
while(isspace(*line)) line++;
if(*line == '\0')
break;
argv[argc++] = lp;
if(*line == '"' || *line == '\'') {
delim = *line++;
while(*line && *line != delim) {
if(*line == '\\')
line++;
*lp++ = *line++;
}
} else {
while(!isdelim(*line)) {
if(*line == '\\')
line++;
*lp++ = *line++;
}
}
*lp++ = '\0';
}
if(pargc) *pargc = argc;
argv[argc++] = nil;
return argv;
}
#include <poll.h>
struct FDmsg
{
int msg;
FD *fd;
};
static FD *fds[100];
static struct pollfd pfds[100];
static int nfds;
static int pollpipe[2] = { -1, -1 };
static void
removeslot(int i)
{
pfds[i].revents = 0;
pfds[i].events = 0;
pfds[i].fd = -1;
fds[i]->fd = -1;
fds[i]->id = -1;
fds[i]->ready = 0;
fds[i] = nil;
}
static void
pollfds(void)
{
struct FDmsg msg;
FD *fd;
int i, n;
pipe(pollpipe);
pfds[0].fd = pollpipe[0];
pfds[0].events = POLLIN;
nfds = 1;
for(;;) {
n = poll(pfds, nfds, -1);
if(n < 0) {
perror("error poll");
return;
}
/* someone wants us to watch their fd or has closed it */
if(pfds[0].revents & POLLIN) {
read(pfds[0].fd, &msg, sizeof(msg));
fd = msg.fd;
switch(msg.msg) {
// wait
case 1:
if(fd->id >= 0) {
/* already in list */
assert(fds[fd->id] == fd);
//printf("polling fd %d in slot %d\n", fd->fd, fd->id);
} else {
/* add to list */
for(i = 1; i < nfds; i++)
if(fds[i] == nil)
break;
assert(i < nelem(pfds));
if(i == nfds) nfds++;
fd->id = i;
fds[fd->id] = fd;
//printf("adding fd %d in slot %d\n", fd->fd, fd->id);
}
pfds[fd->id].fd = fd->fd;
pfds[fd->id].events = POLLIN;
pfds[fd->id].revents = 0;
fd->ready = 0;
break;
// close
case 2:
/* fd was closed */
//printf("received close for fd %d slot %d\n", pfds[fd->id].fd, fd->id);
assert(fd->id >= 0);
close(pfds[fd->id].fd);
removeslot(fd->id);
break;
}
}
for(i = 1; i < nfds; i++) {
/* fd was closed on other side */
if(pfds[i].revents & POLLHUP) {
//printf("fd %d hung up\n", pfds[i].fd);
close(fds[i]->fd);
removeslot(i);
}
if(pfds[i].revents & POLLIN) {
//printf("fd %d became ready\n", pfds[i].fd);
/* ignore this fd for now */
pfds[i].fd = -1;
pfds[i].events = 0;
pfds[i].revents = 0;
fds[i]->ready = 1;
}
if(pfds[i].revents)
printf("more events on fd %d: %d\n", pfds[i].fd, pfds[i].revents);
}
}
}
static void *pollthread(void *arg) { pollfds(); return nil; }
void
startpolling(void)
{
pthread_t th;
pthread_create(&th, nil, pollthread, nil);
// wait for thread to start so waitfd can do its thing
while(((volatile int*)pollpipe)[0] < 0) usleep(1000);
}
void
waitfd(FD *fd)
{
struct FDmsg msg;
if(fd->fd < 0)
return;
fd->ready = 0;
msg.msg = 1;
msg.fd = fd;
write(pollpipe[1], &msg, sizeof(msg));
}
void
closefd(FD *fd)
{
struct FDmsg msg;
printf("closing fd %d\n", fd->fd);
int i = fd->id;
msg.msg = 2;
msg.fd = fd;
write(pollpipe[1], &msg, sizeof(msg));
// wait for thread to notice
while(((volatile FD**)fds)[i] != nil) usleep(1);
}