首页 > 其他分享 >gcc_to_use

gcc_to_use

时间:2023-01-28 18:33:31浏览次数:35  
标签:SRC gcc 文件 LIST use 编译 hello

gcc

目录

概要

  • GCC:GNU Compiler Collection(GUN 编译器集合),是GNU项目中符合ANSI C标准的编译系统,它可以编译C、C++、JAV、Fortran、Pascal、Object-C、Ada等语言。并且GCC是一个交叉平台编译器,能够在当前CPU平台上为多种不同体系结构的硬件平台开发软件,因此尤其适合在嵌入式领域的开发编译。
    • gcc是GCC中的GUN C Compiler(C 编译器)。
    • g++是GCC中的GUN C++ Compiler(C++编译器)。
  • GCC编译流程:
    • 预处理(Pre-Processing)
    • 编译(Compiling)
    • 汇编(Assembling)
    • 链接(Linking)
      img
  • GCC特点:
    • GCC是一个可移植的编译器,支持多种硬件平台。例如ARM、X86等等。
    • GCC不仅是个本地编译器,它还能跨平台交叉编译。所谓的本地编译器,是指编译出来的程序只能够在本地环境进行运行。而GCC编译出来的程序能够在其他平台进行运行。例如嵌入式程序可在x86上编译,然后在arm上运行。
    • GCC有多种语言前端,用于解析不同的语言。
    • GCC是按模块化设计的,可以加入新语言和新CPU架构的支持。
    • GCC是自由软件。任何人都可以使用或更改这个软件。

基本指令及功能(以gcc为例)

  • gcc编译文件
后缀 解析 后缀 解析
.c C原始程序 .s/.S 汇编语言
.C/.cc/.cxx C++原始程序 .h 预编译头文件
.m Objective-C原始程序 .o 目标文件
.i 已预处理C原始程序 .a/.so 库文件
.ii 已预处理C++原始程序
  • gcc编译选项
选项 作用
-o file 产生目标file文件.i/.s/.o/可执行文件等
-E 只运行C预编译器
-S 产生汇编文件.s/.S后停止编译
-c 产生目标文件.o但不链接形成可执行文件
-Wall 使gcc对源程序代码有问题的地方发出警告
-I dir 将dir目录加入搜索头文件目录路径
-L dir 将dir目录加入搜索库文件目录路径
-llib 链接到名为lib的库
-g 在目标文件嵌入调试信息,方便gdb/cgdb程序调试
-ansi 只支持ANSI标准编译,禁止GNU C的某些特色(如asm/typeof关键字)
-DMACRO 以字符串"1"定义MACRO宏
-DMACRO=DEFN 以字符串"DEFN"定义MACRO宏
-IDIRECTORY 指定额外的头文件搜索路径DIRECTORY
-LDIRECTORY 指定额外的函数库搜索路径DIRECTORY
-m486 针对486进行代码优化
-o FILE 生成指定的输出文件,用在生成可执行文件时
-O0 不进行优化处理
-O/-O1/-O2/-O3 优化生成代码
-shared 生成共享目标文件,通常用在建立共享库时
-static 禁止使用共享连接
-UMACRO 取消对MACRO宏的定义
-w 不生成任何警告信息
  • 编译案例
指令一般格式:gcc [选项] 要编译的文件 [选项] [目标文件]

# 对hello.c文件进行预处理,生成了hello.i 文件
gcc -E hello.c -o hello.i   
# 对预处理文件进行编译,生成了汇编文件
gcc -S hello.i -o hello.s    
# 对汇编文件进行编译,生成了目标文件
gcc -c hello.s -o hello.o  
# 对目标文件进行链接,生成可执行文件
gcc hello.o -o hello 
# 直接编译链接成可执行目标文件
gcc hello.c -o hello 
# 编译生成可重定位目标文件
gcc -c hello.c or gcc -c hello.c -o hello.o 
# gcc输出警告选项
gcc -Wall hello.c -o hello
  • 独立编译和一次编译
