WinCE獲取SD卡序列號(hào)
//=====================================================================
//TITLE:
// WinCE獲取SD卡序列號(hào)
//AUTHOR:
// norains
//DATE:
// Thursday 25-February-2011
//Environment:
// Visual Studio 2005
// Windows CE 6.0
// Telechips TCC89x
//=====================================================================
WinCE的設(shè)備,估計(jì)會(huì)和SD卡打交道的應(yīng)該不在少數(shù)。特別是一些軟件,比如導(dǎo)航地圖之類,加密數(shù)據(jù)用的就是SD卡的序列號(hào)。不過(guò),嚴(yán)格來(lái)說(shuō),在WinCE下面并沒(méi)有專門(mén)針對(duì)于SD卡序列號(hào)的獲取函數(shù),而是針對(duì)Storage的。只不過(guò)SD卡也是Storage的一種,所以自然也能夠被獲取。
SD卡序列號(hào)的獲取,是需要通過(guò)驅(qū)動(dòng)的的。這么一說(shuō)的話,熟悉的朋友可能就明白流程了:首先調(diào)用CreateFile打開(kāi)驅(qū)動(dòng),接著使用DeviceIoControl來(lái)獲取序列號(hào),最后則是調(diào)用CloseHandle進(jìn)行關(guān)閉。
一步一步來(lái),先看看CreateFile的調(diào)用,如:
[cpp] view plaincopyHANDLE hDisk = CreateFile(TEXT("DSK2:"),
GENERIC_READ,
FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL,
OPEN_EXISTING,
0,
NULL);
這段代碼沒(méi)什么問(wèn)題,可能大家比較關(guān)心的是"DSK2:"這個(gè)參數(shù)的來(lái)源?;蚴钦f(shuō),我如何確定這個(gè)參數(shù)。很多朋友可能會(huì)錯(cuò)認(rèn)為,當(dāng)我們SD卡插入到設(shè)備中,在"我的設(shè)備"會(huì)出現(xiàn)"Storage Card"分區(qū),那么CreateFile的第一個(gè)形參就應(yīng)該是它。但實(shí)際上這樣是錯(cuò)誤的,傳入的形參并不是分區(qū)名,而應(yīng)該是驅(qū)動(dòng)名。而這個(gè)驅(qū)動(dòng)名的確認(rèn),可以通過(guò)"控制面板"的"存儲(chǔ)器"確認(rèn),如圖:
圖中的"DSK1:"是設(shè)備中NAND FLASH的驅(qū)動(dòng),而"DSK2:"則是SD卡的??赡苡信笥褑?wèn)了,我如何判斷哪個(gè)是SD卡的呢?其實(shí)這是一個(gè)很簡(jiǎn)單的事情:沒(méi)插入SD卡的時(shí)候查看一次,插入SD卡的時(shí)候再查看一次,多出的那個(gè)就是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è)備請(qǐng)求標(biāo)識(shí)符,這個(gè)應(yīng)該也好理解。如果你的代碼編譯時(shí)該標(biāo)識(shí)符沒(méi)有定義,那么你應(yīng)該是沒(méi)有包含Diskio.h文件??赡鼙容^費(fèi)解的是pStoreInfo,為什么要new個(gè)300byte的空間,然后又轉(zhuǎn)換為PSTORAGE_IDENTIFICATION指針呢?這個(gè)我們必須要從STORAGE_IDENTIFICATION結(jié)構(gòu)說(shuō)起。
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)識(shí)符的大小,dwFlags標(biāo)志著標(biāo)識(shí)符是否可用,dwManufactureIDOffset是標(biāo)識(shí)符的中的廠家ID偏移位置,最后的dwSerialNumOffset則是標(biāo)識(shí)符中的序列號(hào)偏移位置。完了?是的,完了,就是這幾樣?xùn)|西。那么,疑問(wèn)來(lái)了,那標(biāo)識(shí)符在哪里?別急,我們來(lái)看看返回回來(lái)數(shù)據(jù)的存儲(chǔ)方式,如圖:
圖中的灰色部分為STORAGE_IDENTIFICATION的成員,其大小為sizeof(STORAGE_IDENTIFICATION),是固定值;而藍(lán)色的部分,則是我們分配的內(nèi)存減去結(jié)構(gòu)體后的大小,容量隨著我們分配的內(nèi)存而改變。在這里稍微返回來(lái)看一下代碼,為什么我們代碼中分配了個(gè)300Byte的空間呢?其實(shí)300這個(gè)數(shù)值是隨意的,如果你設(shè)備的標(biāo)識(shí)符大于這個(gè)數(shù)值,可以進(jìn)行修改。但對(duì)于SD卡來(lái)說(shuō),標(biāo)識(shí)符也就是10位,加上STORAGE_IDENTIFICATION的大小,300的空間完全足夠了。
接著從圖中可以知道,在緊隨著結(jié)構(gòu)體之后的是ManufactureID和SerialNumber的數(shù)值。那么這兩個(gè)數(shù)值的位置應(yīng)該如何確定呢?那就是通過(guò)dwManufactureIDOffset和dwSerialNumOffset的數(shù)值來(lái)確定,這也就是為什么圖中dwManufactureIDOffset方塊延伸出來(lái)的箭頭會(huì)指向ManufactureID前端的原因。
對(duì)于ManufactureID和SerialNumber還有一些地方需要注意的。這兩個(gè)號(hào)碼合起來(lái)的長(zhǎng)度是不固定的,但我們能通過(guò)末尾是否為‘/0‘來(lái)判斷標(biāo)識(shí)符是否結(jié)束。另外還有一點(diǎn),ManufactureID和SerialNumber之間是沒(méi)有分隔符的,我們只能通過(guò)(dwSerialNumOffset - dwManufactureIDOffset)來(lái)確定。但這個(gè)還是需要有前提條件的,就是dwSerialNumOffset和dwManufactureIDOffset都不能為0,因?yàn)闉?時(shí),就意味著該標(biāo)識(shí)符無(wú)效。簡(jiǎn)單點(diǎn)來(lái)說(shuō),如果dwSerialNumOffset這個(gè)偏移量的數(shù)值為0,那么意味著返回的數(shù)值中并沒(méi)有ManufactureID。
結(jié)構(gòu)明白之后,我們就很容易獲取這兩個(gè)標(biāo)志的起始地址了:
[cpp] view plaincopychar *pManufactureID = reinterpret_cast
char *pSerialNum = reinterpret_cast
接下來(lái)的事情可能就不用細(xì)說(shuō)了,無(wú)非就是根據(jù)起始地址來(lái)復(fù)制字符串。最后,就是調(diào)用CloseHandle來(lái)關(guān)閉驅(qū)動(dòng)句柄了。
流程介紹完畢,但還不是文章的結(jié)尾。本文的最后,讓我們來(lái)看看一個(gè)完整的獲取標(biāo)識(shí)符的函數(shù),代碼如下所示:
[cpp] view plaincopyBOOL GetStorageIdentification(std::wstring &strDiskName,std::string &strManufactureID,std::string &strSerialNum)[!--empirenews.page--]
{
BOOL bRes = FALSE;
PSTORAGE_IDENTIFICATION pStoreInfo = NULL;
HANDLE hDisk = INVALID_HANDLE_VALUE;
__try
{
//The buffer for storing data
const DWORD BUFFER_SIZE = 300;
//Allocate the size for the struct
pStoreInfo = (PSTORAGE_IDENTIFICATION) new BYTE[BUFFER_SIZE];
if(pStoreInfo == NULL)
{
__leave;
}
//Open the driver
hDisk = CreateFile(strDiskName.c_str(), GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
if(hDisk == INVALID_HANDLE_VALUE)
{
__leave;
}
//Get the ID from the driver
DWORD dwBytesRet = 0;
if (DeviceIoControl(hDisk, IOCTL_DISK_GET_STORAGEID, NULL, 0, pStoreInfo, BUFFER_SIZE, &dwBytesRet, NULL) == FALSE)
{
__leave;
}
//Get the manufacture ID
if (pStoreInfo->dwManufactureIDOffset != 0)
{
char *pManufactureID = reinterpret_cast
if(pStoreInfo->dwSerialNumOffset != 0)
{
strManufactureID.assign(pManufactureID,pStoreInfo->dwSerialNumOffset - pStoreInfo->dwManufactureIDOffset);
}
else
{
strManufactureID = pManufactureID;
}
}
if(pStoreInfo->dwSerialNumOffset != 0)
{
char *pSerialNum = reinterpret_cast
strSerialNum = pSerialNum;
}
bRes = TRUE;
}
__finally
{
if(pStoreInfo != NULL)
{
delete []pStoreInfo;
}
if(hDisk != INVALID_HANDLE_VALUE)
{
CloseHandle(hDisk);
}
}
return bRes;
}