blob: 18cb34ca8e9ed103c5ae93939ec9f9364330e6eb [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0-or-later
/*
* Utility functions for libfsverity
*
* Copyright 2020 Google LLC
*/
#define _GNU_SOURCE /* for asprintf() and strerror_r() */
#include "lib_private.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
static void *xmalloc(size_t size)
{
void *p = malloc(size);
if (!p)
libfsverity_error_msg("out of memory (tried to allocate %zu bytes)",
size);
return p;
}
void *libfsverity_zalloc(size_t size)
{
void *p = xmalloc(size);
if (!p)
return NULL;
return memset(p, 0, size);
}
void *libfsverity_memdup(const void *mem, size_t size)
{
void *p = xmalloc(size);
if (!p)
return NULL;
return memcpy(p, mem, size);
}
static void (*libfsverity_error_cb)(const char *msg);
LIBEXPORT void
libfsverity_set_error_callback(void (*cb)(const char *msg))
{
libfsverity_error_cb = cb;
}
void libfsverity_do_error_msg(const char *format, va_list va, int err)
{
int saved_errno = errno;
char *msg = NULL;
if (!libfsverity_error_cb)
return;
if (vasprintf(&msg, format, va) < 0)
goto out;
if (err) {
char *msg2 = NULL;
char errbuf[64];
if (asprintf(&msg2, "%s: %s", msg,
strerror_r(err, errbuf, sizeof(errbuf))) < 0)
goto out2;
free(msg);
msg = msg2;
}
(*libfsverity_error_cb)(msg);
out2:
free(msg);
out:
errno = saved_errno;
}
void libfsverity_error_msg(const char *format, ...)
{
va_list va;
va_start(va, format);
libfsverity_do_error_msg(format, va, 0);
va_end(va);
}
void libfsverity_error_msg_errno(const char *format, ...)
{
va_list va;
va_start(va, format);
libfsverity_do_error_msg(format, va, errno);
va_end(va);
}
void libfsverity_warn_on(const char *condition, const char *file, int line)
{
fprintf(stderr, "libfsverity internal error! %s at %s:%d\n",
condition, file, line);
}
void libfsverity_bug_on(const char *condition, const char *file, int line)
{
fprintf(stderr, "libfsverity internal error! %s at %s:%d\n"
"Non-recoverable, aborting program.\n", condition, file, line);
abort();
}
bool libfsverity_mem_is_zeroed(const void *mem, size_t size)
{
const u8 *p = mem;
size_t i;
for (i = 0; i < size; i++) {
if (p[i])
return false;
}
return true;
}