新聞中心

WinCE獲取SD卡序列號

作者: 時間:2016-09-12 來源:網(wǎng)絡(luò) 收藏

//=====================================================================

本文引用地址:http://www.2s4d.com/article/201609/304827.htm

//TITLE:

// WinCE獲取SD卡序列號

//AUTHOR:

// norains

//DATE:

// Thursday 25-February-2011

//Environment:

// Visual Studio 2005

// Windows CE 6.0

// Telechips TCC89x

//=====================================================================

WinCE的設(shè)備,估計會和SD卡打交道的應(yīng)該不在少數(shù)。特別是一些軟件,比如導(dǎo)航地圖之類,加密數(shù)據(jù)用的就是SD卡的序列號。不過,嚴(yán)格來說,在WinCE下面并沒有專門針對于SD卡序列號的獲取函數(shù),而是針對Storage的。只不過SD卡也是Storage的一種,所以自然也能夠被獲取。

SD卡序列號的獲取,是需要通過驅(qū)動的的。這么一說的話,熟悉的朋友可能就明白流程了:首先調(diào)用CreateFile打開驅(qū)動,接著使用DeviceIoControl來獲取序列號,最后則是調(diào)用CloseHandle進(jìn)行關(guān)閉。

一步一步來,先看看CreateFile的調(diào)用,如:

[cpp] view plaincopyHANDLE hDisk = CreateFile(TEXT(DSK2:),

GENERIC_READ,

FILE_SHARE_READ | FILE_SHARE_WRITE,

NULL,

OPEN_EXISTING,

0,

NULL);

這段代碼沒什么問題,可能大家比較關(guān)心的是DSK2:這個參數(shù)的來源?;蚴钦f,我如何確定這個參數(shù)。很多朋友可能會錯認(rèn)為,當(dāng)我們SD卡插入到設(shè)備中,在我的設(shè)備會出現(xiàn)Storage Card分區(qū),那么CreateFile的第一個形參就應(yīng)該是它。但實際上這樣是錯誤的,傳入的形參并不是分區(qū)名,而應(yīng)該是驅(qū)動名。而這個驅(qū)動名的確認(rèn),可以通過控制面板的存儲器確認(rèn),如圖:

圖中的DSK1:是設(shè)備中NAND FLASH的驅(qū)動,而DSK2:則是SD卡的??赡苡信笥褑柫耍胰绾闻袛嗄膫€是SD卡的呢?其實這是一個很簡單的事情:沒插入SD卡的時候查看一次,插入SD卡的時候再查看一次,多出的那個就是SD卡了。~

接著我們就必須調(diào)用DeviceIoControl函數(shù)了,如下代碼所示:

[cpp] view plaincopyPSTORAGE_IDENTIFICATION pStoreInfo = (PSTORAGE_IDENTIFICATION) new BYTE[300];

DeviceIoControl(hDisk,

IOCTL_DISK_GET_STORAGEID,

NULL,

0,

pStoreInfo,

BUFFER_SIZE,

dwBytesRet,

NULL);

IOCTL_DISK_GET_STORAGEID是設(shè)備請求標(biāo)識符,這個應(yīng)該也好理解。如果你的代碼編譯時該標(biāo)識符沒有定義,那么你應(yīng)該是沒有包含Diskio.h文件??赡鼙容^費解的是pStoreInfo,為什么要new個300byte的空間,然后又轉(zhuǎn)換為PSTORAGE_IDENTIFICATION指針呢?這個我們必須要從STORAGE_IDENTIFICATION結(jié)構(gòu)說起。

STORAGE_IDENTIFICATION結(jié)構(gòu)定義如下:

[cpp] view plaincopytypedef struct _STORAGE_IDENTIFICATION {

DWORD dwSize;

DWORD dwFlags;

DWORD dwManufactureIDOffset;

DWORD dwSerialNumOffset;

} STORAGE_IDENTIFICATION, *PSTORAGE_IDENTIFICATION;

dwSize是結(jié)構(gòu)體和標(biāo)識符的大小,dwFlags標(biāo)志著標(biāo)識符是否可用,dwManufactureIDOffset是標(biāo)識符的中的廠家ID偏移位置,最后的dwSerialNumOffset則是標(biāo)識符中的序列號偏移位置。完了?是的,完了,就是這幾樣?xùn)|西。那么,疑問來了,那標(biāo)識符在哪里?別急,我們來看看返回回來數(shù)據(jù)的存儲方式,如圖:

圖中的灰色部分為STORAGE_IDENTIFICATION的成員,其大小為sizeof(STORAGE_IDENTIFICATION),是固定值;而藍(lán)色的部分,則是我們分配的內(nèi)存減去結(jié)構(gòu)體后的大小,容量隨著我們分配的內(nèi)存而改變。在這里稍微返回來看一下代碼,為什么我們代碼中分配了個300Byte的空間呢?其實300這個數(shù)值是隨意的,如果你設(shè)備的標(biāo)識符大于這個數(shù)值,可以進(jìn)行修改。但對于SD卡來說,標(biāo)識符也就是10位,加上STORAGE_IDENTIFICATION的大小,300的空間完全足夠了。

接著從圖中可以知道,在緊隨著結(jié)構(gòu)體之后的是ManufactureID和SerialNumber的數(shù)值。那么這兩個數(shù)值的位置應(yīng)該如何確定呢?那就是通過dwManufactureIDOffset和dwSerialNumOffset的數(shù)值來確定,這也就是為什么圖中dwManufactureIDOffset方塊延伸出來的箭頭會指向ManufactureID前端的原因。

對于ManufactureID和SerialNumber還有一些地方需要注意的。這兩個號碼合起來的長度是不固定的,但我們能通過末尾是否為'/0'來判斷標(biāo)識符是否結(jié)束。另外還有一點,ManufactureID和SerialNumber之間是沒有分隔符的,我們只能通過(dwSerialNumOffset - dwManufactureIDOffset)來確定。但這個還是需要有前提條件的,就是dwSerialNumOffset和dwManufactureIDOffset都不能為0,因為為0時,就意味著該標(biāo)識符無效。簡單點來說,如果dwSerialNumOffset這個偏移量的數(shù)值為0,那么意味著返回的數(shù)值中并沒有ManufactureID。

結(jié)構(gòu)明白之后,我們就很容易獲取這兩個標(biāo)志的起始地址了:

[cpp] view plaincopychar *pManufactureID = reinterpret_cast(pStoreInfo) + pStoreInfo->dwManufactureIDOffset;

char *pSerialNum = reinterpret_cast(pStoreInfo) + pStoreInfo->dwSerialNumOffset;

接下來的事情可能就不用細(xì)說了,無非就是根據(jù)起始地址來復(fù)制字符串。最后,就是調(diào)用CloseHandle來關(guān)閉驅(qū)動句柄了。

流程介紹完畢,但還不是文章的結(jié)尾。本文的最后,讓我們來看看一個完整的獲取標(biāo)識符的函數(shù),代碼如下所示:

[cpp] view plaincopyBOOL GetStorageIdentification(std::wstring strDiskName,std::string strManufactureID,std::string strSerialNum)


上一頁 1 2 下一頁

關(guān)鍵詞:

評論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