| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250 |
- /*
- Bullet Continuous Collision Detection and Physics Library
- Copyright (c) 2003-2012 Erwin Coumans http://bulletphysics.org
- This software is provided 'as-is', without any express or implied warranty.
- In no event will the authors be held liable for any damages arising from the use of this software.
- Permission is granted to anyone to use this software for any purpose,
- including commercial applications, and to alter it and redistribute it freely,
- subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source distribution.
- */
- #include <assert.h>
- //#include <stdbool.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include "string_split.h"
- ///todo: remove stl dependency
- namespace bullet_utils
- {
- void split( btAlignedObjectArray<std::string>&pieces, const std::string& vector_str, const std::string& separator)
- {
- char** strArray = str_split(vector_str.c_str(),separator.c_str());
- int numSubStr = str_array_len(strArray);
- for (int i=0;i<numSubStr;i++)
- pieces.push_back(std::string(strArray[i]));
- str_array_free(strArray);
- }
- };
- /* Append an item to a dynamically allocated array of strings. On failure,
- return NULL, in which case the original array is intact. The item
- string is dynamically copied. If the array is NULL, allocate a new
- array. Otherwise, extend the array. Make sure the array is always
- NULL-terminated. Input string might not be '\0'-terminated. */
- char **str_array_append(char **array, size_t nitems, const char *item,
- size_t itemlen)
- {
- /* Make a dynamic copy of the item. */
- char *copy;
- if (item == NULL)
- copy = NULL;
- else {
- copy = (char*)malloc(itemlen + 1);
- if (copy == NULL)
- return NULL;
- memcpy(copy, item, itemlen);
- copy[itemlen] = '\0';
- }
-
- /* Extend array with one element. Except extend it by two elements,
- in case it did not yet exist. This might mean it is a teeny bit
- too big, but we don't care. */
- array = (char**)realloc(array, (nitems + 2) * sizeof(array[0]));
- if (array == NULL) {
- free(copy);
- return NULL;
- }
-
- /* Add copy of item to array, and return it. */
- array[nitems] = copy;
- array[nitems+1] = NULL;
- return array;
- }
- /* Free a dynamic array of dynamic strings. */
- void str_array_free(char **array)
- {
- if (array == NULL)
- return;
- for (size_t i = 0; array[i] != NULL; ++i)
- free(array[i]);
- free(array);
- }
- /* Split a string into substrings. Return dynamic array of dynamically
- allocated substrings, or NULL if there was an error. Caller is
- expected to free the memory, for example with str_array_free. */
- char **str_split(const char *input, const char *sep)
- {
- size_t nitems = 0;
- char **array = NULL;
- const char *start = input;
- const char *next = strstr(start, sep);
- size_t seplen = strlen(sep);
- const char *item;
- size_t itemlen;
-
- for (;;) {
- next = strstr(start, sep);
- if (next == NULL) {
- /* Add the remaining string (or empty string, if input ends with
- separator. */
- char **newstr = str_array_append(array, nitems, start, strlen(start));
- if (newstr == NULL) {
- str_array_free(array);
- return NULL;
- }
- array = newstr;
- ++nitems;
- break;
- } else if (next == input) {
- /* Input starts with separator. */
- item = "";
- itemlen = 0;
- } else {
- item = start;
- itemlen = next - item;
- }
- char **newstr = str_array_append(array, nitems, item, itemlen);
- if (newstr == NULL) {
- str_array_free(array);
- return NULL;
- }
- array = newstr;
- ++nitems;
- start = next + seplen;
- }
-
- if (nitems == 0) {
- /* Input does not contain separator at all. */
- assert(array == NULL);
- array = str_array_append(array, nitems, input, strlen(input));
- }
-
- return array;
- }
- /* Return length of a NULL-delimited array of strings. */
- size_t str_array_len(char **array)
- {
- size_t len;
-
- for (len = 0; array[len] != NULL; ++len)
- continue;
- return len;
- }
- #ifdef UNIT_TEST_STRING
- #define MAX_OUTPUT 20
- int main(void)
- {
- struct {
- const char *input;
- const char *sep;
- char *output[MAX_OUTPUT];
- } tab[] = {
- /* Input is empty string. Output should be a list with an empty
- string. */
- {
- "",
- "and",
- {
- "",
- NULL,
- },
- },
- /* Input is exactly the separator. Output should be two empty
- strings. */
- {
- "and",
- "and",
- {
- "",
- "",
- NULL,
- },
- },
- /* Input is non-empty, but does not have separator. Output should
- be the same string. */
- {
- "foo",
- "and",
- {
- "foo",
- NULL,
- },
- },
- /* Input is non-empty, and does have separator. */
- {
- "foo bar 1 and foo bar 2",
- " and ",
- {
- "foo bar 1",
- "foo bar 2",
- NULL,
- },
- },
- };
- const int tab_len = sizeof(tab) / sizeof(tab[0]);
- bool errors;
-
- errors = false;
-
- for (int i = 0; i < tab_len; ++i) {
- printf("test %d\n", i);
-
- char **output = str_split(tab[i].input, tab[i].sep);
- if (output == NULL) {
- fprintf(stderr, "output is NULL\n");
- errors = true;
- break;
- }
- size_t num_output = str_array_len(output);
- printf("num_output %lu\n", (unsigned long) num_output);
-
- size_t num_correct = str_array_len(tab[i].output);
- if (num_output != num_correct) {
- fprintf(stderr, "wrong number of outputs (%lu, not %lu)\n",
- (unsigned long) num_output, (unsigned long) num_correct);
- errors = true;
- } else {
- for (size_t j = 0; j < num_output; ++j) {
- if (strcmp(tab[i].output[j], output[j]) != 0) {
- fprintf(stderr, "output[%lu] is '%s' not '%s'\n",
- (unsigned long) j, output[j], tab[i].output[j]);
- errors = true;
- break;
- }
- }
- }
-
- str_array_free(output);
- printf("\n");
- }
-
- if (errors)
- return EXIT_FAILURE;
- return 0;
- }
- #endif//
|