When using 'perf record's option '-I' or '--user-regs=' along with
argument '?' to list available register names, memory of variable 'os'
allocated by strdup() needs to be released before __parse_regs()
returns, otherwise memory leak will occur.
Fixes: bcc84ec65a ("perf record: Add ability to name registers to record")
Signed-off-by: Zheng Zengkai <zhengzengkai@huawei.com>
Acked-by: Jiri Olsa <jolsa@redhat.com>
Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com>
Cc: Li Bin <huawei.libin@huawei.com>
Cc: Mark Rutland <mark.rutland@arm.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Link: https://lore.kernel.org/r/20200703093344.189450-1-zhengzengkai@huawei.com
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
		
	
			
		
			
				
	
	
		
			100 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			100 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| #include <stdbool.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdint.h>
 | |
| #include <string.h>
 | |
| #include <stdio.h>
 | |
| #include "util/debug.h"
 | |
| #include <subcmd/parse-options.h>
 | |
| #include "util/perf_regs.h"
 | |
| #include "util/parse-regs-options.h"
 | |
| 
 | |
| static int
 | |
| __parse_regs(const struct option *opt, const char *str, int unset, bool intr)
 | |
| {
 | |
| 	uint64_t *mode = (uint64_t *)opt->value;
 | |
| 	const struct sample_reg *r = NULL;
 | |
| 	char *s, *os = NULL, *p;
 | |
| 	int ret = -1;
 | |
| 	uint64_t mask;
 | |
| 
 | |
| 	if (unset)
 | |
| 		return 0;
 | |
| 
 | |
| 	/*
 | |
| 	 * cannot set it twice
 | |
| 	 */
 | |
| 	if (*mode)
 | |
| 		return -1;
 | |
| 
 | |
| 	if (intr)
 | |
| 		mask = arch__intr_reg_mask();
 | |
| 	else
 | |
| 		mask = arch__user_reg_mask();
 | |
| 
 | |
| 	/* str may be NULL in case no arg is passed to -I */
 | |
| 	if (str) {
 | |
| 		/* because str is read-only */
 | |
| 		s = os = strdup(str);
 | |
| 		if (!s)
 | |
| 			return -1;
 | |
| 
 | |
| 		for (;;) {
 | |
| 			p = strchr(s, ',');
 | |
| 			if (p)
 | |
| 				*p = '\0';
 | |
| 
 | |
| 			if (!strcmp(s, "?")) {
 | |
| 				fprintf(stderr, "available registers: ");
 | |
| #ifdef HAVE_PERF_REGS_SUPPORT
 | |
| 				for (r = sample_reg_masks; r->name; r++) {
 | |
| 					if (r->mask & mask)
 | |
| 						fprintf(stderr, "%s ", r->name);
 | |
| 				}
 | |
| #endif
 | |
| 				fputc('\n', stderr);
 | |
| 				/* just printing available regs */
 | |
| 				goto error;
 | |
| 			}
 | |
| #ifdef HAVE_PERF_REGS_SUPPORT
 | |
| 			for (r = sample_reg_masks; r->name; r++) {
 | |
| 				if ((r->mask & mask) && !strcasecmp(s, r->name))
 | |
| 					break;
 | |
| 			}
 | |
| #endif
 | |
| 			if (!r || !r->name) {
 | |
| 				ui__warning("Unknown register \"%s\", check man page or run \"perf record %s?\"\n",
 | |
| 					    s, intr ? "-I" : "--user-regs=");
 | |
| 				goto error;
 | |
| 			}
 | |
| 
 | |
| 			*mode |= r->mask;
 | |
| 
 | |
| 			if (!p)
 | |
| 				break;
 | |
| 
 | |
| 			s = p + 1;
 | |
| 		}
 | |
| 	}
 | |
| 	ret = 0;
 | |
| 
 | |
| 	/* default to all possible regs */
 | |
| 	if (*mode == 0)
 | |
| 		*mode = mask;
 | |
| error:
 | |
| 	free(os);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| int
 | |
| parse_user_regs(const struct option *opt, const char *str, int unset)
 | |
| {
 | |
| 	return __parse_regs(opt, str, unset, false);
 | |
| }
 | |
| 
 | |
| int
 | |
| parse_intr_regs(const struct option *opt, const char *str, int unset)
 | |
| {
 | |
| 	return __parse_regs(opt, str, unset, true);
 | |
| }
 |