Junio C Hamano | d1df574 | 2005-04-25 18:26:45 -0700 | [diff] [blame] | 1 | #ifndef STRBUF_H |
| 2 | #define STRBUF_H |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 3 | |
Calvin Wan | 4557779 | 2023-06-06 19:48:37 +0000 | [diff] [blame] | 4 | /* |
| 5 | * NOTE FOR STRBUF DEVELOPERS |
| 6 | * |
| 7 | * strbuf is a low-level primitive; as such it should interact only |
| 8 | * with other low-level primitives. Do not introduce new functions |
| 9 | * which interact with higher-level APIs. |
| 10 | */ |
| 11 | |
Elijah Newren | f6f7755 | 2018-04-19 10:58:08 -0700 | [diff] [blame] | 12 | struct string_list; |
| 13 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 14 | /** |
| 15 | * strbuf's are meant to be used with all the usual C string and memory |
| 16 | * APIs. Given that the length of the buffer is known, it's often better to |
| 17 | * use the mem* functions than a str* one (memchr vs. strchr e.g.). |
| 18 | * Though, one has to be careful about the fact that str* functions often |
| 19 | * stop on NULs and that strbufs may have embedded NULs. |
| 20 | * |
| 21 | * A strbuf is NUL terminated for convenience, but no function in the |
| 22 | * strbuf API actually relies on the string being free of NULs. |
| 23 | * |
| 24 | * strbufs have some invariants that are very important to keep in mind: |
| 25 | * |
Jeff King | aa07cac | 2015-01-16 04:05:10 -0500 | [diff] [blame] | 26 | * - The `buf` member is never NULL, so it can be used in any usual C |
| 27 | * string operations safely. strbuf's _have_ to be initialized either by |
| 28 | * `strbuf_init()` or by `= STRBUF_INIT` before the invariants, though. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 29 | * |
Jeff King | aa07cac | 2015-01-16 04:05:10 -0500 | [diff] [blame] | 30 | * Do *not* assume anything on what `buf` really is (e.g. if it is |
| 31 | * allocated memory or not), use `strbuf_detach()` to unwrap a memory |
| 32 | * buffer from its strbuf shell in a safe way. That is the sole supported |
| 33 | * way. This will give you a malloced buffer that you can later `free()`. |
| 34 | * |
| 35 | * However, it is totally safe to modify anything in the string pointed by |
| 36 | * the `buf` member, between the indices `0` and `len-1` (inclusive). |
| 37 | * |
| 38 | * - The `buf` member is a byte array that has at least `len + 1` bytes |
| 39 | * allocated. The extra byte is used to store a `'\0'`, allowing the |
| 40 | * `buf` member to be a valid C-string. Every strbuf function ensure this |
| 41 | * invariant is preserved. |
| 42 | * |
| 43 | * NOTE: It is OK to "play" with the buffer directly if you work it this |
| 44 | * way: |
| 45 | * |
Jeff King | 088c9a8 | 2015-01-16 04:05:16 -0500 | [diff] [blame] | 46 | * strbuf_grow(sb, SOME_SIZE); <1> |
| 47 | * strbuf_setlen(sb, sb->len + SOME_OTHER_SIZE); |
| 48 | * |
Jeff King | aa07cac | 2015-01-16 04:05:10 -0500 | [diff] [blame] | 49 | * <1> Here, the memory array starting at `sb->buf`, and of length |
| 50 | * `strbuf_avail(sb)` is all yours, and you can be sure that |
| 51 | * `strbuf_avail(sb)` is at least `SOME_SIZE`. |
| 52 | * |
| 53 | * NOTE: `SOME_OTHER_SIZE` must be smaller or equal to `strbuf_avail(sb)`. |
| 54 | * |
| 55 | * Doing so is safe, though if it has to be done in many places, adding the |
| 56 | * missing API to the strbuf module is the way to go. |
| 57 | * |
| 58 | * WARNING: Do _not_ assume that the area that is yours is of size `alloc |
| 59 | * - 1` even if it's true in the current implementation. Alloc is somehow a |
| 60 | * "private" member that should not be messed with. Use `strbuf_avail()` |
| 61 | * instead. |
| 62 | */ |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 63 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 64 | /** |
| 65 | * Data Structures |
| 66 | * --------------- |
| 67 | */ |
| 68 | |
| 69 | /** |
| 70 | * This is the string buffer structure. The `len` member can be used to |
| 71 | * determine the current length of the string, and `buf` member provides |
| 72 | * access to the string itself. |
| 73 | */ |
Junio C Hamano | d1df574 | 2005-04-25 18:26:45 -0700 | [diff] [blame] | 74 | struct strbuf { |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 75 | size_t alloc; |
| 76 | size_t len; |
Brian Gerst | bf0f910 | 2005-05-18 08:14:09 -0400 | [diff] [blame] | 77 | char *buf; |
Junio C Hamano | d1df574 | 2005-04-25 18:26:45 -0700 | [diff] [blame] | 78 | }; |
| 79 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 80 | extern char strbuf_slopbuf[]; |
Ævar Arnfjörð Bjarmason | 608cfd3 | 2021-09-27 14:54:26 +0200 | [diff] [blame] | 81 | #define STRBUF_INIT { .buf = strbuf_slopbuf } |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 82 | |
brian m. carlson | 30e677e | 2018-03-12 02:27:28 +0000 | [diff] [blame] | 83 | struct object_id; |
| 84 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 85 | /** |
Jeff King | 14e2177 | 2015-01-16 04:05:28 -0500 | [diff] [blame] | 86 | * Life Cycle Functions |
| 87 | * -------------------- |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 88 | */ |
| 89 | |
| 90 | /** |
| 91 | * Initialize the structure. The second parameter can be zero or a bigger |
| 92 | * number to allocate memory, in case you want to prevent further reallocs. |
| 93 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 94 | void strbuf_init(struct strbuf *sb, size_t alloc); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 95 | |
| 96 | /** |
Jonathan Nieder | e022215 | 2017-10-03 19:39:54 -0700 | [diff] [blame] | 97 | * Release a string buffer and the memory it used. After this call, the |
| 98 | * strbuf points to an empty string that does not need to be free()ed, as |
| 99 | * if it had been set to `STRBUF_INIT` and never modified. |
| 100 | * |
| 101 | * To clear a strbuf in preparation for further use without the overhead |
| 102 | * of free()ing and malloc()ing again, use strbuf_reset() instead. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 103 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 104 | void strbuf_release(struct strbuf *sb); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 105 | |
| 106 | /** |
| 107 | * Detach the string from the strbuf and returns it; you now own the |
| 108 | * storage the string occupies and it is your responsibility from then on |
| 109 | * to release it with `free(3)` when you are done with it. |
Jonathan Nieder | e022215 | 2017-10-03 19:39:54 -0700 | [diff] [blame] | 110 | * |
| 111 | * The strbuf that previously held the string is reset to `STRBUF_INIT` so |
| 112 | * it can be reused after calling this function. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 113 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 114 | char *strbuf_detach(struct strbuf *sb, size_t *sz); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 115 | |
| 116 | /** |
| 117 | * Attach a string to a buffer. You should specify the string to attach, |
| 118 | * the current length of the string and the amount of allocated memory. |
| 119 | * The amount must be larger than the string length, because the string you |
| 120 | * pass is supposed to be a NUL-terminated string. This string _must_ be |
| 121 | * malloc()ed, and after attaching, the pointer cannot be relied upon |
| 122 | * anymore, and neither be free()d directly. |
| 123 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 124 | void strbuf_attach(struct strbuf *sb, void *str, size_t len, size_t mem); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 125 | |
| 126 | /** |
| 127 | * Swap the contents of two string buffers. |
| 128 | */ |
Nguyễn Thái Ngọc Duy | 187e290 | 2014-03-01 09:50:55 +0700 | [diff] [blame] | 129 | static inline void strbuf_swap(struct strbuf *a, struct strbuf *b) |
| 130 | { |
René Scharfe | 35d803b | 2017-01-28 22:40:58 +0100 | [diff] [blame] | 131 | SWAP(*a, *b); |
Pierre Habouzit | c76689d | 2007-09-20 00:42:12 +0200 | [diff] [blame] | 132 | } |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 133 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 134 | |
| 135 | /** |
Jeff King | 14e2177 | 2015-01-16 04:05:28 -0500 | [diff] [blame] | 136 | * Functions related to the size of the buffer |
| 137 | * ------------------------------------------- |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 138 | */ |
| 139 | |
| 140 | /** |
| 141 | * Determine the amount of allocated but unused memory. |
| 142 | */ |
Nguyễn Thái Ngọc Duy | 187e290 | 2014-03-01 09:50:55 +0700 | [diff] [blame] | 143 | static inline size_t strbuf_avail(const struct strbuf *sb) |
| 144 | { |
Pierre Habouzit | c76689d | 2007-09-20 00:42:12 +0200 | [diff] [blame] | 145 | return sb->alloc ? sb->alloc - sb->len - 1 : 0; |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 146 | } |
Junio C Hamano | a8f3e22 | 2007-09-26 02:26:06 -0700 | [diff] [blame] | 147 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 148 | /** |
| 149 | * Ensure that at least this amount of unused memory is available after |
| 150 | * `len`. This is used when you know a typical size for what you will add |
| 151 | * and want to avoid repetitive automatic resizing of the underlying buffer. |
| 152 | * This is never a needed operation, but can be critical for performance in |
| 153 | * some cases. |
| 154 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 155 | void strbuf_grow(struct strbuf *sb, size_t amount); |
Junio C Hamano | a8f3e22 | 2007-09-26 02:26:06 -0700 | [diff] [blame] | 156 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 157 | /** |
| 158 | * Set the length of the buffer to a given value. This function does *not* |
| 159 | * allocate new memory, so you should not perform a `strbuf_setlen()` to a |
| 160 | * length that is larger than `len + strbuf_avail()`. `strbuf_setlen()` is |
| 161 | * just meant as a 'please fix invariants from this strbuf I just messed |
| 162 | * with'. |
| 163 | */ |
Nguyễn Thái Ngọc Duy | 187e290 | 2014-03-01 09:50:55 +0700 | [diff] [blame] | 164 | static inline void strbuf_setlen(struct strbuf *sb, size_t len) |
| 165 | { |
René Scharfe | 7141efa | 2011-04-27 19:24:50 +0200 | [diff] [blame] | 166 | if (len > (sb->alloc ? sb->alloc - 1 : 0)) |
Ævar Arnfjörð Bjarmason | 46d699f | 2021-12-07 12:05:52 +0100 | [diff] [blame] | 167 | BUG("strbuf_setlen() beyond buffer"); |
Pierre Habouzit | c76689d | 2007-09-20 00:42:12 +0200 | [diff] [blame] | 168 | sb->len = len; |
Martin Ågren | 65961d5 | 2017-08-21 19:43:47 +0200 | [diff] [blame] | 169 | if (sb->buf != strbuf_slopbuf) |
| 170 | sb->buf[len] = '\0'; |
| 171 | else |
| 172 | assert(!strbuf_slopbuf[0]); |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 173 | } |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 174 | |
| 175 | /** |
| 176 | * Empty the buffer by setting the size of it to zero. |
| 177 | */ |
Pierre Habouzit | b315c5c | 2007-09-27 12:58:23 +0200 | [diff] [blame] | 178 | #define strbuf_reset(sb) strbuf_setlen(sb, 0) |
Pierre Habouzit | b449f4c | 2007-09-06 13:20:05 +0200 | [diff] [blame] | 179 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 180 | |
| 181 | /** |
Jeff King | 14e2177 | 2015-01-16 04:05:28 -0500 | [diff] [blame] | 182 | * Functions related to the contents of the buffer |
| 183 | * ----------------------------------------------- |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 184 | */ |
| 185 | |
| 186 | /** |
Jeff King | d468fa2 | 2015-01-16 04:06:04 -0500 | [diff] [blame] | 187 | * Strip whitespace from the beginning (`ltrim`), end (`rtrim`), or both side |
| 188 | * (`trim`) of a string. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 189 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 190 | void strbuf_trim(struct strbuf *sb); |
| 191 | void strbuf_rtrim(struct strbuf *sb); |
| 192 | void strbuf_ltrim(struct strbuf *sb); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 193 | |
Nguyễn Thái Ngọc Duy | c64a8d2 | 2018-02-12 16:49:37 +0700 | [diff] [blame] | 194 | /* Strip trailing directory separators */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 195 | void strbuf_trim_trailing_dir_sep(struct strbuf *sb); |
Nguyễn Thái Ngọc Duy | c64a8d2 | 2018-02-12 16:49:37 +0700 | [diff] [blame] | 196 | |
Pratik Karki | f957362 | 2018-08-08 20:51:16 +0545 | [diff] [blame] | 197 | /* Strip trailing LF or CR/LF */ |
Junio C Hamano | 39f7331 | 2018-11-02 11:04:54 +0900 | [diff] [blame] | 198 | void strbuf_trim_trailing_newline(struct strbuf *sb); |
Pratik Karki | f957362 | 2018-08-08 20:51:16 +0545 | [diff] [blame] | 199 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 200 | /** |
| 201 | * Replace the contents of the strbuf with a reencoded form. Returns -1 |
| 202 | * on error, 0 on success. |
| 203 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 204 | int strbuf_reencode(struct strbuf *sb, const char *from, const char *to); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 205 | |
| 206 | /** |
| 207 | * Lowercase each character in the buffer using `tolower`. |
| 208 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 209 | void strbuf_tolower(struct strbuf *sb); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 210 | |
| 211 | /** |
| 212 | * Compare two buffers. Returns an integer less than, equal to, or greater |
| 213 | * than zero if the first buffer is found, respectively, to be less than, |
| 214 | * to match, or be greater than the second buffer. |
| 215 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 216 | int strbuf_cmp(const struct strbuf *first, const struct strbuf *second); |
Lukas Sandström | eacd6dc | 2008-07-13 20:29:18 +0200 | [diff] [blame] | 217 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 218 | |
| 219 | /** |
Jeff King | 14e2177 | 2015-01-16 04:05:28 -0500 | [diff] [blame] | 220 | * Adding data to the buffer |
| 221 | * ------------------------- |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 222 | * |
| 223 | * NOTE: All of the functions in this section will grow the buffer as |
| 224 | * necessary. If they fail for some reason other than memory shortage and the |
| 225 | * buffer hadn't been allocated before (i.e. the `struct strbuf` was set to |
| 226 | * `STRBUF_INIT`), then they will free() it. |
| 227 | */ |
| 228 | |
| 229 | /** |
| 230 | * Add a single character to the buffer. |
| 231 | */ |
| 232 | static inline void strbuf_addch(struct strbuf *sb, int c) |
| 233 | { |
Jeff King | fec501d | 2015-04-16 04:53:56 -0400 | [diff] [blame] | 234 | if (!strbuf_avail(sb)) |
| 235 | strbuf_grow(sb, 1); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 236 | sb->buf[sb->len++] = c; |
| 237 | sb->buf[sb->len] = '\0'; |
| 238 | } |
| 239 | |
| 240 | /** |
| 241 | * Add a character the specified number of times to the buffer. |
| 242 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 243 | void strbuf_addchars(struct strbuf *sb, int c, size_t n); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 244 | |
| 245 | /** |
| 246 | * Insert data to the given position of the buffer. The remaining contents |
| 247 | * will be shifted, not overwritten. |
| 248 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 249 | void strbuf_insert(struct strbuf *sb, size_t pos, const void *, size_t); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 250 | |
| 251 | /** |
René Scharfe | a91cc7f | 2020-02-09 14:44:23 +0100 | [diff] [blame] | 252 | * Insert a NUL-terminated string to the given position of the buffer. |
| 253 | * The remaining contents will be shifted, not overwritten. It's an |
| 254 | * inline function to allow the compiler to resolve strlen() calls on |
| 255 | * constants at compile time. |
| 256 | */ |
| 257 | static inline void strbuf_insertstr(struct strbuf *sb, size_t pos, |
| 258 | const char *s) |
| 259 | { |
| 260 | strbuf_insert(sb, pos, s, strlen(s)); |
| 261 | } |
| 262 | |
| 263 | /** |
Paul-Sebastian Ungureanu | 5ef264d | 2019-02-25 23:16:07 +0000 | [diff] [blame] | 264 | * Insert data to the given position of the buffer giving a printf format |
| 265 | * string. The contents will be shifted, not overwritten. |
| 266 | */ |
| 267 | void strbuf_vinsertf(struct strbuf *sb, size_t pos, const char *fmt, |
| 268 | va_list ap); |
| 269 | |
Ævar Arnfjörð Bjarmason | 75d31ce | 2021-07-13 10:05:19 +0200 | [diff] [blame] | 270 | __attribute__((format (printf, 3, 4))) |
Paul-Sebastian Ungureanu | 5ef264d | 2019-02-25 23:16:07 +0000 | [diff] [blame] | 271 | void strbuf_insertf(struct strbuf *sb, size_t pos, const char *fmt, ...); |
| 272 | |
| 273 | /** |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 274 | * Remove given amount of data from a given position of the buffer. |
| 275 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 276 | void strbuf_remove(struct strbuf *sb, size_t pos, size_t len); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 277 | |
| 278 | /** |
| 279 | * Remove the bytes between `pos..pos+len` and replace it with the given |
| 280 | * data. |
| 281 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 282 | void strbuf_splice(struct strbuf *sb, size_t pos, size_t len, |
| 283 | const void *data, size_t data_len); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 284 | |
| 285 | /** |
| 286 | * Add a NUL-terminated string to the buffer. Each line will be prepended |
| 287 | * by a comment character and a blank. |
| 288 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 289 | void strbuf_add_commented_lines(struct strbuf *out, |
Calvin Wan | 787cb8a | 2023-06-06 19:48:43 +0000 | [diff] [blame] | 290 | const char *buf, size_t size, |
| 291 | char comment_line_char); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 292 | |
| 293 | |
| 294 | /** |
| 295 | * Add data of given length to the buffer. |
| 296 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 297 | void strbuf_add(struct strbuf *sb, const void *data, size_t len); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 298 | |
| 299 | /** |
| 300 | * Add a NUL-terminated string to the buffer. |
| 301 | * |
| 302 | * NOTE: This function will *always* be implemented as an inline or a macro |
| 303 | * using strlen, meaning that this is efficient to write things like: |
| 304 | * |
Jeff King | 088c9a8 | 2015-01-16 04:05:16 -0500 | [diff] [blame] | 305 | * strbuf_addstr(sb, "immediate string"); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 306 | * |
| 307 | */ |
| 308 | static inline void strbuf_addstr(struct strbuf *sb, const char *s) |
| 309 | { |
| 310 | strbuf_add(sb, s, strlen(s)); |
| 311 | } |
| 312 | |
| 313 | /** |
| 314 | * Copy the contents of another buffer at the end of the current one. |
| 315 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 316 | void strbuf_addbuf(struct strbuf *sb, const struct strbuf *sb2); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 317 | |
| 318 | /** |
Paul-Sebastian Ungureanu | e71c4a8 | 2019-02-25 23:16:06 +0000 | [diff] [blame] | 319 | * Join the arguments into a buffer. `delim` is put between every |
| 320 | * two arguments. |
| 321 | */ |
| 322 | const char *strbuf_join_argv(struct strbuf *buf, int argc, |
| 323 | const char **argv, char delim); |
| 324 | |
| 325 | /** |
René Scharfe | 6f1e2d5 | 2023-06-17 22:43:17 +0200 | [diff] [blame] | 326 | * Used with `strbuf_expand_step` to expand the literals %n and %x |
| 327 | * followed by two hexadecimal digits. Returns the number of recognized |
René Scharfe | 4416b86 | 2023-06-17 22:44:00 +0200 | [diff] [blame] | 328 | * characters. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 329 | */ |
René Scharfe | 4416b86 | 2023-06-17 22:44:00 +0200 | [diff] [blame] | 330 | size_t strbuf_expand_literal(struct strbuf *sb, const char *placeholder); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 331 | |
| 332 | /** |
René Scharfe | 44ccb33 | 2023-06-17 22:41:44 +0200 | [diff] [blame] | 333 | * If the string pointed to by `formatp` contains a percent sign ("%"), |
| 334 | * advance it to point to the character following the next one and |
| 335 | * return 1, otherwise return 0. Append the substring before that |
| 336 | * percent sign to `sb`, or the whole string if there is none. |
Anders Waldenborg | fd2015b | 2019-01-28 22:33:36 +0100 | [diff] [blame] | 337 | */ |
René Scharfe | 44ccb33 | 2023-06-17 22:41:44 +0200 | [diff] [blame] | 338 | int strbuf_expand_step(struct strbuf *sb, const char **formatp); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 339 | |
| 340 | /** |
| 341 | * Append the contents of one strbuf to another, quoting any |
| 342 | * percent signs ("%") into double-percents ("%%") in the |
| 343 | * destination. This is useful for literal data to be fed to either |
| 344 | * strbuf_expand or to the *printf family of functions. |
| 345 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 346 | void strbuf_addbuf_percentquote(struct strbuf *dst, const struct strbuf *src); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 347 | |
brian m. carlson | b44d011 | 2020-04-27 01:18:08 +0000 | [diff] [blame] | 348 | #define STRBUF_ENCODE_SLASH 1 |
| 349 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 350 | /** |
brian m. carlson | 46fd7b3 | 2020-02-20 02:24:13 +0000 | [diff] [blame] | 351 | * Append the contents of a string to a strbuf, percent-encoding any characters |
| 352 | * that are needed to be encoded for a URL. |
brian m. carlson | b44d011 | 2020-04-27 01:18:08 +0000 | [diff] [blame] | 353 | * |
| 354 | * If STRBUF_ENCODE_SLASH is set in flags, percent-encode slashes. Otherwise, |
| 355 | * slashes are not percent-encoded. |
brian m. carlson | 46fd7b3 | 2020-02-20 02:24:13 +0000 | [diff] [blame] | 356 | */ |
brian m. carlson | b44d011 | 2020-04-27 01:18:08 +0000 | [diff] [blame] | 357 | void strbuf_add_percentencode(struct strbuf *dst, const char *src, int flags); |
brian m. carlson | 46fd7b3 | 2020-02-20 02:24:13 +0000 | [diff] [blame] | 358 | |
| 359 | /** |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 360 | * Append the given byte size as a human-readable string (i.e. 12.23 KiB, |
| 361 | * 3.50 MiB). |
| 362 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 363 | void strbuf_humanise_bytes(struct strbuf *buf, off_t bytes); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 364 | |
| 365 | /** |
Dimitriy Ryazantcev | 8f354a1 | 2019-07-02 21:22:48 +0300 | [diff] [blame] | 366 | * Append the given byte rate as a human-readable string (i.e. 12.23 KiB/s, |
| 367 | * 3.50 MiB/s). |
| 368 | */ |
| 369 | void strbuf_humanise_rate(struct strbuf *buf, off_t bytes); |
| 370 | |
| 371 | /** |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 372 | * Add a formatted string to the buffer. |
| 373 | */ |
| 374 | __attribute__((format (printf,2,3))) |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 375 | void strbuf_addf(struct strbuf *sb, const char *fmt, ...); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 376 | |
| 377 | /** |
| 378 | * Add a formatted string prepended by a comment character and a |
| 379 | * blank to the buffer. |
| 380 | */ |
Calvin Wan | 787cb8a | 2023-06-06 19:48:43 +0000 | [diff] [blame] | 381 | __attribute__((format (printf, 3, 4))) |
| 382 | void strbuf_commented_addf(struct strbuf *sb, char comment_line_char, const char *fmt, ...); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 383 | |
| 384 | __attribute__((format (printf,2,0))) |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 385 | void strbuf_vaddf(struct strbuf *sb, const char *fmt, va_list ap); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 386 | |
| 387 | /** |
Jeff King | aa1462c | 2015-06-25 12:55:45 -0400 | [diff] [blame] | 388 | * Add the time specified by `tm`, as formatted by `strftime`. |
René Scharfe | c3fbf81 | 2017-06-15 14:29:53 +0200 | [diff] [blame] | 389 | * `tz_offset` is in decimal hhmm format, e.g. -600 means six hours west |
| 390 | * of Greenwich, and it's used to expand %z internally. However, tokens |
| 391 | * with modifiers (e.g. %Ez) are passed to `strftime`. |
Ævar Arnfjörð Bjarmason | 3b70223 | 2017-07-01 13:15:47 +0000 | [diff] [blame] | 392 | * `suppress_tz_name`, when set, expands %Z internally to the empty |
| 393 | * string rather than passing it to `strftime`. |
Jeff King | aa1462c | 2015-06-25 12:55:45 -0400 | [diff] [blame] | 394 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 395 | void strbuf_addftime(struct strbuf *sb, const char *fmt, |
| 396 | const struct tm *tm, int tz_offset, |
| 397 | int suppress_tz_name); |
Jeff King | aa1462c | 2015-06-25 12:55:45 -0400 | [diff] [blame] | 398 | |
| 399 | /** |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 400 | * Read a given size of data from a FILE* pointer to the buffer. |
| 401 | * |
| 402 | * NOTE: The buffer is rewound if the read fails. If -1 is returned, |
| 403 | * `errno` must be consulted, like you would do for `read(3)`. |
Junio C Hamano | 1a0c8df | 2016-01-13 18:32:23 -0800 | [diff] [blame] | 404 | * `strbuf_read()`, `strbuf_read_file()` and `strbuf_getline_*()` |
| 405 | * family of functions have the same behaviour as well. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 406 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 407 | size_t strbuf_fread(struct strbuf *sb, size_t size, FILE *file); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 408 | |
| 409 | /** |
| 410 | * Read the contents of a given file descriptor. The third argument can be |
| 411 | * used to give a hint about the file size, to avoid reallocs. If read fails, |
| 412 | * any partial read is undone. |
| 413 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 414 | ssize_t strbuf_read(struct strbuf *sb, int fd, size_t hint); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 415 | |
| 416 | /** |
Stefan Beller | b4e04fb | 2015-12-15 16:04:08 -0800 | [diff] [blame] | 417 | * Read the contents of a given file descriptor partially by using only one |
| 418 | * attempt of xread. The third argument can be used to give a hint about the |
| 419 | * file size, to avoid reallocs. Returns the number of new bytes appended to |
| 420 | * the sb. |
| 421 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 422 | ssize_t strbuf_read_once(struct strbuf *sb, int fd, size_t hint); |
Stefan Beller | b4e04fb | 2015-12-15 16:04:08 -0800 | [diff] [blame] | 423 | |
| 424 | /** |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 425 | * Read the contents of a file, specified by its path. The third argument |
| 426 | * can be used to give a hint about the file size, to avoid reallocs. |
Pranit Bauva | ed008d7 | 2016-06-14 11:44:11 +0530 | [diff] [blame] | 427 | * Return the number of bytes read or a negative value if some error |
| 428 | * occurred while opening or reading the file. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 429 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 430 | ssize_t strbuf_read_file(struct strbuf *sb, const char *path, size_t hint); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 431 | |
| 432 | /** |
| 433 | * Read the target of a symbolic link, specified by its path. The third |
| 434 | * argument can be used to give a hint about the size, to avoid reallocs. |
| 435 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 436 | int strbuf_readlink(struct strbuf *sb, const char *path, size_t hint); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 437 | |
| 438 | /** |
Stefan Beller | 2dac9b5 | 2016-02-29 18:07:15 -0800 | [diff] [blame] | 439 | * Write the whole content of the strbuf to the stream not stopping at |
| 440 | * NUL bytes. |
| 441 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 442 | ssize_t strbuf_write(struct strbuf *sb, FILE *stream); |
Stefan Beller | 2dac9b5 | 2016-02-29 18:07:15 -0800 | [diff] [blame] | 443 | |
| 444 | /** |
Patrick Steinhardt | af35e56 | 2023-06-06 07:19:37 +0200 | [diff] [blame] | 445 | * Read from a FILE * until the specified terminator is encountered, |
| 446 | * overwriting the existing contents of the strbuf. |
| 447 | * |
| 448 | * Reading stops after the terminator or at EOF. The terminator is |
| 449 | * removed from the buffer before returning. If the terminator is LF |
| 450 | * and if it is preceded by a CR, then the whole CRLF is stripped. |
| 451 | * Returns 0 unless there was nothing left before EOF, in which case |
| 452 | * it returns `EOF`. |
| 453 | */ |
| 454 | int strbuf_getdelim_strip_crlf(struct strbuf *sb, FILE *fp, int term); |
| 455 | |
| 456 | /** |
Junio C Hamano | 1a0c8df | 2016-01-13 18:32:23 -0800 | [diff] [blame] | 457 | * Read a line from a FILE *, overwriting the existing contents of |
| 458 | * the strbuf. The strbuf_getline*() family of functions share |
| 459 | * this signature, but have different line termination conventions. |
| 460 | * |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 461 | * Reading stops after the terminator or at EOF. The terminator |
| 462 | * is removed from the buffer before returning. Returns 0 unless |
| 463 | * there was nothing left before EOF, in which case it returns `EOF`. |
| 464 | */ |
Junio C Hamano | 8f309ae | 2016-01-13 15:31:17 -0800 | [diff] [blame] | 465 | typedef int (*strbuf_getline_fn)(struct strbuf *, FILE *); |
| 466 | |
| 467 | /* Uses LF as the line terminator */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 468 | int strbuf_getline_lf(struct strbuf *sb, FILE *fp); |
Junio C Hamano | 8f309ae | 2016-01-13 15:31:17 -0800 | [diff] [blame] | 469 | |
| 470 | /* Uses NUL as the line terminator */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 471 | int strbuf_getline_nul(struct strbuf *sb, FILE *fp); |
Junio C Hamano | 8f309ae | 2016-01-13 15:31:17 -0800 | [diff] [blame] | 472 | |
Junio C Hamano | c8aa9fd | 2015-10-28 13:17:29 -0700 | [diff] [blame] | 473 | /* |
Junio C Hamano | 8f309ae | 2016-01-13 15:31:17 -0800 | [diff] [blame] | 474 | * Similar to strbuf_getline_lf(), but additionally treats a CR that |
| 475 | * comes immediately before the LF as part of the terminator. |
Junio C Hamano | 1a0c8df | 2016-01-13 18:32:23 -0800 | [diff] [blame] | 476 | * This is the most friendly version to be used to read "text" files |
| 477 | * that can come from platforms whose native text format is CRLF |
| 478 | * terminated. |
Junio C Hamano | c8aa9fd | 2015-10-28 13:17:29 -0700 | [diff] [blame] | 479 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 480 | int strbuf_getline(struct strbuf *sb, FILE *file); |
Junio C Hamano | c8aa9fd | 2015-10-28 13:17:29 -0700 | [diff] [blame] | 481 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 482 | |
| 483 | /** |
| 484 | * Like `strbuf_getline`, but keeps the trailing terminator (if |
| 485 | * any) in the buffer. |
| 486 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 487 | int strbuf_getwholeline(struct strbuf *sb, FILE *file, int term); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 488 | |
| 489 | /** |
Patrick Steinhardt | bd021f3 | 2020-03-30 15:46:27 +0200 | [diff] [blame] | 490 | * Like `strbuf_getwholeline`, but appends the line instead of |
| 491 | * resetting the buffer first. |
| 492 | */ |
| 493 | int strbuf_appendwholeline(struct strbuf *sb, FILE *file, int term); |
| 494 | |
| 495 | /** |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 496 | * Like `strbuf_getwholeline`, but operates on a file descriptor. |
| 497 | * It reads one character at a time, so it is very slow. Do not |
| 498 | * use it unless you need the correct position in the file |
| 499 | * descriptor. |
| 500 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 501 | int strbuf_getwholeline_fd(struct strbuf *sb, int fd, int term); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 502 | |
| 503 | /** |
| 504 | * Set the buffer to the path of the current working directory. |
| 505 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 506 | int strbuf_getcwd(struct strbuf *sb); |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 507 | |
| 508 | /** |
Jeff King | 670c359 | 2016-10-03 16:34:17 -0400 | [diff] [blame] | 509 | * Normalize in-place the path contained in the strbuf. See |
| 510 | * normalize_path_copy() for details. If an error occurs, the contents of "sb" |
| 511 | * are left untouched, and -1 is returned. |
| 512 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 513 | int strbuf_normalize_path(struct strbuf *sb); |
Jeff King | 670c359 | 2016-10-03 16:34:17 -0400 | [diff] [blame] | 514 | |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 515 | /** |
Calvin Wan | 787cb8a | 2023-06-06 19:48:43 +0000 | [diff] [blame] | 516 | * Strip whitespace from a buffer. If comment_line_char is non-NUL, |
| 517 | * then lines beginning with that character are considered comments, |
| 518 | * thus removed. |
Jeff King | bdfdaa4 | 2015-01-16 04:04:04 -0500 | [diff] [blame] | 519 | */ |
Calvin Wan | 787cb8a | 2023-06-06 19:48:43 +0000 | [diff] [blame] | 520 | void strbuf_stripspace(struct strbuf *buf, char comment_line_char); |
Tobias Klauser | 63af4a8 | 2015-10-16 17:16:42 +0200 | [diff] [blame] | 521 | |
Jeff King | 6dda4e6 | 2014-06-30 13:01:51 -0400 | [diff] [blame] | 522 | static inline int strbuf_strip_suffix(struct strbuf *sb, const char *suffix) |
| 523 | { |
| 524 | if (strip_suffix_mem(sb->buf, &sb->len, suffix)) { |
| 525 | strbuf_setlen(sb, sb->len); |
| 526 | return 1; |
| 527 | } else |
| 528 | return 0; |
| 529 | } |
| 530 | |
Stefan Beller | 6afbbdd | 2015-01-16 04:04:51 -0500 | [diff] [blame] | 531 | /** |
Michael Haggerty | 06379a6 | 2012-11-04 07:46:54 +0100 | [diff] [blame] | 532 | * Split str (of length slen) at the specified terminator character. |
| 533 | * Return a null-terminated array of pointers to strbuf objects |
| 534 | * holding the substrings. The substrings include the terminator, |
| 535 | * except for the last substring, which might be unterminated if the |
| 536 | * original string did not end with a terminator. If max is positive, |
| 537 | * then split the string into at most max substrings (with the last |
| 538 | * substring containing everything following the (max-1)th terminator |
| 539 | * character). |
| 540 | * |
Jeff King | f20e56e | 2015-01-16 04:05:57 -0500 | [diff] [blame] | 541 | * The most generic form is `strbuf_split_buf`, which takes an arbitrary |
| 542 | * pointer/len buffer. The `_str` variant takes a NUL-terminated string, |
| 543 | * the `_max` variant takes a strbuf, and just `strbuf_split` is a convenience |
| 544 | * wrapper to drop the `max` parameter. |
| 545 | * |
Michael Haggerty | 06379a6 | 2012-11-04 07:46:54 +0100 | [diff] [blame] | 546 | * For lighter-weight alternatives, see string_list_split() and |
| 547 | * string_list_split_in_place(). |
| 548 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 549 | struct strbuf **strbuf_split_buf(const char *str, size_t len, |
| 550 | int terminator, int max); |
Michael Haggerty | 06379a6 | 2012-11-04 07:46:54 +0100 | [diff] [blame] | 551 | |
Jeff King | 2f1d9e2 | 2011-06-09 11:54:58 -0400 | [diff] [blame] | 552 | static inline struct strbuf **strbuf_split_str(const char *str, |
Michael Haggerty | 17b73dc | 2012-11-04 07:46:53 +0100 | [diff] [blame] | 553 | int terminator, int max) |
Jeff King | 2f1d9e2 | 2011-06-09 11:54:58 -0400 | [diff] [blame] | 554 | { |
Michael Haggerty | 17b73dc | 2012-11-04 07:46:53 +0100 | [diff] [blame] | 555 | return strbuf_split_buf(str, strlen(str), terminator, max); |
Jeff King | 2f1d9e2 | 2011-06-09 11:54:58 -0400 | [diff] [blame] | 556 | } |
Michael Haggerty | 06379a6 | 2012-11-04 07:46:54 +0100 | [diff] [blame] | 557 | |
Jeff King | 2f1d9e2 | 2011-06-09 11:54:58 -0400 | [diff] [blame] | 558 | static inline struct strbuf **strbuf_split_max(const struct strbuf *sb, |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 559 | int terminator, int max) |
Jeff King | 2f1d9e2 | 2011-06-09 11:54:58 -0400 | [diff] [blame] | 560 | { |
Michael Haggerty | 17b73dc | 2012-11-04 07:46:53 +0100 | [diff] [blame] | 561 | return strbuf_split_buf(sb->buf, sb->len, terminator, max); |
Jeff King | 2f1d9e2 | 2011-06-09 11:54:58 -0400 | [diff] [blame] | 562 | } |
Michael Haggerty | 06379a6 | 2012-11-04 07:46:54 +0100 | [diff] [blame] | 563 | |
Michael Haggerty | 17b73dc | 2012-11-04 07:46:53 +0100 | [diff] [blame] | 564 | static inline struct strbuf **strbuf_split(const struct strbuf *sb, |
| 565 | int terminator) |
Jeff King | 28fc3a6 | 2011-06-09 11:51:22 -0400 | [diff] [blame] | 566 | { |
Michael Haggerty | 17b73dc | 2012-11-04 07:46:53 +0100 | [diff] [blame] | 567 | return strbuf_split_max(sb, terminator, 0); |
Jeff King | 28fc3a6 | 2011-06-09 11:51:22 -0400 | [diff] [blame] | 568 | } |
Michael Haggerty | 06379a6 | 2012-11-04 07:46:54 +0100 | [diff] [blame] | 569 | |
Elijah Newren | f6f7755 | 2018-04-19 10:58:08 -0700 | [diff] [blame] | 570 | /* |
| 571 | * Adds all strings of a string list to the strbuf, separated by the given |
| 572 | * separator. For example, if sep is |
| 573 | * ', ' |
| 574 | * and slist contains |
| 575 | * ['element1', 'element2', ..., 'elementN'], |
| 576 | * then write: |
| 577 | * 'element1, element2, ..., elementN' |
| 578 | * to str. If only one element, just write "element1" to str. |
| 579 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 580 | void strbuf_add_separated_string_list(struct strbuf *str, |
| 581 | const char *sep, |
| 582 | struct string_list *slist); |
Elijah Newren | f6f7755 | 2018-04-19 10:58:08 -0700 | [diff] [blame] | 583 | |
Stefan Beller | 6afbbdd | 2015-01-16 04:04:51 -0500 | [diff] [blame] | 584 | /** |
Michael Haggerty | 06379a6 | 2012-11-04 07:46:54 +0100 | [diff] [blame] | 585 | * Free a NULL-terminated list of strbufs (for example, the return |
| 586 | * values of the strbuf_split*() functions). |
| 587 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 588 | void strbuf_list_free(struct strbuf **list); |
Pierre Habouzit | f1696ee | 2007-09-10 12:35:04 +0200 | [diff] [blame] | 589 | |
Derrick Stolee | 9ea5796 | 2022-12-22 15:14:14 +0000 | [diff] [blame] | 590 | /* |
| 591 | * Remove the filename from the provided path string. If the path |
| 592 | * contains a trailing separator, then the path is considered a directory |
| 593 | * and nothing is modified. |
| 594 | * |
| 595 | * Examples: |
| 596 | * - "/path/to/file" -> "/path/to/" |
| 597 | * - "/path/to/dir/" -> "/path/to/dir/" |
| 598 | */ |
| 599 | void strbuf_strip_file_from_path(struct strbuf *sb); |
| 600 | |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 601 | void strbuf_add_lines(struct strbuf *sb, |
| 602 | const char *prefix, |
| 603 | const char *buf, |
| 604 | size_t size); |
Junio C Hamano | 895680f | 2011-11-04 21:06:30 -0700 | [diff] [blame] | 605 | |
Stefan Beller | 6afbbdd | 2015-01-16 04:04:51 -0500 | [diff] [blame] | 606 | /** |
Michael Haggerty | 5963c03 | 2012-11-25 12:08:34 +0100 | [diff] [blame] | 607 | * Append s to sb, with the characters '<', '>', '&' and '"' converted |
| 608 | * into XML entities. |
| 609 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 610 | void strbuf_addstr_xml_quoted(struct strbuf *sb, |
| 611 | const char *s); |
Michael Haggerty | 5963c03 | 2012-11-25 12:08:34 +0100 | [diff] [blame] | 612 | |
Jeff King | 399ad55 | 2015-09-24 17:05:43 -0400 | [diff] [blame] | 613 | /** |
| 614 | * "Complete" the contents of `sb` by ensuring that either it ends with the |
| 615 | * character `term`, or it is empty. This can be used, for example, |
| 616 | * to ensure that text ends with a newline, but without creating an empty |
| 617 | * blank line if there is no content in the first place. |
| 618 | */ |
| 619 | static inline void strbuf_complete(struct strbuf *sb, char term) |
| 620 | { |
| 621 | if (sb->len && sb->buf[sb->len - 1] != term) |
| 622 | strbuf_addch(sb, term); |
| 623 | } |
| 624 | |
Junio C Hamano | 895680f | 2011-11-04 21:06:30 -0700 | [diff] [blame] | 625 | static inline void strbuf_complete_line(struct strbuf *sb) |
| 626 | { |
Jeff King | 399ad55 | 2015-09-24 17:05:43 -0400 | [diff] [blame] | 627 | strbuf_complete(sb, '\n'); |
Junio C Hamano | 895680f | 2011-11-04 21:06:30 -0700 | [diff] [blame] | 628 | } |
| 629 | |
Jeff King | 0705fe2 | 2017-03-02 03:21:30 -0500 | [diff] [blame] | 630 | /* |
| 631 | * Copy "name" to "sb", expanding any special @-marks as handled by |
Ævar Arnfjörð Bjarmason | c7c33f5 | 2023-03-28 15:58:57 +0200 | [diff] [blame] | 632 | * repo_interpret_branch_name(). The result is a non-qualified branch name |
Jeff King | 0705fe2 | 2017-03-02 03:21:30 -0500 | [diff] [blame] | 633 | * (so "foo" or "origin/master" instead of "refs/heads/foo" or |
| 634 | * "refs/remotes/origin/master"). |
| 635 | * |
| 636 | * Note that the resulting name may not be a syntactically valid refname. |
Jeff King | 0e9f62d | 2017-03-02 03:23:01 -0500 | [diff] [blame] | 637 | * |
| 638 | * If "allowed" is non-zero, restrict the set of allowed expansions. See |
Ævar Arnfjörð Bjarmason | c7c33f5 | 2023-03-28 15:58:57 +0200 | [diff] [blame] | 639 | * repo_interpret_branch_name() for details. |
Jeff King | 0705fe2 | 2017-03-02 03:21:30 -0500 | [diff] [blame] | 640 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 641 | void strbuf_branchname(struct strbuf *sb, const char *name, |
| 642 | unsigned allowed); |
Jeff King | 0705fe2 | 2017-03-02 03:21:30 -0500 | [diff] [blame] | 643 | |
| 644 | /* |
| 645 | * Like strbuf_branchname() above, but confirm that the result is |
| 646 | * syntactically valid to be used as a local branch name in refs/heads/. |
| 647 | * |
| 648 | * The return value is "0" if the result is valid, and "-1" otherwise. |
| 649 | */ |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 650 | int strbuf_check_branch_ref(struct strbuf *sb, const char *name); |
Junio C Hamano | a552de7 | 2009-03-21 13:17:30 -0700 | [diff] [blame] | 651 | |
Matthew DeVore | c269495 | 2019-06-27 15:54:11 -0700 | [diff] [blame] | 652 | typedef int (*char_predicate)(char ch); |
| 653 | |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 654 | void strbuf_addstr_urlencode(struct strbuf *sb, const char *name, |
Matthew DeVore | c269495 | 2019-06-27 15:54:11 -0700 | [diff] [blame] | 655 | char_predicate allow_unencoded_fn); |
René Scharfe | 679eebe | 2014-07-28 20:33:55 +0200 | [diff] [blame] | 656 | |
Nguyễn Thái Ngọc Duy | 9a0a30a | 2012-04-23 19:30:22 +0700 | [diff] [blame] | 657 | __attribute__((format (printf,1,2))) |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 658 | int printf_ln(const char *fmt, ...); |
Nguyễn Thái Ngọc Duy | 9a0a30a | 2012-04-23 19:30:22 +0700 | [diff] [blame] | 659 | __attribute__((format (printf,2,3))) |
Stefan Beller | c7e5fe7 | 2018-09-28 10:30:33 -0700 | [diff] [blame] | 660 | int fprintf_ln(FILE *fp, const char *fmt, ...); |
Nguyễn Thái Ngọc Duy | 9a0a30a | 2012-04-23 19:30:22 +0700 | [diff] [blame] | 661 | |
Jeff King | 88d5a6f | 2014-05-22 05:44:09 -0400 | [diff] [blame] | 662 | char *xstrdup_tolower(const char *); |
Lars Schneider | 13ecb463 | 2018-02-15 16:27:06 +0100 | [diff] [blame] | 663 | char *xstrdup_toupper(const char *); |
Jeff King | 88d5a6f | 2014-05-22 05:44:09 -0400 | [diff] [blame] | 664 | |
Stefan Beller | 6afbbdd | 2015-01-16 04:04:51 -0500 | [diff] [blame] | 665 | /** |
Jeff King | 30a0ddb | 2014-06-18 16:01:34 -0400 | [diff] [blame] | 666 | * Create a newly allocated string using printf format. You can do this easily |
| 667 | * with a strbuf, but this provides a shortcut to save a few lines. |
| 668 | */ |
| 669 | __attribute__((format (printf, 1, 0))) |
| 670 | char *xstrvfmt(const char *fmt, va_list ap); |
| 671 | __attribute__((format (printf, 1, 2))) |
| 672 | char *xstrfmt(const char *fmt, ...); |
| 673 | |
Calvin Wan | fda5d95 | 2023-07-05 17:09:19 +0000 | [diff] [blame] | 674 | int starts_with(const char *str, const char *prefix); |
| 675 | int istarts_with(const char *str, const char *prefix); |
| 676 | |
| 677 | /* |
| 678 | * If the string "str" is the same as the string in "prefix", then the "arg" |
| 679 | * parameter is set to the "def" parameter and 1 is returned. |
| 680 | * If the string "str" begins with the string found in "prefix" and then a |
| 681 | * "=" sign, then the "arg" parameter is set to "str + strlen(prefix) + 1" |
| 682 | * (i.e., to the point in the string right after the prefix and the "=" sign), |
| 683 | * and 1 is returned. |
| 684 | * |
| 685 | * Otherwise, return 0 and leave "arg" untouched. |
| 686 | * |
| 687 | * When we accept both a "--key" and a "--key=<val>" option, this function |
| 688 | * can be used instead of !strcmp(arg, "--key") and then |
| 689 | * skip_prefix(arg, "--key=", &arg) to parse such an option. |
| 690 | */ |
| 691 | int skip_to_optional_arg_default(const char *str, const char *prefix, |
| 692 | const char **arg, const char *def); |
| 693 | |
| 694 | static inline int skip_to_optional_arg(const char *str, const char *prefix, |
| 695 | const char **arg) |
| 696 | { |
| 697 | return skip_to_optional_arg_default(str, prefix, arg, ""); |
| 698 | } |
| 699 | |
| 700 | static inline int ends_with(const char *str, const char *suffix) |
| 701 | { |
| 702 | size_t len; |
| 703 | return strip_suffix(str, suffix, &len); |
| 704 | } |
| 705 | |
Junio C Hamano | d1df574 | 2005-04-25 18:26:45 -0700 | [diff] [blame] | 706 | #endif /* STRBUF_H */ |