# 一次编译
gcc hello.c main.c -o main
# 独立编译
gcc -Wall -c main.c -o main.o
gcc -Wall -c hello.c -o hello.o
gcc -Wall main.o hello.o -o main
独立编译运用到复杂项目时,当其中某个模块代码发生改变,只需编译该模块代码,
不必重新编译所有程序文件,这样可以节省编译时间。
    • 可重用函数关联功能函数
    • 模块化开发
    • 可维护性
  • 静态库(.a):程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库。静态库比较占用磁盘空间,而且程序不可以共享静态库。运行时也是比较占内存的,因为每个程序都包含了一份静态库。
  • 动态库(.so):程序在运行的时候才去链接共享库的代码,多个程序共享使用库的代码,这样就减少了程序的体积。
  • 头文件及库文件位置
    • /usr/include及其子目录底下的include文件夹
    • /usr/local/include及其子目录底下的include文件夹
    • /usr/lib
    • /usr/local/lib
    • /lib
  • 库生成及使用
# rcs(replace and create)生成静态库
ar rcs libhello.a hell.o
# 使用静态库
gcc -Wall main.c libhello.a -o main
# 其中lhello为libhello的缩写
gcc -Wall -L main.c -o main -lhello

# 生成共享库(动态库)
gcc -shared -fPIC hello.o -o libhello.so
# 使用动态库
gcc -Wall main.o -o main -L. -lhello

# 库路径的搜索原则:库的搜索路径遵循几个搜索原则:从左到右搜索-I -l指定的目录,如果在这些目录中找不到,
那么gcc会从由环境 变量指定的目录进行查找。头文件的环境变量是C_INCLUDE_PATH,库的环境变量是LIBRARY_PATH.如果还是找不到,那么会从系统指定指定的目录进行搜索。
# 共享库有时候并不在当前目录下,需要配置到gcc可寻找的路径下(3种方式)
# 1. copy .so文件到系统共享库路径/usr/lib下
# 2. 在~/.bash_profile文件中,配置LD_LIBRARY_PATH变量
# 3. 配置/etc/ld.so.conf,配置完成后调用ldconfig更新ld.so.cache

gcc -gdb

gcc -cmake

  • cmake基本语法规则
    • 变量使用${}取值,IF控制语句中直接使用变量名
    • 指令(参数1 参数2 ...)
    • 指令大小写无关,参数和变量大小写相关,推荐全部使用大写
  • 指定cmake最小版本
CMAKE_MINIMUM_REQUIRED(VERSION 3.5.1)
  • 设置项目名称(此处自动引入两个变量 hello_BINARY_DIR 和 hello_SOURCE_DIR)
PROJECT(hello)
  • 生成可执行文件
ADD_EXECUTABLE(hello hello_world.cpp) 
  • 设置编译类型
ADD_LIBRARY(hello hello.cpp) # 默认生成静态库
ADD_LIBRARY(hello STATIC hello.cpp) # 指定为静态库
ADD_LIBRARY(helloshared SHARED hello.cpp) # 指定为共享库

假设 hello.cpp 文件中没有 main 函数,意味着这个库中没有可执行文件。添加上述命令,cmake 后会依次生成下面几个文件:
libhello.a
libhello.a
libhello_shared.so
静态库以 .a 为后缀名,共享库以 .so 为后缀名,所有库都是一些函数打包后的集合,静态库每次被调用都会生成一个副本,而共享库则只有一个副本,更省空间。
库文件是一个压缩包,里面有编译好的二进制函数,仅有 .a 或 .so 库文件,别人并不知道里面的函数是什么样的,或如何调用;这就需要提供一个头文件,有了库文件和头文件,就可以调用这个库。

  • 设置变量
    • set直接设置变量值
    SET(SRC_LIST a.cpp b.cpp)
    ADD_EXECTUABLE(demo ${SRC_LIST}) 
    
    • set追加设置变量值
    SET(SRC_LIST a.cpp)
    SET(SRC_LIST ${SRC_LIST} b.cpp)
    ADD_EXECTUABLE(demo ${SRC_LIST})
    
    • LIST追加或删除变量值
    SET(SRC_LIST a.cpp)
    LIST(APPEND SRC_LIST b.cpp)
    LIST(REMOVE_ITEM SRC_LIST b.cpp)
    ADD_EXECTUABLE(demo ${SRC_LIST})
    
  • 指定源文件
    • 明确指定(空格或分好隔离开)
    ADD_LIBRARY(hello hello.cpp a.cpp b.cpp)
    
    • 所有源文件
    # AUX_SOURCE_DIRECTORY(dir VAR) 发现一个目录下所有的源代码文件并将列表存储在一个变量中
    AUX_SOURCE_DIRECTORY(. SRC_LIST) # 搜索当前目录下的所有.cpp文件
    ADD_LIBRARY(hello ${SRC_LIST})
    
  • 查找指定库文件
    FIND_LIBRARY(VAR name path)查找到指定的预编译库,并将它的路径存储在变量中。默认的搜索路径为 cmake 包含的系统库。
    类似的命令还有 find_file()、find_path()、find_program()、find_package()等。
  • 设置包含目录
