天天看點

Linux下patch打更新檔指令

轉自http://blog.csdn.net/maotianwang/article/details/11107083

基本指令文法:

patch [-R] {-p(n)} [--dry-run] < patch_file_name

p:為path的縮寫。

n:指将patch檔案中的path第n條’/’及其左邊部分取消。

-R:解除安裝patch包。

--dry-run:嘗試patch軟體,并不真正修改軟體。

使用執行個體:

/home/sense/patch-2.6.19.1

=============================================================

diff --git a/arch/i386/kernel/acpi/boot.c b/arch/i386/kernel/acpi/boot.c

index d12fb97..c8f96cf 100644

--- a/arch/i386/kernel/acpi/boot.c

+++ b/arch/i386/kernel/acpi/boot.c

@@ -333,7 +333,7 @@ acpi_parse_ioapic(acpi_table_entry_heade

/*

   * Parse Interrupt Source Override for the ACPI SCI

   */

-static void acpi_sci_ioapic_setup(u32 bus_irq, u32 gsi, u16 polarity, u16 trigger)

+static void acpi_sci_ioapic_setup(u32 gsi, u16 polarity, u16 trigger)

此patch檔案的path=a/arch/i386/kernel/acpi

[sense@ssn src]$ pwd

/usr/src

[sense@ssn src]$ tree

.

`-- a

`-- arch

       `-- i386

         `-- kernel

            `-- acpi

和a/同目錄時:

[sense@ssn src]$ patch -p0 < /home/sense/patch-2.6.19.1   安裝更新檔

[sense@ssn src]$ patch -p0 -R < /home/sense/patch-2.6.19.1   解除安裝更新檔

和a/arch/同目錄時:

/usr/src/a/

[sense@ssn src]$ patch -p1 < /home/sense/patch-2.6.19.1   安裝更新檔

[sense@ssn src]$ patch -p1 -R < /home/sense/patch-2.6.19.1   解除安裝更新檔

和a/arch/i386/同目錄時:

/usr/src/a/arch/

[sense@ssn src]$ patch -p2 < /home/sense/patch-2.6.19.1   安裝更新檔

[sense@ssn src]$ patch -p2 -R < /home/sense/patch-2.6.19.1   解除安裝更新檔

作為程式員,了解diff&patch命 令是非常必要的。比如說我們發現某個項目有bug代碼,而自己又沒有svn的送出權限,那麼此時最合适的解決方法就是用diff指令做一個更新檔發給項目成 員。項目成員通過patch指令可以立刻知道你的意圖。有人會說直接傳一個新檔案不是更簡單?不要忘了,一個patch檔案尺寸更小傳輸更快,而且可以明 顯的看到都做了哪些修改。

保證目前目錄是demo名錄:

# mkdir demo

# cd demo

先模拟一個項目目錄old:

# mkdir -p old/a/b

# vi old/a/b/foo.txt

old_line_1

old_line_2

假設我們發現項目old有bug代碼,下面我們先拷貝一個新目錄new,并在此修改bug代碼:

# cp -r old new

# vi new/a/b/foo.txt

new_line_1

new_line_2

保證old和new兩個目錄都在目前目錄下,下面就可以使用diff指令了,不要使用絕對路徑,而應該使用相對路徑,至于原因,看到文章結尾你就清楚了:

# LC_ALL=C TZ=UTC0 diff -Naur old new > foo.patch

如果不在意字元集,時差等問題,也可以省略LC_ALL=C TZ=UTC0環境變量:

# diff -Naur old new > foo.patch

其中-Naur參數屬于固定打法,不管是對一個檔案,還是對一個目錄,在使用這個參數基本就可以了。

大概浏覽一下更新檔檔案:

# cat foo.patch

diff -Naur old/a/b/foo.txt new/a/b/foo.txt

--- old/a/b/foo.txt 2009-12-07 20:40:07.000000000 +0800

+++ new/a/b/foo.txt 2009-12-07 20:41:51.000000000 +0800

@@ -1,2 +1,2 @@

-old_line_1

-old_line_2

+new_line_1

+new_line_2

加減号後面的内容是有用的内容,其他的内容是友善你查閱的相關資訊内容,更新檔制作完成。

此時的檔案目錄結構大概如下所示:

#tree

demo

|-- old

| `-- a

| `-- b

| `-- foo.txt

|-- new

`-- foo.patch

下面看看如何使用patch來應用更新檔,要注意的是目前目錄是demo,試試下面指令:

# patch -p0 < foo.patch

patching file old/a/b/foo.txt

這裡唯一需要說明的是p0的含義,因為在foo.patch更新檔檔案裡的路徑資訊是這樣的:

--- old/a/b/foo.txt

p表示跳過幾級目錄,因為是在demo目錄下使用的patch指令,old目錄就在demo目錄下,是以不必跳過任何目錄,而應該使用old/a/b/foo.txt完整路徑,是以此時使用的是p0。

檢視一下目标檔案,你會發現内容已經修改成新的了:

# cat old/a/b/foo.txt

此時如果你再次使用patch指令,系統會問你是否想還原:

Reversed (or previously applied) patch detected! Assume -R? [n] y

檢視一下目标檔案,你會發現内容已經還原成舊的了:

如果你想嚴格指定是應用更新檔可以使用下面指令(就是增加N參數):

# patch -Np0 < foo.patch

如果你想嚴格指定是還原更新檔可以使用下面指令(就是增加R參數):

# patch -Rp0 < foo.patch

注釋:在本例中,每次應用更新檔後,自己還原更新檔,以備後用繼續試驗,我就不多說了。

看到這裡如果你對patch的p參數還不太清楚的話,接着往下看,我們改變一下目前路徑:

# cd old

此時就應該是p1,而不是p0了,引用foo.patch檔案的路徑也要相對變一下,因為目前目錄已經是old了:

# patch -p1 < ../foo.patch

patching file a/b/foo.txt

因為此時我們是在old下使用patch指令,和a子目錄平級,而更新檔檔案foo.patch裡的路徑聲明是:

也就是說第一個斜線左邊的old/部分已經沒用了,這就是p1的含義!

繼續往深度變換路徑,依次測試使用p2,p3參數:

# cd a

# patch -p2 < ../../foo.patch

patching file b/foo.txt

# cd b

# patch -p3 < ../../../foo.patch

patching file foo.txt 

在本例中,p3已經是最深目錄了,此時可以省略p參數:

# patch < ../../../foo.patch

patching file foo.txt

也就是說,不使用p參數的時候,patch指令會忽略任何目錄,直接使用檔案。

下面接着文章前面說的為什麼使用diff指令時最好不要使用絕對路徑,而應該使用相對路徑?

答:如果你在使用diff的時候使用的是絕對路徑,那麼更新檔檔案裡的檔案路徑資訊會類似下面的樣子:

--- /a/b/c/d/e/f/g/bar.txt

如此一來,當别人想應用你的更新檔時,因為目錄結構肯定有差異,是以就不得不費力判斷到底使用p幾。這樣一來就很容易出錯,相反,如果使用相對路徑的話,大多數時候,p0或者p1就足夠了,不易出錯。

跟着本文的步驟操作一下,肯定能掌握diff&patch用法,基本上使用diff時就是"diff -Naur FROM TO"(FROM, TO為變量)這樣的固定打法,然後在使用patch的時候,先看看更新檔檔案的大緻内容,結合目前目錄以确定需要跳過的目錄數,然後套用"patch -pN < patch.file"(N為變量)即可。

------------------- 

總結一下: 

單個檔案 

diff –uN from-file to-file >to-file.patch 

patch –p0 < to-file.patch 

patch –RE –p0 < to-file.patch 

  多個檔案 

diff –uNr from-docu to-docu >to-docu.patch 

patch –p1 < to-docu.patch 

patch –R –p1 

  -------------------

應用 

為核心打更新檔。前面在建立交叉編譯工具鍊時,其中有一步就是為核心打更新檔。當時還不是特别了解,現在很清晰了。參考前面的文章《基于ARM+Linux嵌入式開發的開發工具鍊的建立》。

1、首先是解壓,因為釋出的更新檔檔案都是使用gzip壓縮的。 

$gunzip ../setup-dir/ patch-2.4.21-rmk1.gz 

2、然後進入你的核心源代碼目錄 

$cd linux-2.4.21 

3、打更新檔 

$patch –p1 < ../../setup-dir/patch-2.4.21-rmk1 

打完更新檔後,需要檢查一下有沒有拒絕執行的檔案,即檢查.rej檔案的存在。使用指令: 

$find . -name *.rej 

如果發現,會将其輸出到标準輸出終端,預設螢幕。當然,你也可以采用重定向,輸出到指定檔案,比如reject。 

$fine . -name *.rej >reject 

然後可以檢視reject的内容了。

本文轉自 wdy198622 51CTO部落格,原文連結:http://blog.51cto.com/weimouren/1754320

繼續閱讀