| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #ifndef CARBON_TOOLCHAIN_SEM_IR_FORMATTER_CHUNKS_H_
- #define CARBON_TOOLCHAIN_SEM_IR_FORMATTER_CHUNKS_H_
- #include <string>
- #include "common/check.h"
- #include "llvm/ADT/SmallVector.h"
- #include "llvm/Support/raw_ostream.h"
- namespace Carbon::SemIR {
- // Manages the chunks created by the formatter.
- //
- // There are two kinds of `Chunk`s:
- // - Parent `Chunk`s, with children in a vector.
- // - Content `Chunk`s, with content in a string.
- //
- // The high level usage is:
- // 1. Calls to `AddParent` to prepare parent `Chunk`s.
- // 2. One call to `StartContent` to switch modes.
- // 3. Calls to `FormatChildContent`, `AppendChildToCurrentParent`, and `out`.
- // 4. Calls to `Write`.
- class FormatterChunks {
- public:
- // A type-safe index into `chunks_`.
- struct ChunkId {
- auto operator==(const ChunkId& other) const -> bool = default;
- size_t index;
- };
- // An empty `ChunkId`.
- static constexpr ChunkId None = ChunkId(-1);
- // Reserves space for at least `count` chunks.
- auto Reserve(size_t count) -> void { chunks_.reserve(count); }
- // Adds a parent `Chunk` and returns its `ChunkId`. If `child_chunk_id` isn't
- // `None`, it's added as a child. Must be called before `StartContent`.
- //
- // By default the parent `Chunk` will not be included in the output, and
- // `AppendChildToCurrentParent` must be called to include it.
- auto AddParent(ChunkId child_chunk_id = None) -> ChunkId;
- // Switches from adding parents to adding content.
- auto StartContent() -> void;
- // Calls `format` to add content conditionally included when `parent_chunk_id`
- // is included. Must be called after `StartContent`.
- //
- // During a `FormatChildContent` call where the `parent_chunk_id` is not
- // already included in output, a new content `Chunk` is created, marked as a
- // child of `parent_chunk_id`, and `out` is temporarily directed to it during
- // the duration of `format. Otherwise, `out` lazily creates a content `Chunk`
- // which is always included in output.
- auto FormatChildContent(ChunkId parent_chunk_id,
- llvm::function_ref<auto()->void> format) -> void;
- // Adds `child_chunk_id` to the children of a `FormatChildContent`'s
- // `parent_chunk_id` if called during `format`, or otherwise includes the
- // chunk in output.
- auto AppendChildToCurrentParent(ChunkId child_chunk_id) -> void;
- // Writes included chunks to the given stream.
- auto Write(llvm::raw_ostream& stream) -> void;
- // Returns a stream to write to a content `Chunk`. The returned reference is
- // only valid until the next `Chunk` starts. Must be called after
- // `StartContent`.
- //
- // See `FormatChildContent` for details of the target content `Chunk`.
- auto out() -> llvm::raw_ostream& {
- CARBON_CHECK(content_start_id_ != None);
- if (!out_) {
- AddContent(/*include_in_output=*/true);
- }
- return *out_;
- }
- auto size() -> size_t { return chunks_.size(); }
- private:
- // Either a parent or content.
- struct Chunk {
- // Whether this chunk is known to be included in the output.
- bool include_in_output;
- // Either children or content.
- std::variant<llvm::SmallVector<ChunkId>, std::string> data;
- };
- // Adds a `Chunk` that will have `content`, and directs `out_` to it.
- auto AddContent(bool include_in_output) -> ChunkId;
- // Adds `child_chunk_id` to the children of `parent_chunk_id`.
- auto AppendChildToParent(ChunkId child_chunk_id, ChunkId parent_chunk_id)
- -> void;
- // Indexes into `chunks_`.
- auto Get(ChunkId chunk_id) -> Chunk& { return chunks_[chunk_id.index]; }
- // An output stream pointing at the current content `Chunk`.
- std::unique_ptr<llvm::raw_string_ostream> out_;
- // The location where content started. Set by `StartContent`.
- ChunkId content_start_id_ = None;
- // The current parent `Chunk`. This is only set during calls to
- // `FormatChildContent`.
- ChunkId current_parent_id_ = None;
- // A sequential ordering of `Chunk`s. This will have all parent `Chunk`s
- // first, followed by content `Chunk`s at `content_start_`.
- llvm::SmallVector<Chunk> chunks_;
- };
- } // namespace Carbon::SemIR
- #endif // CARBON_TOOLCHAIN_SEM_IR_FORMATTER_CHUNKS_H_
|