drgn/libdrgn/string_builder.h
Omar Sandoval 87b7292aa5 Relicense drgn from GPLv3+ to LGPLv2.1+
drgn is currently licensed as GPLv3+. Part of the long term vision for
drgn is that other projects can use it as a library providing
programmatic interfaces for debugger functionality. A more permissive
license is better suited to this goal. We decided on LGPLv2.1+ as a good
balance between software freedom and permissiveness.

All contributors not employed by Meta were contacted via email and
consented to the license change. The only exception was the author of
commit c4fbf7e589 ("libdrgn: fix for compilation error"), who did not
respond. That commit reverted a single line of code to one originally
written by me in commit 640b1c011d ("libdrgn: embed DWARF index in
DWARF info cache").

Signed-off-by: Omar Sandoval <osandov@osandov.com>
2022-11-01 17:05:16 -07:00

211 lines
5.8 KiB
C

// Copyright (c) Meta Platforms, Inc. and affiliates.
// SPDX-License-Identifier: LGPL-2.1-or-later
/**
* @file
*
* String builder interface.
*
* See @ref StringBuilding.
*/
#ifndef DRGN_STRING_BUILDER_H
#define DRGN_STRING_BUILDER_H
#include <stdarg.h>
#include <stdbool.h>
#include <stddef.h>
#include <string.h>
/**
* @ingroup Internals
*
* @defgroup StringBuilding String building
*
* String builder interface.
*
* @ref string_builder provides an append-only way to build a string piece by
* piece. @ref string_callback provides an alternative to prepending pieces.
*
* @{
*/
/**
* String builder.
*
* A string builder consists of a buffer and a length. The buffer is resized as
* needed. The buffer can only be appended to; see @ref string_callback for an
* alternative to insertion.
*/
struct string_builder {
/**
* Current string buffer.
*
* This may be reallocated when appending. It must be freed with @c
* free() when it will no longer be used.
*/
char *str;
/** Length of @c str. */
size_t len;
/** Allocated size of @c str. */
size_t capacity;
};
/** String builder initializer. */
#define STRING_BUILDER_INIT { 0 }
/**
* Null-terminate and return a string from a @ref string_builder.
*
* This appends a null character without incrementing @ref string_builder::len.
*
* @return @ref string_builder::str on success, @c NULL on error (if we couldn't
* allocate memory).
*/
char *string_builder_null_terminate(struct string_builder *sb);
/**
* Resize the buffer of a @ref string_builder to a given capacity.
*
* On success, the allocated size of the string buffer is at least @p capacity.
*
* @param[in] sb String builder.
* @param[in] capacity New minimum allocated size of the string buffer.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
bool string_builder_reserve(struct string_builder *sb, size_t capacity);
/**
* Resize the buffer of a @ref string_builder to accomodate appending
* characters.
*
* On success, the allocated size of the string buffer is at least
* `sb->len + n`. This will also allocate extra space so that appends have
* amortized constant time complexity.
*
* @param[in] sb String builder.
* @param[in] n Minimum number of additional characters to reserve.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
bool string_builder_reserve_for_append(struct string_builder *sb, size_t n);
/**
* Append a character to a @ref string_builder.
*
* @param[in] sb String builder.
* @param[in] c Character to append.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
bool string_builder_appendc(struct string_builder *sb, char c);
/**
* Append a number of characters from a string to a @ref string_builder.
*
* @param[in] sb String builder.
* @param[in] str String to append.
* @param[in] len Number of characters from @c str to append.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
bool string_builder_appendn(struct string_builder *sb, const char *str,
size_t len);
/**
* Append a null-terminated string to a @ref string_builder.
*
* @param[in] sb String builder.
* @param[in] str String to append.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
static inline bool string_builder_append(struct string_builder *sb,
const char *str)
{
return string_builder_appendn(sb, str, strlen(str));
}
/**
* Append a string to a @ref string_builder from a printf-style format.
*
* @param[in] sb String builder.
* @param[in] format printf-style format string.
* @param[in] ... Arguments for the format string.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
bool string_builder_appendf(struct string_builder *sb, const char *format, ...)
__attribute__((__format__(__printf__, 2, 3)));
/**
* Append a string to a @ref string_builder from vprintf-style arguments.
*
* @sa string_builder_appendf()
*
* @param[in] sb String builder.
* @param[in] format printf-style format string.
* @param[in] ap Arguments for the format string.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
bool string_builder_vappendf(struct string_builder *sb, const char *format,
va_list ap);
/**
* Append a newline character to a @ref string_builder if the string isn't empty
* and doesn't already end in a newline.
*
* @param[in] sb String builder.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
bool string_builder_line_break(struct string_builder *sb);
/**
* Callback to append to a string later.
*
* Instead of providing functionality to prepend to a @ref string_builder, we
* achieve the same thing by passing around a callback until all prefixes have
* been appended, then calling the callback to append the "infix". This avoids
* the O(n) array shift required for prepend.
*/
struct string_callback {
/** Callback function. */
struct drgn_error *(*fn)(struct string_callback *str, void *arg,
struct string_builder *sb);
/**
* Another string callback to be passed to the callback.
*
* This is useful for strings that need to be built recursively.
*/
struct string_callback *str;
/** Callback argument. */
void *arg;
};
/**
* Call a string callback.
*
* The callback function will be passed @ref string_callback::str and @ref
* string_callback::arg.
*
* @param[in] str String callback. If @c NULL, this is a no-op.
* @param[in] sb String builder to append to.
* @return @c true on success, @c false on error (if we couldn't allocate
* memory).
*/
static inline struct drgn_error *string_callback_call(struct string_callback *str,
struct string_builder *sb)
{
if (str)
return str->fn(str->str, str->arg, sb);
else
return NULL;
}
/** @} */
#endif /* DRGN_STRING_BUILDER_H */