2008-12-20 10:02:00 +00:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
2010-02-12 13:34:04 +00:00
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2008-12-20 10:02:00 +00:00
|
|
|
*
|
|
|
|
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
|
|
|
|
* All rights reserved.
|
2011-10-20 09:47:05 +00:00
|
|
|
*/
|
2008-12-20 10:02:00 +00:00
|
|
|
|
2012-02-17 18:59:41 +00:00
|
|
|
#pragma once
|
2008-12-20 10:02:00 +00:00
|
|
|
|
2019-02-18 08:08:12 +11:00
|
|
|
/** \file
|
|
|
|
* \ingroup bli
|
2011-02-18 13:58:08 +00:00
|
|
|
*/
|
|
|
|
|
2018-11-19 15:24:32 +01:00
|
|
|
#include <inttypes.h>
|
2020-03-19 09:33:03 +01:00
|
|
|
#include <stdarg.h>
|
2012-08-26 11:01:14 +00:00
|
|
|
|
2020-05-08 18:16:39 +02:00
|
|
|
#include "BLI_compiler_attrs.h"
|
|
|
|
#include "BLI_utildefines.h"
|
|
|
|
|
2008-12-20 10:02:00 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
char *BLI_strdupn(const char *str, const size_t len) ATTR_MALLOC ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL();
|
2008-12-20 10:02:00 +00:00
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
char *BLI_strdup(const char *str) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL() ATTR_MALLOC;
|
2010-11-29 14:29:30 +00:00
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
char *BLI_strdupcat(const char *__restrict str1,
|
|
|
|
const char *__restrict str2) ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL() ATTR_MALLOC;
|
2008-12-20 10:02:00 +00:00
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
char *BLI_strncpy(char *__restrict dst, const char *__restrict src, const size_t maxncpy)
|
|
|
|
ATTR_NONNULL();
|
2013-03-14 10:07:05 +00:00
|
|
|
|
2015-01-06 18:21:46 +11:00
|
|
|
char *BLI_strncpy_ensure_pad(char *__restrict dst,
|
|
|
|
const char *__restrict src,
|
|
|
|
const char pad,
|
|
|
|
size_t maxncpy) ATTR_NONNULL();
|
2015-01-03 10:13:02 +01:00
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
size_t BLI_strncpy_rlen(char *__restrict dst,
|
|
|
|
const char *__restrict src,
|
|
|
|
const size_t maxncpy) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
2013-06-16 08:29:02 +00:00
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
size_t BLI_strcpy_rlen(char *__restrict dst, const char *__restrict src) ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL();
|
2009-10-12 11:27:34 +00:00
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
char *BLI_str_quoted_substrN(const char *__restrict str,
|
|
|
|
const char *__restrict prefix) ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL() ATTR_MALLOC;
|
2009-10-09 12:18:32 +00:00
|
|
|
|
2015-06-30 14:50:34 +10:00
|
|
|
char *BLI_str_replaceN(const char *__restrict str,
|
|
|
|
const char *__restrict substr_old,
|
|
|
|
const char *__restrict substr_new) ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL() ATTR_MALLOC;
|
2008-12-20 10:02:00 +00:00
|
|
|
|
2015-06-30 15:18:03 +10:00
|
|
|
void BLI_str_replace_char(char *string, char src, char dst) ATTR_NONNULL();
|
|
|
|
|
2014-05-20 21:41:57 +10:00
|
|
|
size_t BLI_snprintf(char *__restrict dst, size_t maxncpy, const char *__restrict format, ...)
|
|
|
|
ATTR_NONNULL(1, 3) ATTR_PRINTF_FORMAT(3, 4);
|
2015-04-22 05:37:22 +10:00
|
|
|
size_t BLI_snprintf_rlen(char *__restrict dst, size_t maxncpy, const char *__restrict format, ...)
|
|
|
|
ATTR_NONNULL(1, 3) ATTR_PRINTF_FORMAT(3, 4);
|
2009-03-25 20:29:01 +00:00
|
|
|
|
2020-09-04 20:59:13 +02:00
|
|
|
size_t BLI_vsnprintf(char *__restrict buffer,
|
2013-09-01 15:01:15 +00:00
|
|
|
size_t maxncpy,
|
|
|
|
const char *__restrict format,
|
|
|
|
va_list arg) ATTR_PRINTF_FORMAT(3, 0);
|
2015-04-22 05:37:22 +10:00
|
|
|
size_t BLI_vsnprintf_rlen(char *__restrict buffer,
|
|
|
|
size_t maxncpy,
|
|
|
|
const char *__restrict format,
|
|
|
|
va_list arg) ATTR_PRINTF_FORMAT(3, 0);
|
2011-08-23 15:08:54 +00:00
|
|
|
|
2014-05-20 21:41:57 +10:00
|
|
|
char *BLI_sprintfN(const char *__restrict format, ...) ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL(1) ATTR_MALLOC ATTR_PRINTF_FORMAT(1, 2);
|
2008-12-20 10:02:00 +00:00
|
|
|
|
2020-12-10 13:33:55 +11:00
|
|
|
size_t BLI_str_escape(char *__restrict dst, const char *__restrict src, const size_t dst_maxncpy)
|
2013-09-01 15:01:15 +00:00
|
|
|
ATTR_NONNULL();
|
2020-12-10 13:42:59 +11:00
|
|
|
size_t BLI_str_unescape(char *__restrict dst, const char *__restrict src, const size_t src_maxncpy)
|
|
|
|
ATTR_NONNULL();
|
2008-12-20 10:02:00 +00:00
|
|
|
|
2014-07-17 14:54:12 +10:00
|
|
|
size_t BLI_str_format_int_grouped(char dst[16], int num) ATTR_NONNULL();
|
2018-11-19 15:24:32 +01:00
|
|
|
size_t BLI_str_format_uint64_grouped(char dst[16], uint64_t num) ATTR_NONNULL();
|
2020-09-04 20:59:13 +02:00
|
|
|
void BLI_str_format_byte_unit(char dst[15], long long int bytes, const bool base_10)
|
|
|
|
ATTR_NONNULL();
|
2014-07-17 14:54:12 +10:00
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
int BLI_strcaseeq(const char *a, const char *b) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
|
|
|
char *BLI_strcasestr(const char *s, const char *find) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
2016-03-23 18:45:32 +11:00
|
|
|
char *BLI_strncasestr(const char *s, const char *find, size_t len) ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL();
|
2013-09-01 15:01:15 +00:00
|
|
|
int BLI_strcasecmp(const char *s1, const char *s2) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
|
|
|
int BLI_strncasecmp(const char *s1, const char *s2, size_t len) ATTR_WARN_UNUSED_RESULT
|
|
|
|
ATTR_NONNULL();
|
2019-08-31 17:28:48 +10:00
|
|
|
int BLI_strcasecmp_natural(const char *s1, const char *s2) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
2015-01-03 10:13:02 +01:00
|
|
|
int BLI_strcmp_ignore_pad(const char *str1,
|
|
|
|
const char *str2,
|
|
|
|
const char pad) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
|
|
|
|
2013-09-01 15:01:15 +00:00
|
|
|
size_t BLI_strnlen(const char *str, const size_t maxlen) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
|
|
|
|
2015-07-14 09:17:00 +10:00
|
|
|
void BLI_str_tolower_ascii(char *str, const size_t len) ATTR_NONNULL();
|
|
|
|
void BLI_str_toupper_ascii(char *str, const size_t len) ATTR_NONNULL();
|
2018-08-15 14:47:48 +02:00
|
|
|
void BLI_str_rstrip(char *str) ATTR_NONNULL();
|
2013-09-01 15:01:15 +00:00
|
|
|
int BLI_str_rstrip_float_zero(char *str, const char pad) ATTR_NONNULL();
|
2011-05-26 09:58:22 +00:00
|
|
|
|
2014-06-17 15:58:07 +02:00
|
|
|
int BLI_str_index_in_array_n(const char *__restrict str,
|
|
|
|
const char **__restrict str_array,
|
|
|
|
const int str_array_len) ATTR_NONNULL();
|
|
|
|
int BLI_str_index_in_array(const char *__restrict str, const char **__restrict str_array)
|
|
|
|
ATTR_NONNULL();
|
|
|
|
|
2020-02-12 12:48:44 +01:00
|
|
|
bool BLI_str_startswith(const char *__restrict str, const char *__restrict start) ATTR_NONNULL();
|
2015-01-09 23:38:23 +11:00
|
|
|
bool BLI_str_endswith(const char *__restrict str, const char *__restrict end) ATTR_NONNULL();
|
2015-01-10 19:13:38 +01:00
|
|
|
bool BLI_strn_endswith(const char *__restrict str, const char *__restrict end, size_t length)
|
|
|
|
ATTR_NONNULL();
|
2019-04-17 06:17:24 +02:00
|
|
|
|
2015-06-27 11:00:47 +02:00
|
|
|
size_t BLI_str_partition(const char *str, const char delim[], const char **sep, const char **suf)
|
|
|
|
ATTR_NONNULL();
|
|
|
|
size_t BLI_str_rpartition(const char *str, const char delim[], const char **sep, const char **suf)
|
|
|
|
ATTR_NONNULL();
|
2015-06-27 10:22:29 +02:00
|
|
|
size_t BLI_str_partition_ex(const char *str,
|
2015-06-27 11:00:47 +02:00
|
|
|
const char *end,
|
|
|
|
const char delim[],
|
|
|
|
const char **sep,
|
|
|
|
const char **suf,
|
2015-06-27 10:22:29 +02:00
|
|
|
const bool from_right) ATTR_NONNULL(1, 3, 4, 5);
|
2019-04-17 06:17:24 +02:00
|
|
|
|
2020-07-07 10:08:42 +02:00
|
|
|
int BLI_string_max_possible_word_count(const int str_len);
|
|
|
|
bool BLI_string_has_word_prefix(const char *haystack, const char *needle, size_t needle_len);
|
|
|
|
bool BLI_string_all_words_matched(const char *name,
|
|
|
|
const char *str,
|
|
|
|
int (*words)[2],
|
|
|
|
const int words_len);
|
|
|
|
|
2016-03-23 18:45:32 +11:00
|
|
|
int BLI_string_find_split_words(const char *str,
|
|
|
|
const size_t len,
|
|
|
|
const char delim,
|
|
|
|
int r_words[][2],
|
|
|
|
int words_max) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL();
|
|
|
|
|
2019-04-10 09:24:40 +02:00
|
|
|
/* -------------------------------------------------------------------- */
|
2018-04-05 16:44:48 +02:00
|
|
|
/** \name String Copy/Format Macros
|
|
|
|
* Avoid repeating destination with `sizeof(..)`.
|
|
|
|
* \note `ARRAY_SIZE` allows pointers on some platforms.
|
|
|
|
* \{ */
|
|
|
|
#define STRNCPY(dst, src) BLI_strncpy(dst, src, ARRAY_SIZE(dst))
|
|
|
|
#define STRNCPY_RLEN(dst, src) BLI_strncpy_rlen(dst, src, ARRAY_SIZE(dst))
|
|
|
|
#define SNPRINTF(dst, format, ...) BLI_snprintf(dst, ARRAY_SIZE(dst), format, __VA_ARGS__)
|
|
|
|
#define SNPRINTF_RLEN(dst, format, ...) \
|
|
|
|
BLI_snprintf_rlen(dst, ARRAY_SIZE(dst), format, __VA_ARGS__)
|
2019-03-07 12:01:32 +01:00
|
|
|
#define STR_CONCAT(dst, len, suffix) \
|
|
|
|
len += BLI_strncpy_rlen(dst + len, suffix, ARRAY_SIZE(dst) - len)
|
|
|
|
#define STR_CONCATF(dst, len, format, ...) \
|
|
|
|
len += BLI_snprintf_rlen(dst + len, ARRAY_SIZE(dst) - len, format, __VA_ARGS__)
|
2018-04-05 16:44:48 +02:00
|
|
|
/** \} */
|
|
|
|
|
2019-04-10 09:24:40 +02:00
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \name Equal to Any Element (STR_ELEM) Macro
|
|
|
|
*
|
|
|
|
* Follows #ELEM macro convention.
|
|
|
|
* \{ */
|
|
|
|
|
2019-04-20 13:02:20 +02:00
|
|
|
/* Manual line breaks for readability. */
|
|
|
|
/* clang-format off */
|
2019-04-10 09:24:40 +02:00
|
|
|
/* STR_ELEM#(v, ...): is the first arg equal any others? */
|
|
|
|
/* Internal helpers. */
|
|
|
|
#define _VA_STR_ELEM2(v, a) (strcmp(v, a) == 0)
|
2019-04-20 13:02:20 +02:00
|
|
|
#define _VA_STR_ELEM3(v, a, b) \
|
|
|
|
(_VA_STR_ELEM2(v, a) || (_VA_STR_ELEM2(v, b)))
|
|
|
|
#define _VA_STR_ELEM4(v, a, b, c) \
|
|
|
|
(_VA_STR_ELEM3(v, a, b) || (_VA_STR_ELEM2(v, c)))
|
|
|
|
#define _VA_STR_ELEM5(v, a, b, c, d) \
|
|
|
|
(_VA_STR_ELEM4(v, a, b, c) || (_VA_STR_ELEM2(v, d)))
|
|
|
|
#define _VA_STR_ELEM6(v, a, b, c, d, e) \
|
|
|
|
(_VA_STR_ELEM5(v, a, b, c, d) || (_VA_STR_ELEM2(v, e)))
|
|
|
|
#define _VA_STR_ELEM7(v, a, b, c, d, e, f) \
|
|
|
|
(_VA_STR_ELEM6(v, a, b, c, d, e) || (_VA_STR_ELEM2(v, f)))
|
2019-04-19 14:49:57 +02:00
|
|
|
#define _VA_STR_ELEM8(v, a, b, c, d, e, f, g) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM7(v, a, b, c, d, e, f) || (_VA_STR_ELEM2(v, g)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM9(v, a, b, c, d, e, f, g, h) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM8(v, a, b, c, d, e, f, g) || (_VA_STR_ELEM2(v, h)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM10(v, a, b, c, d, e, f, g, h, i) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM9(v, a, b, c, d, e, f, g, h) || (_VA_STR_ELEM2(v, i)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM11(v, a, b, c, d, e, f, g, h, i, j) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM10(v, a, b, c, d, e, f, g, h, i) || (_VA_STR_ELEM2(v, j)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM12(v, a, b, c, d, e, f, g, h, i, j, k) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM11(v, a, b, c, d, e, f, g, h, i, j) || (_VA_STR_ELEM2(v, k)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM13(v, a, b, c, d, e, f, g, h, i, j, k, l) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM12(v, a, b, c, d, e, f, g, h, i, j, k) || (_VA_STR_ELEM2(v, l)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM14(v, a, b, c, d, e, f, g, h, i, j, k, l, m) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM13(v, a, b, c, d, e, f, g, h, i, j, k, l) || (_VA_STR_ELEM2(v, m)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM15(v, a, b, c, d, e, f, g, h, i, j, k, l, m, n) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM14(v, a, b, c, d, e, f, g, h, i, j, k, l, m) || (_VA_STR_ELEM2(v, n)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM16(v, a, b, c, d, e, f, g, h, i, j, k, l, m, n, o) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM15(v, a, b, c, d, e, f, g, h, i, j, k, l, m, n) || (_VA_STR_ELEM2(v, o)))
|
2019-04-10 09:24:40 +02:00
|
|
|
#define _VA_STR_ELEM17(v, a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p) \
|
2019-04-20 13:02:20 +02:00
|
|
|
(_VA_STR_ELEM16(v, a, b, c, d, e, f, g, h, i, j, k, l, m, n, o) || (_VA_STR_ELEM2(v, p)))
|
|
|
|
/* clang-format on */
|
2019-04-10 09:24:40 +02:00
|
|
|
|
|
|
|
/* reusable STR_ELEM macro */
|
|
|
|
#define STR_ELEM(...) VA_NARGS_CALL_OVERLOAD(_VA_STR_ELEM, __VA_ARGS__)
|
|
|
|
|
|
|
|
/** \} */
|
|
|
|
|
2008-12-20 10:02:00 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|