12Sjlaskey//===-- Language.h ---------------------------------------------------*- C++
26Sjlaskey//-*-===//
32Sjlaskey//
4877Sattila// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
52Sjlaskey// See https://llvm.org/LICENSE.txt for license information.
62Sjlaskey// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
72Sjlaskey//
8877Sattila//===----------------------------------------------------------------------===//
92Sjlaskey
102Sjlaskey#ifndef LLDB_TARGET_LANGUAGE_H
112Sjlaskey#define LLDB_TARGET_LANGUAGE_H
122Sjlaskey
132Sjlaskey#include <functional>
14877Sattila#include <memory>
152Sjlaskey#include <set>
162Sjlaskey#include <vector>
172Sjlaskey
18877Sattila#include "lldb/Core/Highlighter.h"
192Sjlaskey#include "lldb/Core/PluginInterface.h"
202Sjlaskey#include "lldb/DataFormatters/DumpValueObjectOptions.h"
212Sjlaskey#include "lldb/DataFormatters/FormatClasses.h"
222Sjlaskey#include "lldb/DataFormatters/StringPrinter.h"
232Sjlaskey#include "lldb/Symbol/TypeSystem.h"
242Sjlaskey#include "lldb/lldb-private.h"
252Sjlaskey#include "lldb/lldb-public.h"
262Sjlaskey
272Sjlaskeynamespace lldb_private {
282Sjlaskey
292Sjlaskeyclass Language : public PluginInterface {
302Sjlaskeypublic:
312Sjlaskey  class TypeScavenger {
322Sjlaskey  public:
332Sjlaskey    class Result {
342Sjlaskey    public:
352Sjlaskey      virtual bool IsValid() = 0;
362Sjlaskey
372Sjlaskey      virtual bool DumpToStream(Stream &stream,
382Sjlaskey                                bool print_help_if_available) = 0;
392Sjlaskey
402Sjlaskey      virtual ~Result() = default;
412Sjlaskey    };
422Sjlaskey
432Sjlaskey    typedef std::set<std::unique_ptr<Result>> ResultSet;
442Sjlaskey
452Sjlaskey    virtual ~TypeScavenger() = default;
462Sjlaskey
472Sjlaskey    size_t Find(ExecutionContextScope *exe_scope, const char *key,
482Sjlaskey                ResultSet &results, bool append = true);
492Sjlaskey
502Sjlaskey  protected:
512Sjlaskey    TypeScavenger() = default;
522Sjlaskey
532Sjlaskey    virtual bool Find_Impl(ExecutionContextScope *exe_scope, const char *key,
542Sjlaskey                           ResultSet &results) = 0;
552Sjlaskey  };
562Sjlaskey
572Sjlaskey  class ImageListTypeScavenger : public TypeScavenger {
582Sjlaskey    class Result : public Language::TypeScavenger::Result {
592Sjlaskey    public:
602Sjlaskey      Result(CompilerType type) : m_compiler_type(type) {}
612Sjlaskey
622Sjlaskey      bool IsValid() override { return m_compiler_type.IsValid(); }
632Sjlaskey
642Sjlaskey      bool DumpToStream(Stream &stream, bool print_help_if_available) override {
652Sjlaskey        if (IsValid()) {
662Sjlaskey          m_compiler_type.DumpTypeDescription(&stream);
672Sjlaskey          stream.EOL();
682Sjlaskey          return true;
692Sjlaskey        }
702Sjlaskey        return false;
712Sjlaskey      }
722Sjlaskey
732Sjlaskey      ~Result() override = default;
742Sjlaskey
752Sjlaskey    private:
762Sjlaskey      CompilerType m_compiler_type;
772Sjlaskey    };
782Sjlaskey
792Sjlaskey  protected:
802Sjlaskey    ImageListTypeScavenger() = default;
812Sjlaskey
822Sjlaskey    ~ImageListTypeScavenger() override = default;
832Sjlaskey
842Sjlaskey    // is this type something we should accept? it's usually going to be a
852Sjlaskey    // filter by language + maybe some sugar tweaking
862Sjlaskey    // returning an empty type means rejecting this candidate entirely;
872Sjlaskey    // any other result will be accepted as a valid match
882Sjlaskey    virtual CompilerType AdjustForInclusion(CompilerType &candidate) = 0;
892Sjlaskey
902Sjlaskey    bool Find_Impl(ExecutionContextScope *exe_scope, const char *key,
912Sjlaskey                   ResultSet &results) override;
922Sjlaskey  };
932Sjlaskey
942Sjlaskey  template <typename... ScavengerTypes>
952Sjlaskey  class EitherTypeScavenger : public TypeScavenger {
962Sjlaskey  public:
972Sjlaskey    EitherTypeScavenger() : TypeScavenger() {
982Sjlaskey      for (std::shared_ptr<TypeScavenger> scavenger : { std::shared_ptr<TypeScavenger>(new ScavengerTypes())... }) {
992Sjlaskey        if (scavenger)
1002Sjlaskey          m_scavengers.push_back(scavenger);
1012Sjlaskey      }
1022Sjlaskey    }
1032Sjlaskey  protected:
1042Sjlaskey    bool Find_Impl(ExecutionContextScope *exe_scope, const char *key,
1052Sjlaskey                   ResultSet &results) override {
1062Sjlaskey      const bool append = false;
1072Sjlaskey      for (auto& scavenger : m_scavengers) {
1082Sjlaskey        if (scavenger && scavenger->Find(exe_scope, key, results, append))
1092Sjlaskey          return true;
1102Sjlaskey      }
1112Sjlaskey      return false;
1122Sjlaskey    }
1132Sjlaskey  private:
1142Sjlaskey    std::vector<std::shared_ptr<TypeScavenger>> m_scavengers;
1152Sjlaskey  };
1162Sjlaskey
1172Sjlaskey  template <typename... ScavengerTypes>
1182Sjlaskey  class UnionTypeScavenger : public TypeScavenger {
1192Sjlaskey  public:
1202Sjlaskey    UnionTypeScavenger() : TypeScavenger() {
1212Sjlaskey      for (std::shared_ptr<TypeScavenger> scavenger : { std::shared_ptr<TypeScavenger>(new ScavengerTypes())... }) {
1222Sjlaskey        if (scavenger)
1232Sjlaskey          m_scavengers.push_back(scavenger);
1242Sjlaskey      }
1252Sjlaskey    }
1262Sjlaskey  protected:
127    bool Find_Impl(ExecutionContextScope *exe_scope, const char *key,
128                   ResultSet &results) override {
129      const bool append = true;
130      bool success = false;
131      for (auto& scavenger : m_scavengers) {
132        if (scavenger)
133          success = scavenger->Find(exe_scope, key, results, append) || success;
134      }
135      return success;
136    }
137  private:
138    std::vector<std::shared_ptr<TypeScavenger>> m_scavengers;
139  };
140
141  enum class FunctionNameRepresentation {
142    eName,
143    eNameWithArgs,
144    eNameWithNoArgs
145  };
146
147  ~Language() override;
148
149  static Language *FindPlugin(lldb::LanguageType language);
150
151  /// Returns the Language associated with the given file path or a nullptr
152  /// if there is no known language.
153  static Language *FindPlugin(llvm::StringRef file_path);
154
155  static Language *FindPlugin(lldb::LanguageType language,
156                              llvm::StringRef file_path);
157
158  // return false from callback to stop iterating
159  static void ForEach(std::function<bool(Language *)> callback);
160
161  virtual lldb::LanguageType GetLanguageType() const = 0;
162
163  // Implement this function to return the user-defined entry point name
164  // for the language.
165  virtual llvm::StringRef GetUserEntryPointName() const { return {}; }
166
167  virtual bool IsTopLevelFunction(Function &function);
168
169  virtual bool IsSourceFile(llvm::StringRef file_path) const = 0;
170
171  virtual const Highlighter *GetHighlighter() const { return nullptr; }
172
173  virtual lldb::TypeCategoryImplSP GetFormatters();
174
175  virtual HardcodedFormatters::HardcodedFormatFinder GetHardcodedFormats();
176
177  virtual HardcodedFormatters::HardcodedSummaryFinder GetHardcodedSummaries();
178
179  virtual HardcodedFormatters::HardcodedSyntheticFinder
180  GetHardcodedSynthetics();
181
182  virtual std::vector<FormattersMatchCandidate>
183  GetPossibleFormattersMatches(ValueObject &valobj,
184                               lldb::DynamicValueType use_dynamic);
185
186  virtual std::unique_ptr<TypeScavenger> GetTypeScavenger();
187
188  virtual const char *GetLanguageSpecificTypeLookupHelp();
189
190  class MethodNameVariant {
191    ConstString m_name;
192    lldb::FunctionNameType m_type;
193
194  public:
195    MethodNameVariant(ConstString name, lldb::FunctionNameType type)
196        : m_name(name), m_type(type) {}
197    ConstString GetName() const { return m_name; }
198    lldb::FunctionNameType GetType() const { return m_type; }
199  };
200  // If a language can have more than one possible name for a method, this
201  // function can be used to enumerate them. This is useful when doing name
202  // lookups.
203  virtual std::vector<Language::MethodNameVariant>
204  GetMethodNameVariants(ConstString method_name) const {
205    return std::vector<Language::MethodNameVariant>();
206  };
207
208  /// Returns true iff the given symbol name is compatible with the mangling
209  /// scheme of this language.
210  ///
211  /// This function should only return true if there is a high confidence
212  /// that the name actually belongs to this language.
213  virtual bool SymbolNameFitsToLanguage(Mangled name) const { return false; }
214
215  /// An individual data formatter may apply to several types and cross language
216  /// boundaries. Each of those languages may want to customize the display of
217  /// values of said types by appending proper prefix/suffix information in
218  /// language-specific ways. This function returns that prefix and suffix.
219  ///
220  /// \param[in] type_hint
221  ///   A StringRef used to determine what the prefix and suffix should be. It
222  ///   is called a hint because some types may have multiple variants for which
223  ///   the prefix and/or suffix may vary.
224  ///
225  /// \return
226  ///   A std::pair<StringRef, StringRef>, the first being the prefix and the
227  ///   second being the suffix. They may be empty.
228  virtual std::pair<llvm::StringRef, llvm::StringRef>
229  GetFormatterPrefixSuffix(llvm::StringRef type_hint);
230
231  // When looking up functions, we take a user provided string which may be a
232  // partial match to the full demangled name and compare it to the actual
233  // demangled name to see if it matches as much as the user specified.  An
234  // example of this is if the user provided A::my_function, but the
235  // symbol was really B::A::my_function.  We want that to be
236  // a match.  But we wouldn't want this to match AnotherA::my_function.  The
237  // user is specifying a truncated path, not a truncated set of characters.
238  // This function does a language-aware comparison for those purposes.
239  virtual bool DemangledNameContainsPath(llvm::StringRef path,
240                                         ConstString demangled) const;
241
242  // if a language has a custom format for printing variable declarations that
243  // it wants LLDB to honor it should return an appropriate closure here
244  virtual DumpValueObjectOptions::DeclPrintingHelper GetDeclPrintingHelper();
245
246  virtual LazyBool IsLogicalTrue(ValueObject &valobj, Status &error);
247
248  // for a ValueObject of some "reference type", if the value points to the
249  // nil/null object, this method returns true
250  virtual bool IsNilReference(ValueObject &valobj);
251
252  /// Returns the summary string for ValueObjects for which IsNilReference() is
253  /// true.
254  virtual llvm::StringRef GetNilReferenceSummaryString() { return {}; }
255
256  // for a ValueObject of some "reference type", if the language provides a
257  // technique to decide whether the reference has ever been assigned to some
258  // object, this method will return true if such detection is possible, and if
259  // the reference has never been assigned
260  virtual bool IsUninitializedReference(ValueObject &valobj);
261
262  virtual bool GetFunctionDisplayName(const SymbolContext *sc,
263                                      const ExecutionContext *exe_ctx,
264                                      FunctionNameRepresentation representation,
265                                      Stream &s);
266
267  virtual ConstString
268  GetDemangledFunctionNameWithoutArguments(Mangled mangled) const {
269    if (ConstString demangled = mangled.GetDemangledName())
270      return demangled;
271
272    return mangled.GetMangledName();
273  }
274
275  virtual void GetExceptionResolverDescription(bool catch_on, bool throw_on,
276                                               Stream &s);
277
278  static void GetDefaultExceptionResolverDescription(bool catch_on,
279                                                     bool throw_on, Stream &s);
280
281  // These are accessors for general information about the Languages lldb knows
282  // about:
283
284  static lldb::LanguageType
285  GetLanguageTypeFromString(const char *string) = delete;
286  static lldb::LanguageType GetLanguageTypeFromString(llvm::StringRef string);
287
288  static const char *GetNameForLanguageType(lldb::LanguageType language);
289
290  static void PrintAllLanguages(Stream &s, const char *prefix,
291                                const char *suffix);
292
293  /// Prints to the specified stream 's' each language type that the
294  /// current target supports for expression evaluation.
295  ///
296  /// \param[out] s      Stream to which the language types are written.
297  /// \param[in]  prefix String that is prepended to the language type.
298  /// \param[in]  suffix String that is appended to the language type.
299  static void PrintSupportedLanguagesForExpressions(Stream &s,
300                                                    llvm::StringRef prefix,
301                                                    llvm::StringRef suffix);
302
303  // return false from callback to stop iterating
304  static void ForAllLanguages(std::function<bool(lldb::LanguageType)> callback);
305
306  static bool LanguageIsCPlusPlus(lldb::LanguageType language);
307
308  static bool LanguageIsObjC(lldb::LanguageType language);
309
310  static bool LanguageIsC(lldb::LanguageType language);
311
312  /// Equivalent to \c LanguageIsC||LanguageIsObjC||LanguageIsCPlusPlus.
313  static bool LanguageIsCFamily(lldb::LanguageType language);
314
315  static bool LanguageIsPascal(lldb::LanguageType language);
316
317  // return the primary language, so if LanguageIsC(l), return eLanguageTypeC,
318  // etc.
319  static lldb::LanguageType GetPrimaryLanguage(lldb::LanguageType language);
320
321  static std::set<lldb::LanguageType> GetSupportedLanguages();
322
323  static LanguageSet GetLanguagesSupportingTypeSystems();
324  static LanguageSet GetLanguagesSupportingTypeSystemsForExpressions();
325  static LanguageSet GetLanguagesSupportingREPLs();
326
327  // Given a mangled function name, calculates some alternative manglings since
328  // the compiler mangling may not line up with the symbol we are expecting.
329  virtual std::vector<ConstString>
330  GenerateAlternateFunctionManglings(const ConstString mangled) const {
331    return std::vector<ConstString>();
332  }
333
334  virtual ConstString
335  FindBestAlternateFunctionMangledName(const Mangled mangled,
336                                       const SymbolContext &sym_ctx) const {
337    return ConstString();
338  }
339
340  virtual llvm::StringRef GetInstanceVariableName() { return {}; }
341
342protected:
343  // Classes that inherit from Language can see and modify these
344
345  Language();
346
347private:
348  Language(const Language &) = delete;
349  const Language &operator=(const Language &) = delete;
350};
351
352} // namespace lldb_private
353
354#endif // LLDB_TARGET_LANGUAGE_H
355