参考:

Create a working compiler with the LLVM framework, Part 2

How to parse C programs with Clang: A tutorial

[Clang,libClang] exercise1 : FileManager

注意:此篇笔记各类之构造函数源码来自 llvm/clang doxygen 5.0.0 svn,与现行版本(此时为 3.9.1)有一定出入,其中主要是现行版本有些参数是 llvm::IntrusiveRefCntPtr(llvm 实现的 smart pointer)而不是 std::shared_ptr

一、Preprocessor 的构造函数

前端由许多部分组成,其中第一部分通常是一个 lexer,clang 中 Preprocessor 类是 lexer 的 main interface。出于性能考虑,clang 没有独立的预处理器程序,而是在 lexing 的过程中进行预处理。

Preprocessor 的构造函数如下所示:

 1 Preprocessor::Preprocessor(
2 std::shared_ptr<PreprocessorOptions> PPOpts, // constructor: PreprocessorOptions()
3 DiagnosticsEngine& diags,
4 LangOptions& opts, // constructor: LangOptions()
5 // Keep track of the various options that can be enabled,
6 // which controls the dialect of C or C++ that is accepted
7 SourceManager& SM,
8 HeaderSearch& Headers,
9 ModuleLoader& TheModuleLoader,
10 IdentifierInfoLookup* IILookup = nullptr,
11 bool OwnsHeaderSearch = false,
12 TranslationUnitKind TUKind = TU_Complete
13 )

Preprocessor 的构造函数

DiagnosticsEngine : 用来给用户报告错误和警告信息。构造函数如下:

1 DiagnosticsEngine::DiagnosticsEngine(
2 IntrusiveRefCntPtr<DiagnosticIDs> Diags, // constructor: DiagnosticIDs()
3 // used for handling and querying diagnostic IDs
4 DiagnosticOptions* DiagOpts, // constructor: DiagnosticOptions()
5 // Options for controlling the compiler diagnostics engine
6 DiagnosticConsumer* client = nullptr,
7 bool ShouldOwnClient = true
8 )

DiagnosticsEngine 的构造函数

其中 DiagnosticConsumer 是一个抽象接口,由前端的 clients 实现,用来 formats and prints fully processed diagnostics。clang 内置一个 TextDiagnosticsConsumer 类,将错误和警告信息写到 console 上,clang binary 用的 DiagnosticConsumer 也是这个类。TextDiagnosticsConsumer 的构造函数如下:

1 TextDiagnosticPrinter::TextDiagnosticPrinter(
2 raw_ostream& os, // llvm::outs() returns a reference to a raw_ostream for standard output
3 DiagnosticOptions* diags,
4 bool OwnsOutputStream = false // within destructor:(OS is the member, initialized with os)
5 // if (OwnsOutputStream) delete &OS
6 )

TextDiagnosticPrinter 的构造函数

SourceManager :handles loading and caching of source files into memory。构造函数如下:

1 SourceManager::SourceManager(vim
2 DiagnosticsEngine& Diag,
3 FileManager& FileMgr,
4 bool UserFilesAreVolatile = false
5 )

SourceManager 的构造函数

FileManager :实现了对文件系统查找、文件系统缓存、目录查找管理的支持。构造函数如下:

1 FileManager::FileManager(
2 const FileSystemOptions& FileSystemOpts, // use default constructor
3 IntrusiveRefCntPtr<vfs::FileSystem> FS = nullptr
4 )

FileManager 的构造函数

HeaderSearch :Encapsulates the information needed to find the file referenced by a #include or #include_next, (sub-)framework lookup, etc。构造函数如下:

1 HeaderSearch::HeaderSearch(
2 std::shared_ptr<HeaderSearchOptions> HSOpts, // constructor: HeaderSearchOptions::HeaderSearchOptions(StringRef _Sysroot = "/")
3 SourceManager& SourceMgr,
4 DiagnosticsEngine& Diags,
5 const LangOptions& LangOpts,
6 const TargetInfo* Target
7 )

HeaderSearch 的构造函数

TargetInfo :Exposes information about the current target。 其构造函数为 protected,因此需要调用工厂函数 static TargetInfo* TargetInfo::CreateTargetInfo(DiagnosticsEngine &Diags, const std::shared_ptr<TargetOptions>& Opts) ,其中 TargetOptions 类包含 target 的相关信息,如 CPU、ABI 等。类中有一个属性 Triple 用以定义 target 的架构。Triple 是一个 string,形如 i386-apple-darwin,通过 llvm::sys::getDefaultTargetTriple() 可以获得编译 llvm 的机器的 host triple。

