stub.c
来自「一个很有名的浏览器」· C语言 代码 · 共 379 行
C
379 行
/* Libc stub functions *//* $Id: stub.c,v 1.18.4.2 2005/02/28 01:31:50 jonas Exp $ */#ifdef HAVE_CONFIG_H#include "config.h"#endif#include <errno.h>#include <ctype.h>#include <stdarg.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <sys/types.h>#ifdef HAVE_SYS_SOCKET_H#include <sys/socket.h> /* SunOS needs this after sys/types.h */#endif#ifdef HAVE_UNISTD_H#include <unistd.h>#endif#ifdef HAVE_NETINET_IN_H#include <netinet/in.h> /* OS/2 needs this after sys/types.h */#endif#ifdef HAVE_ARPA_INET_H#include <arpa/inet.h>#endif#include "elinks.h"#include "osdep/stub.h"#include "util/conv.h"/* These stubs are exception to our "Use (unsigned char *)!" rule. This is * because the stubbed functions are defined using (char *), and we could get * in trouble with this. Or when you use (foo ? strstr() : strcasestr()) and * one of these is system and another stub, we're in trouble and get "Pointer * type mismatch in conditional expression", game over. */#define toupper_equal(s1, s2) (toupper(*((char *) s1)) == toupper(*((char *) s2)))#define toupper_delta(s1, s2) (toupper(*((char *) s1)) - toupper(*((char *) s2)))#ifndef HAVE_STRCASECMPinline intelinks_strcasecmp(const char *s1, const char *s2){ while (*s1 != '\0' && toupper_equal(s1, s2)) { s1++; s2++; } return toupper_delta(s1, s2);}#endif /* !HAVE_STRCASECMP */#ifndef HAVE_STRNCASECMPinline intelinks_strncasecmp(const char *s1, const char *s2, size_t len){ if (len == 0) return 0; while (len-- != 0 && toupper_equal(s1, s2)) { if (len == 0 || *s1 == '\0' || *s2 == '\0') return 0; s1++; s2++; } return toupper_delta(s1, s2);}#endif /* !HAVE_STRNCASECMP */#ifndef HAVE_STRCASESTR/* Stub for strcasestr(), GNU extension */inline char *elinks_strcasestr(const char *haystack, const char *needle){ size_t haystack_length = strlen(haystack); size_t needle_length = strlen(needle); int i; if (haystack_length < needle_length) return NULL; for (i = haystack_length - needle_length + 1; i; i--) { if (!strncasecmp(haystack, needle, needle_length)) return (char *) haystack; haystack++; } return NULL;}#endif#ifndef HAVE_STRDUPinline char *elinks_strdup(const char *str){ int str_len = strlen(str); char *new = malloc(str_len + 1); if (new) { if (str_len) memcpy(new, str, str_len); new[str_len] = '\0'; } return new;}#endif#ifndef HAVE_STRERROR/* Many older systems don't have this, but have the global sys_errlist array * instead. */#if 0extern int sys_nerr;extern const char *const sys_errlist[];#endifinline const char *elinks_strerror(int err_no){ if (err_no < 0 || err_no > sys_nerr) return (const char *) "Unknown Error"; else return (const char *) sys_errlist[err_no];}#endif#ifndef HAVE_STRSTR/* From http://www.unixpapa.com/incnote/string.html */inline char *elinks_strstr(const char *s, const char *p){ char *sp, *pp; for (sp = (char *) s, pp = (char *) p; *sp && *pp; ) { if (*sp == *pp) { sp++; pp++; } else { sp = sp - (pp - p) + 1; pp = (char *) p; } } return (*pp ? NULL : sp - (pp - p));}#endif#if !defined(HAVE_MEMMOVE) && !defined(HAVE_BCOPY)/* The memmove() function is a bit rarer than the other memory functions - * some systems that have the others don't have this. It is identical to * memcpy() but is guaranteed to work even if the strings overlap. * Most systems that don't have memmove() do have * the BSD bcopy() though some really old systems have neither. * Note that bcopy() has the order of the source and destination * arguments reversed. * From http://www.unixpapa.com/incnote/string.html *//* Modified for ELinks by Zas. */inline void *elinks_memmove(void *d, const void *s, size_t n){ register char *dst = (char *) d; register char *src = (char *) s; if (!n || src == dst) return (void *) dst; if (src > dst) for (; n > 0; n--) *(dst++) = *(src++); else for (dst += n - 1, src += n - 1; n > 0; n--) *(dst--) = *(src--); return (void *) dst;}#endif#ifndef HAVE_STPCPYinline char *elinks_stpcpy(char *dest, const char *src){ while ((*dest++ = *src++)); return (dest - 1);}#endif#ifndef HAVE_MEMPCPYinline void *elinks_mempcpy(void *dest, const void *src, size_t n){ return (void *) ((char *) memcpy(dest, src, n) + n);}#endif#ifndef HAVE_ISDIGITinline intelinks_isdigit(int i){ return i >= '0' && i <= '9';}#endif#ifndef HAVE_MEMRCHRinline void *elinks_memrchr(const void *s, int c, size_t n){ char *pos = (char *) s; while (n > 0) { if (pos[n - 1] == (char) c) return (void *) &pos[n - 1]; n--; } return NULL;}#endif#ifndef HAVE_RAISE#if !defined(HAVE_KILL) || !defined(HAVE_GETPID)#error The raise() stub function requires kill() and getpid()#endifintelinks_raise(int signal){ return(kill(getpid(), signal));}#endif#ifndef HAVE_INET_NTOP/* Original code by Paul Vixie. Modified by Gisle Vanem. */#define IN6ADDRSZ 16#define INADDRSZ 4#define INT16SZ 2/* TODO: Move and populate. --jonas */#ifdef CONFIG_WIN32#define SET_ERRNO(e) WSASetLastError(errno = (e))#else#define SET_ERRNO(e) errno = e#endif/* Format an IPv4 address, more or less like inet_ntoa(). * * Returns `dst' (as a const) * Note: * - uses no statics * - takes a unsigned char * not an in_addr as input */static const char *elinks_inet_ntop4(const unsigned char *src, unsigned char *dst, size_t size){ const unsigned char *addr = inet_ntoa(*(struct in_addr*)src); if (strlen(addr) >= size) { SET_ERRNO(ENOSPC); return NULL; } return strcpy(dst, addr);}#ifdef CONFIG_IPV6/* Convert IPv6 binary address into presentation (printable) format. */static const char *elinks_inet_ntop6(const unsigned char *src, char *dst, size_t size){ /* Note that int32_t and int16_t need only be "at least" large enough * to contain a value of the specified size. On some systems, like * Crays, there is no such thing as an integer variable with 16 bits. * Keep this in mind if you think this function should have been coded * to use pointer overlays. All the world's not a VAX. */ unsigned char tmp[sizeof("ffff:ffff:ffff:ffff:ffff:ffff:255.255.255.255")]; unsigned char *tp; struct { long base; long len; } best, cur; unsigned long words[IN6ADDRSZ / INT16SZ]; int i; /* Preprocess: * Copy the input (bytewise) array into a wordwise array. * Find the longest run of 0x00's in src[] for :: shorthanding. */ memset(words, 0, sizeof(words)); for (i = 0; i < IN6ADDRSZ; i++) words[i/2] |= (src[i] << ((1 - (i % 2)) << 3)); best.base = -1; cur.base = -1; for (i = 0; i < (IN6ADDRSZ / INT16SZ); i++) { if (words[i] == 0) { if (cur.base == -1) cur.base = i, cur.len = 1; else cur.len++; } else if (cur.base != -1) { if (best.base == -1 || cur.len > best.len) best = cur; cur.base = -1; } } if ((cur.base != -1) && (best.base == -1 || cur.len > best.len)) best = cur; if (best.base != -1 && best.len < 2) best.base = -1; /* Format the result. */ for (tp = tmp, i = 0; i < (IN6ADDRSZ / INT16SZ); i++) { /* Are we inside the best run of 0x00's? */ if (best.base != -1 && i >= best.base && i < (best.base + best.len)) { if (i == best.base) *tp++ = ':'; continue; } /* Are we following an initial run of 0x00s or any real hex? */ if (i != 0) *tp++ = ':'; /* Is this address an encapsulated IPv4? */ if (i == 6 && best.base == 0 && (best.len == 6 || (best.len == 5 && words[5] == 0xffff))) { if (!elinks_inet_ntop4(src + 12, tp, sizeof(tmp) - (tp - tmp))) { SET_ERRNO(ENOSPC); return NULL; } tp += strlen(tp); break; } tp += snprintf(tp, 5, "%lx", words[i]); } /* Was it a trailing run of 0x00's? */ if (best.base != -1 && (best.base + best.len) == (IN6ADDRSZ / INT16SZ)) *tp++ = ':'; *tp++ = '\0'; /* Check for overflow, copy, and we're done. */ if ((size_t) (tp - tmp) > size) { SET_ERRNO(ENOSPC); return NULL; } return strcpy(dst, tmp);}#endif /* CONFIG_IPV6 *//* Convert a network format address to presentation format. * * Returns pointer to presentation format address (`dst'), * Returns NULL on error (see errno). */const char *elinks_inet_ntop(int af, const void *src, char *dst, size_t size){ switch (af) { case AF_INET: return elinks_inet_ntop4((const unsigned char *) src, dst, size);#ifdef CONFIG_IPV6 case AF_INET6: return elinks_inet_ntop6((const unsigned char *) src, dst, size);#endif default: SET_ERRNO(EAFNOSUPPORT); return NULL; }}#endif /* HAVE_INET_NTOP */
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?