Android编译系统之交叉编译器详解

2023-11-27 16:50

本文主要是介绍Android编译系统之交叉编译器详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、引言

Android开发的同僚都知道,Andriod本身有一套很完善的编译机制,也就是在/build 目录下,每当我们想新增一个库、可执行文件等,只需要添加对应的Android.mk文件即可,具体该文件的通法,详见我之前的博客四、安卓源码分析之Android.mk,而其实Android.mk其实是封装了交叉编译器,所以我们编写的文件才能直接在Android终端上运行。我们今天就来详细看下Android中的交叉编译器

小广告

中间打个小广告,是本人的一个小小副业。

大家有需要各种品牌的鞋(耐克、阿迪、斐乐、vans、匡威等),都可以加下面这个微信号,性价比巨高,质量绝对可靠,自己穿的也都是这买的,到手不喜欢,不影响二次销售可直接退货,希望大家能多多支持(暂时不想买的也欢迎添加,首双优惠!),全国包邮!

vx:cp_shop12138

二、Android交叉编译器

Android源码是自动集成了一个的交叉编译器的,低版本在prebuilt/tool,4以上放在了prebulits/gcc下面的。
在这里插入图片描述
对于交叉编译器来说,在于提供了良好的基础库用于应用程序的编程。而对于内核操作系统的源码来说(全部源码,外加需要的库等都存在),其根本不需要编译器提供任何的库支撑,编译器只需要把内核各模块进行编译,然后链接出最终的image就可以。
而对于应用来说,编译器基本是动态链接库的,因此这些库都在基本的文件系统下面。故需要android自己的编译器,因为用的是bionic库来完成,而android源码文件编译时会自动使用自己的交叉编译器,生成NDK层的.so等,这些都用的自带的编译器。故只要内核的编译器可以满足CPU架构的需要如arm-linux等,应该就可以。一句话内核编译对编译器的依赖性很小,编译器只是辅助。编译器对应用比较关键,编译要链接库,跑起来要用自己的编译器带来的系统库文件,故肯定是需要android自己的。

查看编译kernel uboot的交叉编译器

查看kernel、uboot根目录下的Makefile可知,两者用的都是android自带的交叉编译器

# Cross compiling and selecting different set of gcc/bin-utils
# ---------------------------------------------------------------------------
#
# When performing cross compilation for other architectures ARCH shall be set
# to the target architecture. (See arch/* for the possibilities).
# ARCH can be set during invocation of make:
# make ARCH=ia64
# Another way is to have ARCH set in the environment.
# The default ARCH is the host where make is executed.# CROSS_COMPILE specify the prefix used for all executables used
# during compilation. Only gcc and related bin-utils executables
# are prefixed with $(CROSS_COMPILE).
# CROSS_COMPILE can be set on the command line
# make CROSS_COMPILE=ia64-linux-
# Alternatively CROSS_COMPILE can be set in the environment.
# A third alternative is to store a setting in .config so that plain
# "make" in the configured kernel build directory always uses that.
# Default value for CROSS_COMPILE is not to prefix executables
# Note: Some architectures assign CROSS_COMPILE in their arch/*/Makefile
ARCH            ?= arm
ARCH            ?= $(SUBARCH)
ifeq ($(ARCH),arm64)
ifneq ($(wildcard ../prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9),)
CROSS_COMPILE   ?= ../prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9/bin/aarch64-linux-android-
endif
endif

脱离android本身的编译系统编译程序

一般如果不使用android本身的编译系统,是需要下在一个ndk,使用其中的交叉编译器和库完成的,这里就用prebulits目录下的编译器及库代替,只要下在好了ndk后,后续的过程差不多

先编写好测试的程序
#include "stdio.h"
void main()
{printf("first test in android! \n");
}
Makefile

接着再编写Makefile,最重要的东西

