strongswan/src/libstrongswan/chunk.c

396 lines
6.5 KiB
C
Raw Normal View History

2005-11-17 09:05:55 +00:00
/*
2006-07-07 08:49:06 +00:00
* Copyright (C) 2005-2006 Martin Willi
* Copyright (C) 2005 Jan Hutter
2005-11-17 09:05:55 +00:00
* Hochschule fuer Technik Rapperswil
*
* 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. See <http://www.fsf.org/copyleft/gpl.txt>.
*
* 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.
*
* $Id$
2005-11-17 09:05:55 +00:00
*/
2006-07-05 10:09:42 +00:00
#include <stdio.h>
2007-04-12 11:33:43 +00:00
#include <sys/stat.h>
2008-04-17 14:06:37 +00:00
#include <unistd.h>
#include <errno.h>
2005-11-17 09:05:55 +00:00
#include "chunk.h"
2005-11-17 09:05:55 +00:00
2007-04-05 17:07:14 +00:00
#include <debug.h>
#include <printf_hook.h>
2005-11-17 09:05:55 +00:00
2005-12-06 13:07:06 +00:00
/**
* Empty chunk.
*/
chunk_t chunk_empty = { NULL, 0 };
/**
* Described in header.
*/
chunk_t chunk_create(u_char *ptr, size_t len)
{
chunk_t chunk = {ptr, len};
return chunk;
}
/**
* Described in header.
*/
chunk_t chunk_create_clone(u_char *ptr, chunk_t chunk)
{
chunk_t clone = chunk_empty;
if (chunk.ptr && chunk.len > 0)
{
clone.ptr = ptr;
clone.len = chunk.len;
memcpy(clone.ptr, chunk.ptr, chunk.len);
}
return clone;
}
/**
* Decribed in header.
*/
size_t chunk_length(const char* mode, ...)
{
va_list chunks;
size_t length = 0;
va_start(chunks, mode);
while (TRUE)
{
switch (*mode++)
{
case 'm':
case 'c':
{
chunk_t ch = va_arg(chunks, chunk_t);
length += ch.len;
continue;
}
default:
break;
}
break;
}
va_end(chunks);
return length;
}
/**
* Decribed in header.
*/
chunk_t chunk_create_cat(u_char *ptr, const char* mode, ...)
{
va_list chunks;
chunk_t construct = chunk_create(ptr, 0);
va_start(chunks, mode);
while (TRUE)
{
bool free_chunk = FALSE;
switch (*mode++)
{
case 'm':
{
free_chunk = TRUE;
}
case 'c':
{
chunk_t ch = va_arg(chunks, chunk_t);
memcpy(ptr, ch.ptr, ch.len);
ptr += ch.len;
construct.len += ch.len;
if (free_chunk)
{
free(ch.ptr);
}
continue;
}
default:
break;
}
break;
}
va_end(chunks);
return construct;
}
/**
* Decribed in header.
*/
void chunk_split(chunk_t chunk, const char *mode, ...)
{
va_list chunks;
size_t len;
chunk_t *ch;
va_start(chunks, mode);
while (TRUE)
{
if (*mode == '\0')
{
break;
}
len = va_arg(chunks, size_t);
ch = va_arg(chunks, chunk_t*);
/* a null chunk means skip len bytes */
if (ch == NULL)
{
chunk = chunk_skip(chunk, len);
continue;
}
switch (*mode++)
{
case 'm':
{
ch->len = min(chunk.len, len);
if (ch->len)
{
ch->ptr = chunk.ptr;
}
else
{
ch->ptr = NULL;
}
chunk = chunk_skip(chunk, ch->len);
continue;
}
case 'a':
{
ch->len = min(chunk.len, len);
if (ch->len)
{
ch->ptr = malloc(ch->len);
memcpy(ch->ptr, chunk.ptr, ch->len);
}
else
{
ch->ptr = NULL;
}
chunk = chunk_skip(chunk, ch->len);
continue;
}
case 'c':
{
ch->len = min(ch->len, chunk.len);
ch->len = min(ch->len, len);
if (ch->len)
{
memcpy(ch->ptr, chunk.ptr, ch->len);
}
else
{
ch->ptr = NULL;
}
chunk = chunk_skip(chunk, ch->len);
continue;
}
default:
break;
}
break;
}
va_end(chunks);
}
2007-04-05 17:07:14 +00:00
/**
* Described in header.
*/
2008-04-17 14:06:37 +00:00
bool chunk_write(chunk_t chunk, char *path, mode_t mask, bool force)
2007-04-05 17:07:14 +00:00
{
mode_t oldmask;
FILE *fd;
2008-04-17 14:06:37 +00:00
bool good = FALSE;
2007-04-05 17:07:14 +00:00
2008-04-17 14:06:37 +00:00
if (!force && access(path, F_OK) == 0)
2007-04-05 17:07:14 +00:00
{
2008-04-17 14:06:37 +00:00
DBG1(" file '%s' already exists", path);
return FALSE;
2007-04-05 17:07:14 +00:00
}
oldmask = umask(mask);
fd = fopen(path, "w");
if (fd)
{
2008-04-17 14:06:37 +00:00
if (fwrite(chunk.ptr, sizeof(u_char), chunk.len, fd) == chunk.len)
{
good = TRUE;
}
else
{
DBG1(" writing to file '%s' failed: %s", path, strerror(errno));
}
2007-04-05 17:07:14 +00:00
fclose(fd);
return TRUE;
}
else
{
2008-04-17 14:06:37 +00:00
DBG1(" could not open file '%s': %s", path, strerror(errno));
2007-04-05 17:07:14 +00:00
}
2008-04-17 14:06:37 +00:00
umask(oldmask);
return good;
2007-04-05 17:07:14 +00:00
}
/** hex conversion digits */
static char hexdig_upper[] = "0123456789ABCDEF";
static char hexdig_lower[] = "0123456789abcdef";
/**
* Described in header.
*/
char *chunk_to_hex(chunk_t chunk, bool uppercase)
{
int i;
char *str;
char *hexdig = hexdig_lower;
if (uppercase)
{
hexdig = hexdig_upper;
}
str = malloc(chunk.len * 2 + 1);
str[chunk.len * 2] = '\0';
for (i = 0; i < chunk.len; i ++)
{
str[i*2] = hexdig[(chunk.ptr[i] >> 4) & 0xF];
str[i*2+1] = hexdig[(chunk.ptr[i] ) & 0xF];
}
return str;
}
/**
* Described in header.
*/
void chunk_free(chunk_t *chunk)
{
free(chunk->ptr);
chunk->ptr = NULL;
chunk->len = 0;
}
/**
* Described in header.
*/
void chunk_clear(chunk_t *chunk)
{
memset(chunk->ptr, 0, chunk->len);
chunk_free(chunk);
}
/**
* Described in header.
*/
chunk_t chunk_skip(chunk_t chunk, size_t bytes)
{
if (chunk.len > bytes)
{
chunk.ptr += bytes;
chunk.len -= bytes;
return chunk;
}
return chunk_empty;
}
/**
* Described in header.
*/
int chunk_compare(chunk_t a, chunk_t b)
{
int compare_len = a.len - b.len;
int len = (compare_len < 0)? a.len : b.len;
if (compare_len != 0 || len == 0)
{
return compare_len;
}
return memcmp(a.ptr, b.ptr, len);
};
/**
* Described in header.
*/
bool chunk_equals(chunk_t a, chunk_t b)
{
2006-06-16 05:53:47 +00:00
return a.ptr != NULL && b.ptr != NULL &&
a.len == b.len && memeq(a.ptr, b.ptr, a.len);
2006-06-16 05:53:47 +00:00
}
/**
* Described in header.
*/
bool chunk_equals_or_null(chunk_t a, chunk_t b)
{
if (a.ptr == NULL || b.ptr == NULL)
return TRUE;
return a.len == b.len && memeq(a.ptr, b.ptr, a.len);
}
2006-09-27 14:15:49 +00:00
/**
* output handler in printf() for chunks
*/
static int chunk_print(FILE *stream, const struct printf_info *info,
2006-09-27 14:15:49 +00:00
const void *const *args)
{
chunk_t *chunk = *((chunk_t**)(args[0]));
bool first = TRUE;
chunk_t copy = *chunk;
int written = 0;
printf_hook_functions_t mem = mem_get_printf_hooks();
2006-09-27 14:15:49 +00:00
if (!info->alt)
{
const void *new_args[] = {&chunk->ptr, &chunk->len};
return mem.print(stream, info, new_args);
}
while (copy.len > 0)
{
if (first)
{
first = FALSE;
}
else
{
written += fprintf(stream, ":");
}
written += fprintf(stream, "%02x", *copy.ptr++);
copy.len--;
}
return written;
2006-09-27 14:15:49 +00:00
}
/**
* arginfo handler for printf() mem ranges
*/
static int chunk_arginfo(const struct printf_info *info, size_t n, int *argtypes)
{
if (n > 0)
{
argtypes[0] = PA_POINTER;
}
return 1;
}
/**
* return printf hook functions for a chunk
2006-05-29 07:02:12 +00:00
*/
printf_hook_functions_t chunk_get_printf_hooks()
2006-05-29 07:02:12 +00:00
{
printf_hook_functions_t hooks = {chunk_print, chunk_arginfo};
return hooks;
2006-05-29 07:02:12 +00:00
}