aboutsummaryrefslogtreecommitdiff
path: root/src/traced/probes/ftrace/format_parser.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/traced/probes/ftrace/format_parser.cc')
-rw-r--r--src/traced/probes/ftrace/format_parser.cc199
1 files changed, 199 insertions, 0 deletions
diff --git a/src/traced/probes/ftrace/format_parser.cc b/src/traced/probes/ftrace/format_parser.cc
new file mode 100644
index 000000000..2be687d3a
--- /dev/null
+++ b/src/traced/probes/ftrace/format_parser.cc
@@ -0,0 +1,199 @@
+/*
+ * Copyright (C) 2017 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#include "src/traced/probes/ftrace/format_parser.h"
+
+#include <string.h>
+
+#include <iosfwd>
+#include <iostream>
+#include <memory>
+#include <string>
+#include <vector>
+
+#include "perfetto/base/logging.h"
+#include "perfetto/ext/base/string_splitter.h"
+#include "perfetto/ext/base/utils.h"
+
+namespace perfetto {
+namespace {
+
+#define MAX_FIELD_LENGTH 127
+#define STRINGIFY(x) STRINGIFY2(x)
+#define STRINGIFY2(x) #x
+
+const char* kCommonFieldPrefix = "common_";
+
+bool IsCommonFieldName(const std::string& name) {
+ return name.compare(0, strlen(kCommonFieldPrefix), kCommonFieldPrefix) == 0;
+}
+
+bool IsCIdentifier(const std::string& s) {
+ for (const char c : s) {
+ if (!(std::isalnum(c) || c == '_'))
+ return false;
+ }
+ return !s.empty() && !std::isdigit(s[0]);
+}
+
+bool ParseFtraceEventBody(base::StringSplitter* ss,
+ std::vector<FtraceEvent::Field>* common_fields,
+ std::vector<FtraceEvent::Field>* fields,
+ bool disable_logging_for_testing) {
+ PERFETTO_DCHECK(common_fields || fields);
+ char buffer[MAX_FIELD_LENGTH + 1];
+ while (ss->Next()) {
+ const char* line = ss->cur_token();
+ uint16_t offset = 0;
+ uint16_t size = 0;
+ int is_signed = 0;
+ if (sscanf(line,
+ "\tfield:%" STRINGIFY(MAX_FIELD_LENGTH) "[^;];\toffset: "
+ "%hu;\tsize: "
+ "%hu;\tsigned: %d;",
+ buffer, &offset, &size, &is_signed) == 4) {
+ std::string type_and_name(buffer);
+
+ FtraceEvent::Field field{type_and_name, offset, size, is_signed == 1};
+
+ if (IsCommonFieldName(GetNameFromTypeAndName(type_and_name))) {
+ if (common_fields)
+ common_fields->push_back(field);
+ } else if (fields) {
+ fields->push_back(field);
+ }
+ continue;
+ }
+
+ if (strncmp(line, "print fmt:", 10) == 0) {
+ break;
+ }
+
+ if (!disable_logging_for_testing)
+ PERFETTO_DLOG("Cannot parse line: \"%s\"\n", line);
+ return false;
+ }
+ return true;
+}
+
+} // namespace
+
+// For example:
+// "int foo" -> "foo"
+// "u8 foo[(int)sizeof(struct blah)]" -> "foo"
+// "char[] foo[16]" -> "foo"
+// "something_went_wrong" -> ""
+// "" -> ""
+std::string GetNameFromTypeAndName(const std::string& type_and_name) {
+ size_t right = type_and_name.size();
+ if (right == 0)
+ return "";
+
+ if (type_and_name[type_and_name.size() - 1] == ']') {
+ right = type_and_name.rfind('[');
+ if (right == std::string::npos)
+ return "";
+ }
+
+ size_t left = type_and_name.rfind(' ', right);
+ if (left == std::string::npos)
+ return "";
+ left++;
+
+ std::string result = type_and_name.substr(left, right - left);
+ if (!IsCIdentifier(result))
+ return "";
+
+ return result;
+}
+
+bool ParseFtraceEventBody(std::string input,
+ std::vector<FtraceEvent::Field>* common_fields,
+ std::vector<FtraceEvent::Field>* fields,
+ bool disable_logging_for_testing) {
+ base::StringSplitter ss(std::move(input), '\n');
+ return ParseFtraceEventBody(&ss, common_fields, fields,
+ disable_logging_for_testing);
+}
+
+bool ParseFtraceEvent(std::string input, FtraceEvent* output) {
+ char buffer[MAX_FIELD_LENGTH + 1];
+
+ bool has_id = false;
+ bool has_name = false;
+
+ uint32_t id = 0;
+ std::string name;
+ std::vector<FtraceEvent::Field> common_fields;
+ std::vector<FtraceEvent::Field> fields;
+
+ for (base::StringSplitter ss(std::move(input), '\n'); ss.Next();) {
+ const char* line = ss.cur_token();
+ if (!has_id && sscanf(line, "ID: %u", &id) == 1) {
+ has_id = true;
+ continue;
+ }
+
+ if (!has_name &&
+ sscanf(line, "name: %" STRINGIFY(MAX_FIELD_LENGTH) "s", buffer) == 1) {
+ name = std::string(buffer);
+ has_name = true;
+ continue;
+ }
+
+ if (strcmp("format:", line) == 0) {
+ ParseFtraceEventBody(&ss, &common_fields, &fields,
+ /*disable_logging_for_testing=*/output == nullptr);
+ break;
+ }
+
+ if (output)
+ PERFETTO_DLOG("Cannot parse line: \"%s\"\n", line);
+ return false;
+ }
+
+ if (!has_id || !has_name || fields.empty()) {
+ if (output)
+ PERFETTO_DLOG("Could not parse format file: %s.\n",
+ !has_id ? "no ID found"
+ : !has_name ? "no name found" : "no fields found");
+ return false;
+ }
+
+ if (!output)
+ return true;
+
+ output->id = id;
+ output->name = name;
+ output->fields = std::move(fields);
+ output->common_fields = std::move(common_fields);
+
+ return true;
+}
+
+::std::ostream& operator<<(::std::ostream& os,
+ const FtraceEvent::Field& field) {
+ PrintTo(field, &os);
+ return os;
+}
+
+// Allow gtest to pretty print FtraceEvent::Field.
+void PrintTo(const FtraceEvent::Field& field, ::std::ostream* os) {
+ *os << "FtraceEvent::Field(" << field.type_and_name << ", " << field.offset
+ << ", " << field.size << ", " << field.is_signed << ")";
+}
+
+} // namespace perfetto