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)

  • 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

gcc_to_use的更多相关文章

随机推荐

  1. 借助Radamsa变异数据(初探)

    Radamsa 介绍 Radamsa是一款测试用例生成器,通常用来测试程序对格式错误和潜在恶意输入的承受能力(对程序进行模糊测试).它通过你的输入来返回变异后的数据.它的主要卖点是,它已经在真正重要的 ...

  2. 创建型模式 - 简单工厂模式StaticFactoryMethod

    简单工厂模式的定义         创建型模式:         我们把被创建的对象称为产品,把创建产品的对象称为工厂.如果要创建的产品不多,只要一个工厂类就可以完成,这种模式叫简单工厂模式. 在简单 ...

  3. 【分析笔记】NXP PCF85263 设备驱动分析笔记

    驱动移植 供应商无法提供相应的驱动程序,不过在 linux 最新的内核倒是有一份 pcf85363 的驱动,看代码并核对寄存器功能,是可以兼容 pcf85263 芯片.只是我们用的内核比较老 linu ...

  4. immutable.js学习笔记(二)----- List

    一.List list与数组是兼容的,大多数的api与数组是类似的 注意 List.of(),不需要写中括号 二.List的API (一)size:取得 List 的长度 (二)set:设定指定下标的 ...

  5. C#,Winform软件防破译-源代码加密简单方法之.NET REACTOR(一)

    一..NET Reactor介绍 .NET Reactor是一个功能强大的代码保护和软件许可系统,适用于为.NET Framework编写的软件,并支持生成.NET程序集的所有语言. 支持Blazor ...

  6. drf-序列化字段及参数、序列化和反序列化高级用法、ModelSerializer使用

    1.序列化类常用字段和字段参数 1.1 常用字段类 1.BooleanField 2 NullBooleanField 3 CharField CharField(max_length=None, m ...

  7. Arm-Linux子系统的互相Notify

    前言: Linux下面不同的子系统一个个的组成了整个系统的运行环节,为了让这些子系统能够互相通讯,有一种叫做:notify chain(通知链)的东西.本篇看下. 概括 所谓通知链,有通知,就有执行的 ...

  8. ASP.NET Core - IStartupFilter 与 IHostingStartup

    1. IStartupFilter   上面讲到的方式虽然能够根据不同环境将Startup中的启动逻辑进行分离,但是有些时候我们还会可以根据应用中的功能点将将一系列相关中间件的注册封装到一起,从 St ...

  9. 分布式机器学习:异步SGD和Hogwild!算法(Pytorch)

    1 导引 我们在博客<分布式机器学习:同步并行SGD算法的实现与复杂度分析(PySpark)>和博客<分布式机器学习:模型平均MA与弹性平均EASGD(PySpark) >中介 ...

  10. .net core 阿里云接口之将指定的OSS文件下载到流

    紧接上文,.net core 阿里云接口之获取临时访问凭证_SunshineGGB的博客-CSDN博客 本文继续阿里云接口调用,将指定的OSS文件下载到流. 直接上代码: /// <summar ...