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

相关文章

Android 12解决push framework.jar无法开机的方法小结

《Android12解决pushframework.jar无法开机的方法小结》:本文主要介绍在Android12中解决pushframework.jar无法开机的方法,包括编译指令、框架层和s... 目录1. android 编译指令1.1 framework层的编译指令1.2 替换framework.ja

Redis中6种缓存更新策略详解

《Redis中6种缓存更新策略详解》Redis作为一款高性能的内存数据库,已经成为缓存层的首选解决方案,然而,使用缓存时最大的挑战在于保证缓存数据与底层数据源的一致性,本文将介绍Redis中6种缓存更... 目录引言策略一:Cache-Aside(旁路缓存)策略工作原理代码示例优缺点分析适用场景策略二:Re

Android开发环境配置避坑指南

《Android开发环境配置避坑指南》本文主要介绍了Android开发环境配置过程中遇到的问题及解决方案,包括VPN注意事项、工具版本统一、Gerrit邮箱配置、Git拉取和提交代码、MergevsR... 目录网络环境:VPN 注意事项工具版本统一:android Studio & JDKGerrit的邮

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

Java注解之超越Javadoc的元数据利器详解

《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

MySQL数据库约束深入详解

《MySQL数据库约束深入详解》:本文主要介绍MySQL数据库约束,在MySQL数据库中,约束是用来限制进入表中的数据类型的一种技术,通过使用约束,可以确保数据的准确性、完整性和可靠性,需要的朋友... 目录一、数据库约束的概念二、约束类型三、NOT NULL 非空约束四、DEFAULT 默认值约束五、UN

Python使用Matplotlib绘制3D曲面图详解

《Python使用Matplotlib绘制3D曲面图详解》:本文主要介绍Python使用Matplotlib绘制3D曲面图,在Python中,使用Matplotlib库绘制3D曲面图可以通过mpl... 目录准备工作绘制简单的 3D 曲面图绘制 3D 曲面图添加线框和透明度控制图形视角Matplotlib

MySQL中的分组和多表连接详解

《MySQL中的分组和多表连接详解》:本文主要介绍MySQL中的分组和多表连接的相关操作,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录mysql中的分组和多表连接一、MySQL的分组(group javascriptby )二、多表连接(表连接会产生大量的数据垃圾)MySQL中的

Java 实用工具类Spring 的 AnnotationUtils详解

《Java实用工具类Spring的AnnotationUtils详解》Spring框架提供了一个强大的注解工具类org.springframework.core.annotation.Annot... 目录前言一、AnnotationUtils 的常用方法二、常见应用场景三、与 JDK 原生注解 API 的

redis中使用lua脚本的原理与基本使用详解

《redis中使用lua脚本的原理与基本使用详解》在Redis中使用Lua脚本可以实现原子性操作、减少网络开销以及提高执行效率,下面小编就来和大家详细介绍一下在redis中使用lua脚本的原理... 目录Redis 执行 Lua 脚本的原理基本使用方法使用EVAL命令执行 Lua 脚本使用EVALSHA命令