1*a9643ea8Slogwang /* 2*a9643ea8Slogwang * Copyright (c) 2013 Patrick Kelsey. All rights reserved. 3*a9643ea8Slogwang * Copyright (C) 2017 THL A29 Limited, a Tencent company. 4*a9643ea8Slogwang * All rights reserved. 5*a9643ea8Slogwang * 6*a9643ea8Slogwang * Redistribution and use in source and binary forms, with or without 7*a9643ea8Slogwang * modification, are permitted provided that the following conditions are met: 8*a9643ea8Slogwang * 9*a9643ea8Slogwang * 1. Redistributions of source code must retain the above copyright notice, this 10*a9643ea8Slogwang * list of conditions and the following disclaimer. 11*a9643ea8Slogwang * 2. Redistributions in binary form must reproduce the above copyright notice, 12*a9643ea8Slogwang * this list of conditions and the following disclaimer in the documentation 13*a9643ea8Slogwang * and/or other materials provided with the distribution. 14*a9643ea8Slogwang * 15*a9643ea8Slogwang * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND 16*a9643ea8Slogwang * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 17*a9643ea8Slogwang * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 18*a9643ea8Slogwang * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR 19*a9643ea8Slogwang * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 20*a9643ea8Slogwang * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 21*a9643ea8Slogwang * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 22*a9643ea8Slogwang * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 23*a9643ea8Slogwang * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 24*a9643ea8Slogwang * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 25*a9643ea8Slogwang * 26*a9643ea8Slogwang * Derived in part from libuinet's uinet_host_interface.c. 27*a9643ea8Slogwang */ 28*a9643ea8Slogwang 29*a9643ea8Slogwang #include <assert.h> 30*a9643ea8Slogwang #include <errno.h> 31*a9643ea8Slogwang #include <string.h> 32*a9643ea8Slogwang #include <stdarg.h> 33*a9643ea8Slogwang #include <stdint.h> 34*a9643ea8Slogwang #include <stdio.h> 35*a9643ea8Slogwang #include <stdlib.h> 36*a9643ea8Slogwang #include <sys/mman.h> 37*a9643ea8Slogwang #include <sys/stat.h> 38*a9643ea8Slogwang #include <pthread.h> 39*a9643ea8Slogwang #include <sched.h> 40*a9643ea8Slogwang #include <time.h> 41*a9643ea8Slogwang 42*a9643ea8Slogwang #include <openssl/rand.h> 43*a9643ea8Slogwang #include <rte_malloc.h> 44*a9643ea8Slogwang 45*a9643ea8Slogwang #include "ff_host_interface.h" 46*a9643ea8Slogwang #include "ff_errno.h" 47*a9643ea8Slogwang 48*a9643ea8Slogwang void * 49*a9643ea8Slogwang ff_mmap(void *addr, uint64_t len, int prot, int flags, int fd, uint64_t offset) 50*a9643ea8Slogwang { 51*a9643ea8Slogwang //return rte_malloc("", len, 4096); 52*a9643ea8Slogwang int host_prot; 53*a9643ea8Slogwang int host_flags; 54*a9643ea8Slogwang 55*a9643ea8Slogwang assert(ff_PROT_NONE == PROT_NONE); 56*a9643ea8Slogwang host_prot = 0; 57*a9643ea8Slogwang if ((prot & ff_PROT_READ) == ff_PROT_READ) host_prot |= PROT_READ; 58*a9643ea8Slogwang if ((prot & ff_PROT_WRITE) == ff_PROT_WRITE) host_prot |= PROT_WRITE; 59*a9643ea8Slogwang 60*a9643ea8Slogwang host_flags = 0; 61*a9643ea8Slogwang if ((flags & ff_MAP_SHARED) == ff_MAP_SHARED) host_flags |= MAP_SHARED; 62*a9643ea8Slogwang if ((flags & ff_MAP_PRIVATE) == ff_MAP_PRIVATE) host_flags |= MAP_PRIVATE; 63*a9643ea8Slogwang if ((flags & ff_MAP_ANON) == ff_MAP_ANON) host_flags |= MAP_ANON; 64*a9643ea8Slogwang 65*a9643ea8Slogwang void *ret = (mmap(addr, len, host_prot, host_flags, fd, offset)); 66*a9643ea8Slogwang 67*a9643ea8Slogwang if ((uint64_t)ret == -1) { 68*a9643ea8Slogwang printf("fst mmap failed:%s\n", strerror(errno)); 69*a9643ea8Slogwang exit(1); 70*a9643ea8Slogwang } 71*a9643ea8Slogwang return ret; 72*a9643ea8Slogwang } 73*a9643ea8Slogwang 74*a9643ea8Slogwang int 75*a9643ea8Slogwang ff_munmap(void *addr, uint64_t len) 76*a9643ea8Slogwang { 77*a9643ea8Slogwang //rte_free(addr); 78*a9643ea8Slogwang //return 0; 79*a9643ea8Slogwang return (munmap(addr, len)); 80*a9643ea8Slogwang } 81*a9643ea8Slogwang 82*a9643ea8Slogwang 83*a9643ea8Slogwang void * 84*a9643ea8Slogwang ff_malloc(uint64_t size) 85*a9643ea8Slogwang { 86*a9643ea8Slogwang //return rte_malloc("", size, 0); 87*a9643ea8Slogwang return (malloc(size)); 88*a9643ea8Slogwang } 89*a9643ea8Slogwang 90*a9643ea8Slogwang 91*a9643ea8Slogwang void * 92*a9643ea8Slogwang ff_calloc(uint64_t number, uint64_t size) 93*a9643ea8Slogwang { 94*a9643ea8Slogwang //return rte_calloc("", number, size, 0); 95*a9643ea8Slogwang return (calloc(number, size)); 96*a9643ea8Slogwang } 97*a9643ea8Slogwang 98*a9643ea8Slogwang 99*a9643ea8Slogwang void * 100*a9643ea8Slogwang ff_realloc(void *p, uint64_t size) 101*a9643ea8Slogwang { 102*a9643ea8Slogwang if (size) { 103*a9643ea8Slogwang //return rte_realloc(p, size, 0); 104*a9643ea8Slogwang return (realloc(p, size)); 105*a9643ea8Slogwang } 106*a9643ea8Slogwang 107*a9643ea8Slogwang return (p); 108*a9643ea8Slogwang } 109*a9643ea8Slogwang 110*a9643ea8Slogwang 111*a9643ea8Slogwang void 112*a9643ea8Slogwang ff_free(void *p) 113*a9643ea8Slogwang { 114*a9643ea8Slogwang //rte_free(p); 115*a9643ea8Slogwang free(p); 116*a9643ea8Slogwang } 117*a9643ea8Slogwang 118*a9643ea8Slogwang void panic(const char *, ...) __attribute__((__noreturn__)); 119*a9643ea8Slogwang 120*a9643ea8Slogwang const char *panicstr = NULL; 121*a9643ea8Slogwang 122*a9643ea8Slogwang void 123*a9643ea8Slogwang panic(const char *fmt, ...) 124*a9643ea8Slogwang { 125*a9643ea8Slogwang va_list ap; 126*a9643ea8Slogwang 127*a9643ea8Slogwang va_start(ap, fmt); 128*a9643ea8Slogwang vprintf(fmt, ap); 129*a9643ea8Slogwang va_end(ap); 130*a9643ea8Slogwang 131*a9643ea8Slogwang abort(); 132*a9643ea8Slogwang } 133*a9643ea8Slogwang 134*a9643ea8Slogwang void 135*a9643ea8Slogwang ff_clock_gettime(int id, int64_t *sec, long *nsec) 136*a9643ea8Slogwang { 137*a9643ea8Slogwang struct timespec ts; 138*a9643ea8Slogwang int host_id; 139*a9643ea8Slogwang int rv; 140*a9643ea8Slogwang 141*a9643ea8Slogwang switch (id) { 142*a9643ea8Slogwang case ff_CLOCK_REALTIME: 143*a9643ea8Slogwang host_id = CLOCK_REALTIME; 144*a9643ea8Slogwang break; 145*a9643ea8Slogwang #ifdef CLOCK_MONOTONIC_FAST 146*a9643ea8Slogwang case ff_CLOCK_MONOTONIC_FAST: 147*a9643ea8Slogwang host_id = CLOCK_MONOTONIC_FAST; 148*a9643ea8Slogwang break; 149*a9643ea8Slogwang #endif 150*a9643ea8Slogwang case ff_CLOCK_MONOTONIC: 151*a9643ea8Slogwang default: 152*a9643ea8Slogwang host_id = CLOCK_MONOTONIC; 153*a9643ea8Slogwang break; 154*a9643ea8Slogwang } 155*a9643ea8Slogwang 156*a9643ea8Slogwang rv = clock_gettime(host_id, &ts); 157*a9643ea8Slogwang assert(0 == rv); 158*a9643ea8Slogwang 159*a9643ea8Slogwang *sec = (int64_t)ts.tv_sec; 160*a9643ea8Slogwang *nsec = (long)ts.tv_nsec; 161*a9643ea8Slogwang } 162*a9643ea8Slogwang 163*a9643ea8Slogwang uint64_t 164*a9643ea8Slogwang ff_clock_gettime_ns(int id) 165*a9643ea8Slogwang { 166*a9643ea8Slogwang int64_t sec; 167*a9643ea8Slogwang long nsec; 168*a9643ea8Slogwang 169*a9643ea8Slogwang ff_clock_gettime(id, &sec, &nsec); 170*a9643ea8Slogwang 171*a9643ea8Slogwang return ((uint64_t)sec * ff_NSEC_PER_SEC + nsec); 172*a9643ea8Slogwang } 173*a9643ea8Slogwang 174*a9643ea8Slogwang /* 175*a9643ea8Slogwang * Sleeps for at least the given number of nanoseconds and returns 0, 176*a9643ea8Slogwang * unless there is a non-EINTR failure, in which case a non-zero value is 177*a9643ea8Slogwang * returned. 178*a9643ea8Slogwang */ 179*a9643ea8Slogwang int 180*a9643ea8Slogwang ff_nanosleep(uint64_t nsecs) 181*a9643ea8Slogwang { 182*a9643ea8Slogwang struct timespec ts; 183*a9643ea8Slogwang struct timespec rts; 184*a9643ea8Slogwang int rv; 185*a9643ea8Slogwang 186*a9643ea8Slogwang ts.tv_sec = nsecs / ff_NSEC_PER_SEC; 187*a9643ea8Slogwang ts.tv_nsec = nsecs % ff_NSEC_PER_SEC; 188*a9643ea8Slogwang while ((-1 == (rv = nanosleep(&ts, &rts))) && (EINTR == errno)) { 189*a9643ea8Slogwang ts = rts; 190*a9643ea8Slogwang } 191*a9643ea8Slogwang if (-1 == rv) { 192*a9643ea8Slogwang rv = errno; 193*a9643ea8Slogwang } 194*a9643ea8Slogwang 195*a9643ea8Slogwang return (rv); 196*a9643ea8Slogwang } 197*a9643ea8Slogwang 198*a9643ea8Slogwang void 199*a9643ea8Slogwang ff_arc4rand(void *ptr, unsigned int len, int reseed) 200*a9643ea8Slogwang { 201*a9643ea8Slogwang (void)reseed; 202*a9643ea8Slogwang 203*a9643ea8Slogwang RAND_pseudo_bytes(ptr, len); 204*a9643ea8Slogwang } 205*a9643ea8Slogwang 206*a9643ea8Slogwang uint32_t 207*a9643ea8Slogwang ff_arc4random(void) 208*a9643ea8Slogwang { 209*a9643ea8Slogwang uint32_t ret; 210*a9643ea8Slogwang ff_arc4rand(&ret, sizeof ret, 0); 211*a9643ea8Slogwang return ret; 212*a9643ea8Slogwang } 213*a9643ea8Slogwang 214*a9643ea8Slogwang int ff_setenv(const char *name, const char *value) 215*a9643ea8Slogwang { 216*a9643ea8Slogwang return setenv(name, value, 1); 217*a9643ea8Slogwang } 218*a9643ea8Slogwang 219*a9643ea8Slogwang char *ff_getenv(const char *name) 220*a9643ea8Slogwang { 221*a9643ea8Slogwang return getenv(name); 222*a9643ea8Slogwang } 223*a9643ea8Slogwang 224*a9643ea8Slogwang void ff_os_errno(int error) 225*a9643ea8Slogwang { 226*a9643ea8Slogwang switch (error) { 227*a9643ea8Slogwang case ff_EPERM: errno = EPERM; break; 228*a9643ea8Slogwang case ff_ENOENT: errno = ENOENT; break; 229*a9643ea8Slogwang case ff_ESRCH: errno = ESRCH; break; 230*a9643ea8Slogwang case ff_EINTR: errno = EINTR; break; 231*a9643ea8Slogwang case ff_EIO: errno = EIO; break; 232*a9643ea8Slogwang case ff_ENXIO: errno = ENXIO; break; 233*a9643ea8Slogwang case ff_E2BIG: errno = E2BIG; break; 234*a9643ea8Slogwang case ff_ENOEXEC: errno = ENOEXEC; break; 235*a9643ea8Slogwang case ff_EBADF: errno = EBADF; break; 236*a9643ea8Slogwang case ff_ECHILD: errno = ECHILD; break; 237*a9643ea8Slogwang case ff_EDEADLK: errno = EDEADLK; break; 238*a9643ea8Slogwang case ff_ENOMEM: errno = ENOMEM; break; 239*a9643ea8Slogwang case ff_EACCES: errno = EACCES; break; 240*a9643ea8Slogwang case ff_EFAULT: errno = EFAULT; break; 241*a9643ea8Slogwang case ff_ENOTBLK: errno = ENOTBLK; break; 242*a9643ea8Slogwang case ff_EBUSY: errno = EBUSY; break; 243*a9643ea8Slogwang case ff_EEXIST: errno = EEXIST; break; 244*a9643ea8Slogwang case ff_EXDEV: errno = EXDEV; break; 245*a9643ea8Slogwang case ff_ENODEV: errno = ENODEV; break; 246*a9643ea8Slogwang case ff_ENOTDIR: errno = ENOTDIR; break; 247*a9643ea8Slogwang case ff_EISDIR: errno = EISDIR; break; 248*a9643ea8Slogwang case ff_EINVAL: errno = EINVAL; break; 249*a9643ea8Slogwang case ff_ENFILE: errno = ENFILE; break; 250*a9643ea8Slogwang case ff_EMFILE: errno = EMFILE; break; 251*a9643ea8Slogwang case ff_ENOTTY: errno = ENOTTY; break; 252*a9643ea8Slogwang case ff_ETXTBSY: errno = ETXTBSY; break; 253*a9643ea8Slogwang case ff_EFBIG: errno = EFBIG; break; 254*a9643ea8Slogwang case ff_ENOSPC: errno = ENOSPC; break; 255*a9643ea8Slogwang case ff_ESPIPE: errno = ESPIPE; break; 256*a9643ea8Slogwang case ff_EROFS: errno = EROFS; break; 257*a9643ea8Slogwang case ff_EMLINK: errno = EMLINK; break; 258*a9643ea8Slogwang case ff_EPIPE: errno = EPIPE; break; 259*a9643ea8Slogwang case ff_EDOM: errno = EDOM; break; 260*a9643ea8Slogwang case ff_ERANGE: errno = ERANGE; break; 261*a9643ea8Slogwang 262*a9643ea8Slogwang /* case ff_EAGAIN: same as EWOULDBLOCK */ 263*a9643ea8Slogwang case ff_EWOULDBLOCK: errno = EWOULDBLOCK; break; 264*a9643ea8Slogwang 265*a9643ea8Slogwang case ff_EINPROGRESS: errno = EINPROGRESS; break; 266*a9643ea8Slogwang case ff_EALREADY: errno = EALREADY; break; 267*a9643ea8Slogwang case ff_ENOTSOCK: errno = ENOTSOCK; break; 268*a9643ea8Slogwang case ff_EDESTADDRREQ: errno = EDESTADDRREQ; break; 269*a9643ea8Slogwang case ff_EMSGSIZE: errno = EMSGSIZE; break; 270*a9643ea8Slogwang case ff_EPROTOTYPE: errno = EPROTOTYPE; break; 271*a9643ea8Slogwang case ff_ENOPROTOOPT: errno = ENOPROTOOPT; break; 272*a9643ea8Slogwang case ff_EPROTONOSUPPORT: errno = EPROTONOSUPPORT; break; 273*a9643ea8Slogwang case ff_ESOCKTNOSUPPORT: errno = ESOCKTNOSUPPORT; break; 274*a9643ea8Slogwang 275*a9643ea8Slogwang /* case ff_EOPNOTSUPP: same as ENOTSUP */ 276*a9643ea8Slogwang case ff_ENOTSUP: errno = ENOTSUP; break; 277*a9643ea8Slogwang 278*a9643ea8Slogwang case ff_EPFNOSUPPORT: errno = EPFNOSUPPORT; break; 279*a9643ea8Slogwang case ff_EAFNOSUPPORT: errno = EAFNOSUPPORT; break; 280*a9643ea8Slogwang case ff_EADDRINUSE: errno = EADDRINUSE; break; 281*a9643ea8Slogwang case ff_EADDRNOTAVAIL: errno = EADDRNOTAVAIL; break; 282*a9643ea8Slogwang case ff_ENETDOWN: errno = ENETDOWN; break; 283*a9643ea8Slogwang case ff_ENETUNREACH: errno = ENETUNREACH; break; 284*a9643ea8Slogwang case ff_ENETRESET: errno = ENETRESET; break; 285*a9643ea8Slogwang case ff_ECONNABORTED: errno = ECONNABORTED; break; 286*a9643ea8Slogwang case ff_ECONNRESET: errno = ECONNRESET; break; 287*a9643ea8Slogwang case ff_ENOBUFS: errno = ENOBUFS; break; 288*a9643ea8Slogwang case ff_EISCONN: errno = EISCONN; break; 289*a9643ea8Slogwang case ff_ENOTCONN: errno = ENOTCONN; break; 290*a9643ea8Slogwang case ff_ESHUTDOWN: errno = ESHUTDOWN; break; 291*a9643ea8Slogwang case ff_ETOOMANYREFS: errno = ETOOMANYREFS; break; 292*a9643ea8Slogwang case ff_ETIMEDOUT: errno = ETIMEDOUT; break; 293*a9643ea8Slogwang case ff_ECONNREFUSED: errno = ECONNREFUSED; break; 294*a9643ea8Slogwang case ff_ELOOP: errno = ELOOP; break; 295*a9643ea8Slogwang case ff_ENAMETOOLONG: errno = ENAMETOOLONG; break; 296*a9643ea8Slogwang case ff_EHOSTDOWN: errno = EHOSTDOWN; break; 297*a9643ea8Slogwang case ff_EHOSTUNREACH: errno = EHOSTUNREACH; break; 298*a9643ea8Slogwang case ff_ENOTEMPTY: errno = ENOTEMPTY; break; 299*a9643ea8Slogwang case ff_EUSERS: errno = EUSERS; break; 300*a9643ea8Slogwang case ff_EDQUOT: errno = EDQUOT; break; 301*a9643ea8Slogwang case ff_ESTALE: errno = ESTALE; break; 302*a9643ea8Slogwang case ff_EREMOTE: errno = EREMOTE; break; 303*a9643ea8Slogwang case ff_ENOLCK: errno = ENOLCK; break; 304*a9643ea8Slogwang case ff_ENOSYS: errno = ENOSYS; break; 305*a9643ea8Slogwang case ff_EIDRM: errno = EIDRM; break; 306*a9643ea8Slogwang case ff_ENOMSG: errno = ENOMSG; break; 307*a9643ea8Slogwang case ff_EOVERFLOW: errno = EOVERFLOW; break; 308*a9643ea8Slogwang case ff_ECANCELED: errno = ECANCELED; break; 309*a9643ea8Slogwang case ff_EILSEQ: errno = EILSEQ; break; 310*a9643ea8Slogwang case ff_EBADMSG: errno = EBADMSG; break; 311*a9643ea8Slogwang case ff_EMULTIHOP: errno = EMULTIHOP; break; 312*a9643ea8Slogwang case ff_ENOLINK: errno = ENOLINK; break; 313*a9643ea8Slogwang case ff_EPROTO: errno = EPROTO; break; 314*a9643ea8Slogwang default: errno = error; break; 315*a9643ea8Slogwang } 316*a9643ea8Slogwang 317*a9643ea8Slogwang } 318*a9643ea8Slogwang 319