blob: 1f3494f86f49cf1e7f3c2e89b077d5e6390ac258 [file] [log] [blame]
/*
* 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.
*/
#ifndef SRC_TRACING_CORE_TRACE_WRITER_IMPL_H_
#define SRC_TRACING_CORE_TRACE_WRITER_IMPL_H_
#include <forward_list>
#include "perfetto/protozero/protozero_message_handle.h"
#include "perfetto/protozero/scattered_stream_writer.h"
#include "perfetto/tracing/core/basic_types.h"
#include "perfetto/tracing/core/shared_memory_abi.h"
#include "perfetto/tracing/core/trace_writer.h"
namespace perfetto {
class SharedMemoryArbiterImpl;
// See //include/perfetto/tracing/core/trace_writer.h for docs.
class TraceWriterImpl : public TraceWriter,
public protozero::ScatteredStreamWriter::Delegate {
public:
// TracePacketHandle is defined in trace_writer.h
TraceWriterImpl(SharedMemoryArbiterImpl*, WriterID, BufferID);
~TraceWriterImpl() override;
// TraceWriter implementation. See documentation in trace_writer.h .
TracePacketHandle NewTracePacket() override;
WriterID writer_id() const override;
private:
// Used to handle the backfilling of the headers (the |size_field|) of nested
// messages when a proto is fragmented over several chunks. This patchlist is
// sent out of band to the tracing service, after having returned the initial
// chunks of the fragment.
struct Patch {
Patch(uint16_t chunk_id, uint16_t offset);
Patch(const Patch&) = delete;
Patch& operator=(const Patch&) = delete;
Patch(Patch&&) noexcept = delete;
Patch& operator=(Patch&&) = delete;
const uint16_t chunk_id;
const uint16_t offset_in_chunk;
uint8_t size_field[SharedMemoryABI::kPacketHeaderSize] = {};
};
TraceWriterImpl(const TraceWriterImpl&) = delete;
TraceWriterImpl& operator=(const TraceWriterImpl&) = delete;
// ScatteredStreamWriter::Delegate implementation.
protozero::ContiguousMemoryRange GetNewBuffer() override;
// The per-producer arbiter that coordinates access to the shared memory
// buffer from several threads.
SharedMemoryArbiterImpl* const shmem_arbiter_;
// ID of the current writer.
const WriterID id_;
// This is just copied back into the chunk header.
// See comments in data_source_config.proto for |target_buffer|.
const BufferID target_buffer_;
// Monotonic (% wrapping) sequence id of the chunk. Together with the WriterID
// this allows the Service to reconstruct the linear sequence of packets.
uint16_t cur_chunk_id_ = 0;
// The chunk we are holding onto (if any).
SharedMemoryABI::Chunk cur_chunk_;
// Passed to protozero message to write directly into |cur_chunk_|. It
// keeps track of the write pointer. It calls us back (GetNewBuffer()) when
// |cur_chunk_| is filled.
protozero::ScatteredStreamWriter protobuf_stream_writer_;
// The packet returned via NewTracePacket(). Its owned by this class,
// TracePacketHandle has just a pointer to it.
std::unique_ptr<protos::pbzero::TracePacket> cur_packet_;
// The start address of |cur_packet_| within |cur_chunk_|. Used to figure out
// fragments sizes when a TracePacket write is interrupted by GetNewBuffer().
uint8_t* cur_fragment_start_ = nullptr;
// true if we received a call to GetNewBuffer() after NewTracePacket(),
// false if GetNewBuffer() happened during NewTracePacket() prologue, while
// starting the TracePacket header.
bool fragmenting_packet_ = false;
// When a packet is fragmented across different chunks, the |size_field| of
// the outstanding nested protobuf messages is redirected onto Patch entries
// in this list at the time the Chunk is returned (because at that point we
// have to release the ownership of the current Chunk). This list will be
// later sent out-of-band to the tracing service, who will patch the required
// chunks, if they are still around.
// Note: the ProtoZeroMessage will take pointers to the |size_field| of these
// entries. This container must guarantee that the Patch objects are never
// moved around (i.e. cannot be a vector because of reallocations can change
// addresses of pre-existing entries).
std::forward_list<Patch> patch_list_;
};
} // namespace perfetto
#endif // SRC_TRACING_CORE_TRACE_WRITER_IMPL_H_