|  |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright © 2012 Kristian Høgsberg
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining
 | 
					
						
							|  |  |  |  * a copy of this software and associated documentation files (the
 | 
					
						
							|  |  |  |  * "Software"), to deal in the Software without restriction, including
 | 
					
						
							|  |  |  |  * without limitation the rights to use, copy, modify, merge, publish,
 | 
					
						
							|  |  |  |  * distribute, sublicense, and/or sell copies of the Software, and to
 | 
					
						
							|  |  |  |  * permit persons to whom the Software is furnished to do so, subject to
 | 
					
						
							|  |  |  |  * the following conditions:
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice (including the
 | 
					
						
							|  |  |  |  * next paragraph) shall be included in all copies or substantial
 | 
					
						
							|  |  |  |  * portions of the Software.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 | 
					
						
							|  |  |  |  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 | 
					
						
							|  |  |  |  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 | 
					
						
							|  |  |  |  * NONINFRINGEMENT.  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
 | 
					
						
							|  |  |  |  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
 | 
					
						
							|  |  |  |  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 | 
					
						
							|  |  |  |  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | 
					
						
							|  |  |  |  * SOFTWARE.
 | 
					
						
							|  |  |  |  */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "config.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <stdint.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "config-parser.h"
 | 
					
						
							|  |  |  | #include "string-helpers.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int
 | 
					
						
							|  |  |  | handle_option(const struct weston_option *option, char *value)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	char* p;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (option->type) {
 | 
					
						
							|  |  |  | 	case WESTON_OPTION_INTEGER:
 | 
					
						
							|  |  |  | 		if (!safe_strtoint(value, option->data))
 | 
					
						
							|  |  |  | 			return 0;
 | 
					
						
							|  |  |  | 		return 1;
 | 
					
						
							|  |  |  | 	case WESTON_OPTION_UNSIGNED_INTEGER:
 | 
					
						
							|  |  |  | 		errno = 0;
 | 
					
						
							| 
									
										
											  
											
												option-parser: Require integer option string values to be base-10
The third arg to strtol() specifies the base to assume for the number.
When 0 is passed, as is currently done in option-parser.c, hexadecimal
and octal numbers are permitted and automatically detected and
converted.
In weston and the weston clients and tests using option-parser.c, the
options are all things that can be expected to be specified in base 10:
widths, heights, counts, scales, font sizes, ports, ttys, connectors,
etc.  The subsurfaces client uses two modes, limited to values 0 and 1
only.  The zuc testsuite has a --random parameter for specifying a seed,
which is the only option where using hexadecimal or octal numbers might
conceivably happen.
The benefit of limiting this to base-10 is to eliminate surprises when
parsing numbers from the command line.  Also, by making the code
consistent with other usages of strtol/strtoul, it may make it possible
to factor out the common code in the future.
Signed-off-by: Bryce Harrington <bryce@osg.samsung.com>
											
										 
											9 years ago
										 |  |  | 		* (uint32_t *) option->data = strtoul(value, &p, 10);
 | 
					
						
							|  |  |  | 		if (errno != 0 || p == value || *p != '\0')
 | 
					
						
							|  |  |  | 			return 0;
 | 
					
						
							|  |  |  | 		return 1;
 | 
					
						
							|  |  |  | 	case WESTON_OPTION_STRING:
 | 
					
						
							|  |  |  | 		* (char **) option->data = strdup(value);
 | 
					
						
							|  |  |  | 		return 1;
 | 
					
						
							|  |  |  | 	default:
 | 
					
						
							|  |  |  | 		assert(0);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int
 | 
					
						
							|  |  |  | long_option(const struct weston_option *options, int count, char *arg)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	int k, len;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (k = 0; k < count; k++) {
 | 
					
						
							|  |  |  | 		if (!options[k].name)
 | 
					
						
							|  |  |  | 			continue;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		len = strlen(options[k].name);
 | 
					
						
							|  |  |  | 		if (strncmp(options[k].name, arg + 2, len) != 0)
 | 
					
						
							|  |  |  | 			continue;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (options[k].type == WESTON_OPTION_BOOLEAN) {
 | 
					
						
							|  |  |  | 			if (!arg[len + 2]) {
 | 
					
						
							|  |  |  | 				* (int32_t *) options[k].data = 1;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				return 1;
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		} else if (arg[len+2] == '=') {
 | 
					
						
							|  |  |  | 			return handle_option(options + k, arg + len + 3);
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int
 | 
					
						
							|  |  |  | short_option(const struct weston_option *options, int count, char *arg)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	int k;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!arg[1])
 | 
					
						
							|  |  |  | 		return 0;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (k = 0; k < count; k++) {
 | 
					
						
							|  |  |  | 		if (options[k].short_name != arg[1])
 | 
					
						
							|  |  |  | 			continue;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (options[k].type == WESTON_OPTION_BOOLEAN) {
 | 
					
						
							|  |  |  | 			if (!arg[2]) {
 | 
					
						
							|  |  |  | 				* (int32_t *) options[k].data = 1;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				return 1;
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		} else if (arg[2]) {
 | 
					
						
							|  |  |  | 			return handle_option(options + k, arg + 2);
 | 
					
						
							|  |  |  | 		} else {
 | 
					
						
							|  |  |  | 			return 0;
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int
 | 
					
						
							|  |  |  | short_option_with_arg(const struct weston_option *options, int count, char *arg, char *param)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	int k;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!arg[1])
 | 
					
						
							|  |  |  | 		return 0;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (k = 0; k < count; k++) {
 | 
					
						
							|  |  |  | 		if (options[k].short_name != arg[1])
 | 
					
						
							|  |  |  | 			continue;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (options[k].type == WESTON_OPTION_BOOLEAN)
 | 
					
						
							|  |  |  | 			continue;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return handle_option(options + k, param);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int
 | 
					
						
							|  |  |  | parse_options(const struct weston_option *options,
 | 
					
						
							|  |  |  | 	      int count, int *argc, char *argv[])
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	int i, j;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 1, j = 1; i < *argc; i++) {
 | 
					
						
							|  |  |  | 		if (argv[i][0] == '-') {
 | 
					
						
							|  |  |  | 			if (argv[i][1] == '-') {
 | 
					
						
							|  |  |  | 				/* Long option, e.g. --foo or --foo=bar */
 | 
					
						
							|  |  |  | 				if (long_option(options, count, argv[i]))
 | 
					
						
							|  |  |  | 					continue;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			} else {
 | 
					
						
							|  |  |  | 				/* Short option, e.g -f or -f42 */
 | 
					
						
							|  |  |  | 				if (short_option(options, count, argv[i]))
 | 
					
						
							|  |  |  | 					continue;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* ...also handle -f 42 */
 | 
					
						
							|  |  |  | 				if (i+1 < *argc &&
 | 
					
						
							|  |  |  | 				    short_option_with_arg(options, count, argv[i], argv[i+1])) {
 | 
					
						
							|  |  |  | 					i++;
 | 
					
						
							|  |  |  | 					continue;
 | 
					
						
							|  |  |  | 				}
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		argv[j++] = argv[i];
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	argv[j] = NULL;
 | 
					
						
							|  |  |  | 	*argc = j;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return j;
 | 
					
						
							|  |  |  | }
 |