Coverage Report

Created: 2023-09-12 09:32

/Users/buildslave/jenkins/workspace/coverage/llvm-project/lldb/include/lldb/Target/Language.h
Line
Count
Source (jump to first uncovered line)
1
//===-- Language.h ---------------------------------------------------*- C++
2
//-*-===//
3
//
4
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5
// See https://llvm.org/LICENSE.txt for license information.
6
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7
//
8
//===----------------------------------------------------------------------===//
9
10
#ifndef LLDB_TARGET_LANGUAGE_H
11
#define LLDB_TARGET_LANGUAGE_H
12
13
#include <functional>
14
#include <memory>
15
#include <set>
16
#include <vector>
17
18
#include "lldb/Core/Highlighter.h"
19
#include "lldb/Core/PluginInterface.h"
20
#include "lldb/DataFormatters/DumpValueObjectOptions.h"
21
#include "lldb/DataFormatters/FormatClasses.h"
22
#include "lldb/DataFormatters/StringPrinter.h"
23
#include "lldb/Symbol/TypeSystem.h"
24
#include "lldb/lldb-private.h"
25
#include "lldb/lldb-public.h"
26
27
namespace lldb_private {
28
29
class Language : public PluginInterface {
30
public:
31
  class TypeScavenger {
32
  public:
33
    class Result {
34
    public:
35
      virtual bool IsValid() = 0;
36
37
      virtual bool DumpToStream(Stream &stream,
38
                                bool print_help_if_available) = 0;
39
40
16
      virtual ~Result() = default;
41
    };
42
43
    typedef std::set<std::unique_ptr<Result>> ResultSet;
44
45
158
    virtual ~TypeScavenger() = default;
46
47
    size_t Find(ExecutionContextScope *exe_scope, const char *key,
48
                ResultSet &results, bool append = true);
49
50
  protected:
51
158
    TypeScavenger() = default;
52
53
    virtual bool Find_Impl(ExecutionContextScope *exe_scope, const char *key,
54
                           ResultSet &results) = 0;
55
  };
56
57
  class ImageListTypeScavenger : public TypeScavenger {
58
    class Result : public Language::TypeScavenger::Result {
59
    public:
60
10
      Result(CompilerType type) : m_compiler_type(type) {}
61
62
20
      bool IsValid() override { return m_compiler_type.IsValid(); }
63
64
10
      bool DumpToStream(Stream &stream, bool print_help_if_available) override {
65
10
        if (IsValid()) {
66
10
          m_compiler_type.DumpTypeDescription(&stream);
67
10
          stream.EOL();
68
10
          return true;
69
10
        }
70
0
        return false;
71
10
      }
72
73
10
      ~Result() override = default;
74
75
    private:
76
      CompilerType m_compiler_type;
77
    };
78
79
  protected:
80
113
    ImageListTypeScavenger() = default;
81
82
113
    ~ImageListTypeScavenger() override = default;
83
84
    // is this type something we should accept? it's usually going to be a
85
    // filter by language + maybe some sugar tweaking
86
    // returning an empty type means rejecting this candidate entirely;
87
    // any other result will be accepted as a valid match
88
    virtual CompilerType AdjustForInclusion(CompilerType &candidate) = 0;
89
90
    bool Find_Impl(ExecutionContextScope *exe_scope, const char *key,
91
                   ResultSet &results) override;
92
  };
93
94
  template <typename... ScavengerTypes>
95
  class EitherTypeScavenger : public TypeScavenger {
96
  public:
97
15
    EitherTypeScavenger() : TypeScavenger() {
98
45
      for (std::shared_ptr<TypeScavenger> scavenger : { std::shared_ptr<TypeScavenger>(new ScavengerTypes())... }) {
99
45
        if (scavenger)
100
45
          m_scavengers.push_back(scavenger);
101
45
      }
102
15
    }
103
  protected:
104
    bool Find_Impl(ExecutionContextScope *exe_scope, const char *key,
105
15
                   ResultSet &results) override {
106
15
      const bool append = false;
107
39
      for (auto& scavenger : m_scavengers) {
108
39
        if (scavenger && scavenger->Find(exe_scope, key, results, append))
109
8
          return true;
110
39
      }
111
7
      return false;
112
15
    }
113
  private:
114
    std::vector<std::shared_ptr<TypeScavenger>> m_scavengers;
115
  };
116
117
  template <typename... ScavengerTypes>
118
  class UnionTypeScavenger : public TypeScavenger {
119
  public:
120
    UnionTypeScavenger() : TypeScavenger() {
121
      for (std::shared_ptr<TypeScavenger> scavenger : { std::shared_ptr<TypeScavenger>(new ScavengerTypes())... }) {
122
        if (scavenger)
123
          m_scavengers.push_back(scavenger);
124
      }
125
    }
126
  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
  virtual bool IsTopLevelFunction(Function &function);
164
165
  virtual bool IsSourceFile(llvm::StringRef file_path) const = 0;
166
167
0
  virtual const Highlighter *GetHighlighter() const { return nullptr; }
168
169
  virtual lldb::TypeCategoryImplSP GetFormatters();
170
171
  virtual HardcodedFormatters::HardcodedFormatFinder GetHardcodedFormats();
172
173
  virtual HardcodedFormatters::HardcodedSummaryFinder GetHardcodedSummaries();
174
175
  virtual HardcodedFormatters::HardcodedSyntheticFinder
176
  GetHardcodedSynthetics();
177
178
  virtual std::vector<FormattersMatchCandidate>
179
  GetPossibleFormattersMatches(ValueObject &valobj,
180
                               lldb::DynamicValueType use_dynamic);
181
182
  virtual std::unique_ptr<TypeScavenger> GetTypeScavenger();
183
184
  virtual const char *GetLanguageSpecificTypeLookupHelp();
185
186
  class MethodNameVariant {
187
    ConstString m_name;
188
    lldb::FunctionNameType m_type;
189
190
  public:
191
    MethodNameVariant(ConstString name, lldb::FunctionNameType type)
192
7.15M
        : m_name(name), m_type(type) {}
193
7.15M
    ConstString GetName() const { return m_name; }
194
7.82M
    lldb::FunctionNameType GetType() const { return m_type; }
195
  };
196
  // If a language can have more than one possible name for a method, this
197
  // function can be used to enumerate them. This is useful when doing name
198
  // lookups.
199
  virtual std::vector<Language::MethodNameVariant>
200
642M
  GetMethodNameVariants(ConstString method_name) const {
201
642M
    return std::vector<Language::MethodNameVariant>();
202
642M
  };
203
204
  /// Returns true iff the given symbol name is compatible with the mangling
205
  /// scheme of this language.
206
  ///
207
  /// This function should only return true if there is a high confidence
208
  /// that the name actually belongs to this language.
209
124k
  virtual bool SymbolNameFitsToLanguage(Mangled name) const { return false; }
210
211
  /// An individual data formatter may apply to several types and cross language
212
  /// boundaries. Each of those languages may want to customize the display of
213
  /// values of said types by appending proper prefix/suffix information in
214
  /// language-specific ways. This function returns that prefix and suffix.
215
  ///
216
  /// \param[in] type_hint
217
  ///   A StringRef used to determine what the prefix and suffix should be. It
218
  ///   is called a hint because some types may have multiple variants for which
219
  ///   the prefix and/or suffix may vary.
220
  ///
221
  /// \return
222
  ///   A std::pair<StringRef, StringRef>, the first being the prefix and the
223
  ///   second being the suffix. They may be empty.
224
  virtual std::pair<llvm::StringRef, llvm::StringRef>
225
  GetFormatterPrefixSuffix(llvm::StringRef type_hint);
226
227
  // When looking up functions, we take a user provided string which may be a
228
  // partial match to the full demangled name and compare it to the actual
229
  // demangled name to see if it matches as much as the user specified.  An
230
  // example of this is if the user provided A::my_function, but the
231
  // symbol was really B::A::my_function.  We want that to be
232
  // a match.  But we wouldn't want this to match AnotherA::my_function.  The
233
  // user is specifying a truncated path, not a truncated set of characters.
234
  // This function does a language-aware comparison for those purposes.
235
  virtual bool DemangledNameContainsPath(llvm::StringRef path, 
236
                                         ConstString demangled) const;
237
238
  // if a language has a custom format for printing variable declarations that
239
  // it wants LLDB to honor it should return an appropriate closure here
240
  virtual DumpValueObjectOptions::DeclPrintingHelper GetDeclPrintingHelper();
241
242
  virtual LazyBool IsLogicalTrue(ValueObject &valobj, Status &error);
243
244
  // for a ValueObject of some "reference type", if the value points to the
245
  // nil/null object, this method returns true
246
  virtual bool IsNilReference(ValueObject &valobj);
247
248
  /// Returns the summary string for ValueObjects for which IsNilReference() is
249
  /// true.
250
0
  virtual llvm::StringRef GetNilReferenceSummaryString() { return {}; }
251
252
  // for a ValueObject of some "reference type", if the language provides a
253
  // technique to decide whether the reference has ever been assigned to some
254
  // object, this method will return true if such detection is possible, and if
255
  // the reference has never been assigned
256
  virtual bool IsUninitializedReference(ValueObject &valobj);
257
258
  virtual bool GetFunctionDisplayName(const SymbolContext *sc,
259
                                      const ExecutionContext *exe_ctx,
260
                                      FunctionNameRepresentation representation,
261
                                      Stream &s);
262
263
  virtual ConstString
264
14.3k
  GetDemangledFunctionNameWithoutArguments(Mangled mangled) const {
265
14.3k
    if (ConstString demangled = mangled.GetDemangledName())
266
14.3k
      return demangled;
267
268
0
    return mangled.GetMangledName();
269
14.3k
  }
270
271
  virtual void GetExceptionResolverDescription(bool catch_on, bool throw_on,
272
                                               Stream &s);
273
274
  static void GetDefaultExceptionResolverDescription(bool catch_on,
275
                                                     bool throw_on, Stream &s);
276
277
  // These are accessors for general information about the Languages lldb knows
278
  // about:
279
280
  static lldb::LanguageType
281
  GetLanguageTypeFromString(const char *string) = delete;
282
  static lldb::LanguageType GetLanguageTypeFromString(llvm::StringRef string);
283
284
  static const char *GetNameForLanguageType(lldb::LanguageType language);
285
286
  static void PrintAllLanguages(Stream &s, const char *prefix,
287
                                const char *suffix);
288
289
  /// Prints to the specified stream 's' each language type that the
290
  /// current target supports for expression evaluation.
291
  ///
292
  /// \param[out] s      Stream to which the language types are written.
293
  /// \param[in]  prefix String that is prepended to the language type.
294
  /// \param[in]  suffix String that is appended to the language type.
295
  static void PrintSupportedLanguagesForExpressions(Stream &s,
296
                                                    llvm::StringRef prefix,
297
                                                    llvm::StringRef suffix);
298
299
  // return false from callback to stop iterating
300
  static void ForAllLanguages(std::function<bool(lldb::LanguageType)> callback);
301
302
  static bool LanguageIsCPlusPlus(lldb::LanguageType language);
303
304
  static bool LanguageIsObjC(lldb::LanguageType language);
305
306
  static bool LanguageIsC(lldb::LanguageType language);
307
308
  /// Equivalent to \c LanguageIsC||LanguageIsObjC||LanguageIsCPlusPlus.
309
  static bool LanguageIsCFamily(lldb::LanguageType language);
310
311
  static bool LanguageIsPascal(lldb::LanguageType language);
312
313
  // return the primary language, so if LanguageIsC(l), return eLanguageTypeC,
314
  // etc.
315
  static lldb::LanguageType GetPrimaryLanguage(lldb::LanguageType language);
316
317
  static std::set<lldb::LanguageType> GetSupportedLanguages();
318
319
  static LanguageSet GetLanguagesSupportingTypeSystems();
320
  static LanguageSet GetLanguagesSupportingTypeSystemsForExpressions();
321
  static LanguageSet GetLanguagesSupportingREPLs();
322
323
  // Given a mangled function name, calculates some alternative manglings since
324
  // the compiler mangling may not line up with the symbol we are expecting.
325
  virtual std::vector<ConstString>
326
0
  GenerateAlternateFunctionManglings(const ConstString mangled) const {
327
0
    return std::vector<ConstString>();
328
0
  }
329
330
  virtual ConstString
331
  FindBestAlternateFunctionMangledName(const Mangled mangled,
332
0
                                       const SymbolContext &sym_ctx) const {
333
0
    return ConstString();
334
0
  }
335
336
0
  virtual llvm::StringRef GetInstanceVariableName() { return {}; }
337
338
protected:
339
  // Classes that inherit from Language can see and modify these
340
341
  Language();
342
343
private:
344
  Language(const Language &) = delete;
345
  const Language &operator=(const Language &) = delete;
346
};
347
348
} // namespace lldb_private
349
350
#endif // LLDB_TARGET_LANGUAGE_H