Android编译系统中头文件搜索路径的顺序问题

移动开发 Android
Android编译系统本身设置了一堆公共的头文件搜索路径,然后允许每个项目在自己的Android.mk中通过LOCAL_C_INCLUSES 来添加独特的搜索路径。

今天在编译一个代码时发现Android编译系统在设置头文件搜索路径的顺序上好像有些问题。Android编译系统本身设置了一堆公共的头文件搜索路径(参见pathmap.mk中pathmap_INCL的定义),然后允许每个项目在自己的Android.mk中通过LOCAL_C_INCLUSES 来添加独特的搜索路径。按照一般的想法,在最后的编译参数中,项目自己独特的搜索路径应该放在公共搜索路径之前,这样,一旦出现头文件名冲突的情况,会优 先使用项目自己指定的头文件。但是在Android的编译系统中情况并非如此,项目自定义的头文件搜索路径反而被放在了最后。参见 definitions.mk文件里的下面这个定义:

  1. define transform-cpp-to-o 
  2. @mkdir -p $(dir $@) 
  3. @echo "target $(PRIVATE_ARM_MODE) C++: $(PRIVATE_MODULE) <= $<" 
  4. $(hide) $(PRIVATE_CXX) \ 
  5.     $(foreach incdir, \ 
  6.         $(if $(PRIVATE_NO_DEFAULT_COMPILER_FLAGS),, \ 
  7.         $(TARGET_PROJECT_INCLUDES) \ 
  8.         $(TARGET_C_INCLUDES) \ 
  9.          ) \ 
  10.         $(PRIVATE_C_INCLUDES) \ 
  11.       , \ 
  12.         -I $(incdir) \ 
  13.      ) \ 
  14.     -c \ 
  15.     $(if $(PRIVATE_NO_DEFAULT_COMPILER_FLAGS),, \ 
  16.         $(TARGET_GLOBAL_CFLAGS) \ 
  17.         $(TARGET_GLOBAL_CPPFLAGS) \ 
  18.         $(PRIVATE_ARM_CFLAGS) \ 
  19.      ) \ 
  20.     -fno-rtti \ 
  21.     $(PRIVATE_CFLAGS) \ 
  22.     $(PRIVATE_CPPFLAGS) \ 
  23.     $(PRIVATE_DEBUG_CFLAGS) \ 
  24.     -MD -o $@ $< 
  25. $(hide) $(transform-d-to-p) 
  26. endef 

这个定义就是编译C++文件使用的命令行。注意红字部分,PRIVATE_C_INCLUDES中包含了项目的LOCAL_C_INCLUDES的定义 (参见binary.mk)。明显项目自定义的搜索路径被放在了最后。后面还有C文件的编译命令行的定义(define transform-c-or-s-to-o-no-deps)也同样如此。

不知道Android这样设计是出于什么考虑。我尝试把顺序调整一下,看是否会影响Android的编译。上述定义调整之后如下:

  1. define transform-cpp-to-o 
  2. @mkdir -p $(dir $@) 
  3. @echo "target $(PRIVATE_ARM_MODE) C++: $(PRIVATE_MODULE) <= $<" 
  4. $(hide) $(PRIVATE_CXX) \ 
  5.     $(foreach incdir, \ 
  6.         $(PRIVATE_C_INCLUDES) \  
  7. $(if $(PRIVATE_NO_DEFAULT_COMPILER_FLAGS),, \ 
  8.         $(TARGET_PROJECT_INCLUDES) \ 
  9.         $(TARGET_C_INCLUDES) \ 
  10.          ) \ 
  11.       , \ 
  12.         -I $(incdir) \ 
  13.      ) \ 
  14.     -c \ 
  15.     $(if $(PRIVATE_NO_DEFAULT_COMPILER_FLAGS),, \ 
  16.         $(TARGET_GLOBAL_CFLAGS) \ 
  17.         $(TARGET_GLOBAL_CPPFLAGS) \ 
  18.         $(PRIVATE_ARM_CFLAGS) \ 
  19.      ) \ 
  20.     -fno-rtti \ 
  21.     $(PRIVATE_CFLAGS) \ 
  22.     $(PRIVATE_CPPFLAGS) \ 
  23.     $(PRIVATE_DEBUG_CFLAGS) \ 
  24.     -MD -o $@ $< 
  25. $(hide) $(transform-d-to-p) 
  26. endef  

对define transform-c-or-s-to-o-no-deps也做类似调整。重新编译后发现只有webkit的编译有问题。原因在于webkit的代码中 包含了几个STL的头文件(WebKit/android/stl),特别是其中的strings与bionic定义的头文件冲突,在调整头文件搜索顺序 后,优先选择了这个文件。这个strings文件其实是一个空文件(除了注释没有任何语句),干脆删除了它,果然编译顺利进行了,一直到编译完成再也没有 出现问题。看来上述调整时可行的。我用的android源代码版本是2.0,其它版本没有试过不知道怎么样。BTW,如果想在编译时打印出编译命令,在make的参数中加上SHOW_COMMANDS=1即可。

责任编辑:闫佳明 来源: oschina
相关推荐

2009-10-23 15:53:00

linux库文件

2011-07-14 22:36:37

C++

2009-06-17 17:00:03

2010-01-25 17:55:38

C++头文件

2021-06-08 08:52:18

Makefile编译c 文件

2013-06-28 14:00:28

Android

2011-07-06 11:20:09

WPS行政公文

2010-02-06 14:48:37

C++头文件

2023-10-06 23:56:42

顺序查找Python

2016-09-19 10:54:36

C语言静态连接语言

2010-08-05 10:42:41

Android开发Android高级编程

2011-09-14 15:23:00

Android 2.2

2013-05-21 10:10:29

Hadoop文件系统

2013-05-24 09:57:28

搜索流程搜索产品设计搜索设计

2010-02-02 13:04:03

C++头文件

2009-07-10 17:16:39

MyEclipse不编

2009-11-30 16:38:30

Android

2023-06-24 10:44:34

Linux文件搜索

2013-04-23 09:31:52

SQL Server

2010-03-03 17:43:12

Android系统
点赞
收藏

51CTO技术栈公众号