mirror of
https://github.com/haiwen/ccnet-server.git
synced 2025-09-05 23:00:21 +00:00
Initial commit of Ccnet server.
This commit is contained in:
882
lib/libccnet_utils.c
Normal file
882
lib/libccnet_utils.c
Normal file
@@ -0,0 +1,882 @@
|
||||
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
||||
|
||||
#include <config.h>
|
||||
|
||||
#include "libccnet_utils.h"
|
||||
|
||||
#ifdef WIN32
|
||||
#include <winsock2.h>
|
||||
#include <ws2tcpip.h>
|
||||
#include <Rpc.h>
|
||||
#include <shlobj.h>
|
||||
#include <psapi.h>
|
||||
#else
|
||||
#include <arpa/inet.h>
|
||||
#endif
|
||||
|
||||
#ifndef WIN32
|
||||
#include <pwd.h>
|
||||
#include <uuid/uuid.h>
|
||||
#endif
|
||||
|
||||
#include <unistd.h>
|
||||
#include <sys/time.h>
|
||||
#include <sys/types.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/stat.h>
|
||||
#include <dirent.h>
|
||||
#include <errno.h>
|
||||
#include <limits.h>
|
||||
#include <stdarg.h>
|
||||
#include <ctype.h>
|
||||
|
||||
#include <string.h>
|
||||
#include <openssl/sha.h>
|
||||
#include <openssl/hmac.h>
|
||||
#include <openssl/evp.h>
|
||||
#include <openssl/bio.h>
|
||||
#include <openssl/buffer.h>
|
||||
|
||||
|
||||
#include <glib.h>
|
||||
#include <glib/gstdio.h>
|
||||
#include <searpc-utils.h>
|
||||
|
||||
#ifdef WIN32
|
||||
int
|
||||
ccnet_util_pgpipe (ccnet_pipe_t handles[2])
|
||||
{
|
||||
SOCKET s;
|
||||
struct sockaddr_in serv_addr;
|
||||
int len = sizeof( serv_addr );
|
||||
|
||||
handles[0] = handles[1] = INVALID_SOCKET;
|
||||
|
||||
if ( ( s = socket( AF_INET, SOCK_STREAM, 0 ) ) == INVALID_SOCKET )
|
||||
{
|
||||
g_warning("pgpipe failed to create socket: %d\n", WSAGetLastError());
|
||||
return -1;
|
||||
}
|
||||
|
||||
memset( &serv_addr, 0, sizeof( serv_addr ) );
|
||||
serv_addr.sin_family = AF_INET;
|
||||
serv_addr.sin_port = htons(0);
|
||||
serv_addr.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
|
||||
if (bind(s, (SOCKADDR *) & serv_addr, len) == SOCKET_ERROR)
|
||||
{
|
||||
g_warning("pgpipe failed to bind: %d\n", WSAGetLastError());
|
||||
closesocket(s);
|
||||
return -1;
|
||||
}
|
||||
if (listen(s, 1) == SOCKET_ERROR)
|
||||
{
|
||||
g_warning("pgpipe failed to listen: %d\n", WSAGetLastError());
|
||||
closesocket(s);
|
||||
return -1;
|
||||
}
|
||||
if (getsockname(s, (SOCKADDR *) & serv_addr, &len) == SOCKET_ERROR)
|
||||
{
|
||||
g_warning("pgpipe failed to getsockname: %d\n", WSAGetLastError());
|
||||
closesocket(s);
|
||||
return -1;
|
||||
}
|
||||
if ((handles[1] = socket(PF_INET, SOCK_STREAM, 0)) == INVALID_SOCKET)
|
||||
{
|
||||
g_warning("pgpipe failed to create socket 2: %d\n", WSAGetLastError());
|
||||
closesocket(s);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (connect(handles[1], (SOCKADDR *) & serv_addr, len) == SOCKET_ERROR)
|
||||
{
|
||||
g_warning("pgpipe failed to connect socket: %d\n", WSAGetLastError());
|
||||
closesocket(s);
|
||||
return -1;
|
||||
}
|
||||
if ((handles[0] = accept(s, (SOCKADDR *) & serv_addr, &len)) == INVALID_SOCKET)
|
||||
{
|
||||
g_warning("pgpipe failed to accept socket: %d\n", WSAGetLastError());
|
||||
closesocket(handles[1]);
|
||||
handles[1] = INVALID_SOCKET;
|
||||
closesocket(s);
|
||||
return -1;
|
||||
}
|
||||
closesocket(s);
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
struct timeval
|
||||
ccnet_util_timeval_from_msec (uint64_t milliseconds)
|
||||
{
|
||||
struct timeval ret;
|
||||
const uint64_t microseconds = milliseconds * 1000;
|
||||
ret.tv_sec = microseconds / 1000000;
|
||||
ret.tv_usec = microseconds % 1000000;
|
||||
return ret;
|
||||
}
|
||||
|
||||
int
|
||||
ccnet_util_checkdir (const char *dir)
|
||||
{
|
||||
struct stat st;
|
||||
|
||||
#ifdef WIN32
|
||||
/* remove trailing '\\' */
|
||||
char *path = g_strdup(dir);
|
||||
char *p = (char *)path + strlen(path) - 1;
|
||||
while (*p == '\\' || *p == '/') *p-- = '\0';
|
||||
if ((g_stat(dir, &st) < 0) || !S_ISDIR(st.st_mode)) {
|
||||
g_free (path);
|
||||
return -1;
|
||||
}
|
||||
g_free (path);
|
||||
return 0;
|
||||
#else
|
||||
if ((g_stat(dir, &st) < 0) || !S_ISDIR(st.st_mode))
|
||||
return -1;
|
||||
return 0;
|
||||
#endif
|
||||
}
|
||||
|
||||
int
|
||||
ccnet_util_checkdir_with_mkdir (const char *dir)
|
||||
{
|
||||
#ifdef WIN32
|
||||
int ret;
|
||||
char *path = g_strdup(dir);
|
||||
char *p = (char *)path + strlen(path) - 1;
|
||||
while (*p == '\\' || *p == '/') *p-- = '\0';
|
||||
ret = g_mkdir_with_parents(path, 0755);
|
||||
g_free (path);
|
||||
return ret;
|
||||
#else
|
||||
return g_mkdir_with_parents(dir, 0755);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
ssize_t /* Read "n" bytes from a descriptor. */
|
||||
ccnet_util_recvn(evutil_socket_t fd, void *vptr, size_t n)
|
||||
{
|
||||
size_t nleft;
|
||||
ssize_t nread;
|
||||
char *ptr;
|
||||
|
||||
ptr = vptr;
|
||||
nleft = n;
|
||||
while (nleft > 0) {
|
||||
#ifndef WIN32
|
||||
if ( (nread = read(fd, ptr, nleft)) < 0)
|
||||
#else
|
||||
if ( (nread = recv(fd, ptr, nleft, 0)) < 0)
|
||||
#endif
|
||||
{
|
||||
if (errno == EINTR)
|
||||
nread = 0; /* and call read() again */
|
||||
else
|
||||
return(-1);
|
||||
} else if (nread == 0)
|
||||
break; /* EOF */
|
||||
|
||||
nleft -= nread;
|
||||
ptr += nread;
|
||||
}
|
||||
return(n - nleft); /* return >= 0 */
|
||||
}
|
||||
|
||||
ssize_t /* Write "n" bytes to a descriptor. */
|
||||
ccnet_util_sendn (evutil_socket_t fd, const void *vptr, size_t n)
|
||||
{
|
||||
size_t nleft;
|
||||
ssize_t nwritten;
|
||||
const char *ptr;
|
||||
|
||||
ptr = vptr;
|
||||
nleft = n;
|
||||
while (nleft > 0) {
|
||||
#ifndef WIN32
|
||||
if ( (nwritten = write(fd, ptr, nleft)) <= 0)
|
||||
#else
|
||||
if ( (nwritten = send(fd, ptr, nleft, 0)) <= 0)
|
||||
#endif
|
||||
{
|
||||
if (nwritten < 0 && errno == EINTR)
|
||||
nwritten = 0; /* and call write() again */
|
||||
else
|
||||
return(-1); /* error */
|
||||
}
|
||||
|
||||
nleft -= nwritten;
|
||||
ptr += nwritten;
|
||||
}
|
||||
return(n);
|
||||
}
|
||||
|
||||
char*
|
||||
ccnet_util_expand_path (const char *src)
|
||||
{
|
||||
#ifdef WIN32
|
||||
char new_path[PATH_MAX + 1];
|
||||
char *p = new_path;
|
||||
const char *q = src;
|
||||
|
||||
memset(new_path, 0, sizeof(new_path));
|
||||
if (*src == '~') {
|
||||
const char *home = g_get_home_dir();
|
||||
memcpy(new_path, home, strlen(home));
|
||||
p += strlen(new_path);
|
||||
q++;
|
||||
}
|
||||
memcpy(p, q, strlen(q));
|
||||
|
||||
/* delete the charactor '\' or '/' at the end of the path
|
||||
* because the function stat faied to deal with directory names
|
||||
* with '\' or '/' in the end */
|
||||
p = new_path + strlen(new_path) - 1;
|
||||
while(*p == '\\' || *p == '/') *p-- = '\0';
|
||||
|
||||
return strdup (new_path);
|
||||
#else
|
||||
const char *next_in, *ntoken;
|
||||
char new_path[PATH_MAX + 1];
|
||||
char *next_out;
|
||||
int len;
|
||||
|
||||
/* special cases */
|
||||
if (!src || *src == '\0')
|
||||
return NULL;
|
||||
if (strlen(src) > PATH_MAX)
|
||||
return NULL;
|
||||
|
||||
next_in = src;
|
||||
next_out = new_path;
|
||||
*next_out = '\0';
|
||||
|
||||
if (*src == '~') {
|
||||
/* handle src start with '~' or '~<user>' like '~plt' */
|
||||
struct passwd *pw = NULL;
|
||||
|
||||
for ( ; *next_in != '/' && *next_in != '\0'; next_in++) ;
|
||||
|
||||
len = next_in - src;
|
||||
if (len == 1) {
|
||||
pw = getpwuid (geteuid());
|
||||
} else {
|
||||
/* copy '~<user>' to new_path */
|
||||
memcpy (new_path, src, len);
|
||||
new_path[len] = '\0';
|
||||
pw = getpwnam (new_path + 1);
|
||||
}
|
||||
if (pw == NULL)
|
||||
return NULL;
|
||||
|
||||
len = strlen (pw->pw_dir);
|
||||
memcpy (new_path, pw->pw_dir, len);
|
||||
next_out = new_path + len;
|
||||
*next_out = '\0';
|
||||
|
||||
if (*next_in == '\0')
|
||||
return strdup (new_path);
|
||||
} else if (*src != '/') {
|
||||
getcwd (new_path, PATH_MAX);
|
||||
for ( ; *next_out; next_out++) ; /* to '\0' */
|
||||
}
|
||||
|
||||
while (*next_in != '\0') {
|
||||
/* move ntoken to the next not '/' char */
|
||||
for (ntoken = next_in; *ntoken == '/'; ntoken++) ;
|
||||
|
||||
for (next_in = ntoken; *next_in != '/'
|
||||
&& *next_in != '\0'; next_in++) ;
|
||||
|
||||
len = next_in - ntoken;
|
||||
|
||||
if (len == 0) {
|
||||
/* the path ends with '/', keep it */
|
||||
*next_out++ = '/';
|
||||
*next_out = '\0';
|
||||
break;
|
||||
}
|
||||
|
||||
if (len == 2 && ntoken[0] == '.' && ntoken[1] == '.')
|
||||
{
|
||||
/* '..' */
|
||||
for (; next_out > new_path && *next_out != '/'; next_out--)
|
||||
;
|
||||
*next_out = '\0';
|
||||
} else if (ntoken[0] != '.' || len != 1) {
|
||||
/* not '.' */
|
||||
*next_out++ = '/';
|
||||
memcpy (next_out, ntoken, len);
|
||||
next_out += len;
|
||||
*next_out = '\0';
|
||||
}
|
||||
}
|
||||
|
||||
/* the final special case */
|
||||
if (new_path[0] == '\0') {
|
||||
new_path[0] = '/';
|
||||
new_path[1] = '\0';
|
||||
}
|
||||
return strdup (new_path);
|
||||
#endif
|
||||
}
|
||||
|
||||
#ifndef WIN32
|
||||
char* ccnet_util_gen_uuid ()
|
||||
{
|
||||
char *uuid_str = g_malloc (37);
|
||||
uuid_t uuid;
|
||||
|
||||
uuid_generate (uuid);
|
||||
uuid_unparse_lower (uuid, uuid_str);
|
||||
|
||||
return uuid_str;
|
||||
}
|
||||
|
||||
#else
|
||||
char* ccnet_util_gen_uuid ()
|
||||
{
|
||||
char *uuid_str = g_malloc (37);
|
||||
unsigned char *str = NULL;
|
||||
UUID uuid;
|
||||
|
||||
UuidCreate(&uuid);
|
||||
UuidToString(&uuid, &str);
|
||||
memcpy(uuid_str, str, 37);
|
||||
RpcStringFree(&str);
|
||||
return uuid_str;
|
||||
}
|
||||
#endif
|
||||
|
||||
char* ccnet_util_strjoin_n (const char *seperator, int argc, char **argv)
|
||||
{
|
||||
GString *buf;
|
||||
int i;
|
||||
char *str;
|
||||
|
||||
if (argc == 0)
|
||||
return NULL;
|
||||
|
||||
buf = g_string_new (argv[0]);
|
||||
for (i = 1; i < argc; ++i) {
|
||||
g_string_append (buf, seperator);
|
||||
g_string_append (buf, argv[i]);
|
||||
}
|
||||
|
||||
str = buf->str;
|
||||
g_string_free (buf, FALSE);
|
||||
return str;
|
||||
}
|
||||
|
||||
/**
|
||||
* handle the empty string problem.
|
||||
*/
|
||||
gchar*
|
||||
ccnet_util_key_file_get_string (GKeyFile *keyf,
|
||||
const char *category,
|
||||
const char *key)
|
||||
{
|
||||
gchar *v;
|
||||
|
||||
if (!g_key_file_has_key (keyf, category, key, NULL))
|
||||
return NULL;
|
||||
|
||||
v = g_key_file_get_string (keyf, category, key, NULL);
|
||||
if (v != NULL && v[0] == '\0') {
|
||||
g_free(v);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return v;
|
||||
}
|
||||
|
||||
void
|
||||
ccnet_util_string_list_free (GList *str_list)
|
||||
{
|
||||
GList *ptr = str_list;
|
||||
|
||||
while (ptr) {
|
||||
g_free (ptr->data);
|
||||
ptr = ptr->next;
|
||||
}
|
||||
|
||||
g_list_free (str_list);
|
||||
}
|
||||
|
||||
void
|
||||
ccnet_util_string_list_join (GList *str_list, GString *str, const char *seperator)
|
||||
{
|
||||
GList *ptr;
|
||||
if (!str_list)
|
||||
return;
|
||||
|
||||
ptr = str_list;
|
||||
g_string_append (str, ptr->data);
|
||||
|
||||
for (ptr = ptr->next; ptr; ptr = ptr->next) {
|
||||
g_string_append (str, seperator);
|
||||
g_string_append (str, (char *)ptr->data);
|
||||
}
|
||||
}
|
||||
|
||||
static GList *
|
||||
string_list_parse (const char *list_in_str, const char *seperator)
|
||||
{
|
||||
if (!list_in_str)
|
||||
return NULL;
|
||||
|
||||
GList *list = NULL;
|
||||
char **array = g_strsplit (list_in_str, seperator, 0);
|
||||
char **ptr;
|
||||
|
||||
for (ptr = array; *ptr; ptr++) {
|
||||
list = g_list_prepend (list, g_strdup(*ptr));
|
||||
}
|
||||
list = g_list_reverse (list);
|
||||
|
||||
g_strfreev (array);
|
||||
return list;
|
||||
}
|
||||
|
||||
GList *
|
||||
ccnet_util_string_list_parse_sorted (const char *list_in_str, const char *seperator)
|
||||
{
|
||||
GList *list = string_list_parse (list_in_str, seperator);
|
||||
|
||||
return g_list_sort (list, (GCompareFunc)g_strcmp0);
|
||||
}
|
||||
|
||||
static unsigned hexval(char c)
|
||||
{
|
||||
if (c >= '0' && c <= '9')
|
||||
return c - '0';
|
||||
if (c >= 'a' && c <= 'f')
|
||||
return c - 'a' + 10;
|
||||
if (c >= 'A' && c <= 'F')
|
||||
return c - 'A' + 10;
|
||||
return ~0;
|
||||
}
|
||||
|
||||
int
|
||||
ccnet_util_hex_to_rawdata (const char *hex_str,
|
||||
unsigned char *rawdata,
|
||||
int n_bytes)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < n_bytes; i++) {
|
||||
unsigned int val = (hexval(hex_str[0]) << 4) | hexval(hex_str[1]);
|
||||
if (val & ~0xff)
|
||||
return -1;
|
||||
*rawdata++ = val;
|
||||
hex_str += 2;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
#ifdef WIN32
|
||||
|
||||
#ifndef EAFNOSUPPORT
|
||||
#define EAFNOSUPPORT WSAEAFNOSUPPORT
|
||||
#endif
|
||||
|
||||
#ifndef IN6ADDRSZ
|
||||
#define IN6ADDRSZ 16
|
||||
#endif
|
||||
|
||||
#ifndef INT16SZ
|
||||
#define INT16SZ 2
|
||||
#endif
|
||||
|
||||
#ifndef INADDRSZ
|
||||
#define INADDRSZ 4
|
||||
#endif
|
||||
|
||||
#ifndef inet_ntop
|
||||
static const char *
|
||||
inet_ntop4 (const u_char *src, char *dst, size_t size)
|
||||
{
|
||||
static const char fmt[] = "%u.%u.%u.%u";
|
||||
char tmp[sizeof("255.255.255.255")];
|
||||
int l;
|
||||
l = _snprintf(tmp, size, fmt, src[0], src[1], src[2], src[3]);
|
||||
if (l <= 0 || l >= size) {
|
||||
return (NULL);
|
||||
}
|
||||
strncpy(dst, tmp, size);
|
||||
return (dst);
|
||||
}
|
||||
|
||||
static const char *
|
||||
inet_ntop6 (const u_char *src, char *dst, size_t size)
|
||||
{
|
||||
char tmp[sizeof "ffff:ffff:ffff:ffff:ffff:ffff:255.255.255.255"];
|
||||
char *tp, *ep;
|
||||
struct
|
||||
{
|
||||
int base, len;
|
||||
} best, cur;
|
||||
u_int words[IN6ADDRSZ / INT16SZ];
|
||||
int i;
|
||||
int advance;
|
||||
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;
|
||||
best.len = -1;
|
||||
cur.len = -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) {
|
||||
if (best.base == -1 || cur.len > best.len)
|
||||
best = cur;
|
||||
}
|
||||
if (best.base != -1 && best.len < 2)
|
||||
best.base = -1;
|
||||
tp = tmp;
|
||||
ep = tmp + sizeof(tmp);
|
||||
for (i = 0; i < (IN6ADDRSZ / INT16SZ) && tp < ep; 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) {
|
||||
if (tp + 1 >= ep)
|
||||
return (NULL);
|
||||
*tp++ = ':';
|
||||
}
|
||||
continue;
|
||||
}
|
||||
/** Are we following an initial run of 0x00s or any real hex? */
|
||||
if (i != 0) {
|
||||
if (tp + 1 >= ep)
|
||||
return (NULL);
|
||||
*tp++ = ':';
|
||||
}
|
||||
/** Is this address an encapsulated IPv4? */
|
||||
if (i == 6 && best.base == 0 &&
|
||||
(best.len == 6 || (best.len == 5 && words[5] == 0xffff))) {
|
||||
if (!inet_ntop4(src+12, tp, (size_t)(ep - tp)))
|
||||
return (NULL);
|
||||
tp += strlen(tp);
|
||||
break;
|
||||
}
|
||||
advance = snprintf(tp, ep - tp, "%x", words[i]);
|
||||
if (advance <= 0 || advance >= ep - tp)
|
||||
return (NULL);
|
||||
tp += advance;
|
||||
}
|
||||
/** Was it a trailing run of 0x00's? */
|
||||
if (best.base != -1 && (best.base + best.len) == (IN6ADDRSZ / INT16SZ)) {
|
||||
if (tp + 1 >= ep)
|
||||
return (NULL);
|
||||
*tp++ = ':';
|
||||
}
|
||||
if (tp + 1 >= ep)
|
||||
return (NULL);
|
||||
*tp++ = '\0';
|
||||
|
||||
/**
|
||||
* Check for overflow, copy, and we're done.
|
||||
*/
|
||||
if ((size_t)(tp - tmp) > size) {
|
||||
errno = ENOSPC;
|
||||
return (NULL);
|
||||
}
|
||||
strncpy(dst, tmp, size);
|
||||
dst[size] = '\0';
|
||||
return (dst);
|
||||
}
|
||||
|
||||
const char *
|
||||
ccnet_util_inet_ntop(int af, const void *src, char *dst, size_t size)
|
||||
{
|
||||
switch (af) {
|
||||
case AF_INET:
|
||||
return (inet_ntop4(src, dst, size));
|
||||
case AF_INET6:
|
||||
return (inet_ntop6(src, dst, size));
|
||||
default:
|
||||
return (NULL);
|
||||
}
|
||||
/** NOTREACHED */
|
||||
}
|
||||
#endif //inet_ntop
|
||||
|
||||
#ifndef inet_aton
|
||||
int
|
||||
ccnet_util_inet_aton (const char *string, struct in_addr *addr)
|
||||
{
|
||||
addr->s_addr = inet_addr(string);
|
||||
if (addr->s_addr != -1 || strcmp("255.255.255.255", string) == 0)
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifndef inet_pton
|
||||
/*
|
||||
* Don't even consider trying to compile this on a system where
|
||||
* sizeof(int) < 4. sizeof(int) > 4 is fine; all the world's not a VAX.
|
||||
*/
|
||||
|
||||
/* int
|
||||
* inet_pton4(src, dst, pton)
|
||||
* when last arg is 0: inet_aton(). with hexadecimal, octal and shorthand.
|
||||
* when last arg is 1: inet_pton(). decimal dotted-quad only.
|
||||
* return:
|
||||
* 1 if `src' is a valid input, else 0.
|
||||
* notice:
|
||||
* does not touch `dst' unless it's returning 1.
|
||||
* author:
|
||||
* Paul Vixie, 1996.
|
||||
*/
|
||||
static int
|
||||
inet_pton4 (const char *src, u_char *dst, int pton)
|
||||
{
|
||||
u_int val;
|
||||
u_int digit;
|
||||
int base, n;
|
||||
unsigned char c;
|
||||
u_int parts[4];
|
||||
register u_int *pp = parts;
|
||||
|
||||
c = *src;
|
||||
for (;;) {
|
||||
/*
|
||||
* Collect number up to ``.''.
|
||||
* Values are specified as for C:
|
||||
* 0x=hex, 0=octal, isdigit=decimal.
|
||||
*/
|
||||
if (!isdigit(c))
|
||||
return (0);
|
||||
val = 0; base = 10;
|
||||
if (c == '0') {
|
||||
c = *++src;
|
||||
if (c == 'x' || c == 'X')
|
||||
base = 16, c = *++src;
|
||||
else if (isdigit(c) && c != '9')
|
||||
base = 8;
|
||||
}
|
||||
/* inet_pton() takes decimal only */
|
||||
if (pton && base != 10)
|
||||
return (0);
|
||||
for (;;) {
|
||||
if (isdigit(c)) {
|
||||
digit = c - '0';
|
||||
if (digit >= base)
|
||||
break;
|
||||
val = (val * base) + digit;
|
||||
c = *++src;
|
||||
} else if (base == 16 && isxdigit(c)) {
|
||||
digit = c + 10 - (islower(c) ? 'a' : 'A');
|
||||
if (digit >= 16)
|
||||
break;
|
||||
val = (val << 4) | digit;
|
||||
c = *++src;
|
||||
} else
|
||||
break;
|
||||
}
|
||||
if (c == '.') {
|
||||
/*
|
||||
* Internet format:
|
||||
* a.b.c.d
|
||||
* a.b.c (with c treated as 16 bits)
|
||||
* a.b (with b treated as 24 bits)
|
||||
* a (with a treated as 32 bits)
|
||||
*/
|
||||
if (pp >= parts + 3)
|
||||
return (0);
|
||||
*pp++ = val;
|
||||
c = *++src;
|
||||
} else
|
||||
break;
|
||||
}
|
||||
/*
|
||||
* Check for trailing characters.
|
||||
*/
|
||||
if (c != '\0' && !isspace(c))
|
||||
return (0);
|
||||
/*
|
||||
* Concoct the address according to
|
||||
* the number of parts specified.
|
||||
*/
|
||||
n = pp - parts + 1;
|
||||
/* inet_pton() takes dotted-quad only. it does not take shorthand. */
|
||||
if (pton && n != 4)
|
||||
return (0);
|
||||
switch (n) {
|
||||
|
||||
case 0:
|
||||
return (0); /* initial nondigit */
|
||||
|
||||
case 1: /* a -- 32 bits */
|
||||
break;
|
||||
|
||||
case 2: /* a.b -- 8.24 bits */
|
||||
if (parts[0] > 0xff || val > 0xffffff)
|
||||
return (0);
|
||||
val |= parts[0] << 24;
|
||||
break;
|
||||
|
||||
case 3: /* a.b.c -- 8.8.16 bits */
|
||||
if ((parts[0] | parts[1]) > 0xff || val > 0xffff)
|
||||
return (0);
|
||||
val |= (parts[0] << 24) | (parts[1] << 16);
|
||||
break;
|
||||
|
||||
case 4: /* a.b.c.d -- 8.8.8.8 bits */
|
||||
if ((parts[0] | parts[1] | parts[2] | val) > 0xff)
|
||||
return (0);
|
||||
val |= (parts[0] << 24) | (parts[1] << 16) | (parts[2] << 8);
|
||||
break;
|
||||
}
|
||||
if (dst) {
|
||||
val = htonl(val);
|
||||
memcpy(dst, &val, INADDRSZ);
|
||||
}
|
||||
return (1);
|
||||
}
|
||||
|
||||
/* int
|
||||
* inet_pton6(src, dst)
|
||||
* convert presentation level address to network order binary form.
|
||||
* return:
|
||||
* 1 if `src' is a valid [RFC1884 2.2] address, else 0.
|
||||
* notice:
|
||||
* (1) does not touch `dst' unless it's returning 1.
|
||||
* (2) :: in a full address is silently ignored.
|
||||
* credit:
|
||||
* inspired by Mark Andrews.
|
||||
* author:
|
||||
* Paul Vixie, 1996.
|
||||
*/
|
||||
static int
|
||||
inet_pton6 (const char *src, u_char *dst)
|
||||
{
|
||||
static const char xdigits_l[] = "0123456789abcdef",
|
||||
xdigits_u[] = "0123456789ABCDEF";
|
||||
u_char tmp[IN6ADDRSZ], *tp, *endp, *colonp;
|
||||
const char *xdigits, *curtok;
|
||||
int ch, saw_xdigit;
|
||||
u_int val;
|
||||
|
||||
memset((tp = tmp), '\0', IN6ADDRSZ);
|
||||
endp = tp + IN6ADDRSZ;
|
||||
colonp = NULL;
|
||||
/* Leading :: requires some special handling. */
|
||||
if (*src == ':')
|
||||
if (*++src != ':')
|
||||
return (0);
|
||||
curtok = src;
|
||||
saw_xdigit = 0;
|
||||
val = 0;
|
||||
while ((ch = *src++) != '\0') {
|
||||
const char *pch;
|
||||
|
||||
if ((pch = strchr((xdigits = xdigits_l), ch)) == NULL)
|
||||
pch = strchr((xdigits = xdigits_u), ch);
|
||||
if (pch != NULL) {
|
||||
val <<= 4;
|
||||
val |= (pch - xdigits);
|
||||
if (val > 0xffff)
|
||||
return (0);
|
||||
saw_xdigit = 1;
|
||||
continue;
|
||||
}
|
||||
if (ch == ':') {
|
||||
curtok = src;
|
||||
if (!saw_xdigit) {
|
||||
if (colonp)
|
||||
return (0);
|
||||
colonp = tp;
|
||||
continue;
|
||||
} else if (*src == '\0')
|
||||
return (0);
|
||||
if (tp + INT16SZ > endp)
|
||||
return (0);
|
||||
*tp++ = (u_char) (val >> 8) & 0xff;
|
||||
*tp++ = (u_char) val & 0xff;
|
||||
saw_xdigit = 0;
|
||||
val = 0;
|
||||
continue;
|
||||
}
|
||||
if (ch == '.' && ((tp + INADDRSZ) <= endp) &&
|
||||
inet_pton4(curtok, tp, 1) > 0) {
|
||||
tp += INADDRSZ;
|
||||
saw_xdigit = 0;
|
||||
break; /* '\0' was seen by inet_pton4(). */
|
||||
}
|
||||
return (0);
|
||||
}
|
||||
if (saw_xdigit) {
|
||||
if (tp + INT16SZ > endp)
|
||||
return (0);
|
||||
*tp++ = (u_char) (val >> 8) & 0xff;
|
||||
*tp++ = (u_char) val & 0xff;
|
||||
}
|
||||
if (colonp != NULL) {
|
||||
/*
|
||||
* Since some memmove()'s erroneously fail to handle
|
||||
* overlapping regions, we'll do the shift by hand.
|
||||
*/
|
||||
const int n = tp - colonp;
|
||||
int i;
|
||||
|
||||
if (tp == endp)
|
||||
return (0);
|
||||
for (i = 1; i <= n; i++) {
|
||||
endp[- i] = colonp[n - i];
|
||||
colonp[n - i] = 0;
|
||||
}
|
||||
tp = endp;
|
||||
}
|
||||
if (tp != endp)
|
||||
return (0);
|
||||
memcpy(dst, tmp, IN6ADDRSZ);
|
||||
return (1);
|
||||
}
|
||||
|
||||
/* int
|
||||
* inet_pton(af, src, dst)
|
||||
* convert from presentation format (which usually means ASCII printable)
|
||||
* to network format (which is usually some kind of binary format).
|
||||
* return:
|
||||
* 1 if the address was valid for the specified address family
|
||||
* 0 if the address wasn't valid (`dst' is untouched in this case)
|
||||
* -1 if some other error occurred (`dst' is untouched in this case, too)
|
||||
* author:
|
||||
* Paul Vixie, 1996.
|
||||
*/
|
||||
int
|
||||
ccnet_util_inet_pton (int af, const char *src, void *dst)
|
||||
{
|
||||
switch (af) {
|
||||
case AF_INET:
|
||||
return (inet_pton4(src, dst, 1));
|
||||
case AF_INET6:
|
||||
return (inet_pton6(src, dst));
|
||||
default:
|
||||
errno = EAFNOSUPPORT;
|
||||
return (-1);
|
||||
}
|
||||
/* NOTREACHED */
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif //WIN32
|
Reference in New Issue
Block a user