Path: blob/main/crypto/krb5/src/util/profile/argv_parse.c
34889 views
/* -*- mode: c; c-basic-offset: 4; indent-tabs-mode: nil -*- */1/*2* Copyright 1999 by Theodore Ts'o.3*4* Permission to use, copy, modify, and distribute this software for5* any purpose with or without fee is hereby granted, provided that6* the above copyright notice and this permission notice appear in all7* copies. THE SOFTWARE IS PROVIDED "AS IS" AND THEODORE TS'O (THE8* AUTHOR) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,9* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS.10* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,11* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER12* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION13* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR14* IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. (Isn't15* it sick that the U.S. culture of lawsuit-happy lawyers requires16* this kind of disclaimer?)17*/1819/*20* Version 1.1, modified 2/27/199921*22* argv_parse.c --- utility function for parsing a string into a23* argc, argv array.24*25* This file defines a function argv_parse() which parsing a26* passed-in string, handling double quotes and backslashes, and27* creates an allocated argv vector which can be freed using the28* argv_free() function.29*30* See argv_parse.h for the formal definition of the functions.31*/3233#include "prof_int.h"3435#ifdef HAVE_STDLIB_H36#include <stdlib.h>37#endif38#include <ctype.h>39#include <string.h>40#include "argv_parse.h"4142#define STATE_WHITESPACE 143#define STATE_TOKEN 244#define STATE_QUOTED 34546/*47* Returns 0 on success, -1 on failure.48*/49int argv_parse(char *in_buf, int *ret_argc, char ***ret_argv)50{51int argc = 0, max_argc = 0;52char **argv, **new_argv, *buf, ch;53char *cp = 0, *outcp = 0;54int state = STATE_WHITESPACE;5556buf = malloc(strlen(in_buf)+1);57if (!buf)58return -1;5960max_argc = 0; argc = 0; argv = 0;61outcp = buf;62for (cp = in_buf; (ch = *cp); cp++) {63if (state == STATE_WHITESPACE) {64if (isspace((int) ch))65continue;66/* Not whitespace, so start a new token */67state = STATE_TOKEN;68if (argc >= max_argc) {69max_argc += 3;70new_argv = realloc(argv,71(max_argc+1)*sizeof(char *));72if (!new_argv) {73if (argv) free(argv);74free(buf);75return -1;76}77argv = new_argv;78}79argv[argc++] = outcp;80}81if (state == STATE_QUOTED) {82if (ch == '"')83state = STATE_TOKEN;84else85*outcp++ = ch;86continue;87}88/* Must be processing characters in a word */89if (isspace((int) ch)) {90/*91* Terminate the current word and start92* looking for the beginning of the next word.93*/94*outcp++ = 0;95state = STATE_WHITESPACE;96continue;97}98if (ch == '"') {99state = STATE_QUOTED;100continue;101}102if (ch == '\\') {103ch = *++cp;104switch (ch) {105case '\0':106ch = '\\'; cp--; break;107case 'n':108ch = '\n'; break;109case 't':110ch = '\t'; break;111case 'b':112ch = '\b'; break;113}114}115*outcp++ = ch;116}117if (state != STATE_WHITESPACE)118*outcp++ = '\0';119if (argv == 0) {120argv = malloc(sizeof(char *));121free(buf);122}123argv[argc] = 0;124if (ret_argc)125*ret_argc = argc;126if (ret_argv)127*ret_argv = argv;128return 0;129}130131void argv_free(char **argv)132{133if (*argv)134free(*argv);135free(argv);136}137138#ifdef DEBUG139/*140* For debugging141*/142143#include <stdio.h>144145int main(int argc, char **argv)146{147int ac, ret;148char **av, **cpp;149char buf[256];150151while (!feof(stdin)) {152if (fgets(buf, sizeof(buf), stdin) == NULL)153break;154ret = argv_parse(buf, &ac, &av);155if (ret != 0) {156printf("Argv_parse returned %d!\n", ret);157continue;158}159printf("Argv_parse returned %d arguments...\n", ac);160for (cpp = av; *cpp; cpp++) {161if (cpp != av)162printf(", ");163printf("'%s'", *cpp);164}165printf("\n");166argv_free(av);167}168exit(0);169}170#endif /* DEBUG */171172173