mirror of
https://github.com/aap/pdp6.git
synced 2026-01-11 23:53:31 +00:00
190 lines
3.3 KiB
C
190 lines
3.3 KiB
C
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <ctype.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 <unistd.h>
|
|
#include <netdb.h>
|
|
|
|
|
|
void
|
|
strtolower(char *s)
|
|
{
|
|
for(; *s != '\0'; s++)
|
|
*s = tolower(*s);
|
|
}
|
|
|
|
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
|
|
writen(int fd, void *data, int n)
|
|
{
|
|
int m;
|
|
|
|
while(n > 0){
|
|
m = write(fd, data, n);
|
|
if(m == -1)
|
|
return -1;
|
|
data += m;
|
|
n -= m;
|
|
}
|
|
return 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;
|
|
}
|
|
|
|
void
|
|
serve(int port, void (*handlecon)(int, void*), void *arg)
|
|
{
|
|
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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
listen(sockfd, 5);
|
|
len = sizeof(client);
|
|
while(confd = accept(sockfd, (struct sockaddr*)&client, &len),
|
|
confd >= 0)
|
|
handlecon(confd, arg);
|
|
perror("error: accept");
|
|
return;
|
|
}
|
|
|
|
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 NULL;
|
|
}
|
|
ftruncate(fd, sz);
|
|
p = mmap(NULL, sz, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
|
|
if(p == MAP_FAILED) {
|
|
fprintf(stderr, "couldn't mmap file\n");
|
|
return NULL;
|
|
}
|
|
|
|
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 NULL;
|
|
}
|
|
p = mmap(NULL, sz, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
|
|
if(p == MAP_FAILED) {
|
|
fprintf(stderr, "couldn't mmap file\n");
|
|
return NULL;
|
|
}
|
|
|
|
return p;
|
|
}
|