| /* |
| * This file is part of the UCB release of Plan 9. It is subject to the license |
| * terms in the LICENSE file found in the top-level directory of this |
| * distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No |
| * part of the UCB release of Plan 9, including this file, may be copied, |
| * modified, propagated, or distributed except according to the terms contained |
| * in the LICENSE file. |
| */ |
| /* |
| |
| * Copyright 2013 Google Inc. |
| * Copyright (c) 1989-2003 by Lucent Technologies, Bell Laboratories. |
| */ |
| |
| #include <string.h> |
| static char qsep[] = " \t\r\n"; |
| |
| int sep(char *s) |
| { |
| if (strchr(qsep, *s)) |
| return 1; |
| return 0; |
| } |
| /* s is output string, t is input string. |
| * warning: modifies data in place. |
| */ |
| static char *qtoken(char *s, char *sep) |
| { |
| int quoting; |
| char *t; |
| |
| quoting = 0; |
| t = s; |
| while (*t != '\0' && (quoting || (strchr(sep, *t) == NULL))) { |
| if (*t != '\'') { |
| *s++ = *t++; |
| continue; |
| } |
| /* *t is a quote */ |
| if (!quoting) { |
| quoting = 1; |
| t++; |
| continue; |
| } |
| /* quoting and we're on a quote */ |
| if (t[1] != '\'') { |
| /* end of quoted section; absorb closing quote */ |
| t++; |
| quoting = 0; |
| continue; |
| } |
| /* doubled quote; fold one quote into two */ |
| t++; |
| *s++ = *t++; |
| } |
| if (*s != '\0') { |
| *s = '\0'; |
| if (t == s) |
| t++; |
| } |
| return t; |
| } |
| |
| static char *etoken(char *t, char *sep) |
| { |
| int quoting; |
| |
| /* move to end of next token */ |
| quoting = 0; |
| while (*t != '\0' && (quoting || (strchr(sep, *t) == NULL))) { |
| if (*t != '\'') { |
| t++; |
| continue; |
| } |
| /* *t is a quote */ |
| if (!quoting) { |
| quoting = 1; |
| t++; |
| continue; |
| } |
| /* quoting and we're on a quote */ |
| if (t[1] != '\'') { |
| /* end of quoted section; absorb closing quote */ |
| t++; |
| quoting = 0; |
| continue; |
| } |
| /* doubled quote; fold one quote into two */ |
| t += 2; |
| } |
| return t; |
| } |
| |
| int gettokens(char *s, char **args, int maxargs, char *sep) |
| { |
| int nargs; |
| |
| for (nargs = 0; nargs < maxargs; nargs++) { |
| while ((*s != '\0') && (strchr(sep, *s) != NULL)) |
| *s++ = '\0'; |
| if (*s == '\0') |
| break; |
| args[nargs] = s; |
| s = etoken(s, sep); |
| } |
| |
| return nargs; |
| } |
| |
| int tokenize(char *s, char **args, int maxargs) |
| { |
| int nargs; |
| |
| for (nargs = 0; nargs < maxargs; nargs++) { |
| while ((*s != '\0') && sep(s)) |
| s++; |
| if (*s == '\0') |
| break; |
| args[nargs] = s; |
| s = qtoken(s, qsep); |
| } |
| |
| return nargs; |
| } |