diff options
Diffstat (limited to 'kernel/trace/trace.c')
| -rw-r--r-- | kernel/trace/trace.c | 86 |
1 files changed, 71 insertions, 15 deletions
diff --git a/kernel/trace/trace.c b/kernel/trace/trace.c index c2e2c2310374..4f1dade56981 100644 --- a/kernel/trace/trace.c +++ b/kernel/trace/trace.c | |||
| @@ -704,7 +704,7 @@ __update_max_tr(struct trace_array *tr, struct task_struct *tsk, int cpu) | |||
| 704 | void | 704 | void |
| 705 | update_max_tr(struct trace_array *tr, struct task_struct *tsk, int cpu) | 705 | update_max_tr(struct trace_array *tr, struct task_struct *tsk, int cpu) |
| 706 | { | 706 | { |
| 707 | struct ring_buffer *buf = tr->buffer; | 707 | struct ring_buffer *buf; |
| 708 | 708 | ||
| 709 | if (trace_stop_count) | 709 | if (trace_stop_count) |
| 710 | return; | 710 | return; |
| @@ -719,6 +719,7 @@ update_max_tr(struct trace_array *tr, struct task_struct *tsk, int cpu) | |||
| 719 | 719 | ||
| 720 | arch_spin_lock(&ftrace_max_lock); | 720 | arch_spin_lock(&ftrace_max_lock); |
| 721 | 721 | ||
| 722 | buf = tr->buffer; | ||
| 722 | tr->buffer = max_tr.buffer; | 723 | tr->buffer = max_tr.buffer; |
| 723 | max_tr.buffer = buf; | 724 | max_tr.buffer = buf; |
| 724 | 725 | ||
| @@ -2400,6 +2401,27 @@ static void test_ftrace_alive(struct seq_file *m) | |||
| 2400 | seq_printf(m, "# MAY BE MISSING FUNCTION EVENTS\n"); | 2401 | seq_printf(m, "# MAY BE MISSING FUNCTION EVENTS\n"); |
| 2401 | } | 2402 | } |
| 2402 | 2403 | ||
| 2404 | #ifdef CONFIG_TRACER_MAX_TRACE | ||
| 2405 | static void print_snapshot_help(struct seq_file *m, struct trace_iterator *iter) | ||
| 2406 | { | ||
| 2407 | if (iter->trace->allocated_snapshot) | ||
| 2408 | seq_printf(m, "#\n# * Snapshot is allocated *\n#\n"); | ||
| 2409 | else | ||
| 2410 | seq_printf(m, "#\n# * Snapshot is freed *\n#\n"); | ||
| 2411 | |||
| 2412 | seq_printf(m, "# Snapshot commands:\n"); | ||
| 2413 | seq_printf(m, "# echo 0 > snapshot : Clears and frees snapshot buffer\n"); | ||
| 2414 | seq_printf(m, "# echo 1 > snapshot : Allocates snapshot buffer, if not already allocated.\n"); | ||
| 2415 | seq_printf(m, "# Takes a snapshot of the main buffer.\n"); | ||
| 2416 | seq_printf(m, "# echo 2 > snapshot : Clears snapshot buffer (but does not allocate)\n"); | ||
| 2417 | seq_printf(m, "# (Doesn't have to be '2' works with any number that\n"); | ||
| 2418 | seq_printf(m, "# is not a '0' or '1')\n"); | ||
| 2419 | } | ||
| 2420 | #else | ||
| 2421 | /* Should never be called */ | ||
| 2422 | static inline void print_snapshot_help(struct seq_file *m, struct trace_iterator *iter) { } | ||
| 2423 | #endif | ||
| 2424 | |||
| 2403 | static int s_show(struct seq_file *m, void *v) | 2425 | static int s_show(struct seq_file *m, void *v) |
| 2404 | { | 2426 | { |
| 2405 | struct trace_iterator *iter = v; | 2427 | struct trace_iterator *iter = v; |
| @@ -2411,7 +2433,9 @@ static int s_show(struct seq_file *m, void *v) | |||
| 2411 | seq_puts(m, "#\n"); | 2433 | seq_puts(m, "#\n"); |
| 2412 | test_ftrace_alive(m); | 2434 | test_ftrace_alive(m); |
| 2413 | } | 2435 | } |
| 2414 | if (iter->trace && iter->trace->print_header) | 2436 | if (iter->snapshot && trace_empty(iter)) |
| 2437 | print_snapshot_help(m, iter); | ||
| 2438 | else if (iter->trace && iter->trace->print_header) | ||
| 2415 | iter->trace->print_header(m); | 2439 | iter->trace->print_header(m); |
| 2416 | else | 2440 | else |
| 2417 | trace_default_header(m); | 2441 | trace_default_header(m); |
| @@ -2857,11 +2881,25 @@ static int set_tracer_option(struct tracer *trace, char *cmp, int neg) | |||
| 2857 | return -EINVAL; | 2881 | return -EINVAL; |
| 2858 | } | 2882 | } |
| 2859 | 2883 | ||
| 2860 | static void set_tracer_flags(unsigned int mask, int enabled) | 2884 | /* Some tracers require overwrite to stay enabled */ |
| 2885 | int trace_keep_overwrite(struct tracer *tracer, u32 mask, int set) | ||
| 2886 | { | ||
| 2887 | if (tracer->enabled && (mask & TRACE_ITER_OVERWRITE) && !set) | ||
| 2888 | return -1; | ||
| 2889 | |||
| 2890 | return 0; | ||
| 2891 | } | ||
| 2892 | |||
| 2893 | int set_tracer_flag(unsigned int mask, int enabled) | ||
| 2861 | { | 2894 | { |
| 2862 | /* do nothing if flag is already set */ | 2895 | /* do nothing if flag is already set */ |
| 2863 | if (!!(trace_flags & mask) == !!enabled) | 2896 | if (!!(trace_flags & mask) == !!enabled) |
| 2864 | return; | 2897 | return 0; |
| 2898 | |||
| 2899 | /* Give the tracer a chance to approve the change */ | ||
| 2900 | if (current_trace->flag_changed) | ||
| 2901 | if (current_trace->flag_changed(current_trace, mask, !!enabled)) | ||
| 2902 | return -EINVAL; | ||
| 2865 | 2903 | ||
| 2866 | if (enabled) | 2904 | if (enabled) |
| 2867 | trace_flags |= mask; | 2905 | trace_flags |= mask; |
| @@ -2871,18 +2909,24 @@ static void set_tracer_flags(unsigned int mask, int enabled) | |||
| 2871 | if (mask == TRACE_ITER_RECORD_CMD) | 2909 | if (mask == TRACE_ITER_RECORD_CMD) |
| 2872 | trace_event_enable_cmd_record(enabled); | 2910 | trace_event_enable_cmd_record(enabled); |
| 2873 | 2911 | ||
| 2874 | if (mask == TRACE_ITER_OVERWRITE) | 2912 | if (mask == TRACE_ITER_OVERWRITE) { |
| 2875 | ring_buffer_change_overwrite(global_trace.buffer, enabled); | 2913 | ring_buffer_change_overwrite(global_trace.buffer, enabled); |
| 2914 | #ifdef CONFIG_TRACER_MAX_TRACE | ||
| 2915 | ring_buffer_change_overwrite(max_tr.buffer, enabled); | ||
| 2916 | #endif | ||
| 2917 | } | ||
| 2876 | 2918 | ||
| 2877 | if (mask == TRACE_ITER_PRINTK) | 2919 | if (mask == TRACE_ITER_PRINTK) |
| 2878 | trace_printk_start_stop_comm(enabled); | 2920 | trace_printk_start_stop_comm(enabled); |
| 2921 | |||
| 2922 | return 0; | ||
| 2879 | } | 2923 | } |
| 2880 | 2924 | ||
| 2881 | static int trace_set_options(char *option) | 2925 | static int trace_set_options(char *option) |
| 2882 | { | 2926 | { |
| 2883 | char *cmp; | 2927 | char *cmp; |
| 2884 | int neg = 0; | 2928 | int neg = 0; |
| 2885 | int ret = 0; | 2929 | int ret = -ENODEV; |
| 2886 | int i; | 2930 | int i; |
| 2887 | 2931 | ||
| 2888 | cmp = strstrip(option); | 2932 | cmp = strstrip(option); |
| @@ -2892,19 +2936,20 @@ static int trace_set_options(char *option) | |||
| 2892 | cmp += 2; | 2936 | cmp += 2; |
| 2893 | } | 2937 | } |
| 2894 | 2938 | ||
| 2939 | mutex_lock(&trace_types_lock); | ||
| 2940 | |||
| 2895 | for (i = 0; trace_options[i]; i++) { | 2941 | for (i = 0; trace_options[i]; i++) { |
| 2896 | if (strcmp(cmp, trace_options[i]) == 0) { | 2942 | if (strcmp(cmp, trace_options[i]) == 0) { |
| 2897 | set_tracer_flags(1 << i, !neg); | 2943 | ret = set_tracer_flag(1 << i, !neg); |
| 2898 | break; | 2944 | break; |
| 2899 | } | 2945 | } |
| 2900 | } | 2946 | } |
| 2901 | 2947 | ||
| 2902 | /* If no option could be set, test the specific tracer options */ | 2948 | /* If no option could be set, test the specific tracer options */ |
| 2903 | if (!trace_options[i]) { | 2949 | if (!trace_options[i]) |
| 2904 | mutex_lock(&trace_types_lock); | ||
| 2905 | ret = set_tracer_option(current_trace, cmp, neg); | 2950 | ret = set_tracer_option(current_trace, cmp, neg); |
| 2906 | mutex_unlock(&trace_types_lock); | 2951 | |
| 2907 | } | 2952 | mutex_unlock(&trace_types_lock); |
| 2908 | 2953 | ||
| 2909 | return ret; | 2954 | return ret; |
| 2910 | } | 2955 | } |
| @@ -2914,6 +2959,7 @@ tracing_trace_options_write(struct file *filp, const char __user *ubuf, | |||
| 2914 | size_t cnt, loff_t *ppos) | 2959 | size_t cnt, loff_t *ppos) |
| 2915 | { | 2960 | { |
| 2916 | char buf[64]; | 2961 | char buf[64]; |
| 2962 | int ret; | ||
| 2917 | 2963 | ||
| 2918 | if (cnt >= sizeof(buf)) | 2964 | if (cnt >= sizeof(buf)) |
| 2919 | return -EINVAL; | 2965 | return -EINVAL; |
| @@ -2923,7 +2969,9 @@ tracing_trace_options_write(struct file *filp, const char __user *ubuf, | |||
| 2923 | 2969 | ||
| 2924 | buf[cnt] = 0; | 2970 | buf[cnt] = 0; |
| 2925 | 2971 | ||
| 2926 | trace_set_options(buf); | 2972 | ret = trace_set_options(buf); |
| 2973 | if (ret < 0) | ||
| 2974 | return ret; | ||
| 2927 | 2975 | ||
| 2928 | *ppos += cnt; | 2976 | *ppos += cnt; |
| 2929 | 2977 | ||
| @@ -3227,6 +3275,9 @@ static int tracing_set_tracer(const char *buf) | |||
| 3227 | goto out; | 3275 | goto out; |
| 3228 | 3276 | ||
| 3229 | trace_branch_disable(); | 3277 | trace_branch_disable(); |
| 3278 | |||
| 3279 | current_trace->enabled = false; | ||
| 3280 | |||
| 3230 | if (current_trace->reset) | 3281 | if (current_trace->reset) |
| 3231 | current_trace->reset(tr); | 3282 | current_trace->reset(tr); |
| 3232 | 3283 | ||
| @@ -3271,6 +3322,7 @@ static int tracing_set_tracer(const char *buf) | |||
| 3271 | } | 3322 | } |
| 3272 | 3323 | ||
| 3273 | current_trace = t; | 3324 | current_trace = t; |
| 3325 | current_trace->enabled = true; | ||
| 3274 | trace_branch_enable(tr); | 3326 | trace_branch_enable(tr); |
| 3275 | out: | 3327 | out: |
| 3276 | mutex_unlock(&trace_types_lock); | 3328 | mutex_unlock(&trace_types_lock); |
| @@ -4144,8 +4196,6 @@ tracing_snapshot_write(struct file *filp, const char __user *ubuf, size_t cnt, | |||
| 4144 | default: | 4196 | default: |
| 4145 | if (current_trace->allocated_snapshot) | 4197 | if (current_trace->allocated_snapshot) |
| 4146 | tracing_reset_online_cpus(&max_tr); | 4198 | tracing_reset_online_cpus(&max_tr); |
| 4147 | else | ||
| 4148 | ret = -EINVAL; | ||
| 4149 | break; | 4199 | break; |
| 4150 | } | 4200 | } |
| 4151 | 4201 | ||
| @@ -4759,7 +4809,13 @@ trace_options_core_write(struct file *filp, const char __user *ubuf, size_t cnt, | |||
| 4759 | 4809 | ||
| 4760 | if (val != 0 && val != 1) | 4810 | if (val != 0 && val != 1) |
| 4761 | return -EINVAL; | 4811 | return -EINVAL; |
| 4762 | set_tracer_flags(1 << index, val); | 4812 | |
| 4813 | mutex_lock(&trace_types_lock); | ||
| 4814 | ret = set_tracer_flag(1 << index, val); | ||
| 4815 | mutex_unlock(&trace_types_lock); | ||
| 4816 | |||
| 4817 | if (ret < 0) | ||
| 4818 | return ret; | ||
| 4763 | 4819 | ||
| 4764 | *ppos += cnt; | 4820 | *ppos += cnt; |
| 4765 | 4821 | ||
