目录
Openwrt package Makefile
在”Openwrt main Makefile”章节里面有说道主Makefile会通过include package/Makefile调用package下的Makefile,package下的Makefile又会调用调用$(call subdir,package)遍历package子目录下的Makefile。package下的Makefile是源码里面就提供的,不会修改,但package子目录下的Makefile确是我们经常要打交道的,本章节将对其进行说明。
我们随便打开package下面的子目录,通常会发现几样东西:
package/$(PKG_NAME)/Makefile [必备]
package/$(PKG_NAME)/src/ [可选]
package/$(PKG_NAME)/patches/ [可选]
package/$(PKG_NAME)/files/ [可选]
src目录、patches目录、files目录都是可选的,src目录存放的是该功能模块的源代码,pactches目录通常包括bug修复和对可执行文件体积的优化,files目录通常是运行脚本包括配置文件等。你也可能看到其它目录,因为只要在Makefile文件中指明,目录名字是可以任取的。
Makefile文件最关键,一般来说它提供了下载、编译、安装这个软件包的步骤。
当我们打开package子目录的Makefile文件,很难认出这是一个Makefile。它的格式跟一般的Makefile不一样,因为它的功能跟普通Makefile就是不一样的,它是一种编写方便的模板。
这里,以package/bridge/Makefile文件为例:
include $(TOPDIR)/rules.mk
PKG_NAME:=bridge
PKG_VERSION:=1.0.6
PKG_RELEASE:=1
PKG_BUILD_DIR:=$(BUILD_DIR)/bridge-utils-$(PKG_VERSION)
PKG_SOURCE:=bridge-utils-$(PKG_VERSION).tar.gz
PKG_SOURCE_URL:=@SF/bridge
PKG_MD5SUM:=9b7dc52656f5cbec846a7ba3299f73bd
PKG_CAT:=zcat
include $(INCLUDE_DIR)/package.mk
define Package/bridge
SECTION:=base
CATEGORY:=Network
DEFAULT:=y
TITLE:=Ethernet bridging configuration utility
DESCRIPTION:=Ethernet bridging configuration utility\\\
Manage ethernet bridging; a way to connect networks together to\\\
form a larger network.
URL:=http://bridge.sourceforge.net/
endef
define Build/Configure
$(call Build/Configure/Default,--with-linux-headers=$(LINUX_DIR))
endef
define Package/bridge/install
$(INSTALL_DIR) $(1)/usr/sbin
$(INSTALL_BIN) $(PKG_BUILD_DIR)/brctl/brctl $(1)/usr/sbin/
endef
$(eval $(call BuildPackage,bridge))
1.包含全局变量
首先在Makefile中的第一行一定要包含下面这个命令,这是Makefile的一些全局变量的相关定义
include $(TOPDIR)/rules.mk
2.软件包变量
建立一个软件包不需要太多工作;大部分工作都隐藏在其它的makefiles中,编写工作被抽象成对几个变量的赋值。
PKG_NAME : 软件包的名字, 在 menuconfig 和 ipkg 显示
PKG_VERSION :软件包的版本,主干分支的版本正是我们要下载的
PKG_RELEASE :这个 makefile 的擦写版本
PKG_BUILD_DIR :编译软件包的目录
PKG_SOURCE :要下载的软件包的名字,一般是由 PKG_NAME 和 PKG_VERSION 组成
PKG_SOURCE_URL :下载这个软件包的链接
PKG_MD5SUM :软件包的 MD5 值
PKG_CAT :解压软件包的方法 (zcat, bzcat, unzip)
PKG_BUILD_DEPENDS :需要预先构建的软件包,但只是在构建本软件包时,而不是运行的时候。它的语法和下面的DEPENDS一样。
3.BuildPackage相关的宏定义
1.描述软件包在menuconfig和ipkg中的信息,可以定义如下变量:
define Package/< PKG_NAME >
SECTION : 软件包类型
CATEGORY : 软件包在menuconfig里的位置,如Network, Utilities
SUBMENU : menuconfig中软件包所属的二级目录,如dial-in/up
DEFAULT: 默认的编译模式,m=编译成模块,y=编译到镜像,n或者不加不编译,[依赖
包 两个之间通过空格分隔 前面加+为默认显示 选中该软件包自动选中依
赖包 不加+为默认不显示 选中依赖包才显示]
TITLE : 软件包标题
DESCRIPTION : 软件包的详细说明,由于存在bug,现在已经放弃
URL : 软件的原始位置,一般是软件作者的主页
MAINTAINER : (optional) 软件包维护人员
DEPENDS : (optional) 依赖项,运行本软件依赖的其他包
endif
2.配置说明
define Build/Configure (可选)
在Automake中需要进行./configure,所以本配置方法主要针对需要配置的软件
包而设计,一般自行开发的软件包可以不在这里说明。
endif
3.软件包安装
define Package/install
软件包的安装方法,包括一系列拷贝编译好的文件到指定位置。調用時會帶一
個參數,就是嵌入系統的鏡像文件系統目錄,因此$(1)表示嵌入系统的镜像目
录。一般可以采用下面的方法:
$(INSTALL_DIR) $(1)/usr/bin
$(INSTALL_BIN) $(PKG_BUILD_DIR)/$(PKG_NAME) $(1)/usr/bin/
endif
4.其他BuildPackage相关的宏定义
上面列出的三个是在bridge这个模块下面用的宏定义,其他还有很多其他的定义,我们也可以了解下,再其他情况可能也会用到。
1.编译准备
define Build/Prepare (可选)
对于网上下载的软件包不需要再描述,对于非网上下载或自行开发的软件包
必须说明编译准备方法,如下:
mkdir -p $(PKG_BUILD_DIR)
创建编译目录,也就是$(TOPDIR)/build_dir/target-<ARCH>*/$(PKG_NAME)-%(PKG_VERSION)
$(CP) ./src/* $(PKG_BUILD_DIR)/
将软件包的src的源码文件拷贝到编译目录去
endif
2.编译源代码
define Build/Compile (可选)
默认是编译源码里面的Makefile,如果你想传递一些参数比如环
境变量什么的,那就可以定义,编译方法,没有特别说明的可以不予以定义。如果不
定义将使用默认的编译方法Build/Compile/Default
自行开发的软件包可以考虑使用下面的定义。
$(MAKE) -C $(PKG_BUILD_DIR) \
$(TARGET_CONFIGURE_OPTS) \
CFLAGS="$(TARGET_CFLAGS) -I$(LINUX_DIR)/include"
endif
3.安装之前执行的脚本
define Package/ $(PKG_NAME)/ preinst (可选)
软件安装之前被执行的脚本,别忘了在第一句加上#!/bin/sh,如果脚本执行完毕要取消 安装过程,直接让它返回false即可。
#!/bin/sh
.........
exit 0
endif
4.安装之后执行的脚本
define Package/ $(PKG_NAME)/ postinst (可选)
软件安装之后被执行的脚本,别忘了在第一句加上#!/bin/sh。
#!/bin/sh
.........
exit 0
endif
5.删除之前被执行的脚本
define Package/ $(PKG_NAME)/ prerm (可选)
软件删除之前被执行的脚本,别忘了在第一句加上#!/bin/sh。如果脚本执行完毕要取消 删除过程,直接让它返回false即可。
#!/bin/sh
.........
exit 0
endif
6.删除之后被执行的脚本
define Package/ $(PKG_NAME)/ postrm (可选)
软件删除之后被执行的脚本,别忘了在第一句加上#!/bin/sh。
#!/bin/sh
.........
exit 0
endif
为什么一些定义是”Package/”前缀,另一些定义却是”Build”前缀?这是因为Openwrt支持一个特性:从单个源代码构建多个软件包。OpenWrt工作在一个Makefile对应一个源代码的假设之上,但是你可以把编译生成的程序分割成任意多个软件包。因为编译只要一次,所以使用全局的”Build”定义是最合适的。然后你可以增加很多“Package/”定义,为各软件包分别指定安装方法。
5.软件包的实现
完成前面定义后,必须使用eval函数实现各种定义。这是最为关键的BuildPackage宏,它是在$(INCLUDE_DIR)/package.mk文件里定义的。BuildPackage宏只要求一个参数,即要编译的软件包名,在本例中是”bridge”。
对于一般软件包$(eval $(call Package,$(PKG_NAME)))
对于内核模块$(eval $(call KernelPackage,$(PKG_NAME)))
如果一个软件包有多个程序,eval函数也可以设计多个软件包处理。
在编辑好Makefile文件,并放到指定目录后,这个新的软件包将在下次执行make menuconfig时出现,你可以选择这个软件包,保存退出,编译,就把一个软件包成功移植到OpenWrt中了,具体例子将在“Openwrt add function module to package”章节就行说明。