ModuleLoader:描述了 module loader 的抽象接口。Module loader 负责解析一个 module name(如“std”),将其与实际的 module file 联系起来,并加载该 module。CompilerInstance 便是一个实现了该接口的 module loader。

二、通过 CompilerInstance 创建 Preprocessor

比起手写 Preprocessor,CompilerInstance 更加实用一些。CompilerInstance 主要有两个作用:(1)管理运行编译器所必须的各个对象,如 preprocessor、target information、AST context 等;(2)提供创建和操作常用 Clang 对象的有用方法。下面是其类定义的一部分:

 1 class CompilerInstance : public ModuleLoader {
2 /// The options used in this compiler instance.
3 std::shared_ptr<CompilerInvocation> Invocation;
4 /// The diagnostics engine instance.
5 IntrusiveRefCntPtr<DiagnosticsEngine> Diagnostics;
6 /// The target being compiled for.
7 IntrusiveRefCntPtr<TargetInfo> Target;
8 /// The file manager.
9 IntrusiveRefCntPtr<FileManager> FileMgr;
10 /// The source manager.
11 IntrusiveRefCntPtr<SourceManager> SourceMgr;
12 /// The preprocessor.
13 std::shared_ptr<Preprocessor> PP;
14 /// The AST context.
15 IntrusiveRefCntPtr<ASTContext> Context;
16 /// An optional sema source that will be attached to sema.
17 IntrusiveRefCntPtr<ExternalSemaSource> ExternalSemaSrc;
18 /// The AST consumer.
19 std::unique_ptr<ASTConsumer> Consumer;
20 /// The semantic analysis object.
21 std::unique_ptr<Sema> TheSema;
22 /// ...
23 };

CompilerInstance 类定义的一部分

下列代码通过 CompilerInstance 来创建 Preprocessor:

 1 #include <memory>
2
3 #include "clang/Basic/LangOptions.h"
4 #include "clang/Basic/TargetInfo.h"
5 #include "clang/Frontend/CompilerInstance.h"
6
7 int main() {
8 clang::CompilerInstance ci;
9
10 ci.createDiagnostics();
11
12 std::shared_ptr<clang::TargetOptions> pTargetOptions =
13 std::make_shared<clang::TargetOptions>();
14 pTargetOptions->Triple = llvm::sys::getDefaultTargetTriple();
15 clang::TargetInfo *pTargetInfo =
16 clang::TargetInfo::CreateTargetInfo(ci.getDiagnostics(), pTargetOptions);
17 ci.setTarget(pTargetInfo);
18
19 ci.createFileManager();
20 ci.createSourceManager(ci.getFileManager());
21 ci.createPreprocessor(clang::TU_Complete);
22
23 return 0;
24 }

Use CompilerInstance to construct Preprocessor

首先创建 DiagnosticsEngine(通过 createDiagnostics()),然后创建并设置 TargetInfo,然后依次创建 FileManager(通过 createFileManager()),SourceManager(通过 createSourceManager (FileManager &FileMgr)),最后创建 Preprocessor(createPreprocessor(TranslationUnitKind))。

三、FileManager 与 SourceManager

FileManager:

p, li { white-space: pre-wrap }

