[转] 用Diff和Patch工具维护源码
在Unix系統下,維護源碼版本可以使用很多方法,其中最常用的當然是大名鼎鼎的CVS,但實際上,簡單的版本維護工作并沒有必要使用復雜的CVS等專門的版本維護工具,Unix標配中的diff和patch工具就完全可以完成代碼的簡單備份和升級工作。
diff以"行"為單位比較兩個文本文件(也可以是目錄比較),并將不同之處以某種格式輸出到標準輸出上;patch可以讀入這種輸出,并按照一定指令使源文件(目錄)按照目標文件(目錄)更新。Linux內核源碼就是按照這種方式保持更新的,我們在www.kernel.org上可以下載到最新內核的patch文件的bzip2包。本文以gnudiffutils 2.7和patch 2.5為例介紹diff和patch工具的使用。
1.diff
diff既可以用來比較兩個文件,也可以用來比較兩個目錄中每個文件。使用-r(--recursive)參數時還可以在目錄中嵌套比較。比較目錄時除比較同名文件外,對不同名的文件當成新文件處理。對于比較C程序文件,diff還提供了專門的參數(-p,--show-c- function)來標識不同之處所在的函數名。
diff的輸出格式有三種:列舉方式、命令模式和上下文模式,其中命令模式有分為兩種:ed命令格式和RCS(Revision Control System,版本控制系統)命令格式,上下文模式也按格式分為老版和新版兩種。看下面的例子就能基本清楚各個格式的區別:
命令格式記錄的是從test1更新到test2所需要執行的命令,而上下文模式通常可讀性更好一些,它所記錄的主要是二者的差異,通常還記錄所需修改部分的上下幾行(可配置)內容以供比較。見下面的例子:
新版格式較之老版要緊湊一些,Linux內核源碼的升級就是按照新版上下文格式用diff組織的,比如patch-2.4.16中所用的具體命令為:
diff -Nur linux-2.4.15 linux參數N表示如果某個文件僅在一個目錄中出現,則假定其在另一個目錄中為空文件;u表示unified格式,r表示在目錄中嵌套使用,linux-2.4.15顯然是老核的目錄名,而linux則為新核的目錄名。
回頁首
2.patch
盡管并沒有指定patch和diff的關系,但通常patch都使用diff的結果來完成打補丁的工作,這和patch本身支持多種 diff輸出文件格式有很大關系。patch通過讀入patch命令文件(可以從標準輸入),對目標文件進行修改。通常先用diff命令比較新老版本,patch命令文件則采用diff的輸出文件,從而保持原版本與新版本一致。
patch的標準格式為
patch [options] [originalfile] [patchfile]如果patchfile為空則從標準輸入讀取patchfile內容;如果originalfile也為空,則從 patchfile(肯定來自標準輸入)中讀取需要打補丁的文件名。因此,如果需要修改的是目錄,一般都必須在patchfile中記錄目錄下的各個文件名。絕大多數情況下,patch都用以下這種簡單的方式使用:
patch -p[num] <patchfilepatch命令可以忽略文件中的冗余信息,從中取出diff的格式以及所需要patch的文件名,文件名按照diff參數中的"源文件"、"目標文件"以及冗余信息中的"Index:"行中所指定的文件的順序來決定。也就是說,對于如下diff結果文件(Linux內核源碼 2.4.16升級包,部分):
diff -Nur linux-2.4.15/Makefile linux/Makefile
--- linux-2.4.15/Makefile Thu Nov 22 17:22:58 2001
+++ linux/Makefile Sat Nov 24 16:21:53 2001
@@ -1,7 +1,7 @@
VERSION = 2
PATCHLEVEL = 4
-SUBLEVEL = 15
-EXTRAVERSION =-greased-turkey
+SUBLEVEL = 16
+EXTRAVERSION =
KERNELRELEASE=$(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION)
……
patch首先嘗試當前目錄(或者-d參數指定的目錄)下的linux-2.4.15/Makefile文件是否存在,如果不存在則試圖對linux/Makefile文件操作,僅當兩者都不存在時(或者設置了POSIXLY_CORRECT環境變量)才會讀取Index:的內容(此文件中沒有標識)。
前面提到的-p參數決定了是否使用讀出的源文件名的前綴目錄信息,不提供-p參數,則忽略所有目錄信息,-p0(或者-p 0)表示使用全部的路徑信息,-p1將忽略第一個"/"以前的目錄,依此類推。如/usr/src/linux-2.4.15/Makefile這樣的文件名,在提供-p3參數時將使用linux-2.4.15/Makefile作為所要patch的文件。
對于剛才舉的Linux內核源碼2.4.16升級包的例子,假定源碼目錄位于/usr/src/linux中,則在當前目錄為/usr /src時使用"patch -p0 <patch-2.4.16"可以工作,在當前目錄為/usr/src/linux時,"patch -p1<patch-2.4.16"也可以正常工作。
patch可以直接操作上下文格式以及混合ed格式的diff輸出文件,而將ed格式文件通過管道提交給ed程序操作(暫時不知RCS格式的文件如何處理)。
回頁首
3.配合使用diff和patch升級源碼
在此僅舉一個簡單的例子來說明如何用diff/patch工具維護源碼升級。
假設program-1.0目錄中為老版,現開發完成的新版位于program-2.0目錄中,將兩個目錄置于同一父目錄下,然后在該父目錄上執行:
diff -Nur program-1.0 program-2.0 >program-2.0.patch將生成一個program-2.0.patch的補丁文件,發布該補丁文件(當然可以先壓縮成bzip2格式)。
假設拿到的是program-2.0.patch.bz2文件,則在program-1.0目錄同級執行:
bzcat program-2.0.patch.bz2 | patch -p0如此即完成了從1.0到2.0的升級。
如果希望恢復到原版本,可以使用-R(--reverse)參數,但僅對上下文格式的diff文件有效。還有一個備份參數也可以使用,但簡單應用中,整個目錄備份可能更方便一些。
參考資料
- Patch手冊頁
- Diff手冊頁
關于作者
楊沙洲,目前在國防科技大學計算機學院攻讀軟件方向博士學位。您可以通過 pubb@163.net與他聯系。
轉載于:https://www.cnblogs.com/napoleon_liu/archive/2011/01/31/1948217.html
總結
以上是生活随笔為你收集整理的[转] 用Diff和Patch工具维护源码的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: UA OPTI512R 傅立叶光学导论1
- 下一篇: 嵌入式第11次实验