Merge branch 'jk/parse-options-concat'
authorJunio C Hamano <gitster@pobox.com>
Wed, 3 Aug 2016 22:10:25 +0000 (15:10 -0700)
committerJunio C Hamano <gitster@pobox.com>
Wed, 3 Aug 2016 22:10:25 +0000 (15:10 -0700)
Users of the parse_options_concat() API function need to allocate
extra slots in advance and fill them with OPT_END() when they want
to decide the set of supported options dynamically, which makes the
code error-prone and hard to read. This has been corrected by tweaking
the API to allocate and return a new copy of "struct option" array.

* jk/parse-options-concat:
parse_options: allocate a new array when concatenating

1  2 
parse-options-cb.c
diff --combined parse-options-cb.c
index ba5acf3111d809a3c77314fb2c0852854620cfae,2d875202cd8c8164b6604ea889ee633424fe3b55..9667bc75a08e8b64290f5f44c92b0bac35d1d0fa
@@@ -117,19 -117,24 +117,24 @@@ int parse_opt_tertiary(const struct opt
        return 0;
  }
  
int parse_options_concat(struct option *dst, size_t dst_size, struct option *src)
struct option *parse_options_concat(struct option *a, struct option *b)
  {
-       int i, j;
-       for (i = 0; i < dst_size; i++)
-               if (dst[i].type == OPTION_END)
-                       break;
-       for (j = 0; i < dst_size; i++, j++) {
-               dst[i] = src[j];
-               if (src[j].type == OPTION_END)
-                       return 0;
-       }
-       return -1;
+       struct option *ret;
+       size_t i, a_len = 0, b_len = 0;
+       for (i = 0; a[i].type != OPTION_END; i++)
+               a_len++;
+       for (i = 0; b[i].type != OPTION_END; i++)
+               b_len++;
+       ALLOC_ARRAY(ret, st_add3(a_len, b_len, 1));
+       for (i = 0; i < a_len; i++)
+               ret[i] = a[i];
+       for (i = 0; i < b_len; i++)
+               ret[a_len + i] = b[i];
+       ret[a_len + b_len] = b[b_len]; /* final OPTION_END */
+       return ret;
  }
  
  int parse_opt_string_list(const struct option *opt, const char *arg, int unset)
        if (!arg)
                return -1;
  
 -      string_list_append(v, xstrdup(arg));
 +      string_list_append(v, arg);
        return 0;
  }