blob: 26cea8a62d5dd4a9ffc980ea26cec9efc26cec84 [file] [log] [blame]
/*
*
* Embedded Linux library
*
* Copyright (C) 2011-2014 Intel Corporation. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#define _GNU_SOURCE
#include <stdio.h>
#include <ctype.h>
#include <stdarg.h>
#include <stdlib.h>
#include <limits.h>
#include "util.h"
#include "private.h"
/**
* SECTION:util
* @short_description: Utility functions
*
* Utility functions
*/
#define STRLOC __FILE__ ":" L_STRINGIFY(__LINE__)
/**
* l_malloc:
* @size: memory size to allocate
*
* If for any reason the memory allocation fails, then execution will be
* halted via abort().
*
* In case @size is 0 then #NULL will be returned.
*
* Returns: pointer to allocated memory
**/
LIB_EXPORT void *l_malloc(size_t size)
{
if (likely(size)) {
void *ptr;
ptr = malloc(size);
if (ptr)
return ptr;
fprintf(stderr, "%s:%s(): failed to allocate %zd bytes\n",
STRLOC, __func__, size);
abort();
}
return NULL;
}
/**
* l_realloc:
* @mem: previously allocated memory, or NULL
* @size: memory size to allocate
*
* If for any reason the memory allocation fails, then execution will be
* halted via abort().
*
* In case @mem is NULL, this function acts like l_malloc.
* In case @size is 0 then #NULL will be returned.
*
* Returns: pointer to allocated memory
**/
LIB_EXPORT void *l_realloc(void *mem, size_t size)
{
if (likely(size)) {
void *ptr;
ptr = realloc(mem, size);
if (ptr)
return ptr;
fprintf(stderr, "%s:%s(): failed to re-allocate %zd bytes\n",
STRLOC, __func__, size);
abort();
} else
l_free(mem);
return NULL;
}
/**
* l_memdup:
* @mem: pointer to memory you want to duplicate
* @size: memory size
*
* If for any reason the memory allocation fails, then execution will be
* halted via abort().
*
* In case @size is 0 then #NULL will be returned.
*
* Returns: pointer to duplicated memory buffer
**/
LIB_EXPORT void *l_memdup(const void *mem, size_t size)
{
void *ptr;
ptr = l_malloc(size);
memcpy(ptr, mem, size);
return ptr;
}
/**
* l_free:
* @ptr: memory pointer
*
* Free the allocated memory area.
**/
LIB_EXPORT void l_free(void *ptr)
{
free(ptr);
}
/**
* l_strdup:
* @str: string pointer
*
* Allocates and duplicates sring
*
* Returns: a newly allocated string
**/
LIB_EXPORT char *l_strdup(const char *str)
{
if (likely(str)) {
char *tmp;
tmp = strdup(str);
if (tmp)
return tmp;
fprintf(stderr, "%s:%s(): failed to allocate string\n",
STRLOC, __func__);
abort();
}
return NULL;
}
/**
* l_strndup:
* @str: string pointer
* @max: Maximum number of characters to copy
*
* Allocates and duplicates sring. If the string is longer than @max
* characters, only @max are copied and a null terminating character
* is added.
*
* Returns: a newly allocated string
**/
LIB_EXPORT char *l_strndup(const char *str, size_t max)
{
if (likely(str)) {
char *tmp;
tmp = strndup(str, max);
if (tmp)
return tmp;
fprintf(stderr, "%s:%s(): failed to allocate string\n",
STRLOC, __func__);
abort();
}
return NULL;
}
/**
* l_strdup_printf:
* @format: string format
* @...: parameters to insert into format string
*
* Returns: a newly allocated string
**/
LIB_EXPORT char *l_strdup_printf(const char *format, ...)
{
va_list args;
char *str;
int len;
va_start(args, format);
len = vasprintf(&str, format, args);
va_end(args);
if (len < 0) {
fprintf(stderr, "%s:%s(): failed to allocate string\n",
STRLOC, __func__);
abort();
return NULL;
}
return str;
}
/**
* l_strdup_vprintf:
* @format: string format
* @args: parameters to insert into format string
*
* Returns: a newly allocated string
**/
LIB_EXPORT char *l_strdup_vprintf(const char *format, va_list args)
{
char *str;
int len;
len = vasprintf(&str, format, args);
if (len < 0) {
fprintf(stderr, "%s:%s(): failed to allocate string\n",
STRLOC, __func__);
abort();
return NULL;
}
return str;
}
/**
* l_strlcpy:
* @dst: Destination buffer for string
* @src: Source buffer containing null-terminated string to copy
* @len: Maximum destination buffer space to use
*
* Copies a string from the @src buffer to the @dst buffer, using no
* more than @len bytes in @dst. @dst is guaranteed to be
* null-terminated. The caller can determine if the copy was truncated by
* checking if the return value is greater than or equal to @len.
*
* NOTE: Passing in a NULL string results in a no-op
*
* Returns: The length of the @src string, not including the null
* terminator.
*/
LIB_EXPORT size_t l_strlcpy(char *dst, const char *src, size_t len)
{
size_t src_len = src ? strlen(src) : 0;
if (!src)
goto done;
if (len) {
if (src_len < len) {
len = src_len + 1;
} else {
len -= 1;
dst[len] = '\0';
}
memcpy(dst, src, len);
}
done:
return src_len;
}
/**
* l_str_has_prefix:
* @str: A string to be examined
* @delim: Prefix string
*
* Determines if the string given by @str is prefixed by string given by
* @prefix.
*
* Returns: True if @str was prefixed by @prefix. False otherwise.
*/
LIB_EXPORT bool l_str_has_prefix(const char *str, const char *prefix)
{
size_t str_len;
size_t prefix_len;
if (unlikely(!str))
return false;
if (unlikely(!prefix))
return false;
str_len = strlen(str);
prefix_len = strlen(prefix);
if (str_len < prefix_len)
return false;
return !strncmp(str, prefix, prefix_len);
}
/**
* l_str_has_suffix:
* @str: A string to be examined
* @suffix: Suffix string
*
* Determines if the string given by @str ends with the specified @suffix.
*
* Returns: True if @str ends with the specified @suffix. False otherwise.
*/
LIB_EXPORT bool l_str_has_suffix(const char *str, const char *suffix)
{
size_t str_len;
size_t suffix_len;
size_t len_diff;
if (unlikely(!str))
return false;
if (unlikely(!suffix))
return false;
str_len = strlen(str);
suffix_len = strlen(suffix);
if (str_len < suffix_len)
return false;
len_diff = str_len - suffix_len;
return !strcmp(&str[len_diff], suffix);
}
static char *hexstring_common(const unsigned char *buf, size_t len,
const char hexdigits[static 16])
{
char *str;
size_t i;
if (unlikely(!buf) || unlikely(!len))
return NULL;
str = l_malloc(len * 2 + 1);
for (i = 0; i < len; i++) {
str[(i * 2) + 0] = hexdigits[buf[i] >> 4];
str[(i * 2) + 1] = hexdigits[buf[i] & 0xf];
}
str[len * 2] = '\0';
return str;
}
/**
* l_util_hexstring:
* @buf: buffer pointer
* @len: length of buffer
*
* Returns: a newly allocated hex string. Note that the string will contain
* lower case hex digits a-f. If you require upper case hex digits, use
* @l_util_hexstring_upper
**/
LIB_EXPORT char *l_util_hexstring(const unsigned char *buf, size_t len)
{
static const char hexdigits[] = "0123456789abcdef";
return hexstring_common(buf, len, hexdigits);
}
/**
* l_util_hexstring_upper:
* @buf: buffer pointer
* @len: length of buffer
*
* Returns: a newly allocated hex string. Note that the string will contain
* upper case hex digits a-f. If you require lower case hex digits, use
* @l_util_hexstring
**/
LIB_EXPORT char *l_util_hexstring_upper(const unsigned char *buf, size_t len)
{
static const char hexdigits[] = "0123456789ABCDEF";
return hexstring_common(buf, len, hexdigits);
}
/**
* l_util_from_hexstring:
* @str: Null-terminated string containing the hex-encoded bytes
* @out_len: Number of bytes decoded
*
* Returns: a newly allocated byte array. Empty strings are treated as
* an error condition.
**/
LIB_EXPORT unsigned char *l_util_from_hexstring(const char *str,
size_t *out_len)
{
size_t i, j;
size_t len;
char c;
unsigned char *buf;
if (unlikely(!str))
return NULL;
for (i = 0; str[i]; i++) {
c = toupper(str[i]);
if ((c >= '0' && c <= '9') || (c >= 'A' && c <= 'F'))
continue;
return NULL;
}
if (!i)
return NULL;
if ((i % 2) != 0)
return NULL;
len = i;
buf = l_malloc(i >> 1);
for (i = 0, j = 0; i < len; i++, j++) {
c = toupper(str[i]);
if (c >= '0' && c <= '9')
buf[j] = c - '0';
else if (c >= 'A' && c <= 'F')
buf[j] = 10 + c - 'A';
i += 1;
c = toupper(str[i]);
if (c >= '0' && c <= '9')
buf[j] = buf[j] * 16 + c - '0';
else if (c >= 'A' && c <= 'F')
buf[j] = buf[j] * 16 + 10 + c - 'A';
}
if (out_len)
*out_len = j;
return buf;
}
static void hexdump(const char dir, const unsigned char *buf, size_t len,
l_util_hexdump_func_t function, void *user_data)
{
static const char hexdigits[] = "0123456789abcdef";
char str[68];
size_t i;
if (unlikely(!len))
return;
str[0] = dir;
for (i = 0; i < len; i++) {
str[((i % 16) * 3) + 1] = ' ';
str[((i % 16) * 3) + 2] = hexdigits[buf[i] >> 4];
str[((i % 16) * 3) + 3] = hexdigits[buf[i] & 0xf];
str[(i % 16) + 51] = isprint(buf[i]) ? buf[i] : '.';
if ((i + 1) % 16 == 0) {
str[49] = ' ';
str[50] = ' ';
str[67] = '\0';
function(str, user_data);
str[0] = ' ';
}
}
if (i % 16 > 0) {
size_t j;
for (j = (i % 16); j < 16; j++) {
str[(j * 3) + 1] = ' ';
str[(j * 3) + 2] = ' ';
str[(j * 3) + 3] = ' ';
str[j + 51] = ' ';
}
str[49] = ' ';
str[50] = ' ';
str[67] = '\0';
function(str, user_data);
}
}
LIB_EXPORT void l_util_hexdump(bool in, const void *buf, size_t len,
l_util_hexdump_func_t function, void *user_data)
{
if (likely(!function))
return;
hexdump(in ? '<' : '>', buf, len, function, user_data);
}
LIB_EXPORT void l_util_hexdump_two(bool in, const void *buf1, size_t len1,
const void *buf2, size_t len2,
l_util_hexdump_func_t function, void *user_data)
{
if (likely(!function))
return;
hexdump(in ? '<' : '>', buf1, len1, function, user_data);
hexdump(' ', buf2, len2, function, user_data);
}
LIB_EXPORT void l_util_hexdumpv(bool in, const struct iovec *iov,
size_t n_iov,
l_util_hexdump_func_t function,
void *user_data)
{
static const char hexdigits[] = "0123456789abcdef";
char str[68];
size_t i;
size_t len;
size_t c;
const uint8_t *buf;
if (unlikely(!iov || !n_iov))
return;
str[0] = in ? '<' : '>';
for (i = 0, len = 0; i < n_iov; i++)
len += iov[i].iov_len;
c = 0;
buf = iov[0].iov_base;
for (i = 0; i < len; i++, c++) {
if (c == iov[0].iov_len) {
c = 0;
iov += 1;
buf = iov[0].iov_base;
}
str[((i % 16) * 3) + 1] = ' ';
str[((i % 16) * 3) + 2] = hexdigits[buf[c] >> 4];
str[((i % 16) * 3) + 3] = hexdigits[buf[c] & 0xf];
str[(i % 16) + 51] = isprint(buf[c]) ? buf[c] : '.';
if ((i + 1) % 16 == 0) {
str[49] = ' ';
str[50] = ' ';
str[67] = '\0';
function(str, user_data);
str[0] = ' ';
}
}
if (i % 16 > 0) {
size_t j;
for (j = (i % 16); j < 16; j++) {
str[(j * 3) + 1] = ' ';
str[(j * 3) + 2] = ' ';
str[(j * 3) + 3] = ' ';
str[j + 51] = ' ';
}
str[49] = ' ';
str[50] = ' ';
str[67] = '\0';
function(str, user_data);
}
}
LIB_EXPORT void l_util_debug(l_util_hexdump_func_t function, void *user_data,
const char *format, ...)
{
va_list args;
char *str;
int len;
if (likely(!function))
return;
if (unlikely(!format))
return;
va_start(args, format);
len = vasprintf(&str, format, args);
va_end(args);
if (unlikely(len < 0))
return;
function(str, user_data);
free(str);
}
/**
* l_util_get_debugfs_path:
*
* Returns: a pointer to mount point of debugfs
**/
LIB_EXPORT const char *l_util_get_debugfs_path(void)
{
static char path[PATH_MAX + 1];
static bool found = false;
char type[100];
FILE *fp;
if (found)
return path;
fp = fopen("/proc/mounts", "r");
if (!fp)
return NULL;
while (fscanf(fp, "%*s %" L_STRINGIFY(PATH_MAX) "s %99s %*s %*d %*d\n",
path, type) == 2) {
if (!strcmp(type, "debugfs")) {
found = true;
break;
}
}
fclose(fp);
if (!found)
return NULL;
return path;
}