//===- CTagsEmitter.cpp - Generate ctags-compatible index -----------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // This tablegen backend emits an index of definitions in ctags(1) format. // A helper script, utils/TableGen/tdtags, provides an easier-to-use // interface; run 'tdtags -H' for documentation. // //===----------------------------------------------------------------------===// #include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/SourceMgr.h" #include "llvm/TableGen/Error.h" #include "llvm/TableGen/Record.h" #include "llvm/TableGen/TableGenBackend.h" #include #include using namespace llvm; #define DEBUG_TYPE "ctags-emitter" namespace { class Tag { private: StringRef Id; StringRef BufferIdentifier; unsigned Line; public: Tag(StringRef Name, const SMLoc Location) : Id(Name) { const MemoryBuffer *CurMB = SrcMgr.getMemoryBuffer(SrcMgr.FindBufferContainingLoc(Location)); BufferIdentifier = CurMB->getBufferIdentifier(); auto LineAndColumn = SrcMgr.getLineAndColumn(Location); Line = LineAndColumn.first; } int operator<(const Tag &B) const { return std::tuple(Id, BufferIdentifier, Line) < std::tuple(B.Id, B.BufferIdentifier, B.Line); } void emit(raw_ostream &OS) const { OS << Id << "\t" << BufferIdentifier << "\t" << Line << "\n"; } }; class CTagsEmitter { private: const RecordKeeper &Records; public: CTagsEmitter(const RecordKeeper &R) : Records(R) {} void run(raw_ostream &OS); private: static SMLoc locate(const Record *R); }; } // End anonymous namespace. SMLoc CTagsEmitter::locate(const Record *R) { ArrayRef Locs = R->getLoc(); return !Locs.empty() ? Locs.front() : SMLoc(); } void CTagsEmitter::run(raw_ostream &OS) { const auto &Classes = Records.getClasses(); const auto &Defs = Records.getDefs(); std::vector Tags; // Collect tags. Tags.reserve(Classes.size() + Defs.size()); for (const auto &C : Classes) { Tags.push_back(Tag(C.first, locate(C.second.get()))); for (SMLoc FwdLoc : C.second->getForwardDeclarationLocs()) Tags.push_back(Tag(C.first, FwdLoc)); } for (const auto &D : Defs) Tags.push_back(Tag(D.first, locate(D.second.get()))); // Emit tags. llvm::sort(Tags); OS << "!_TAG_FILE_FORMAT\t1\t/original ctags format/\n"; OS << "!_TAG_FILE_SORTED\t1\t/0=unsorted, 1=sorted, 2=foldcase/\n"; for (const Tag &T : Tags) T.emit(OS); } static TableGen::Emitter::OptClass X("gen-ctags", "Generate ctags-compatible index");