|  | /* | 
|  | * 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; | 
|  | } |