# case 0 
INCLUDE_DIRECTORIES(
    ${CMAKE_CURRENT_SOURCE_DIR}
    ${CMAKE_CURRENT_BINARY_DIR}
    ${CMAKE_CURRENT_SOURCE_DIR}/include

# case 1
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -I${CMAKE_CURRENT_SOURCE_DIR}")
  • 设置链接库搜索目录
# case 0
LINK_DIRECTORIES(
    ${CMAKE_CURRENT_SOURCE_DIR}/libs
)

# case 1
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -L${CMAKE_CURRENT_SOURCE_DIR}/libs")
  • 设置target需要链接的库
# 指令TARGET_LINK_LIBRARIES(target_lib  link_lib_path)

# 指定链接静态库或动态库
TARGET_LINK_LIBRARIES(hello libhello.a)
TARGET_LINK_LIBRARIES(hello libhello_shared.so) 

# 设置全路径
TARGET_LINK_LIBRARIES(hello ${CMAKE_CURRENT_SOURCE_DIR}/libs/libhello.a)
TARGET_LINK_LIBRARIES(hello ${CMAKE_CURRENT_SOURCE_DIR}/libs/libhello_shared.so)
  • 编译流程

    • 编写hello.cpp源文件和CMakeLists.txt文件
    • 新建build目录,在build目录下cmake ..命令,生成了 CMakeFIles 文件夹和CMakeCache.txt, cmake_install.cmake, Makefie 文件
    • 在build路径下执行make命令
  • CMakeLists.txt文件demo

# cmake 版本
cmake_minimum_required(VERSION 2.6)
 
####
# CMake中指令不区分大小写   
# ${} 表示取出变量中的值
####
 
# 项目名称, 名称后面为支持的语言,不写时默认支持所有语言
PROJECT(project_name [CXX][C][java])
# 或者
PROJECT(name )
# PROJECT(...)语句运行之后,会默认定义两个变量,后面可以直接使用
# name_BINARY_DIR  编译目录
# name_SOURCE_DIR  工程目录
 
# set表示设置变量名
SET(LIBRARIES /usr/lib/x86_64-linux-gnu/libm.so)
# 变量中只保存一个源文件
SET(SRC_LIST main.c)  # 或者 SET(SRC_LIST “main.c”)    # 把main.c保存到变量SRC_LIST中
# 也可以是保存多个源文件
SET(SRC_LIST main.c func.c func.h)
# 当然也可以保存所有的源文件,将所有源文件保存到SRC_LIST变量中
AUX_SOURCE_DIRECTORY(directory SRC_LIST)
# 把当前目录下所有源文件全部保存到SRC_LIST变量中
AUX_SOURCE_DIRECTORY(. SRC_LIST)
# 选择指定目录下的源文件保存到变量中
AUX_SOURCE_DIRECTORY(./dir/main.c SRC_LIST)
 
# message 表示输出提示信息
MESSAGE([SEND_ERROR | STATUS | FATAL_ERROR] "message")
# STATUS 参数较为常用,输出前缀为-的信息
# 第一个参数是消息类型,后面的参数是一条或多条要显示的消息。错误类型有3种:
# SEND_ERROR:表示产生错误信息
# STATUS:表示一般的状态信息
# FATAL_ERROR:我们知道肯定是严重错误信息,cmake会立即停止执行
 
 
# 生成可执行文件
ADD_EXECUTABLE(result ${SRC_LIST})
 
# 添加可执行文件所需要的库
TARGET_LINK_LIBRARIES(result ${LIBRARIES})
 
# 更改可执行文件的输出目录
SET(EXECUTABLE_OUTPUT_PATH "${CMAKE_BINARY_DIR}/bin")
# 其中CMAKE_BINARY_DIR 变量中的CMAKE_表示项目名的统称,等价于上面所说的 name_BINARY_DIR  

标签:SRC,gcc,文件,LIST,use,编译,hello
From: https://www.cnblogs.com/kuanghl/p/17071088.html

相关文章