| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * A general argument parsing module | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-03-21 01:14:04 +00:00
										 |  |  |  * $Id$ | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is: all of this file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): none yet. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <ctype.h> /* for tolower */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | #include "BLI_listbase.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | #include "BLI_string.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | #include "BLI_args.h"
 | 
					
						
							|  |  |  | #include "BLI_ghash.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-13 10:52:18 +00:00
										 |  |  | static char NO_DOCS[] = "NO DOCUMENTATION SPECIFIED"; | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct bArgDoc; | 
					
						
							|  |  |  | typedef struct bArgDoc { | 
					
						
							|  |  |  | 	struct bArgDoc *next, *prev; | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | 	const char *short_arg; | 
					
						
							|  |  |  | 	const char *long_arg; | 
					
						
							|  |  |  | 	const char *documentation; | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	int  done; | 
					
						
							|  |  |  | } bArgDoc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | typedef struct bAKey { | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | 	const char *arg; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	uintptr_t pass; /* cast easier */ | 
					
						
							|  |  |  | 	int case_str; /* case specific or not */ | 
					
						
							|  |  |  | } bAKey; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct bArgument { | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | 	bAKey *key; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	BA_ArgCallback func; | 
					
						
							|  |  |  | 	void *data; | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	bArgDoc *doc; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | } bArgument; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct bArgs { | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	ListBase docs; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	GHash  *items; | 
					
						
							|  |  |  | 	int 	argc; | 
					
						
							|  |  |  | 	char  **argv; | 
					
						
							|  |  |  | 	int	  *passes; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static unsigned int case_strhash(const void *ptr) { | 
					
						
							|  |  |  | 	const char *s= ptr; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	unsigned int i= 0; | 
					
						
							|  |  |  | 	unsigned char c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while ( (c= tolower(*s++)) ) | 
					
						
							|  |  |  | 		i= i*37 + c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return i; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static unsigned int	keyhash(const void *ptr) | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | 	const bAKey *k = ptr; | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	return case_strhash(k->arg); // ^ BLI_ghashutil_inthash((void*)k->pass);
 | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static int keycmp(const void *a, const void *b) | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | 	const bAKey *ka = a; | 
					
						
							|  |  |  | 	const bAKey *kb = b; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	if (ka->pass == kb->pass || ka->pass == -1 || kb->pass == -1) { /* -1 is wildcard for pass */ | 
					
						
							|  |  |  | 		if (ka->case_str == 1 || kb->case_str == 1) | 
					
						
							|  |  |  | 			return BLI_strcasecmp(ka->arg, kb->arg); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return strcmp(ka->arg, kb->arg); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | 		return BLI_ghashutil_intcmp((const void*)ka->pass, (const void*)kb->pass); | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static bArgument *lookUp(struct bArgs *ba, const char *arg, int pass, int case_str) | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bAKey key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	key.case_str = case_str; | 
					
						
							|  |  |  | 	key.pass = pass; | 
					
						
							|  |  |  | 	key.arg = arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return BLI_ghash_lookup(ba->items, &key); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | bArgs *BLI_argsInit(int argc, char **argv) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bArgs *ba = MEM_callocN(sizeof(bArgs), "bArgs"); | 
					
						
							|  |  |  | 	ba->passes = MEM_callocN(sizeof(int) * argc, "bArgs passes"); | 
					
						
							| 
									
										
										
										
											2010-05-07 07:54:25 +00:00
										 |  |  | 	ba->items = BLI_ghash_new(keyhash, keycmp, "bArgs passes gh"); | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	ba->docs.first = ba->docs.last = NULL; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	ba->argc = argc; | 
					
						
							|  |  |  | 	ba->argv = argv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ba; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void freeItem(void *val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MEM_freeN(val); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_argsFree(struct bArgs *ba) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_ghash_free(ba->items, freeItem, freeItem); | 
					
						
							|  |  |  | 	MEM_freeN(ba->passes); | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	BLI_freelistN(&ba->docs); | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	MEM_freeN(ba); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_argsPrint(struct bArgs *ba) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	for (i = 0; i < ba->argc; i++) { | 
					
						
							|  |  |  | 		printf("argv[%d] = %s\n", i, ba->argv[i]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char **BLI_argsArgv(struct bArgs *ba) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ba->argv; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static bArgDoc *internalDocs(struct bArgs *ba, const char *short_arg, const char *long_arg, const char *doc) | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bArgDoc *d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	d = MEM_callocN(sizeof(bArgDoc), "bArgDoc"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (doc == NULL) | 
					
						
							|  |  |  | 		doc = NO_DOCS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	d->short_arg = short_arg; | 
					
						
							|  |  |  | 	d->long_arg = long_arg; | 
					
						
							|  |  |  | 	d->documentation = doc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_addtail(&ba->docs, d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return d; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static void internalAdd(struct bArgs *ba, const char *arg, int pass, int case_str, BA_ArgCallback cb, void *data, bArgDoc *d) | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | 	bArgument *a; | 
					
						
							|  |  |  | 	bAKey *key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	a = lookUp(ba, arg, pass, case_str); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (a) { | 
					
						
							|  |  |  | 		printf("WARNING: conflicting argument\n"); | 
					
						
							|  |  |  | 		printf("\ttrying to add '%s' on pass %i, %scase sensitive\n", arg, pass, case_str == 1? "not ": ""); | 
					
						
							|  |  |  | 		printf("\tconflict with '%s' on pass %i, %scase sensitive\n\n", a->key->arg, (int)a->key->pass, a->key->case_str == 1? "not ": ""); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	a = MEM_callocN(sizeof(bArgument), "bArgument"); | 
					
						
							|  |  |  | 	key = MEM_callocN(sizeof(bAKey), "bAKey"); | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	key->arg = arg; | 
					
						
							|  |  |  | 	key->pass = pass; | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | 	key->case_str = case_str; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | 	a->key = key; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 	a->func = cb; | 
					
						
							|  |  |  | 	a->data = data; | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	a->doc = d; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BLI_ghash_insert(ba->items, key, a); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | void BLI_argsAddCase(struct bArgs *ba, int pass, const char *short_arg, int short_case, const char *long_arg, int long_case, const char *doc, BA_ArgCallback cb, void *data) | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	bArgDoc *d = internalDocs(ba, short_arg, long_arg, doc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (short_arg) | 
					
						
							|  |  |  | 		internalAdd(ba, short_arg, pass, short_case, cb, data, d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (long_arg) | 
					
						
							|  |  |  | 		internalAdd(ba, long_arg, pass, long_case, cb, data, d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | void BLI_argsAdd(struct bArgs *ba, int pass, const char *short_arg, const char *long_arg, const char *doc, BA_ArgCallback cb, void *data) | 
					
						
							| 
									
										
										
										
											2010-05-23 20:39:21 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	BLI_argsAddCase(ba, pass, short_arg, 0, long_arg, 0, doc, cb, data); | 
					
						
							| 
									
										
										
										
											2010-05-23 20:39:21 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | static void internalDocPrint(bArgDoc *d) | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | 	if (d->short_arg && d->long_arg) | 
					
						
							|  |  |  | 		printf("%s or %s", d->short_arg, d->long_arg); | 
					
						
							|  |  |  | 	else if (d->short_arg) | 
					
						
							|  |  |  | 		printf("%s", d->short_arg); | 
					
						
							|  |  |  | 	else if (d->long_arg) | 
					
						
							|  |  |  | 		printf("%s", d->long_arg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	printf(" %s\n\n", d->documentation); | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | void BLI_argsPrintArgDoc(struct bArgs *ba, const char *arg) | 
					
						
							| 
									
										
										
										
											2010-05-24 18:53:45 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bArgument *a = lookUp(ba, arg, -1, -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (a) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		bArgDoc *d = a->doc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internalDocPrint(d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		d->done = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_argsPrintOtherDoc(struct bArgs *ba) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bArgDoc *d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for( d = ba->docs.first; d; d = d->next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		if (d->done == 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			internalDocPrint(d); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void BLI_argsParse(struct bArgs *ba, int pass, BA_ArgCallback default_cb, void *default_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for( i = 1; i < ba->argc; i++) { /* skip argv[0] */ | 
					
						
							|  |  |  | 		if (ba->passes[i] == 0) { | 
					
						
							| 
									
										
										
										
											2010-02-07 15:36:20 +00:00
										 |  |  | 			 /* -1 signal what side of the comparison it is */ | 
					
						
							|  |  |  | 			bArgument *a = lookUp(ba, ba->argv[i], pass, -1); | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 			BA_ArgCallback func = NULL; | 
					
						
							|  |  |  | 			void *data = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (a) { | 
					
						
							|  |  |  | 				func = a->func; | 
					
						
							|  |  |  | 				data = a->data; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				func = default_cb; | 
					
						
							|  |  |  | 				data = default_data; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (func) { | 
					
						
							|  |  |  | 				int retval = func(ba->argc - i, ba->argv + i, data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (retval >= 0) { | 
					
						
							|  |  |  | 					int j; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					/* use extra arguments */ | 
					
						
							|  |  |  | 					for (j = 0; j <= retval; j++) { | 
					
						
							|  |  |  | 						ba->passes[i + j] = pass; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					i += retval; | 
					
						
							|  |  |  | 				} else if (retval == -1){ | 
					
						
							| 
									
										
										
										
											2010-06-30 03:19:28 +00:00
										 |  |  | 					if (a->key->pass != -1) | 
					
						
							|  |  |  | 						ba->passes[i] = pass; | 
					
						
							| 
									
										
										
										
											2010-02-01 01:43:31 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |