lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <538E8F38.2070706@hitachi.com>
Date:	Wed, 04 Jun 2014 12:15:04 +0900
From:	Yoshihiro YUNOMAE <yoshihiro.yunomae.ez@...achi.com>
To:	Steven Rostedt <rostedt@...dmis.org>
Cc:	linux-kernel@...r.kernel.org,
	Hidehiro Kawai <hidehiro.kawai.ez@...achi.com>,
	Frederic Weisbecker <fweisbec@...il.com>,
	Masami Hiramatsu <masami.hiramatsu.pt@...achi.com>,
	Ingo Molnar <mingo@...hat.com>, yrl.pp-manager.tt@...achi.com
Subject: Re: Re: [PATCH ftrace/core 2/2] ftrace: Introduce saved_cmdlines_size
 file

Hi Steven,

Thank you for your review.

(2014/06/04 9:30), Steven Rostedt wrote:
> On Tue, 03 Jun 2014 13:28:05 +0900
> Yoshihiro YUNOMAE <yoshihiro.yunomae.ez@...achi.com> wrote:
>
>> ---
>>   kernel/trace/trace.c |  203 ++++++++++++++++++++++++++++++++++++++++++++------
>>   1 file changed, 180 insertions(+), 23 deletions(-)
>>
>> diff --git a/kernel/trace/trace.c b/kernel/trace/trace.c
>> index 135af32..473eb68 100644
>> --- a/kernel/trace/trace.c
>> +++ b/kernel/trace/trace.c
>> @@ -1285,22 +1285,82 @@ void tracing_reset_all_online_cpus(void)
>>   	}
>>   }
>>
>> -#define SAVED_CMDLINES 128
>> +#define SAVED_CMDLINES_DEFAULT 128
>>   #define NO_CMDLINE_MAP UINT_MAX
>> -static unsigned map_pid_to_cmdline[PID_MAX_DEFAULT+1];
>> -static unsigned map_cmdline_to_pid[SAVED_CMDLINES];
>> -static char saved_cmdlines[SAVED_CMDLINES][TASK_COMM_LEN];
>> -static int cmdline_idx;
>>   static arch_spinlock_t trace_cmdline_lock = __ARCH_SPIN_LOCK_UNLOCKED;
>> +struct saved_cmdlines_buffer {
>> +	unsigned map_pid_to_cmdline[PID_MAX_DEFAULT+1];
>> +	unsigned *map_cmdline_to_pid;
>> +	unsigned cmdline_num;
>> +	int cmdline_idx;
>> +	char *saved_cmdlines;
>> +};
>> +static struct saved_cmdlines_buffer *savedcmd;
>>
>>   /* temporary disable recording */
>>   static atomic_t trace_record_cmdline_disabled __read_mostly;
>>
>> -static void trace_init_cmdlines(void)
>> +static inline char *get_saved_cmdlines(int idx)
>> +{
>> +	return &savedcmd->saved_cmdlines[idx * TASK_COMM_LEN];
>> +}
>> +
>> +static inline void set_cmdline(int idx, const char *cmdline)
>> +{
>> +	memcpy(get_saved_cmdlines(idx), cmdline, TASK_COMM_LEN);
>> +}
>> +
>> +static int allocate_cmdlines_buffer(unsigned int val,
>> +				    struct saved_cmdlines_buffer *s)
>>   {
>> -	memset(&map_pid_to_cmdline, NO_CMDLINE_MAP, sizeof(map_pid_to_cmdline));
>> -	memset(&map_cmdline_to_pid, NO_CMDLINE_MAP, sizeof(map_cmdline_to_pid));
>> -	cmdline_idx = 0;
>> +	s->map_cmdline_to_pid = kmalloc(val * sizeof(unsigned), GFP_KERNEL);
>> +	if (!s->map_cmdline_to_pid)
>> +		goto out;
>> +
>> +	s->saved_cmdlines = kmalloc(val * TASK_COMM_LEN, GFP_KERNEL);
>> +	if (!s->saved_cmdlines)
>> +		goto out_free_map_cmdline_to_pid;
>> +
>> +	return 0;
>> +
>> +out_free_map_cmdline_to_pid:
>> +	kfree(s->map_cmdline_to_pid);
>> +out:
>> +	return -ENOMEM;
>> +}
>> +
>> +static void trace_init_cmdlines_buffer(unsigned int val,
>> +				       struct saved_cmdlines_buffer *s)
>> +{
>> +	s->cmdline_idx = 0;
>> +	s->cmdline_num = val;
>> +	memset(&s->map_pid_to_cmdline, NO_CMDLINE_MAP,
>> +	       sizeof(s->map_pid_to_cmdline));
>> +	memset(s->map_cmdline_to_pid, NO_CMDLINE_MAP,
>> +	       val * sizeof(*s->map_cmdline_to_pid));
>> +}
>
> Please combine the above two functions. Get rid of
> trace_init_cmdlines_buffer(). That is, move the contents of
> trace_init_cmdlines_buffer() into allocate_cmdline_buffer() just before
> the return 0. There's no reason that we need to have two functions, one
> to allocate and one to initialize it. The allocation can also
> initialize it.

OK, I'll move the initialization into allocate_cmdline_buffer().

>> +
>> +static int trace_create_savedcmd(void)
>> +{
>> +	int ret;
>> +
>> +	savedcmd = kmalloc(sizeof(struct saved_cmdlines_buffer), GFP_KERNEL);
>> +	if (!savedcmd)
>> +		goto out;
>
> 	if (!savedcmd)
> 		return -ENOMEM;
>
>> +
>> +	ret = allocate_cmdlines_buffer(SAVED_CMDLINES_DEFAULT, savedcmd);
>> +	if (ret < 0)
>> +		goto out_free;
>
> 	if (ret < 0) {
> 		kfree(savedcmd);
> 		savedcmd = NULL;
> 		return -ENOMEM;
> 	}
>
> This function is small enough that we don't need to use gotos to make
> it clean.

OK. I'll remove gotos.

>> +
>> +	trace_init_cmdlines_buffer(SAVED_CMDLINES_DEFAULT, savedcmd);
>> +
>> +	return 0;
>> +
>> +out_free:
>> +	kfree(savedcmd);
>> +	savedcmd = NULL;
>> +out:
>> +	return -ENOMEM;
>>   }
>>
>>   int is_tracing_stopped(void)
>> @@ -1457,9 +1517,9 @@ static int trace_save_cmdline(struct task_struct *tsk)
>>   	if (!arch_spin_trylock(&trace_cmdline_lock))
>>   		return 0;
>>
>> -	idx = map_pid_to_cmdline[tsk->pid];
>> +	idx = savedcmd->map_pid_to_cmdline[tsk->pid];
>>   	if (idx == NO_CMDLINE_MAP) {
>> -		idx = (cmdline_idx + 1) % SAVED_CMDLINES;
>> +		idx = (savedcmd->cmdline_idx + 1) % savedcmd->cmdline_num;
>>
>>   		/*
>>   		 * Check whether the cmdline buffer at idx has a pid
>> @@ -1467,17 +1527,17 @@ static int trace_save_cmdline(struct task_struct *tsk)
>>   		 * need to clear the map_pid_to_cmdline. Otherwise we
>>   		 * would read the new comm for the old pid.
>>   		 */
>> -		pid = map_cmdline_to_pid[idx];
>> +		pid = savedcmd->map_cmdline_to_pid[idx];
>>   		if (pid != NO_CMDLINE_MAP)
>> -			map_pid_to_cmdline[pid] = NO_CMDLINE_MAP;
>> +			savedcmd->map_pid_to_cmdline[pid] = NO_CMDLINE_MAP;
>>
>> -		map_cmdline_to_pid[idx] = tsk->pid;
>> -		map_pid_to_cmdline[tsk->pid] = idx;
>> +		savedcmd->map_cmdline_to_pid[idx] = tsk->pid;
>> +		savedcmd->map_pid_to_cmdline[tsk->pid] = idx;
>>
>> -		cmdline_idx = idx;
>> +		savedcmd->cmdline_idx = idx;
>>   	}
>>
>> -	memcpy(&saved_cmdlines[idx], tsk->comm, TASK_COMM_LEN);
>> +	set_cmdline(idx, tsk->comm);
>>
>>   	arch_spin_unlock(&trace_cmdline_lock);
>>
>> @@ -1503,9 +1563,9 @@ static void __trace_find_cmdline(int pid, char comm[])
>>   		return;
>>   	}
>>
>> -	map = map_pid_to_cmdline[pid];
>> +	map = savedcmd->map_pid_to_cmdline[pid];
>>   	if (map != NO_CMDLINE_MAP)
>> -		strcpy(comm, saved_cmdlines[map]);
>> +		strcpy(comm, get_saved_cmdlines(map));
>>   	else
>>   		strcpy(comm, "<...>");
>>   }
>> @@ -3593,6 +3653,7 @@ static const char readme_msg[] =
>>   	"  trace_options\t\t- Set format or modify how tracing happens\n"
>>   	"\t\t\t  Disable an option by adding a suffix 'no' to the\n"
>>   	"\t\t\t  option name\n"
>> +	"  saved_cmdlines_size\t- echo command number in here to store comm-pid list\n"
>>   #ifdef CONFIG_DYNAMIC_FTRACE
>>   	"\n  available_filter_functions - list of functions that can be filtered on\n"
>>   	"  set_ftrace_filter\t- echo function name in here to only trace these\n"
>> @@ -3715,7 +3776,8 @@ static void *saved_cmdlines_next(struct seq_file *m, void *v, loff_t *pos)
>>
>>   	(*pos)++;
>>
>> -	for (; ptr < &map_cmdline_to_pid[SAVED_CMDLINES]; ptr++) {
>> +	for (; ptr < &savedcmd->map_cmdline_to_pid[savedcmd->cmdline_num];
>> +	     ptr++) {
>>   		if (*ptr == -1 || *ptr == NO_CMDLINE_MAP)
>>   			continue;
>>
>> @@ -3733,7 +3795,7 @@ static void *saved_cmdlines_start(struct seq_file *m, loff_t *pos)
>>   	preempt_disable();
>>   	arch_spin_lock(&trace_cmdline_lock);
>>
>> -	v = &map_cmdline_to_pid[0];
>> +	v = &savedcmd->map_cmdline_to_pid[0];
>>   	while (l <= *pos) {
>>   		v = saved_cmdlines_next(m, v, &l);
>>   		if (!v)
>> @@ -3774,11 +3836,95 @@ static int tracing_saved_cmdlines_open(struct inode *inode, struct file *filp)
>>   	return seq_open(filp, &tracing_saved_cmdlines_seq_ops);
>>   }
>>
>> +static int tracing_saved_cmdlines_close(struct inode *inode, struct file *filp)
>> +{
>> +	return seq_release(inode, filp);
>> +}
>> +
>>   static const struct file_operations tracing_saved_cmdlines_fops = {
>>   	.open		= tracing_saved_cmdlines_open,
>>   	.read		= seq_read,
>>   	.llseek		= seq_lseek,
>> -	.release	= seq_release,
>> +	.release	= tracing_saved_cmdlines_close,
>
> Confused. tracing_saved_cmdlines_close() just does a seq_release()
> passing in the same arguments. Why this change? Is there some clean up
> missing here?
>
> Oh, with the updates in ftrace/core, you don't need it anymore. Remove
> that helper function and just call seq_release() via .release.

Oh, I forgot to fix here.
As you say, we should call just seq_release here.

>> +};
>> +
>> +static ssize_t
>> +tracing_saved_cmdlines_size_read(struct file *filp, char __user *ubuf,
>> +				 size_t cnt, loff_t *ppos)
>> +{
>> +	char buf[64];
>> +	int r;
>> +
>> +	arch_spin_lock(&trace_cmdline_lock);
>> +	r = sprintf(buf, "%u\n", savedcmd->cmdline_num);
>> +	arch_spin_unlock(&trace_cmdline_lock);
>> +
>> +	return simple_read_from_buffer(ubuf, cnt, ppos, buf, r);
>> +}
>> +
>> +static void free_saved_cmdlines_buffer(struct saved_cmdlines_buffer *s)
>> +{
>> +	kfree(s->saved_cmdlines);
>> +	kfree(s->map_cmdline_to_pid);
>> +	kfree(s);
>> +}
>> +
>> +static int tracing_resize_saved_cmdlines(unsigned int val)
>> +{
>> +	struct saved_cmdlines_buffer *s, *savedcmd_temp;
>> +	int err = -ENOMEM;
>
> You don't need err.
>
>> +
>> +	s = kmalloc(sizeof(struct saved_cmdlines_buffer), GFP_KERNEL);
>> +	if (!s)
> 		return -ENOMEM;
>
>> +		goto out;
>> +
>> +	if (allocate_cmdlines_buffer(val, s) < 0)
> 					{
> 		kfree(s);
> 		return -ENOMEM;
> 	}
>
> Again, this function is small enough not to need the extra gotos.

OK, I'll remove gotos.

>> +		goto out_free;
>> +
>> +	trace_init_cmdlines_buffer(val, s);
>> +
>> +	arch_spin_lock(&trace_cmdline_lock);
>> +	savedcmd_temp = savedcmd;
>> +	savedcmd = s;
>> +	arch_spin_unlock(&trace_cmdline_lock);
>> +	free_saved_cmdlines_buffer(savedcmd_temp);
>> +
>> +	return 0;
>> +
>> +out_free:
>> +	kfree(s);
>> +out:
>> +	return err;
>> +}
>> +
>> +static ssize_t
>> +tracing_saved_cmdlines_size_write(struct file *filp, const char __user *ubuf,
>> +				  size_t cnt, loff_t *ppos)
>> +{
>> +	unsigned long val;
>> +	int ret;
>> +
>> +	ret = kstrtoul_from_user(ubuf, cnt, 10, &val);
>> +	if (ret)
>> +		return ret;
>> +
>> +	/* must have at least 1 entry or less than PID_MAX_DEFAULT */
>> +	if (!val || val > PID_MAX_DEFAULT)
>> +		return -EINVAL;
>> +
>> +	ret = tracing_resize_saved_cmdlines((unsigned int)val);
>> +	if (ret < 0)
>> +		return ret;
>> +
>> +	*ppos += cnt;
>> +
>> +	return cnt;
>> +}
>> +
>> +static const struct file_operations tracing_saved_cmdlines_size_fops = {
>> +	.open		= tracing_open_generic,
>> +	.read		= tracing_saved_cmdlines_size_read,
>> +	.write		= tracing_saved_cmdlines_size_write,
>>   };
>>
>>   static ssize_t
>> @@ -6375,6 +6521,9 @@ static __init int tracer_init_debugfs(void)
>>   	trace_create_file("saved_cmdlines", 0444, d_tracer,
>>   			NULL, &tracing_saved_cmdlines_fops);
>>
>> +	trace_create_file("saved_cmdlines_size", 0644, d_tracer,
>> +			  NULL, &tracing_saved_cmdlines_size_fops);
>> +
>>   #ifdef CONFIG_DYNAMIC_FTRACE
>>   	trace_create_file("dyn_ftrace_total_info", 0444, d_tracer,
>>   			&ftrace_update_tot_cnt, &tracing_dyn_info_fops);
>> @@ -6621,7 +6770,8 @@ __init static int tracer_alloc_buffers(void)
>>   	if (global_trace.buffer_disabled)
>>   		tracing_off();
>>
>> -	trace_init_cmdlines();
>> +	if (trace_create_savedcmd() < 0)
>> +		goto out_free_trace_buffers;
>
> Actually, since the freeing of saved_cmdlines is easier, move the
> creation of it before the buffers, and if it fails, we don't need to
> worry about freeing the buffers. Then if the buffers fail, we just need
> to free the cmdlines.

OK, I'll move trace_create_savedcmd() before allocate_trace_buffers().

Thank you,
Yoshihiro YUNOMAE

-- 
Yoshihiro YUNOMAE
Software Platform Research Dept. Linux Technology Center
Hitachi, Ltd., Yokohama Research Laboratory
E-mail: yoshihiro.yunomae.ez@...achi.com


--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