1 /*
2  * Copyright (C) 2019 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "src/trace_processor/sqlite/sqlite_raw_table.h"
18 
19 #include <inttypes.h>
20 
21 #include "perfetto/base/compiler.h"
22 #include "perfetto/ext/base/string_utils.h"
23 #include "src/trace_processor/importers/common/system_info_tracker.h"
24 #include "src/trace_processor/importers/ftrace/ftrace_descriptors.h"
25 #include "src/trace_processor/sqlite/sqlite_utils.h"
26 #include "src/trace_processor/types/gfp_flags.h"
27 #include "src/trace_processor/types/softirq_action.h"
28 #include "src/trace_processor/types/task_state.h"
29 #include "src/trace_processor/types/variadic.h"
30 
31 #include "protos/perfetto/trace/ftrace/binder.pbzero.h"
32 #include "protos/perfetto/trace/ftrace/clk.pbzero.h"
33 #include "protos/perfetto/trace/ftrace/dpu.pbzero.h"
34 #include "protos/perfetto/trace/ftrace/filemap.pbzero.h"
35 #include "protos/perfetto/trace/ftrace/ftrace.pbzero.h"
36 #include "protos/perfetto/trace/ftrace/ftrace_event.pbzero.h"
37 #include "protos/perfetto/trace/ftrace/g2d.pbzero.h"
38 #include "protos/perfetto/trace/ftrace/irq.pbzero.h"
39 #include "protos/perfetto/trace/ftrace/power.pbzero.h"
40 #include "protos/perfetto/trace/ftrace/sched.pbzero.h"
41 #include "protos/perfetto/trace/ftrace/workqueue.pbzero.h"
42 
43 namespace perfetto {
44 namespace trace_processor {
45 
46 namespace {
47 
48 struct FtraceTime {
FtraceTimeperfetto::trace_processor::__anon94b96ab40111::FtraceTime49   FtraceTime(int64_t ns)
50       : secs(ns / 1000000000LL), micros((ns - secs * 1000000000LL) / 1000) {}
51 
52   const int64_t secs;
53   const int64_t micros;
54 };
55 
56 class ArgsSerializer {
57  public:
58   ArgsSerializer(TraceProcessorContext*,
59                  ArgSetId arg_set_id,
60                  NullTermStringView event_name,
61                  std::vector<uint32_t>* field_id_to_arg_index,
62                  base::StringWriter*);
63 
64   void SerializeArgs();
65 
66  private:
67   using ValueWriter = std::function<void(const Variadic&)>;
68   using SerializerValueWriter = void (ArgsSerializer::*)(const Variadic&);
69 
WriteArgForField(uint32_t field_id)70   void WriteArgForField(uint32_t field_id) {
71     WriteArgForField(field_id,
72                      [this](const Variadic& v) { return WriteValue(v); });
73   }
74 
WriteArgForField(uint32_t field_id,SerializerValueWriter writer)75   void WriteArgForField(uint32_t field_id, SerializerValueWriter writer) {
76     WriteArgForField(field_id, [this, writer](const Variadic& variadic) {
77       (this->*writer)(variadic);
78     });
79   }
80 
WriteArgForField(uint32_t field_id,ValueWriter writer)81   void WriteArgForField(uint32_t field_id, ValueWriter writer) {
82     WriteArgAtRow(FieldIdToRow(field_id), writer);
83   }
84 
WriteValueForField(uint32_t field_id)85   void WriteValueForField(uint32_t field_id) {
86     WriteValueForField(field_id,
87                        [this](const Variadic& v) { return WriteValue(v); });
88   }
89 
WriteValueForField(uint32_t field_id,SerializerValueWriter writer)90   void WriteValueForField(uint32_t field_id, SerializerValueWriter writer) {
91     WriteValueForField(field_id, [this, writer](const Variadic& variadic) {
92       (this->*writer)(variadic);
93     });
94   }
95 
WriteValueForField(uint32_t field_id,ValueWriter writer)96   void WriteValueForField(uint32_t field_id, ValueWriter writer) {
97     writer(storage_->GetArgValue(FieldIdToRow(field_id)));
98   }
99 
WriteArgAtRow(uint32_t arg_index)100   void WriteArgAtRow(uint32_t arg_index) {
101     WriteArgAtRow(arg_index,
102                   [this](const Variadic& v) { return WriteValue(v); });
103   }
104 
105   void WriteArgAtRow(uint32_t arg_index, ValueWriter writer);
106 
WriteKernelFnValue(const Variadic & value)107   void WriteKernelFnValue(const Variadic& value) {
108     if (value.type == Variadic::Type::kUint) {
109       writer_->AppendHexInt(value.uint_value);
110     } else if (value.type == Variadic::Type::kString) {
111       WriteValue(value);
112     } else {
113       PERFETTO_DFATAL("Invalid field type %d", static_cast<int>(value.type));
114     }
115   }
116 
117   void WriteValue(const Variadic& variadic);
118 
FieldIdToRow(uint32_t field_id)119   uint32_t FieldIdToRow(uint32_t field_id) {
120     PERFETTO_DCHECK(field_id > 0);
121     PERFETTO_DCHECK(field_id < field_id_to_arg_index_->size());
122     uint32_t index_in_arg_set = (*field_id_to_arg_index_)[field_id];
123     return start_row_ + index_in_arg_set;
124   }
125 
126   const TraceStorage* storage_ = nullptr;
127   TraceProcessorContext* context_ = nullptr;
128   ArgSetId arg_set_id_ = kInvalidArgSetId;
129   NullTermStringView event_name_;
130   std::vector<uint32_t>* field_id_to_arg_index_;
131 
132   RowMap row_map_;
133   uint32_t start_row_ = 0;
134 
135   base::StringWriter* writer_ = nullptr;
136 };
137 
ArgsSerializer(TraceProcessorContext * context,ArgSetId arg_set_id,NullTermStringView event_name,std::vector<uint32_t> * field_id_to_arg_index,base::StringWriter * writer)138 ArgsSerializer::ArgsSerializer(TraceProcessorContext* context,
139                                ArgSetId arg_set_id,
140                                NullTermStringView event_name,
141                                std::vector<uint32_t>* field_id_to_arg_index,
142                                base::StringWriter* writer)
143     : context_(context),
144       arg_set_id_(arg_set_id),
145       event_name_(event_name),
146       field_id_to_arg_index_(field_id_to_arg_index),
147       writer_(writer) {
148   storage_ = context_->storage.get();
149   const auto& args = storage_->arg_table();
150   const auto& set_ids = args.arg_set_id();
151 
152   // We assume that the row map is a contiguous range (which is always the case
153   // because arg_set_ids are contiguous by definition).
154   row_map_ = args.FilterToRowMap({set_ids.eq(arg_set_id_)});
155   start_row_ = row_map_.empty() ? 0 : row_map_.Get(0);
156 
157   // If the vector already has entries, we've previously cached the mapping
158   // from field id to arg index.
159   if (!field_id_to_arg_index->empty())
160     return;
161 
162   auto* descriptor = GetMessageDescriptorForName(event_name);
163   if (!descriptor) {
164     // If we don't have a descriptor, this event must be a generic ftrace event.
165     // As we can't possibly have any special handling for generic events, just
166     // add a row to the vector (for the invalid field id 0) to remove future
167     // lookups for this event name.
168     field_id_to_arg_index->resize(1);
169     return;
170   }
171 
172   // If we have a descriptor, try and create the mapping from proto field id
173   // to the index in the arg set.
174   size_t max = descriptor->max_field_id;
175 
176   // We need to reserve an index for the invalid field id 0.
177   field_id_to_arg_index_->resize(max + 1);
178 
179   // Go through each field id and find the entry in the args table for that
180   for (uint32_t i = 1; i <= max; ++i) {
181     for (auto it = row_map_.IterateRows(); it; it.Next()) {
182       base::StringView key = args.key().GetString(it.row());
183       if (key == descriptor->fields[i].name) {
184         (*field_id_to_arg_index)[i] = it.index();
185         break;
186       }
187     }
188   }
189 }
190 
SerializeArgs()191 void ArgsSerializer::SerializeArgs() {
192   if (row_map_.empty())
193     return;
194 
195   if (event_name_ == "sched_switch") {
196     using SS = protos::pbzero::SchedSwitchFtraceEvent;
197 
198     WriteArgForField(SS::kPrevCommFieldNumber);
199     WriteArgForField(SS::kPrevPidFieldNumber);
200     WriteArgForField(SS::kPrevPrioFieldNumber);
201     WriteArgForField(SS::kPrevStateFieldNumber, [this](const Variadic& value) {
202       PERFETTO_DCHECK(value.type == Variadic::Type::kInt);
203       auto state = static_cast<uint16_t>(value.int_value);
204       auto kernel_version =
205           SystemInfoTracker::GetOrCreate(context_)->GetKernelVersion();
206       writer_->AppendString(
207           ftrace_utils::TaskState(state, kernel_version).ToString('|').data());
208     });
209     writer_->AppendLiteral(" ==>");
210     WriteArgForField(SS::kNextCommFieldNumber);
211     WriteArgForField(SS::kNextPidFieldNumber);
212     WriteArgForField(SS::kNextPrioFieldNumber);
213     return;
214   } else if (event_name_ == "sched_wakeup") {
215     using SW = protos::pbzero::SchedWakeupFtraceEvent;
216     WriteArgForField(SW::kCommFieldNumber);
217     WriteArgForField(SW::kPidFieldNumber);
218     WriteArgForField(SW::kPrioFieldNumber);
219     WriteArgForField(SW::kTargetCpuFieldNumber, [this](const Variadic& value) {
220       PERFETTO_DCHECK(value.type == Variadic::Type::kInt);
221       writer_->AppendPaddedInt<'0', 3>(value.int_value);
222     });
223     return;
224   } else if (event_name_ == "clock_set_rate") {
225     using CSR = protos::pbzero::ClockSetRateFtraceEvent;
226     writer_->AppendLiteral(" ");
227     WriteValueForField(CSR::kNameFieldNumber);
228     WriteArgForField(CSR::kStateFieldNumber);
229     WriteArgForField(CSR::kCpuIdFieldNumber);
230     return;
231   } else if (event_name_ == "clk_set_rate") {
232     using CSR = protos::pbzero::ClkSetRateFtraceEvent;
233     writer_->AppendLiteral(" ");
234     WriteValueForField(CSR::kNameFieldNumber);
235     writer_->AppendLiteral(" ");
236     WriteValueForField(CSR::kRateFieldNumber);
237     return;
238   } else if (event_name_ == "clock_enable") {
239     using CE = protos::pbzero::ClockEnableFtraceEvent;
240     WriteValueForField(CE::kNameFieldNumber);
241     WriteArgForField(CE::kStateFieldNumber);
242     WriteArgForField(CE::kCpuIdFieldNumber);
243     return;
244   } else if (event_name_ == "clock_disable") {
245     using CD = protos::pbzero::ClockDisableFtraceEvent;
246     WriteValueForField(CD::kNameFieldNumber);
247     WriteArgForField(CD::kStateFieldNumber);
248     WriteArgForField(CD::kCpuIdFieldNumber);
249     return;
250   } else if (event_name_ == "binder_transaction") {
251     using BT = protos::pbzero::BinderTransactionFtraceEvent;
252     writer_->AppendString(" transaction=");
253     WriteValueForField(BT::kDebugIdFieldNumber, [this](const Variadic& value) {
254       PERFETTO_DCHECK(value.type == Variadic::Type::kInt);
255       writer_->AppendUnsignedInt(static_cast<uint32_t>(value.int_value));
256     });
257 
258     writer_->AppendString(" dest_node=");
259     WriteValueForField(
260         BT::kTargetNodeFieldNumber, [this](const Variadic& value) {
261           PERFETTO_DCHECK(value.type == Variadic::Type::kInt);
262           writer_->AppendUnsignedInt(static_cast<uint32_t>(value.int_value));
263         });
264 
265     writer_->AppendString(" dest_proc=");
266     WriteValueForField(BT::kToProcFieldNumber);
267 
268     writer_->AppendString(" dest_thread=");
269     WriteValueForField(BT::kToThreadFieldNumber);
270 
271     writer_->AppendString(" reply=");
272     WriteValueForField(BT::kReplyFieldNumber);
273 
274     writer_->AppendString(" flags=0x");
275     WriteValueForField(BT::kFlagsFieldNumber, [this](const Variadic& value) {
276       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
277       writer_->AppendHexInt(value.uint_value);
278     });
279 
280     writer_->AppendString(" code=0x");
281     WriteValueForField(BT::kCodeFieldNumber, [this](const Variadic& value) {
282       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
283       writer_->AppendHexInt(value.uint_value);
284     });
285     return;
286   } else if (event_name_ == "binder_transaction_alloc_buf") {
287     using BTAB = protos::pbzero::BinderTransactionAllocBufFtraceEvent;
288     writer_->AppendString(" transaction=");
289     WriteValueForField(
290         BTAB::kDebugIdFieldNumber, [this](const Variadic& value) {
291           PERFETTO_DCHECK(value.type == Variadic::Type::kInt);
292           writer_->AppendUnsignedInt(static_cast<uint32_t>(value.int_value));
293         });
294     WriteArgForField(BTAB::kDataSizeFieldNumber);
295     WriteArgForField(BTAB::kOffsetsSizeFieldNumber);
296     return;
297   } else if (event_name_ == "binder_transaction_received") {
298     using BTR = protos::pbzero::BinderTransactionReceivedFtraceEvent;
299     writer_->AppendString(" transaction=");
300     WriteValueForField(BTR::kDebugIdFieldNumber, [this](const Variadic& value) {
301       PERFETTO_DCHECK(value.type == Variadic::Type::kInt);
302       writer_->AppendUnsignedInt(static_cast<uint32_t>(value.int_value));
303     });
304     return;
305   } else if (event_name_ == "mm_filemap_add_to_page_cache") {
306     using MFA = protos::pbzero::MmFilemapAddToPageCacheFtraceEvent;
307     writer_->AppendString(" dev ");
308     WriteValueForField(MFA::kSDevFieldNumber, [this](const Variadic& value) {
309       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
310       writer_->AppendUnsignedInt(value.uint_value >> 20);
311     });
312     writer_->AppendString(":");
313     WriteValueForField(MFA::kSDevFieldNumber, [this](const Variadic& value) {
314       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
315       writer_->AppendUnsignedInt(value.uint_value & ((1 << 20) - 1));
316     });
317     writer_->AppendString(" ino ");
318     WriteValueForField(MFA::kIInoFieldNumber, [this](const Variadic& value) {
319       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
320       writer_->AppendHexInt(value.uint_value);
321     });
322     writer_->AppendString(" page=0000000000000000");
323     writer_->AppendString(" pfn=");
324     WriteValueForField(MFA::kPfnFieldNumber);
325     writer_->AppendString(" ofs=");
326     WriteValueForField(MFA::kIndexFieldNumber, [this](const Variadic& value) {
327       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
328       writer_->AppendUnsignedInt(value.uint_value << 12);
329     });
330     return;
331   } else if (event_name_ == "print") {
332     using P = protos::pbzero::PrintFtraceEvent;
333 
334     writer_->AppendChar(' ');
335     WriteValueForField(P::kBufFieldNumber, [this](const Variadic& value) {
336       PERFETTO_DCHECK(value.type == Variadic::Type::kString);
337 
338       NullTermStringView str = storage_->GetString(value.string_value);
339       // If the last character is a newline in a print, just drop it.
340       auto chars_to_print = !str.empty() && str.c_str()[str.size() - 1] == '\n'
341                                 ? str.size() - 1
342                                 : str.size();
343       writer_->AppendString(str.c_str(), chars_to_print);
344     });
345     return;
346   } else if (event_name_ == "sched_blocked_reason") {
347     using SBR = protos::pbzero::SchedBlockedReasonFtraceEvent;
348     WriteArgForField(SBR::kPidFieldNumber);
349     WriteArgForField(SBR::kIoWaitFieldNumber);
350     WriteArgForField(SBR::kCallerFieldNumber,
351                      &ArgsSerializer::WriteKernelFnValue);
352     return;
353   } else if (event_name_ == "workqueue_activate_work") {
354     using WAW = protos::pbzero::WorkqueueActivateWorkFtraceEvent;
355     writer_->AppendString(" work struct ");
356     WriteValueForField(WAW::kWorkFieldNumber, [this](const Variadic& value) {
357       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
358       writer_->AppendHexInt(value.uint_value);
359     });
360     return;
361   } else if (event_name_ == "workqueue_execute_start") {
362     using WES = protos::pbzero::WorkqueueExecuteStartFtraceEvent;
363     writer_->AppendString(" work struct ");
364     WriteValueForField(WES::kWorkFieldNumber, [this](const Variadic& value) {
365       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
366       writer_->AppendHexInt(value.uint_value);
367     });
368     writer_->AppendString(": function ");
369     WriteValueForField(WES::kFunctionFieldNumber,
370                        &ArgsSerializer::WriteKernelFnValue);
371     return;
372   } else if (event_name_ == "workqueue_execute_end") {
373     using WE = protos::pbzero::WorkqueueExecuteEndFtraceEvent;
374     writer_->AppendString(" work struct ");
375     WriteValueForField(WE::kWorkFieldNumber, [this](const Variadic& value) {
376       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
377       writer_->AppendHexInt(value.uint_value);
378     });
379     return;
380   } else if (event_name_ == "workqueue_queue_work") {
381     using WQW = protos::pbzero::WorkqueueQueueWorkFtraceEvent;
382     writer_->AppendString(" work struct=");
383     WriteValueForField(WQW::kWorkFieldNumber, [this](const Variadic& value) {
384       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
385       writer_->AppendHexInt(value.uint_value);
386     });
387     WriteArgForField(WQW::kFunctionFieldNumber,
388                      &ArgsSerializer::WriteKernelFnValue);
389     WriteArgForField(WQW::kWorkqueueFieldNumber, [this](const Variadic& value) {
390       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
391       writer_->AppendHexInt(value.uint_value);
392     });
393     WriteValueForField(WQW::kReqCpuFieldNumber);
394     WriteValueForField(WQW::kCpuFieldNumber);
395     return;
396   } else if (event_name_ == "irq_handler_entry") {
397     using IEN = protos::pbzero::IrqHandlerEntryFtraceEvent;
398     WriteArgForField(IEN::kIrqFieldNumber);
399     WriteArgForField(IEN::kNameFieldNumber);
400     return;
401   } else if (event_name_ == "irq_handler_exit") {
402     using IEX = protos::pbzero::IrqHandlerExitFtraceEvent;
403     WriteArgForField(IEX::kIrqFieldNumber);
404     writer_->AppendString(" ret=");
405     WriteValueForField(IEX::kRetFieldNumber, [this](const Variadic& value) {
406       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
407       writer_->AppendString(value.uint_value ? "handled" : "unhandled");
408     });
409     return;
410   } else if (event_name_ == "softirq_entry") {
411     using SIE = protos::pbzero::SoftirqEntryFtraceEvent;
412     WriteArgForField(SIE::kVecFieldNumber);
413     writer_->AppendString(" [action=");
414     WriteValueForField(SIE::kVecFieldNumber, [this](const Variadic& value) {
415       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
416       writer_->AppendString(kActionNames[value.uint_value]);
417     });
418     writer_->AppendString("]");
419     return;
420   } else if (event_name_ == "softirq_exit") {
421     using SIX = protos::pbzero::SoftirqExitFtraceEvent;
422     WriteArgForField(SIX::kVecFieldNumber);
423     writer_->AppendString(" [action=");
424     WriteValueForField(SIX::kVecFieldNumber, [this](const Variadic& value) {
425       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
426       writer_->AppendString(kActionNames[value.uint_value]);
427     });
428     writer_->AppendString("]");
429     return;
430   } else if (event_name_ == "dpu_tracing_mark_write") {
431     using TMW = protos::pbzero::DpuTracingMarkWriteFtraceEvent;
432     WriteValueForField(TMW::kTypeFieldNumber, [this](const Variadic& value) {
433       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
434       writer_->AppendChar(static_cast<char>(value.uint_value));
435     });
436     writer_->AppendString("|");
437     WriteValueForField(TMW::kPidFieldNumber);
438     writer_->AppendString("|");
439     WriteValueForField(TMW::kNameFieldNumber);
440     writer_->AppendString("|");
441     WriteValueForField(TMW::kValueFieldNumber);
442     return;
443   } else if (event_name_ == "g2d_tracing_mark_write") {
444     using TMW = protos::pbzero::G2dTracingMarkWriteFtraceEvent;
445     WriteValueForField(TMW::kTypeFieldNumber, [this](const Variadic& value) {
446       PERFETTO_DCHECK(value.type == Variadic::Type::kUint);
447       writer_->AppendChar(static_cast<char>(value.uint_value));
448     });
449     writer_->AppendString("|");
450     WriteValueForField(TMW::kPidFieldNumber);
451     writer_->AppendString("|");
452     WriteValueForField(TMW::kNameFieldNumber);
453     writer_->AppendString("|");
454     WriteValueForField(TMW::kValueFieldNumber);
455     return;
456   }
457   for (auto it = row_map_.IterateRows(); it; it.Next()) {
458     WriteArgAtRow(it.row());
459   }
460 }
461 
WriteArgAtRow(uint32_t arg_row,ValueWriter writer)462 void ArgsSerializer::WriteArgAtRow(uint32_t arg_row, ValueWriter writer) {
463   const auto& args = storage_->arg_table();
464   const auto& key = storage_->GetString(args.key()[arg_row]);
465   auto value = storage_->GetArgValue(arg_row);
466 
467   writer_->AppendChar(' ');
468   writer_->AppendString(key.c_str(), key.size());
469   writer_->AppendChar('=');
470 
471   if (key == "gfp_flags") {
472     auto kernel_version =
473         SystemInfoTracker::GetOrCreate(context_)->GetKernelVersion();
474     WriteGfpFlag(value.uint_value, kernel_version, writer_);
475     return;
476   }
477   writer(value);
478 }
479 
WriteValue(const Variadic & value)480 void ArgsSerializer::WriteValue(const Variadic& value) {
481   switch (value.type) {
482     case Variadic::kInt:
483       writer_->AppendInt(value.int_value);
484       break;
485     case Variadic::kUint:
486       writer_->AppendUnsignedInt(value.uint_value);
487       break;
488     case Variadic::kString: {
489       const auto& str = storage_->GetString(value.string_value);
490       writer_->AppendString(str.c_str(), str.size());
491       break;
492     }
493     case Variadic::kReal:
494       writer_->AppendDouble(value.real_value);
495       break;
496     case Variadic::kPointer:
497       writer_->AppendUnsignedInt(value.pointer_value);
498       break;
499     case Variadic::kBool:
500       writer_->AppendBool(value.bool_value);
501       break;
502     case Variadic::kJson: {
503       const auto& str = storage_->GetString(value.json_value);
504       writer_->AppendString(str.c_str(), str.size());
505       break;
506     }
507   }
508 }
509 
510 }  // namespace
511 
SqliteRawTable(sqlite3 * db,Context context)512 SqliteRawTable::SqliteRawTable(sqlite3* db, Context context)
513     : DbSqliteTable(
514           db,
515           {context.cache, tables::RawTable::Schema(), TableComputation::kStatic,
516            &context.context->storage->raw_table(), nullptr}),
517       serializer_(context.context) {
__anon94b96ab41e02(sqlite3_context* ctx, int argc, sqlite3_value** argv) 518   auto fn = [](sqlite3_context* ctx, int argc, sqlite3_value** argv) {
519     auto* thiz = static_cast<SqliteRawTable*>(sqlite3_user_data(ctx));
520     thiz->ToSystrace(ctx, argc, argv);
521   };
522   sqlite3_create_function(db, "to_ftrace", 1,
523                           SQLITE_UTF8 | SQLITE_DETERMINISTIC, this, fn, nullptr,
524                           nullptr);
525 }
526 
527 SqliteRawTable::~SqliteRawTable() = default;
528 
RegisterTable(sqlite3 * db,QueryCache * cache,TraceProcessorContext * context)529 void SqliteRawTable::RegisterTable(sqlite3* db,
530                                    QueryCache* cache,
531                                    TraceProcessorContext* context) {
532   SqliteTable::Register<SqliteRawTable, Context>(db, Context{cache, context},
533                                                  "raw");
534 }
535 
ToSystrace(sqlite3_context * ctx,int argc,sqlite3_value ** argv)536 void SqliteRawTable::ToSystrace(sqlite3_context* ctx,
537                                 int argc,
538                                 sqlite3_value** argv) {
539   if (argc != 1 || sqlite3_value_type(argv[0]) != SQLITE_INTEGER) {
540     sqlite3_result_error(ctx, "Usage: to_ftrace(id)", -1);
541     return;
542   }
543   uint32_t row = static_cast<uint32_t>(sqlite3_value_int64(argv[0]));
544 
545   auto str = serializer_.SerializeToString(row);
546   sqlite3_result_text(ctx, str.release(), -1, free);
547 }
548 
SystraceSerializer(TraceProcessorContext * context)549 SystraceSerializer::SystraceSerializer(TraceProcessorContext* context)
550     : context_(context) {
551   storage_ = context_->storage.get();
552 }
553 
SerializeToString(uint32_t raw_row)554 SystraceSerializer::ScopedCString SystraceSerializer::SerializeToString(
555     uint32_t raw_row) {
556   const auto& raw = storage_->raw_table();
557 
558   char line[4096];
559   base::StringWriter writer(line, sizeof(line));
560 
561   SerializePrefix(raw_row, &writer);
562 
563   StringId event_name_id = raw.name()[raw_row];
564   NullTermStringView event_name = storage_->GetString(event_name_id);
565   writer.AppendChar(' ');
566   if (event_name == "print" || event_name == "g2d_tracing_mark_write" ||
567       event_name == "dpu_tracing_mark_write") {
568     writer.AppendString("tracing_mark_write");
569   } else {
570     writer.AppendString(event_name.c_str(), event_name.size());
571   }
572   writer.AppendChar(':');
573 
574   ArgsSerializer serializer(context_, raw.arg_set_id()[raw_row], event_name,
575                             &proto_id_to_arg_index_by_event_[event_name_id],
576                             &writer);
577   serializer.SerializeArgs();
578 
579   return ScopedCString(writer.CreateStringCopy(), free);
580 }
581 
SerializePrefix(uint32_t raw_row,base::StringWriter * writer)582 void SystraceSerializer::SerializePrefix(uint32_t raw_row,
583                                          base::StringWriter* writer) {
584   const auto& raw = storage_->raw_table();
585 
586   int64_t ts = raw.ts()[raw_row];
587   uint32_t cpu = raw.cpu()[raw_row];
588 
589   UniqueTid utid = raw.utid()[raw_row];
590   uint32_t tid = storage_->thread_table().tid()[utid];
591 
592   uint32_t tgid = 0;
593   auto opt_upid = storage_->thread_table().upid()[utid];
594   if (opt_upid.has_value()) {
595     tgid = storage_->process_table().pid()[*opt_upid];
596   }
597   auto name = storage_->GetString(storage_->thread_table().name()[utid]);
598 
599   FtraceTime ftrace_time(ts);
600   if (tid == 0) {
601     name = "<idle>";
602   } else if (name.empty()) {
603     name = "<unknown>";
604   }
605 
606   int64_t padding = 16 - static_cast<int64_t>(name.size());
607   if (padding > 0) {
608     writer->AppendChar(' ', static_cast<size_t>(padding));
609   }
610   for (size_t i = 0; i < name.size(); ++i) {
611     char c = name.data()[i];
612     writer->AppendChar(c == '-' ? '_' : c);
613   }
614   writer->AppendChar('-');
615 
616   size_t pre_pid_pos = writer->pos();
617   writer->AppendInt(tid);
618   size_t pid_chars = writer->pos() - pre_pid_pos;
619   if (PERFETTO_LIKELY(pid_chars < 5)) {
620     writer->AppendChar(' ', 5 - pid_chars);
621   }
622 
623   writer->AppendLiteral(" (");
624   if (tgid == 0) {
625     writer->AppendLiteral("-----");
626   } else {
627     writer->AppendPaddedInt<' ', 5>(tgid);
628   }
629   writer->AppendLiteral(") [");
630   writer->AppendPaddedInt<'0', 3>(cpu);
631   writer->AppendLiteral("] .... ");
632 
633   writer->AppendInt(ftrace_time.secs);
634   writer->AppendChar('.');
635   writer->AppendPaddedInt<'0', 6>(ftrace_time.micros);
636   writer->AppendChar(':');
637 }
638 
639 }  // namespace trace_processor
640 }  // namespace perfetto
641