/* Serval Distributed Numbering Architecture (DNA) Copyright (C) 2012 Serval Project Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include #include #include #include #include #include #include "net.h" #include "serval.h" int _set_nonblock(int fd, struct __sourceloc where) { int flags; if ((flags = fcntl(fd, F_GETFL, NULL)) == -1) { logMessage_perror(LOG_LEVEL_ERROR, where, "set_nonblock: fcntl(%d,F_GETFL,NULL)", fd); return -1; } if (fcntl(fd, F_SETFL, flags | O_NONBLOCK) == -1) { logMessage_perror(LOG_LEVEL_ERROR, where, "set_nonblock: fcntl(%d,F_SETFL,0x%x|O_NONBLOCK)", fd, flags); return -1; } return 0; } int _set_block(int fd, struct __sourceloc where) { int flags; if ((flags = fcntl(fd, F_GETFL, NULL)) == -1) { logMessage_perror(LOG_LEVEL_ERROR, where, "set_block: fcntl(%d,F_GETFL,NULL)", fd); return -1; } if (fcntl(fd, F_SETFL, flags & ~O_NONBLOCK) == -1) { logMessage_perror(LOG_LEVEL_ERROR, where, "set_block: fcntl(%d,F_SETFL,0x%x&~O_NONBLOCK)", fd, flags); return -1; } return 0; } ssize_t _read_nonblock(int fd, void *buf, size_t len, struct __sourceloc where) { ssize_t nread = read(fd, buf, len); if (nread == -1) { switch (errno) { case EINTR: case EAGAIN: #if defined(EWOULDBLOCK) && EWOULDBLOCK != EAGAIN case EWOULDBLOCK: #endif return 0; } logMessage_perror(LOG_LEVEL_ERROR, where, "read_nonblock: read(%d,%p,%lu)", fd, buf, (unsigned long)len); return -1; } return nread; } ssize_t _write_all(int fd, const void *buf, size_t len, struct __sourceloc where) { ssize_t written = write(fd, buf, len); if (written == -1) { logMessage_perror(LOG_LEVEL_ERROR, where, "write_all: write(%d,%p %s,%lu)", fd, buf, alloca_toprint(30, buf, len), (unsigned long)len); return -1; } if (written != len) { logMessage(LOG_LEVEL_ERROR, where, "write_all: write(%d,%p %s,%lu) returned %ld", fd, buf, alloca_toprint(30, buf, len), (unsigned long)len, (long)written); return -1; } return written; } ssize_t _write_nonblock(int fd, const void *buf, size_t len, struct __sourceloc where) { ssize_t written = write(fd, buf, len); if (written == -1) { switch (errno) { case EINTR: case EAGAIN: #if defined(EWOULDBLOCK) && EWOULDBLOCK != EAGAIN case EWOULDBLOCK: #endif return 0; } logMessage_perror(LOG_LEVEL_ERROR, where, "write_nonblock: write(%d,%p %s,%lu)", fd, buf, alloca_toprint(30, buf, len), (unsigned long)len); return -1; } return written; } ssize_t _write_all_nonblock(int fd, const void *buf, size_t len, struct __sourceloc where) { ssize_t written = _write_nonblock(fd, buf, len, where); if (written != -1 && written != len) { logMessage(LOG_LEVEL_ERROR, where, "write_all_nonblock: write(%d,%p %s,%lu) returned %ld", fd, buf, alloca_toprint(30, buf, len), (unsigned long)len, (long)written); return -1; } return written; } ssize_t _write_str(int fd, const char *str, struct __sourceloc where) { return _write_all(fd, str, strlen(str), where); } ssize_t _write_str_nonblock(int fd, const char *str, struct __sourceloc where) { return _write_all_nonblock(fd, str, strlen(str), where); } ssize_t recvwithttl(int sock,unsigned char *buffer, size_t bufferlen,int *ttl, struct sockaddr *recvaddr, socklen_t *recvaddrlen) { struct msghdr msg; struct iovec iov[1]; iov[0].iov_base=buffer; iov[0].iov_len=bufferlen; bzero(&msg,sizeof(msg)); msg.msg_name = recvaddr; msg.msg_namelen = *recvaddrlen; msg.msg_iov = &iov[0]; msg.msg_iovlen = 1; // setting the following makes the data end up in the wrong place // msg.msg_iov->iov_base=iov_buffer; // msg.msg_iov->iov_len=sizeof(iov_buffer); struct cmsghdr cmsgcmsg[16]; msg.msg_control = &cmsgcmsg[0]; msg.msg_controllen = sizeof(struct cmsghdr)*16; msg.msg_flags = 0; ssize_t len = recvmsg(sock,&msg,0); if (len == -1 && errno != EAGAIN && errno != EWOULDBLOCK) return WHY_perror("recvmsg"); if (0&&debug&DEBUG_PACKETRX) { DEBUGF("recvmsg returned %lld (flags=%d, msg_controllen=%d)", (long long) len, msg.msg_flags, msg.msg_controllen); dump("received data", buffer, len); } struct cmsghdr *cmsg; if (len>0) { for (cmsg = CMSG_FIRSTHDR(&msg); cmsg != NULL; cmsg = CMSG_NXTHDR(&msg,cmsg)) { if ((cmsg->cmsg_level == IPPROTO_IP) && ((cmsg->cmsg_type == IP_RECVTTL) ||(cmsg->cmsg_type == IP_TTL)) &&(cmsg->cmsg_len) ){ if (debug&DEBUG_PACKETRX) DEBUGF(" TTL (%p) data location resolves to %p", ttl,CMSG_DATA(cmsg)); if (CMSG_DATA(cmsg)) { *ttl = *(unsigned char *) CMSG_DATA(cmsg); if (debug&DEBUG_PACKETRX) DEBUGF(" TTL of packet is %d", *ttl); } } else { if (debug&DEBUG_PACKETRX) DEBUGF("I didn't expect to see level=%02x, type=%02x", cmsg->cmsg_level,cmsg->cmsg_type); } } } *recvaddrlen=msg.msg_namelen; return len; } int urandombytes(unsigned char *x, unsigned long long xlen) { static int urandomfd = -1; int tries = 0; if (urandomfd == -1) { for (tries = 0; tries < 4; ++tries) { urandomfd = open("/dev/urandom",O_RDONLY); if (urandomfd != -1) break; sleep(1); } if (urandomfd == -1) { WHY_perror("open(/dev/urandom)"); return -1; } } tries = 0; while (xlen > 0) { int i = (xlen < 1048576) ? xlen : 1048576; i = read(urandomfd, x, i); if (i == -1) { if (++tries > 4) { WHY_perror("read(/dev/urandom)"); return -1; } sleep(1); } else { tries = 0; x += i; xlen -= i; } } return 0; } time_ms_t gettime_ms() { struct timeval nowtv; // If gettimeofday() fails or returns an invalid value, all else is lost! if (gettimeofday(&nowtv, NULL) == -1) FATAL_perror("gettimeofday"); if (nowtv.tv_sec < 0 || nowtv.tv_usec < 0 || nowtv.tv_usec >= 1000000) FATALF("gettimeofday returned tv_sec=%ld tv_usec=%ld", nowtv.tv_sec, nowtv.tv_usec); return nowtv.tv_sec * 1000LL + nowtv.tv_usec / 1000; } // Returns sleep time remaining. time_ms_t sleep_ms(time_ms_t milliseconds) { if (milliseconds <= 0) return 0; struct timespec delay; struct timespec remain; delay.tv_sec = milliseconds / 1000; delay.tv_nsec = (milliseconds % 1000) * 1000000; if (nanosleep(&delay, &remain) == -1 && errno != EINTR) FATALF_perror("nanosleep(tv_sec=%ld, tv_nsec=%ld)", delay.tv_sec, delay.tv_nsec); return remain.tv_sec * 1000 + remain.tv_nsec / 1000000; }