本地目錄的只讀控制(禁止寫、刪除、新建))
2005.4.26幾句廢話:
這是我在驅網發的第二篇文章,第一篇是關于利用過濾驅動將U盤設置為只讀的。
這篇是利用文件系統過濾驅動將一個目錄(本地硬盤)設置為只讀,包括禁止修改
文件,禁止刪除文件,禁止新建文件等。 我最終的目的是想實現 利用文件過濾驅動
將任意盤(包括移動,A等)設置為只讀。因此我的路還很長,還需要各位的幫助。
正文:
0 準備工作:
由于是對文件、目錄的攔截。首先要知道如何得到路徑、文件名、盤符等等。否則
你沒有辦法去做判斷。我使用IFS 中的sfilter修改。因此沒有filemon那樣有直接的
函數取得。
在驅動中取得盤符和路徑是分開的。取得盤符使用RtlVolumeDeviceToDosName (file->DeviceObject,&dosname); 取得路徑的方法是:
irpSp = IoGetCurrentIrpStackLocation( Irp );
file = irpSp->FileObject;
RtlCopyUnicodeString(&name,&file->FileName);
DbgPrint(\"%ws\",(&name)->Buffer);// 這個地方直接用name.Buffer什么也打印不出來,奇怪
但是這里要注意的是并不是每一個IRP中的irpSp->FileObject都有,很多時候是NULL。這個值有
文件系統來填寫。所以在取路徑前一定要做相應的判斷,否則會藍屏。
1 禁止訪問目錄:
實現禁止訪問目錄是比較簡單的,有很多的方法。我是在IRP_MJ_DIRECTORY_CONTROL
中判斷是不是要禁止的目錄,然后攔截。攔截的操作我就不多說了,基本一樣:
Irp->IoStatus.Status = STATUS_ACCESS_DENIED;
Irp->IoStatus.Information = 0;
status = Irp->IoStatus.Status;
IoCompleteRequest(Irp, IO_NO_INCREMENT);
return status;
2 設置目錄為只讀
由于禁止目錄訪問的方法很容易,所以很容易讓人覺得禁止寫是不是就是在IRP_MJ_WRITE中
判斷目錄路徑然后直接攔截了。我測試后發現不能這樣實現,論壇上的人告我將目錄設置為只讀
實際上是把目錄下所有文件設置為只讀,即有一目錄C:\\Jason你想設置為只讀,實際的操作是:
對于該目錄下任一文件xx.xx,當該文件想進行寫的時候,驅動會得到路徑\\Jason\\xx.xx,此時
判斷文件的父目錄是不是\\Jason并攔截之,即可實現\\Jaosn\\xx.xx的只讀控制。
到目前我實現就是使用這樣的方法,是不是可以直接對目錄進行攔截,我不知道。
有了這個概念,還需要一些技巧去實現。主要是在實現父目錄匹配的字符串比較上。這個調
試一下就可以解決了。
3 禁止刪除
禁止刪除也還算簡單,但是如果你象我一樣是一個人瞎搞,也許不會知道這個方法。在此感謝
joshua_yu 告訴我這個方法。
在IRP_MJ_SET_INFORMATION中,
PIO_STACK_LOCATION irpSp = IoGetCurrentIrpStackLocation(Irp);
irpSp->Parameters.SetFile.FileInformationClass == FileDispositionInformation
核心的就是判斷irpSp->Parameters.SetFile.FileInformationClass 是不是等于
FileDispositionInformation。
但是我一直很奇怪就是這些技術我怎么沒有看到,而別人看的到?看的什么資料??
4 禁止創建文件
這里主要就是要區別一下新建文件和打開文件。對于這個過程,joshua_yu有他自己的理解:
“我是這樣認為的:
當我們調用CreateFile并且希望創建一個文件的時候,系統會首先發送一個標志為FILE_OPEN的請求,并且判斷底層文件系統的返回值,如果返回成功,則表明文件存在并且已經成功打開,否則如果返回結果是NO SUCH FILE,則緊接著創建一個FILE_OPEN_IF請求,得以將文件創建,所以如果我們在Create的Options當中發現了FILE_CREATE,FILE_OPEN_IF和FILE_OVERWRITE_IF三個標志,則表明一定是在創建而不是打開。”
原理就是這樣,代碼實現我是這樣做的:
CreateDisposition = (irpSp->Parameters.Create.Options>> 24) & 0x000000ff;
if(CreateDisposition==FILE_CREATE||CreateDisposition==FILE_OPEN_IF
||CreateDisposition==FILE_OVERWRITE_IF)
{
DbgPrint(\"It is a CREATE FILE operation\\n\");
Irp->IoStatus.Status = STATUS_ACCESS_DENIED;
Irp->IoStatus.Information = 0;
status = Irp->IoStatus.Status;
IoCompleteRequest(Irp, IO_NO_INCREMENT);
return status;
}
5 一些其他問題:
由于這個是一點一點改好的,所以我能想到的主要技術就是這么多了。如果有問題,請和我交流
zhjie374@hotmail.com
順便問個問題:移動設備動態插拔我怎么才能攔截到它的IRP?
6 最后還是要說一句: 本人畢業于排名300開外的大學,水平有限,有錯誤之處請各位批評指正。
Jason Zhang SCT_SH 2005.4.26
posted on 2007-05-07 23:54 葉子 閱讀(4394) 評論(2) 編輯 收藏 引用 所屬分類: 驅動開發