.PHONY: cleanDIRROOT=../android/prebuiltsSTALIB=../android/prebuilts/ndk/9/platforms/android-4/arch-arm/
CROSS_COMPILE=$(DIRROOT)/gcc/linux-x86/arm/arm-linux-androideabi-4.6/bin/arm-linux-androideabi-CC=$(CROSS_COMPILE)gcc
AR=$(CROSS_COMPILE)ar
LD=$(CROSS_COMPILE)ld#不连接系统标准启动文件和标准库文件,只把指定的文件传递给连接器
#CFLAGS= -I$(STALIB) -L$(LIB) -nostdlib --sysroot $(STALIB)
CFLAGS= --sysroot $(STALIB)TARGET = test1
SRCS = $(wildcard *.c)
OBJS = $(SRCS:.c=.o)all: $(SRCS)$(CC) $(CFLAGS) $(SRCS) -o $(TARGET)clean:rm -f *.o *.a *.so
其中有几个重点:

1、设置交叉编译器的路径
注意:此处必须是绝对路径

CROSS_COMPILE=$(DIRROOT)/gcc/linux-x86/arm/arm-linux-androideabi-4.6/bin/arm-linux-androideabi-

2、设定头文件及库文件
这个简单的c源文件中引入了stdio.h这个头文件,我们需要给编译器指定一个头文件及库文件的查找路径才行

关于头文件及库文件的路径的指定有下面几条常用的指令:

(1)–sysroot=AAA
在AAA这个路径下的 usr/include中查找头文件;在AAA这个路径下的 usr/lib中查找库文件

(2)-isysroot BBB
在BBB这个路径下的usr/include中查找头文件,需要注意的是:这样设置之后,会覆盖–sysroot=AAA中的头文件的查找路径

(3)-isystem CCC
直接在CCC这个路径的目录下去查找头文件,并不是usr/include中了,需要注意

本文使用第一种

STALIB=../android/prebuilts/ndk/9/platforms/android-4/arch-arm/
CFLAGS= --sysroot $(STALIB)

3、“-nostdlib”的使用
-nostdlib作用: 不连接系统标准启动文件和标准库文件,只把指定的文件传递给连接器。这个选项常用于编译内bootloader等程序,它们不需要启动文件、标准库文件。
如果不加上此命令,会出现如下错误

crtbegin_dynamic.o: No such file: No such file or directory
编入固件

如上编译好Makefile后,才此目录下直接make,会生成一个可执行文件,将其放入Android或值直接编进固件,在Android终端上执行此命令,就会出现如下打印,即成功
在这里插入图片描述

移植NDK来编译android应用

关于这个,有一篇文章讲的很好,大家可以去查阅构建Android的交叉编译器、用NDK编译移植

四、使用Android中的编译系统

其实Android中的编译系统已经十分完善,一般的库也都兼容了这个编译系统,都会自带的提供Android.mk。

比如最近刚刚移植的curl库,就是直接使用Andriod中的编译系统完成的
一般我会将外部移植的库放在/external 目录下
在这里插入图片描述
在编译的时候,我们只需要根据我们的android,修改相应的动态库,头文件路径,然后使用mm 编译既可

利用build编译出来的所有文件:动态库,可执行文件等,都会放在out目录下面
在这里插入图片描述
可以看到编译出来的libcurl 动态库文件

这篇关于Android编译系统之交叉编译器详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/428121

相关文章

Linux换行符的使用方法详解

《Linux换行符的使用方法详解》本文介绍了Linux中常用的换行符LF及其在文件中的表示,展示了如何使用sed命令替换换行符,并列举了与换行符处理相关的Linux命令,通过代码讲解的非常详细,需要的... 目录简介检测文件中的换行符使用 cat -A 查看换行符使用 od -c 检查字符换行符格式转换将

详解C#如何提取PDF文档中的图片

《详解C#如何提取PDF文档中的图片》提取图片可以将这些图像资源进行单独保存,方便后续在不同的项目中使用,下面我们就来看看如何使用C#通过代码从PDF文档中提取图片吧... 当 PDF 文件中包含有价值的图片,如艺术画作、设计素材、报告图表等,提取图片可以将这些图像资源进行单独保存,方便后续在不同的项目中使

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

C#数据结构之字符串(string)详解

《C#数据结构之字符串(string)详解》:本文主要介绍C#数据结构之字符串(string),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录转义字符序列字符串的创建字符串的声明null字符串与空字符串重复单字符字符串的构造字符串的属性和常用方法属性常用方法总结摘

Java中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML