aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--kernel/trace/trace.h1
-rw-r--r--kernel/trace/trace_events_filter.c124
2 files changed, 45 insertions, 80 deletions
diff --git a/kernel/trace/trace.h b/kernel/trace/trace.h
index acef8b4636f0..628614532d16 100644
--- a/kernel/trace/trace.h
+++ b/kernel/trace/trace.h
@@ -683,7 +683,6 @@ struct event_filter {
683 int n_preds; 683 int n_preds;
684 struct filter_pred **preds; 684 struct filter_pred **preds;
685 char *filter_string; 685 char *filter_string;
686 bool no_reset;
687}; 686};
688 687
689struct event_subsystem { 688struct event_subsystem {
diff --git a/kernel/trace/trace_events_filter.c b/kernel/trace/trace_events_filter.c
index 92672016da28..9c4f9a0dae2b 100644
--- a/kernel/trace/trace_events_filter.c
+++ b/kernel/trace/trace_events_filter.c
@@ -615,14 +615,7 @@ static int init_subsystem_preds(struct event_subsystem *system)
615 return 0; 615 return 0;
616} 616}
617 617
618enum { 618static void filter_free_subsystem_preds(struct event_subsystem *system)
619 FILTER_DISABLE_ALL,
620 FILTER_INIT_NO_RESET,
621 FILTER_SKIP_NO_RESET,
622};
623
624static void filter_free_subsystem_preds(struct event_subsystem *system,
625 int flag)
626{ 619{
627 struct ftrace_event_call *call; 620 struct ftrace_event_call *call;
628 621
@@ -633,14 +626,6 @@ static void filter_free_subsystem_preds(struct event_subsystem *system,
633 if (strcmp(call->system, system->name) != 0) 626 if (strcmp(call->system, system->name) != 0)
634 continue; 627 continue;
635 628
636 if (flag == FILTER_INIT_NO_RESET) {
637 call->filter->no_reset = false;
638 continue;
639 }
640
641 if (flag == FILTER_SKIP_NO_RESET && call->filter->no_reset)
642 continue;
643
644 filter_disable_preds(call); 629 filter_disable_preds(call);
645 remove_filter_string(call->filter); 630 remove_filter_string(call->filter);
646 } 631 }
@@ -817,44 +802,6 @@ add_pred_fn:
817 return 0; 802 return 0;
818} 803}
819 804
820static int filter_add_subsystem_pred(struct filter_parse_state *ps,
821 struct event_subsystem *system,
822 struct filter_pred *pred,
823 char *filter_string,
824 bool dry_run)
825{
826 struct ftrace_event_call *call;
827 int err = 0;
828 bool fail = true;
829
830 list_for_each_entry(call, &ftrace_events, list) {
831
832 if (!call->define_fields)
833 continue;
834
835 if (strcmp(call->system, system->name))
836 continue;
837
838 if (call->filter->no_reset)
839 continue;
840
841 err = filter_add_pred(ps, call, pred, dry_run);
842 if (err)
843 call->filter->no_reset = true;
844 else
845 fail = false;
846
847 if (!dry_run)
848 replace_filter_string(call->filter, filter_string);
849 }
850
851 if (fail) {
852 parse_error(ps, FILT_ERR_BAD_SUBSYS_FILTER, 0);
853 return err;
854 }
855 return 0;
856}
857
858static void parse_init(struct filter_parse_state *ps, 805static void parse_init(struct filter_parse_state *ps,
859 struct filter_op *ops, 806 struct filter_op *ops,
860 char *infix_string) 807 char *infix_string)
@@ -1209,8 +1156,7 @@ static int check_preds(struct filter_parse_state *ps)
1209 return 0; 1156 return 0;
1210} 1157}
1211 1158
1212static int replace_preds(struct event_subsystem *system, 1159static int replace_preds(struct ftrace_event_call *call,
1213 struct ftrace_event_call *call,
1214 struct filter_parse_state *ps, 1160 struct filter_parse_state *ps,
1215 char *filter_string, 1161 char *filter_string,
1216 bool dry_run) 1162 bool dry_run)
@@ -1257,11 +1203,7 @@ static int replace_preds(struct event_subsystem *system,
1257add_pred: 1203add_pred:
1258 if (!pred) 1204 if (!pred)
1259 return -ENOMEM; 1205 return -ENOMEM;
1260 if (call) 1206 err = filter_add_pred(ps, call, pred, dry_run);
1261 err = filter_add_pred(ps, call, pred, false);
1262 else
1263 err = filter_add_subsystem_pred(ps, system, pred,
1264 filter_string, dry_run);
1265 filter_free_pred(pred); 1207 filter_free_pred(pred);
1266 if (err) 1208 if (err)
1267 return err; 1209 return err;
@@ -1272,6 +1214,44 @@ add_pred:
1272 return 0; 1214 return 0;
1273} 1215}
1274 1216
1217static int replace_system_preds(struct event_subsystem *system,
1218 struct filter_parse_state *ps,
1219 char *filter_string)
1220{
1221 struct ftrace_event_call *call;
1222 int err;
1223 bool fail = true;
1224
1225 list_for_each_entry(call, &ftrace_events, list) {
1226
1227 if (!call->define_fields)
1228 continue;
1229
1230 if (strcmp(call->system, system->name) != 0)
1231 continue;
1232
1233 /* try to see if the filter can be applied */
1234 err = replace_preds(call, ps, filter_string, true);
1235 if (err)
1236 continue;
1237
1238 /* really apply the filter */
1239 filter_disable_preds(call);
1240 err = replace_preds(call, ps, filter_string, false);
1241 if (err)
1242 filter_disable_preds(call);
1243 else
1244 replace_filter_string(call->filter, filter_string);
1245 fail = false;
1246 }
1247
1248 if (fail) {
1249 parse_error(ps, FILT_ERR_BAD_SUBSYS_FILTER, 0);
1250 return err;
1251 }
1252 return 0;
1253}
1254
1275int apply_event_filter(struct ftrace_event_call *call, char *filter_string) 1255int apply_event_filter(struct ftrace_event_call *call, char *filter_string)
1276{ 1256{
1277 int err; 1257 int err;
@@ -1306,7 +1286,7 @@ int apply_event_filter(struct ftrace_event_call *call, char *filter_string)
1306 goto out; 1286 goto out;
1307 } 1287 }
1308 1288
1309 err = replace_preds(NULL, call, ps, filter_string, false); 1289 err = replace_preds(call, ps, filter_string, false);
1310 if (err) 1290 if (err)
1311 append_filter_err(ps, call->filter); 1291 append_filter_err(ps, call->filter);
1312 1292
@@ -1334,7 +1314,7 @@ int apply_subsystem_event_filter(struct event_subsystem *system,
1334 goto out_unlock; 1314 goto out_unlock;
1335 1315
1336 if (!strcmp(strstrip(filter_string), "0")) { 1316 if (!strcmp(strstrip(filter_string), "0")) {
1337 filter_free_subsystem_preds(system, FILTER_DISABLE_ALL); 1317 filter_free_subsystem_preds(system);
1338 remove_filter_string(system->filter); 1318 remove_filter_string(system->filter);
1339 mutex_unlock(&event_mutex); 1319 mutex_unlock(&event_mutex);
1340 return 0; 1320 return 0;
@@ -1354,23 +1334,9 @@ int apply_subsystem_event_filter(struct event_subsystem *system,
1354 goto out; 1334 goto out;
1355 } 1335 }
1356 1336
1357 filter_free_subsystem_preds(system, FILTER_INIT_NO_RESET); 1337 err = replace_system_preds(system, ps, filter_string);
1358 1338 if (err)
1359 /* try to see the filter can be applied to which events */
1360 err = replace_preds(system, NULL, ps, filter_string, true);
1361 if (err) {
1362 append_filter_err(ps, system->filter);
1363 goto out;
1364 }
1365
1366 filter_free_subsystem_preds(system, FILTER_SKIP_NO_RESET);
1367
1368 /* really apply the filter to the events */
1369 err = replace_preds(system, NULL, ps, filter_string, false);
1370 if (err) {
1371 append_filter_err(ps, system->filter); 1339 append_filter_err(ps, system->filter);
1372 filter_free_subsystem_preds(system, 2);
1373 }
1374 1340
1375out: 1341out:
1376 filter_opstack_clear(ps); 1342 filter_opstack_clear(ps);