execute, spawn-pipe: Make multithread-safe on native Windows.
[gnulib.git] / lib / argv-iter.c
blob0632a7cc077a4b01dad9c505aea03a4a2539e27d
1 /* Iterate over arguments from argv or --files0-from=FILE
2 Copyright (C) 2008-2020 Free Software Foundation, Inc.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <https://www.gnu.org/licenses/>. */
17 /* Written by Jim Meyering. */
19 #include <config.h>
20 #include "argv-iter.h"
22 #include <stdlib.h>
23 #include <string.h>
25 struct argv_iterator
27 /* Test FP to determine whether in read-mode or argv-mode. */
28 /* file-mode: fp records position */
29 FILE *fp;
30 size_t item_idx;
31 char *tok;
32 size_t buf_len;
34 /* argv-mode: record just argv and current pointer */
35 char **arg_list;
36 char **p;
39 struct argv_iterator *
40 argv_iter_init_argv (char **argv)
42 struct argv_iterator *ai = malloc (sizeof *ai);
43 if (!ai)
44 return NULL;
45 ai->fp = NULL;
46 ai->arg_list = argv;
47 ai->p = argv;
48 return ai;
51 /* Initialize to read from the stream, FP.
52 The input is expected to contain a list of NUL-delimited tokens. */
53 struct argv_iterator *
54 argv_iter_init_stream (FILE *fp)
56 struct argv_iterator *ai = malloc (sizeof *ai);
57 if (!ai)
58 return NULL;
59 ai->fp = fp;
60 ai->tok = NULL;
61 ai->buf_len = 0;
63 ai->item_idx = 0;
64 ai->arg_list = NULL;
65 return ai;
68 char *
69 argv_iter (struct argv_iterator *ai, enum argv_iter_err *err)
71 if (ai->fp)
73 ssize_t len = getdelim (&ai->tok, &ai->buf_len, '\0', ai->fp);
74 if (len < 0)
76 *err = feof (ai->fp) ? AI_ERR_EOF : AI_ERR_READ;
77 return NULL;
80 *err = AI_ERR_OK;
81 ai->item_idx++;
82 return ai->tok;
84 else
86 if (*(ai->p) == NULL)
88 *err = AI_ERR_EOF;
89 return NULL;
91 else
93 *err = AI_ERR_OK;
94 return *(ai->p++);
99 size_t
100 argv_iter_n_args (struct argv_iterator const *ai)
102 return ai->fp ? ai->item_idx : ai->p - ai->arg_list;
105 void
106 argv_iter_free (struct argv_iterator *ai)
108 if (ai->fp)
109 free (ai->tok);
110 free (ai);