Clang Preprocessor 类的创建的更多相关文章

  1. C# 根据类名称创建类示例

    //获得类所在的程序集名称(此处我选择当前程序集) string bllName = System.IO.Path.GetFileNameWithoutExtension(System.Reflect ...

  2. php简单实用的操作文件工具类(创建、移动、复制、删除)

    php简单实用好用的文件及文件夹复制函数和工具类(创建.移动.复制.删除) function recurse_copy($src,$dst) {  // 原目录,复制到的目录 $dir = opend ...

  3. 李洪强iOS开发之OC[013] -类的创建的练习

    // //  main.m //  12 - 类的创建练习 // //  Created by vic fan on 16/7/9. //  Copyright © 2016年 李洪强. All ri ...

  4. C++:类的创建

    类的创建 #include<iostream> #include<cmath> using namespace std; class Complex //声明一个名为Compl ...

  5. 2--OC -- 类的创建与实例化

    2.OC -- 类的创建与实例化   一.OC类的简述 1.OC类分为2个文件:.h文件用于类的声明,.m文件用于实现.h的函数: 2.类是声明使用关键字:@interface.@end : 3.类是 ...

  6. JAVA类的创建: 创建JAVA的类 ,JAVA的字段,JAVA类的方法

    1. 创建Java的类 如果说Java的一切都是对象,那么类型就是决定了某一类对象的外观与行为.可是类型的关键字不是type,而是class,创建一个新的类型要用下面的代码: 1 2 3 class ...

  7. python 通过元类控制类的创建

    一.python中如何创建类? 1. 直接定义类 class A: a = 'a' 2. 通过type对象创建 在python中一切都是对象 在上面这张图中,A是我们平常在python中写的类,它可以 ...

  8. Day 5-7 exec 和元类的创建.

    exec方法 元类 exec(str_command,globals,locals)参数1:字符串形式的命令参数2:全局作用域(字典形式). 如果不指定,默认globals参数3:局部作用(字典形式) ...

  9. Egret 类的创建和继承--TypeScript

    class test extends egret.DisplayObjectContainer { /** * 类的创建 */ //属性 name: string; age: number; ts: ...

  10. 快速创建SpringBoot2.x应用之工具类自动创建web应用、SpringBoot2.x的依赖默认Maven版本

    快速创建SpringBoot2.x应用之工具类自动创建web应用简介:使用构建工具自动生成项目基本架构 1.工具自动创建:http://start.spring.io/ 2.访问地址:http://l ...

随机推荐

  1. JS leetcode x 的平方根 题解分析

    壹 ❀ 引 这几天心情复杂,也不知道形容.做道题吧,其实是上周的题,一直没整理,比较巧的是,这也是我同学17年去PPTV面试时遇到的一题,题目来自leetcode69. x 的平方根,题目描述如下: ...

  2. NC204871 求和

    题目链接 题目 题目描述 已知有 \(n\) 个节点,有 \(n-1\) 条边,形成一个树的结构. 给定一个根节点 \(k\) ,每个节点都有一个权值,节点i的权值为 \(v_i\) . 给 \(m\ ...

  3. Qt5.15.0 升级至 Qt5.15.9 遇到的一些错误

    按照之前我写的文章教程,可以很简单的编译出静态库(仅供学习交流) 编译 windows 上的 qt 静态库 编译出静态库后,替换旧版本的库,见我另一篇文章教程 VS2019 配置 QT 库 之所以没有 ...

  4. Docker实践之09-高级网络配置

    目录 一.Docker网络原理及默认配置 二.Docker网络定制配置参数 三.容器访问控制原理 1.容器访问外部网络 2.容器之间访问 3.访问所有端口 4.访问指定端口 5.映射容器端口到主机端口 ...

  5. Generating equals/hashCode implementation but without a call to superclass

    Generating equals/hashCode implementation but without a call to superclass1.lombok 警告,没有注入父类的字段当我们给一 ...

  6. flutter打包android的一些配置修改(解决白屏,视频闪退)

    1.打包后视频播放闪退 视频播放器选择了flutter_tencentplayer(https://github.com/qq326646683/flutter_tencentplayer) 解决:不 ...

  7. Rust 开发的高性能 Python 包管理工具,可替换 pip、pip-tools 和 virtualenv

    最近,我在 Python 潮流周刊 中分享了一个超级火爆的项目,这还不到一个月,它在 Github 上已经拿下了 8K star 的亮眼成绩,可见其受欢迎程度极高!国内还未见有更多消息,我趁着周末把一 ...

  8. 3. JVM运行时数据区

    1. 运行时数据区概述 前面的章节中已经将类的加载过程大致过程说清楚了,此时类已经加载到内存中,,后面就是运行时数据区的各个组件的工作了 由上图可以看出来, jvm将class字节码加载完成后,后面运 ...

  9. Java 演示线程的死锁问题

    1 package bytezero.deadlock; 2 3 /** 4 * 演示线程的死锁问题: 5 * 6 * 1.死锁的理解:不同的线程分别占用对方需要的同步资源不放弃,都在等待对方放弃 7 ...

  10. 7、mysql的缓存优化

    概述 开启Mysql的查询缓存,当执行完全相同的SQL语句的时候,服务器就会直接从缓存中读取结果,当数据被修改,之前的缓存会失效,修改比较频繁的表不适合做查询缓存. 操作流程 客户端发送一条查询给服务 ...