新聞中心

        EEPW首頁 > 嵌入式系統 > 設計應用 > WinCE獲取SD卡序列號

        WinCE獲取SD卡序列號

        作者: 時間:2016-09-12 來源:網絡 收藏

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

        本文引用地址:http://www.104case.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的設備,估計會和SD卡打交道的應該不在少數。特別是一些軟件,比如導航地圖之類,加密數據用的就是SD卡的序列號。不過,嚴格來說,在WinCE下面并沒有專門針對于SD卡序列號的獲取函數,而是針對Storage的。只不過SD卡也是Storage的一種,所以自然也能夠被獲取。

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

        一步一步來,先看看CreateFile的調用,如:

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

        GENERIC_READ,

        FILE_SHARE_READ | FILE_SHARE_WRITE,

        NULL,

        OPEN_EXISTING,

        0,

        NULL);

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

        圖中的DSK1:是設備中NAND FLASH的驅動,而DSK2:則是SD卡的??赡苡信笥褑柫?,我如何判斷哪個是SD卡的呢?其實這是一個很簡單的事情:沒插入SD卡的時候查看一次,插入SD卡的時候再查看一次,多出的那個就是SD卡了。~

        接著我們就必須調用DeviceIoControl函數了,如下代碼所示:

        [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是設備請求標識符,這個應該也好理解。如果你的代碼編譯時該標識符沒有定義,那么你應該是沒有包含Diskio.h文件。可能比較費解的是pStoreInfo,為什么要new個300byte的空間,然后又轉換為PSTORAGE_IDENTIFICATION指針呢?這個我們必須要從STORAGE_IDENTIFICATION結構說起。

        STORAGE_IDENTIFICATION結構定義如下:

        [cpp] view plaincopytypedef struct _STORAGE_IDENTIFICATION {

        DWORD dwSize;

        DWORD dwFlags;

        DWORD dwManufactureIDOffset;

        DWORD dwSerialNumOffset;

        } STORAGE_IDENTIFICATION, *PSTORAGE_IDENTIFICATION;

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

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

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

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

        結構明白之后,我們就很容易獲取這兩個標志的起始地址了:

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

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

        接下來的事情可能就不用細說了,無非就是根據起始地址來復制字符串。最后,就是調用CloseHandle來關閉驅動句柄了。

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

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


        上一頁 1 2 下一頁

        關鍵詞:

        評論


        相關推薦

        技術專區

        關閉
        主站蜘蛛池模板: 滁州市| 绍兴县| 南澳县| 白山市| 竹溪县| 固阳县| 台前县| 景宁| 建湖县| 巴塘县| 正宁县| 新宾| 沧州市| 漾濞| 武义县| 文安县| 台中县| 宜川县| 湖南省| 钦州市| 柳林县| 盐亭县| 仪征市| 息烽县| 松原市| 宾阳县| 河曲县| 永州市| 本溪| 济阳县| 崇信县| 江源县| 揭东县| 福泉市| 津南区| 娄烦县| 黄骅市| 门源| 黄平县| 五指山市| 甘孜县|