初识LLVM&Clang-开发Xcode插件

初识LLVM&Clang-开发Xcode插件

LLVM

Xcode现在使用的编译器就是LLVMLLVM比以前使用的GCC编译器速度快好几倍。并且LLVM可以编译 Kotlin,Ruby,Python,Haskell,Java,D,PHP,Pure,Lua 和许多其他语言。

LLVM IR

通过LLVM编译后的产物是LLVM IRLLVM IR是一个区别于源码和机器码的一种中间代码。这里就是LLVM的强大之处,不管编译什么哪种语言,输出的都是LLVM IR

这里就要说一句:LLVM编译器是区分前后端的,而传统的编译器(GCC)是不区分前后端的。这样导致的后果就是传统编译器如果要支持其他的一种语言或硬件平台的话要做大量工作。

传统编译器.png
llvm编译器.png

LLVM如果要支持一种新的语言,那么只需要实现一个新的编译器前端即可,后端可以不变,因为前端的产物都是LLVM IR编译器后端都能识别。如果要改变硬件平台的话,就只要实现一个新的编译器后端即可,通过把前端输出的LLVM IR再次编译成对应硬件平台的代码。从这就可以看出前后端分离,以及LLVM IR的作用了。

LLVM IR 的三种格式:

  • 内存中的编译中间语言
  • 硬盘上存储的可读中间格式(以 .ll 结尾)
  • 硬盘上存储的二进制中间语言(以 .bc 结尾)

这三种中间格式完全是等价的。

Bitcode

这么说LLVM IR可能还不熟悉,但是我们说道bitcode时就熟悉多了。其实bitcode就是LLVM IR第三种格式(硬盘上存储的二进制中间语言)。我们在打包的时候可以选择是否bitcode编译打包。如果选择了bitcode打包方式,上传IPA包时同时也会上传bitcode文件。并且之后Apple就不会使用你的IPA包了,会通过对bitcode文件再次打包。这么做是因为Apple对上传的bitcode可做一些优化工作,并且还可以对安装的目标设备进行二进制优化,减少安装包的大小,比如CPU架构为armv7的就不需要arm64的文件。去除不必要的架构可以加快打包速度。

bitcode.png
CPU 架构.png

Clang

前面说到了LLVM编译器分为前后端,Clang就是编译器的前端。Clang的主要功能是输出代码对应的抽象语法树( AST ),针对用户发生的编译错误准确地给出建议,并将代码编译成LLVM IR

Clang 的主要工作:

  • 预处理: 比如把宏嵌入到对应的位置,头文件的导入,去除注释( clang -E main.m )
  • 词法分析: 这里会把代码切成一个个 Token,比如大小括号,等于号还有字符串等
  • 语法分析: 验证语法是否正确
  • 生成AST: 将所有节点组成抽象语法树AST
  • 静态分析:分析代码是否存在问题,给出错误信息和修复方案
  • 生成LLVM IR: CodeGen 会负责将语法树自顶向下遍历逐步翻译成LLVM IR

以上是其中涉及的一些概念点,想深入了解的话还是要单独去找资料阅读。这里只是皮毛中的皮毛😂。下面就看下如何实现一个Xcode的插件:

LLVM环境搭建

下载LLVM代码到本地
$ git clone https://git.llvm.org/git/llvm.git/

或者直接到GitHub上下载也可以。

下载clang
$ cd llvm/tools
$ git clone https://git.llvm.org/git/clang.git/
安装clang.png

配置和构建LLVM和Clang

CMake

首先我要先安装编译工具CMake这里有一片介绍文档可够了解。

$ brew install cmake
使用ninja编译

1、安装

$ brew install ninja

2、在llvm同级目录下新建一个llvm_build目录,最终会在llvm_build目录下生成build.ninja

3、在llvm同级目录下新建一个llvm_release目录,最终编译文件会在llvm_release文件夹路径下。

$ cd llvm_build

$ cmake -G Ninja ../llvm -DCMAKE_INSTALL_PREFIX=安装路径
//例如:cmake -G Ninja ../llvm -DCMAKE_INSTALL_PREFIX=/Users/zhouqiang/clangPlugin/llvm_release
屏幕快照 2019-08-23 14.37.18.png

4、依次执行编译、安装指令。

$ ninja

$ ninja install

创建插件

1、在/llvm/tools/clang/tools目录下新建插件。

创建插件.png

2、修改/llvm/tools/clang/tools目录下的CMakeLists.txt文件,新增add_clang_subdirectory(QTPlugin)

cmake.list文件.png

3、在QTPlugin目录下新建一个名为QTPlugin.cpp的文件

#include <iostream>
#include "clang/AST/AST.h"
#include "clang/AST/DeclObjC.h"
#include "clang/AST/ASTConsumer.h"
#include "clang/ASTMatchers/ASTMatchers.h"
#include "clang/Frontend/CompilerInstance.h"
#include "clang/ASTMatchers/ASTMatchFinder.h"
#include "clang/Frontend/FrontendPluginRegistry.h"

using namespace clang;
using namespace std;
using namespace llvm;
using namespace clang::ast_matchers;

namespace QTPlugin {
    
    class QTMatchHandler: public MatchFinder::MatchCallback {
    private:
        CompilerInstance &CI;
        
