• Say bye to CMake and Makefile


    用了几年的CMake,最近想试着琢磨如何将C++应用的动态链接全部改成静态链接,发现还需要研究CMake的用法,进入CMake的文档,

    http://www.cmake.org/cmake/help/syntax.html

    看到这句:

    In many ways writing a CMakeLists file is like a writing a program in a simple language. Like most languages CMake provides flow control structures to help you along your way. CMake provides three flow control structures:

    再往下看,里面是什么条件语句,循环,定义过程等,一个简单的script语言用来控制CMake生成Makefile。

    过去CMake帮助了我,因为我对写Makefile很厌烦,CMake让我能够设计良好的目录结构,不要再自己实现目录树递归的Makefile脚本。

    但是时至今天,对于已经熟练掌握newlisp语言的我,还有什么比用newlisp来实现脚本更强大的工具,为什么我还用CMake?我并不需要考虑跨平台编译,我只是在实践Linux服务器上的C++编程。我仅仅使用GCC作为编译器。直接用newlisp来控制GCC,可以让我直接使用GCC,更深层次的理解编译器的行为。

    说做就做,本着仍然不改变原有CMake工程的精神,创建了几个newlisp脚本实现递归目录树,编译文件,最后动态或者静态链接。

    现在介绍一下使用。这个CppCMS程序目录结构是:

     tree -L 4
    .
    ├── builder
    │   ├── build_config.lsp
    │   ├── compile.lsp
    │   ├── compile.lsp~
    │   ├── dlink.lsp
    │   ├── dlink.lsp~
    │   ├── g++.lsp
    │   ├── link.lsp~
    │   ├── rebuild.lsp
    │   ├── slink.lsp
    │   ├── slink.lsp~
    │   └── tmpl.lsp
    ├── codes
    │   ├── main
    │   │   ├── build
    │   │   ├── CMakeLists.txt
    │   │   ├── conf.d
    │   │   │   ├── ca-certs.crt
    │   │   │   ├── site.crt
    │   │   │   ├── ssl.key
    │   │   │   ├── ssl-pwd.key
    │   │   │   ├── site.conf
    │   │   │   ├── default.conf_bk
    │   │   │   ├── example_ssl.conf
    │   │   │   ├── fastcgi.cml
    │   │   │   ├── nginx_signing.key
    │   │   │   ├── server.crt
    │   │   │   ├── server.csr
    │   │   │   ├── server.key
    │   │   │   └── server.key.org
    │   │   ├── config.js
    │   │   ├── create_deploy.sh
    │   │   ├── include
    │   │   │   ├── bean
    │   │   │   ├── configuration.h
    │   │   │   ├── controller
    │   │   │   ├── display.h
    │   │   │   ├── displays.h
    │   │   │   ├── exception
    │   │   │   ├── group.h
    │   │   │   ├── groups.h
    │   │   │   ├── helper
    │   │   │   ├── module
    │   │   │   ├── my_application.h
    │   │   │   ├── response.h
    │   │   │   ├── service
    │   │   │   └── web_user.h
    │   │   ├── install.sh
    │   │   ├── jscheck
    │   │   │   ├── pom.xml
    │   │   │   └── src
    │   │   ├── proto
    │   │   │   ├── build.sh
    │   │   │   ├── input
    │   │   │   └── output
    │   │   ├── resources
    │   │   │   ├── html
    │   │   │   ├── images
    │   │   │   ├── plugin
    │   │   │   ├── script
    │   │   │   └── style
    │   │   ├── src
    │   │   │   ├── bean
    │   │   │   ├── CMakeLists.txt
    │   │   │   ├── controller
    │   │   │   ├── display.cpp
    │   │   │   ├── displays.cpp
    │   │   │   ├── group.cpp
    │   │   │   ├── groups.cpp
    │   │   │   ├── helper
    │   │   │   ├── main.cpp
    │   │   │   ├── module
    │   │   │   ├── my_application.cpp
    │   │   │   ├── service
    │   │   │   ├── view
    │   │   │   └── web_user.cc
    │   │   ├── template
    │   │   │   ├── accounts.tmpl
    │   │   │   ├── default.tmpl
    │   │   │   ├── deny.tmpl
    │   │   │   ├── gprsConfig.tmpl
    │   │   │   ├── gprsManager.tmpl
    │   │   │   ├── groupManagement.tmpl
    │   │   │   ├── home.tmpl
    │   │   │   ├── login.tmpl
    │   │   │   ├── log.tmpl
    │   │   │   ├── message.tmpl
    │   │   │   ├── register_client.tmpl
    │   │   │   ├── updatePassword.tmpl
    │   │   │   └── userTitle.tmpl
    │   │   └── tool.sh
    │   └── test
    └── README
    


    运行方式就是进入builder目录执行

    ./rebuild.lsp

    首先将CppCMS的tmpl模板文件编译成C++类

    然后将所有.cc和.cpp文件编译成.o文件

    最后动态链接成可执行程序。

    先看rebuild.lsp代码:

    #!/usr/bin/newlisp 
    
    ;; init
    (load "/opt/build_config.lsp")
    (set 'armory-folder (env "NEWLISP_ARMORY_HOME"))
    (println (append "newlisp armory home folder: " armory-folder))
    (load (append armory-folder "/codes/file/file.lsp"))
    (file:init)
    
    ;; clean files genereated by CMake
    (set 'cmake-build-dir "../codes/main/build")
    (file:clean-folder cmake-build-dir)
    
    ;; clean view/*.cpp files generated from cppcms_tmpl_cc
    (set 'view-dir "../codes/main/src/view")
    (file:clean-folder view-dir)
    
    (set 'tmpl-dir "../codes/main/template")
    
    ;; generate .cc files in view folder
    (load "tmpl.lsp")
    (tmpl-to-cc tmpl-dir view-dir)
    
    ;; compile all c++ files to .o file in ./o folder
    (load "g++.lsp")
    (set 'include-paths 
         (list "../codes/main/include"
    	    "../codes/main/src/../../loki-0.1.7/include"))
    (set 'o-dir "./o")
    (set 'src-paths 
         (list "../codes/main/src"
    	   "../codes/main/src/bean"
    	   "../codes/main/src/controller"
    	   "../codes/main/src/helper"
    	   "../codes/main/src/module"
    	   "../codes/main/src/service"
    	   "../codes/main/src/view"
    	   ))
    
    (compile include-paths src-paths o-dir)
    
    ;; link all .o files
    (set 'libs 
         (list "pthread"
    	   "cppcms"
    	   "mongoclient"
    	   "booster"
    	   "loki"
    	   "cryptopp"
    	   "boost_system"
    	   "boost_thread"
    	   "boost_filesystem"
    	   ))
    (set 'binary-name "sports_lottery")
    (set 'bin-dir "bin")
    (dynamic-link o-dir bin-dir binary-name libs)
    
    
    (exit)


    该文件用到了newlisp armory模块,参考我的GitHub项目: https://github.com/csfreebird/newlisp_armory

    tmpl.lsp文件专门负责处理CppCMS tmpl文件,代码如下:

    (define (get-extension name)
      (first (regex "[^.]*$" name))
      )
    
    ;; @syntax (remove-extension name)
    ;; @return file name without extension
    ;; @note remove extension name e.g a.b .b is extension, it will be removed
    (define (remove-extension name)
      ((regex "(.*)\.(.*)$" name) 3)
      )
    
    ;; @syntax (tmpl-to-cc)
    ;; @note find all tmpl files in tmpl-folder, translate them into *.cc files
    (define (tmpl-to-cc tmpl-folder cc-folder)
      (set 'tmpl-files (directory tmpl-folder "\.tmpl"))
      (set 'cmd-tmpl (append "cppcms_tmpl_cc " tmpl-folder "/%s -o " cc-folder "/%s.cc"))
      (dolist (tmpl-file tmpl-files)
        (set 'cmd (format cmd-tmpl tmpl-file (remove-extension tmpl-file)))
        (println cmd)
        (exec cmd)
        )
    )


    g++.lsp专门生成g++命令,文件内容如下:

    ;; @syntax (compile include-dirs src-dirs o-dir)
    ;; @parameter include-dir a list contains one or more relative or absolute include directories
    ;; @parameter src-dirs a list contains one or more relative or absolute src dirs
    (define (compile include-dirs src-dirs o-dir)
      (if (directory? o-dir)
          (file:clean-folder o-dir)
        (make-dir o-dir))
      (set 'path1 "")
      (dolist (path include-dirs)
        (set 'path1 (append path1 "-I" path " ")))
    
      (set 'cmd-template (format "/usr/bin/c++ -g %s -Wall -o %s/" path1 o-dir))
      
      (dolist (dir src-dirs)
        (compile-dir dir cmd-template)
        )
      )
    
    ;; @syntax (compile-dir dir cmd-template)
    ;; @parameter dir one folder which contains many .cc or .cpp files
    ;; @parameter cmd-template the command template that has -g, -Wall and -I args 
    (define (compile-dir dir cmd-template)
      (set 'file-list (directory dir "\.cc|\.cpp"))
      (println "dir: " dir)
      (dolist (cc-path file-list)
        (set 'str (append cc-path ".o"))
        (set 'cmd (append cmd-template  str " -c "  dir "/" cc-path))
        (println cmd)
        (exec cmd))
      )
    
    ;; @syntax (dynamic-link o-dir bin-dir binary-name libs)
    ;; @parameter o-dir the direcotry includs all .o files
    ;; @parameter bin-dir the location of linked binary file
    ;; @libs the list of all dependencies
    (define (dynamic-link o-dir bin-dir binary-name libs)
      (if (directory? bin-dir)
          (file:clean-folder bin-dir)
        (make-dir bin-dir))
      (set 'cmd "/usr/bin/c++ -g")
      (set 'o-files (directory o-dir "\.o"))
      (dolist (o-file o-files)
        (set 'cmd (append cmd " " (real-path o-dir) "/" o-file)))
      (set 'cmd (append cmd " -o " bin-dir "/" binary-name " -rdynamic"))
      (dolist (lib libs)
        (set 'cmd (append cmd " -l" lib)))
      (println cmd)
      (exec cmd)
      )
    
    ;; @syntax (static-link o-dir bin-dir binary-name libs)
    ;; @parameter o-dir the direcotry includs all .o files
    ;; @parameter bin-dir the location of linked binary file
    ;; @libs the list of all dependencies
    (define (static-link o-dir bin-dir binary-name libs)
      (if (directory? bin-dir)
          (file:clean-folder bin-dir)
        (make-dir bin-dir))
      (set 'cmd "/usr/bin/c++ -g ")
      (set 'o-files (directory o-dir "\.o"))
      (dolist (o-file o-files)
        (set 'cmd (append cmd " " (real-path o-dir) "/" o-file)))
      (set 'cmd (append cmd " -o " bin-dir "/" binary-name " -static-libgcc -static-libstdc++ -static -L/usr/lib/x86_64-linux-gnu"))
      (dolist (lib libs)
        (set 'cmd (append cmd " -l" lib))
        (println cmd))
        (exec cmd)
      )
    


    为了方便使用,还提供了专门用于编译的文件compile.lsp,可以直接运行。

    ;; compile all c++ files to .o file in ./o folder
    (load "g++.lsp")
    (set 'include-paths 
         (list "../codes/main/include"
    	    "../codes/main/src/../../loki-0.1.7/include"))
    (set 'o-dir "./o")
    (set 'src-paths 
         (list "../codes/main/src"
    	   "../codes/main/src/bean"
    	   "../codes/main/src/controller"
    	   "../codes/main/src/helper"
    	   "../codes/main/src/module"
    	   "../codes/main/src/service"
    	   "../codes/main/src/view"
    	   ))
    
    (compile include-paths src-paths o-dir)
    
    (exit)


    专门动态链接的文件dlink.lsp

    #!/usr/bin/newlisp 
    
    ;; init
    (load "/opt/build_config.lsp")
    (set 'armory-folder (env "NEWLISP_ARMORY_HOME"))
    (println (append "newlisp armory home folder: " armory-folder))
    (load (append armory-folder "/codes/file/file.lsp"))
    (file:init)
    
    
    ;; compile all c++ files to .o file in ./o folder
    (load "g++.lsp")
    (set 'o-dir "./o")
    ;; link all .o files
    (set 'libs 
         (list "pthread"
    	   "cppcms"
    	   "mongoclient"
    	   "booster"
    	   "loki"
    	   "cryptopp"
    	   "boost_system"
    	   "boost_thread"
    	   "boost_filesystem"
    	   ))
    (set 'binary-name "sports_lottery_d")
    (set 'bin-dir "bin")
    (dynamic-link o-dir bin-dir binary-name libs)
    
    
    (exit)


    专门静态链接的文件slink.lsp

    #!/usr/bin/newlisp 
    
    ;; init
    (load "/opt/build_config.lsp")
    (set 'armory-folder (env "NEWLISP_ARMORY_HOME"))
    (println (append "newlisp armory home folder: " armory-folder))
    (load (append armory-folder "/codes/file/file.lsp"))
    (file:init)
    
    
    ;; compile all c++ files to .o file in ./o folder
    (load "g++.lsp")
    (set 'o-dir "./o")
    ;; link all .o files
    (set 'libs 
         (list "mongoclient"
    	   "cppcms"
    	   "booster"
    	   "boost_system"
    	   "loki"
    	   "cryptopp"
    	   "boost_thread"
    	   "pthread"
    	   "boost_filesystem"
    	   "dl"
    	   "gcrypt"
    	   "z"
    	   "pcre"
    	   "icuuc"
    	   "icui18n"
    	   "gpg-error"
    	   "icuuc"
    	   "icudata"
    	   ))
    (set 'binary-name "sports_lottery_s")
    (set 'bin-dir "bin")
    (static-link o-dir bin-dir binary-name libs)
    
    
    (exit)
    


    注意,slink.lsp中这些静态库的名称和顺序是不断尝试出来的。我总结了一个好方法:

    1. A依赖B,A必须出现在B前

    2. 一开始只加一个库mongoclient,然后看连接器报错,看少什么库,然后通过ldd来查找正确的库名称,添加在后面。这样就能找全所有的静态库。

    newlisp真是强大的脚本工具,不断改进我的生活。

  • 相关阅读:
    (转)iOS7界面设计规范(2)
    (转)iOS7界面设计规范(1)
    (转)iOS7人机界面设计规范
    (转)iOS Wow体验
    (转)iOS Wow体验
    (转)iOS Wow体验
    (转)iOS Wow体验
    bzoj1044木棍分割
    bzoj1690开关灯
    终于开了blog了。。
  • 原文地址:https://www.cnblogs.com/aukle/p/3228674.html
Copyright © 2020-2023  润新知