mirror of
				https://github.com/MaskRay/ccls.git
				synced 2025-10-24 17:12:41 +00:00 
			
		
		
		
	This fixes a plethora of "not indexed" errors when the document has not been indexed. * Message handler throws NotIndexed if not overdue * The message is put into backlog and tagged with backlog_path * path2backlog[path] tracks backlog associated with document `path` * The backlog is cleared when the index is merged * backlog[0] is forced to run if it becomes overdue
		
			
				
	
	
		
			273 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			273 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2017-2018 ccls Authors
 | |
| // SPDX-License-Identifier: Apache-2.0
 | |
| 
 | |
| #include "message_handler.hh"
 | |
| #include "pipeline.hh"
 | |
| #include "query.hh"
 | |
| 
 | |
| #include <algorithm>
 | |
| 
 | |
| MAKE_HASHABLE(ccls::SymbolIdx, t.usr, t.kind);
 | |
| 
 | |
| namespace ccls {
 | |
| REFLECT_STRUCT(SymbolInformation, name, kind, location, containerName);
 | |
| 
 | |
| namespace {
 | |
| struct DocumentHighlight {
 | |
|   enum Kind { Text = 1, Read = 2, Write = 3 };
 | |
| 
 | |
|   lsRange range;
 | |
|   int kind = 1;
 | |
| 
 | |
|   // ccls extension
 | |
|   Role role = Role::None;
 | |
| 
 | |
|   bool operator<(const DocumentHighlight &o) const {
 | |
|     return !(range == o.range) ? range < o.range : kind < o.kind;
 | |
|   }
 | |
| };
 | |
| REFLECT_STRUCT(DocumentHighlight, range, kind, role);
 | |
| } // namespace
 | |
| 
 | |
| void MessageHandler::textDocument_documentHighlight(
 | |
|     TextDocumentPositionParam ¶m, ReplyOnce &reply) {
 | |
|   int file_id;
 | |
|   auto [file, wf] = FindOrFail(param.textDocument.uri.GetPath(), reply, &file_id);
 | |
|   if (!wf)
 | |
|     return;
 | |
| 
 | |
|   std::vector<DocumentHighlight> result;
 | |
|   std::vector<SymbolRef> syms =
 | |
|       FindSymbolsAtLocation(wf, file, param.position, true);
 | |
|   for (auto [sym, refcnt] : file->symbol2refcnt) {
 | |
|     if (refcnt <= 0)
 | |
|       continue;
 | |
|     Usr usr = sym.usr;
 | |
|     Kind kind = sym.kind;
 | |
|     if (std::none_of(syms.begin(), syms.end(), [&](auto &sym1) {
 | |
|           return usr == sym1.usr && kind == sym1.kind;
 | |
|         }))
 | |
|       continue;
 | |
|     if (auto loc = GetLsLocation(db, wfiles, sym, file_id)) {
 | |
|       DocumentHighlight highlight;
 | |
|       highlight.range = loc->range;
 | |
|       if (sym.role & Role::Write)
 | |
|         highlight.kind = DocumentHighlight::Write;
 | |
|       else if (sym.role & Role::Read)
 | |
|         highlight.kind = DocumentHighlight::Read;
 | |
|       else
 | |
|         highlight.kind = DocumentHighlight::Text;
 | |
|       highlight.role = sym.role;
 | |
|       result.push_back(highlight);
 | |
|     }
 | |
|   }
 | |
|   std::sort(result.begin(), result.end());
 | |
|   reply(result);
 | |
| }
 | |
| 
 | |
| namespace {
 | |
| struct DocumentLink {
 | |
|   lsRange range;
 | |
|   DocumentUri target;
 | |
| };
 | |
| REFLECT_STRUCT(DocumentLink, range, target);
 | |
| } // namespace
 | |
| 
 | |
| void MessageHandler::textDocument_documentLink(TextDocumentParam ¶m,
 | |
|                                                ReplyOnce &reply) {
 | |
|   auto [file, wf] = FindOrFail(param.textDocument.uri.GetPath(), reply);
 | |
|   if (!wf) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   std::vector<DocumentLink> result;
 | |
|   for (const IndexInclude &include : file->def->includes)
 | |
|     if (std::optional<int> bline =
 | |
|             wf->GetBufferPosFromIndexPos(include.line, nullptr, false)) {
 | |
|       const std::string &line = wf->buffer_lines[*bline];
 | |
|       auto start = line.find_first_of("\"<"), end = line.find_last_of("\">");
 | |
|       if (start < end)
 | |
|         result.push_back({lsRange{{*bline, (int)start + 1}, {*bline, (int)end}},
 | |
|                           DocumentUri::FromPath(include.resolved_path)});
 | |
|     }
 | |
|   reply(result);
 | |
| } // namespace ccls
 | |
| 
 | |
| namespace {
 | |
| struct DocumentSymbolParam : TextDocumentParam {
 | |
|   // false: outline; true: all symbols
 | |
|   bool all = false;
 | |
|   // If >= 0, return Range[] instead of SymbolInformation[] to reduce output.
 | |
|   int startLine = -1;
 | |
|   int endLine = -1;
 | |
| };
 | |
| REFLECT_STRUCT(DocumentSymbolParam, textDocument, all, startLine, endLine);
 | |
| 
 | |
| struct DocumentSymbol {
 | |
|   std::string name;
 | |
|   std::string detail;
 | |
|   SymbolKind kind;
 | |
|   lsRange range;
 | |
|   lsRange selectionRange;
 | |
|   std::vector<std::unique_ptr<DocumentSymbol>> children;
 | |
| };
 | |
| void Reflect(JsonWriter &vis, std::unique_ptr<DocumentSymbol> &v);
 | |
| REFLECT_STRUCT(DocumentSymbol, name, detail, kind, range, selectionRange,
 | |
|                children);
 | |
| void Reflect(JsonWriter &vis, std::unique_ptr<DocumentSymbol> &v) {
 | |
|   Reflect(vis, *v);
 | |
| }
 | |
| 
 | |
| template <typename Def>
 | |
| bool Ignore(const Def *def) {
 | |
|   return false;
 | |
| }
 | |
| template <>
 | |
| bool Ignore(const QueryType::Def *def) {
 | |
|   return !def || def->kind == SymbolKind::TypeParameter;
 | |
| }
 | |
| template<>
 | |
| bool Ignore(const QueryVar::Def *def) {
 | |
|   return !def || def->is_local();
 | |
| }
 | |
| 
 | |
| void Uniquify(std::vector<std::unique_ptr<DocumentSymbol>> &cs) {
 | |
|   std::sort(cs.begin(), cs.end(),
 | |
|             [](auto &l, auto &r) { return l->range < r->range; });
 | |
|   cs.erase(std::unique(cs.begin(), cs.end(),
 | |
|                        [](auto &l, auto &r) { return l->range == r->range; }),
 | |
|            cs.end());
 | |
|   for (auto &c : cs)
 | |
|     Uniquify(c->children);
 | |
| }
 | |
| } // namespace
 | |
| 
 | |
| void MessageHandler::textDocument_documentSymbol(JsonReader &reader,
 | |
|                                                  ReplyOnce &reply) {
 | |
|   DocumentSymbolParam param;
 | |
|   Reflect(reader, param);
 | |
| 
 | |
|   int file_id;
 | |
|   auto [file, wf] = FindOrFail(param.textDocument.uri.GetPath(), reply, &file_id);
 | |
|   if (!wf) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (param.startLine >= 0) {
 | |
|     std::vector<lsRange> result;
 | |
|     for (auto [sym, refcnt] : file->symbol2refcnt)
 | |
|       if (refcnt > 0 && (param.all || sym.extent.Valid()) &&
 | |
|           param.startLine <= sym.range.start.line &&
 | |
|           sym.range.start.line <= param.endLine)
 | |
|         if (auto loc = GetLsLocation(db, wfiles, sym, file_id))
 | |
|           result.push_back(loc->range);
 | |
|     std::sort(result.begin(), result.end());
 | |
|     reply(result);
 | |
|   } else if (g_config->client.hierarchicalDocumentSymbolSupport) {
 | |
|     std::unordered_map<SymbolIdx, std::unique_ptr<DocumentSymbol>> sym2ds;
 | |
|     std::vector<std::pair<std::vector<const void *>, DocumentSymbol *>> funcs,
 | |
|         types;
 | |
|     for (auto [sym, refcnt] : file->symbol2refcnt) {
 | |
|       if (refcnt <= 0 || !sym.extent.Valid())
 | |
|         continue;
 | |
|       auto r = sym2ds.try_emplace(SymbolIdx{sym.usr, sym.kind});
 | |
|       if (!r.second)
 | |
|         continue;
 | |
|       auto &ds = r.first->second;
 | |
|       ds = std::make_unique<DocumentSymbol>();
 | |
|       if (auto range = GetLsRange(wf, sym.range)) {
 | |
|         ds->selectionRange = *range;
 | |
|         ds->range = ds->selectionRange;
 | |
|         // For a macro expansion, M(name), we may use `M` for extent and `name`
 | |
|         // for spell, do the check as selectionRange must be a subrange of
 | |
|         // range.
 | |
|         if (sym.extent.Valid())
 | |
|           if (auto range1 = GetLsRange(wf, sym.extent);
 | |
|               range1 && range1->Includes(*range))
 | |
|             ds->range = *range1;
 | |
|       }
 | |
|       std::vector<const void *> def_ptrs;
 | |
|       WithEntity(db, sym, [&](const auto &entity) {
 | |
|         auto *def = entity.AnyDef();
 | |
|         if (!def)
 | |
|           return;
 | |
|         ds->name = def->Name(false);
 | |
|         ds->detail = def->detailed_name;
 | |
|         for (auto &def : entity.def)
 | |
|           if (def.file_id == file_id && !Ignore(&def)) {
 | |
|             ds->kind = def.kind;
 | |
|             if (def.spell || def.kind == SymbolKind::Namespace)
 | |
|               def_ptrs.push_back(&def);
 | |
|           }
 | |
|       });
 | |
|       if (!(param.all || sym.role & Role::Definition ||
 | |
|             ds->kind == SymbolKind::Function ||
 | |
|             ds->kind == SymbolKind::Method ||
 | |
|             ds->kind == SymbolKind::Namespace)) {
 | |
|         ds.reset();
 | |
|         continue;
 | |
|       }
 | |
|       if (def_ptrs.empty())
 | |
|         continue;
 | |
|       if (sym.kind == Kind::Func)
 | |
|         funcs.emplace_back(std::move(def_ptrs), ds.get());
 | |
|       else if (sym.kind == Kind::Type)
 | |
|         types.emplace_back(std::move(def_ptrs), ds.get());
 | |
|     }
 | |
| 
 | |
|     for (auto &[def_ptrs, ds] : funcs)
 | |
|       for (const void *def_ptr : def_ptrs)
 | |
|         for (Usr usr1 : ((const QueryFunc::Def *)def_ptr)->vars) {
 | |
|           auto it = sym2ds.find(SymbolIdx{usr1, Kind::Var});
 | |
|           if (it != sym2ds.end() && it->second)
 | |
|             ds->children.push_back(std::move(it->second));
 | |
|         }
 | |
|     for (auto &[def_ptrs, ds] : types)
 | |
|       for (const void *def_ptr : def_ptrs) {
 | |
|         auto *def = (const QueryType::Def *)def_ptr;
 | |
|         for (Usr usr1 : def->funcs) {
 | |
|           auto it = sym2ds.find(SymbolIdx{usr1, Kind::Func});
 | |
|           if (it != sym2ds.end() && it->second)
 | |
|             ds->children.push_back(std::move(it->second));
 | |
|         }
 | |
|         for (Usr usr1 : def->types) {
 | |
|           auto it = sym2ds.find(SymbolIdx{usr1, Kind::Type});
 | |
|           if (it != sym2ds.end() && it->second)
 | |
|             ds->children.push_back(std::move(it->second));
 | |
|         }
 | |
|         for (auto [usr1, _] : def->vars) {
 | |
|           auto it = sym2ds.find(SymbolIdx{usr1, Kind::Var});
 | |
|           if (it != sym2ds.end() && it->second)
 | |
|             ds->children.push_back(std::move(it->second));
 | |
|         }
 | |
|       }
 | |
|     std::vector<std::unique_ptr<DocumentSymbol>> result;
 | |
|     for (auto &[_, ds] : sym2ds)
 | |
|       if (ds) {
 | |
|         Uniquify(ds->children);
 | |
|         result.push_back(std::move(ds));
 | |
|       }
 | |
|     Uniquify(result);
 | |
|     reply(result);
 | |
|   } else {
 | |
|     std::vector<SymbolInformation> result;
 | |
|     for (auto [sym, refcnt] : file->symbol2refcnt) {
 | |
|       if (refcnt <= 0 || !sym.extent.Valid() ||
 | |
|           !(param.all || sym.role & Role::Definition))
 | |
|         continue;
 | |
|       if (std::optional<SymbolInformation> info =
 | |
|               GetSymbolInfo(db, sym, false)) {
 | |
|         if ((sym.kind == Kind::Type && Ignore(db->GetType(sym).AnyDef())) ||
 | |
|             (sym.kind == Kind::Var && Ignore(db->GetVar(sym).AnyDef())))
 | |
|           continue;
 | |
|         if (auto loc = GetLsLocation(db, wfiles, sym, file_id)) {
 | |
|           info->location = *loc;
 | |
|           result.push_back(*info);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     reply(result);
 | |
|   }
 | |
| }
 | |
| } // namespace ccls
 |