        bool isUserSourceCode(const string filename) {
            if (filename.empty()) return false;
            
            // 非Xcode中的源码都认为是用户源码
            if (filename.find("/Applications/Xcode.app/") == 0) return false;
            
            return true;
        }
        
        bool isShouldUseCopy(const string typeStr) {
            if (typeStr.find("NSString") != string::npos ||
                typeStr.find("NSArray") != string::npos ||
                typeStr.find("NSDictionary") != string::npos/*...*/) {
                return true;
            }
            return false;
        }
    public:
        QTMatchHandler(CompilerInstance &CI) :CI(CI) {}
        
        void run(const MatchFinder::MatchResult &Result) {
            const ObjCPropertyDecl *propertyDecl = Result.Nodes.getNodeAs<ObjCPropertyDecl>("objcPropertyDecl");
            if (propertyDecl && isUserSourceCode(CI.getSourceManager().getFilename(propertyDecl->getSourceRange().getBegin()).str()) ) {
                ObjCPropertyDecl::PropertyAttributeKind attrKind = propertyDecl->getPropertyAttributes();
                string typeStr = propertyDecl->getType().getAsString();
                
                if (propertyDecl->getTypeSourceInfo() && isShouldUseCopy(typeStr) && !(attrKind & ObjCPropertyDecl::OBJC_PR_copy)) {
                    cout<<"--------- "<<typeStr<<": 不是使用的 copy 修饰--------"<<endl;
                    DiagnosticsEngine &diag = CI.getDiagnostics();
                    diag.Report(propertyDecl->getBeginLoc(), diag.getCustomDiagID(DiagnosticsEngine::Warning, "--------- %0 不是使用的 copy 修饰--------")) << typeStr;
                }
            }
        }
    };
    
    class QTASTConsumer: public ASTConsumer {
    private:
        MatchFinder matcher;
        QTMatchHandler handler;
    public:
        QTASTConsumer(CompilerInstance &CI) :handler(CI) {
            matcher.addMatcher(objcPropertyDecl().bind("objcPropertyDecl"), &handler);
        }
        
        void HandleTranslationUnit(ASTContext &context) {
            matcher.matchAST(context);
        }
    };
    
    class QTASTAction: public PluginASTAction {
    public:
        unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI, StringRef iFile) {
            return unique_ptr<QTASTConsumer> (new QTASTConsumer(CI));
        }
        
        bool ParseArgs(const CompilerInstance &ci, const std::vector<std::string> &args) {
            return true;
        }
    };
}

static FrontendPluginRegistry::Add<QTPlugin::QTASTAction> X("QTPlugin", "The QTPlugin is my first clang-plugin.");

4、在QTPlugin目录下新建一个名为CMakeLists.txt的文件,内容为

add_llvm_library(xxPlugin MODULE xxPlugin.cpp PLUGIN_TOOL clang)

if(LLVM_ENABLE_PLUGINS AND (WIN32 OR CYGWIN))
  target_link_libraries(xxPlugin PRIVATE
    clangAST
    clangBasic
    clangFrontend
    LLVMSupport
    )
endif()

5、目录文件创建完成之后,利用CMake重新生成一下Xcode项目。

$ cd llvm_xcode
$ cmake -G Xcode ../llvm

6、插件源代码在 Xcode 项目中的Loadable modules目录下可以找到,这样就可以直接在 Xcode 里编写插件代码。

7、最后command+B编译生成QTPlugin.dylib文件,找到插件对应的QTPlugin.dylib

QTPlugin.png

Xcode集成QTPlugin

1、创建一个新的Xcode项目

2、打开需要加载插件的Xcode项目,在Build Settings栏目中的OTHER_CFLAGS添加上如下内容:

-Xclang -load -Xclang (.dylib)动态库路径 -Xclang -add-plugin -Xclang 插件名字(namespace 的名字,名字不对则无法使用插件)
例如:
-Xclang -load -Xclang /Users/zhouqiang/clangPlugin/llvm_xcode/Debug/lib/QTPlugin.dylib -Xclang -add-plugin -Xclang QTPlugin
other c flags.png

3、编译报错:由于Clang插件需要使用对应的版本去加载,如果版本不一致则会导致编译错误,会出现如下图所示:

屏幕快照 2019-08-20 19.31.22.png

Build Settings栏目中新增两项用户定义的设置

xcode_add_user_defined_settings.png

分别是CCCXX

user_define.png

CC对应的是自己编译的clang的绝对路径,CXX对应的是自己编译的clang++的绝对路径。

clang&clang++.png

4、编译报错如下

屏幕快照 2019-08-20 19.33.19.png

则可以在Build Settings栏目中搜索index,将Enable Index-Wihle-Building FunctionalityDefault改为NO

屏幕快照 2019-08-23 15.29.25.png

5、最后在新创建的Xcode项目中编译就会有如下警告了。说明你的插件成功导入并生效了。

end.png

学习文档

LLVM & Clang 入门

iOS 编译详解 LLVM Clang

PS

我的博客即将同步至腾讯云+社区,邀请大家一同入驻:https://cloud.tencent.com/developer/support-plan?invite_code=10myh96g1kqry

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容