| #include <lib9.h> | 
 |  | 
 | static char qsep[] = " \t\r\n"; | 
 |  | 
 | static char* | 
 | qtoken(char *s, char *sep) | 
 | { | 
 | 	int quoting; | 
 | 	char *t; | 
 |  | 
 | 	quoting = 0; | 
 | 	t = s;	/* s is output string, t is input string */ | 
 | 	while(*t!='\0' && (quoting || utfrune(sep, *t)==nil)){ | 
 | 		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 || utfrune(sep, *t)==nil)){ | 
 | 		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' && utfrune(sep, *s)!=nil) | 
 | 			*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' && utfrune(qsep, *s)!=nil) | 
 | 			s++; | 
 | 		if(*s == '\0') | 
 | 			break; | 
 | 		args[nargs] = s; | 
 | 		s = qtoken(s, qsep); | 
 | 	} | 
 |  | 
 | 	return nargs; | 
 | } |