www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當前位置:首頁 > 嵌入式 > 嵌入式軟件
[導讀] //=====================================================================//TITLE:// WinCE獲取SD卡序列號//AUTHOR:// norains//DATE:// Thursday 25-February-2011//En

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

//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"分區(qū),那么CreateFile的第一個形參就應該是它。但實際上這樣是錯誤的,傳入的形參并不是分區(qū)名,而應該是驅動名。而這個驅動名的確認,可以通過"控制面板"的"存儲器"確認,如圖:

圖中的"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)[!--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(pStoreInfo) + pStoreInfo->dwManufactureIDOffset;

if(pStoreInfo->dwSerialNumOffset != 0)

{

strManufactureID.assign(pManufactureID,pStoreInfo->dwSerialNumOffset - pStoreInfo->dwManufactureIDOffset);

}

else

{

strManufactureID = pManufactureID;

}

}

if(pStoreInfo->dwSerialNumOffset != 0)

{

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

strSerialNum = pSerialNum;

}

bRes = TRUE;

}

__finally

{

if(pStoreInfo != NULL)

{

delete []pStoreInfo;

}

if(hDisk != INVALID_HANDLE_VALUE)

{

CloseHandle(hDisk);

}

}

return bRes;

}

本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯系該專欄作者,如若文章內容侵犯您的權益,請及時聯系本站刪除。
換一批
延伸閱讀

LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關鍵字: 驅動電源

在工業(yè)自動化蓬勃發(fā)展的當下,工業(yè)電機作為核心動力設備,其驅動電源的性能直接關系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅動電源設計中至關重要的兩個環(huán)節(jié),集成化方案的設計成為提升電機驅動性能的關鍵。

關鍵字: 工業(yè)電機 驅動電源

LED 驅動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設備的使用壽命。然而,在實際應用中,LED 驅動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設計、生...

關鍵字: 驅動電源 照明系統(tǒng) 散熱

根據LED驅動電源的公式,電感內電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關鍵字: LED 設計 驅動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產業(yè)的重要發(fā)展方向。電動汽車的核心技術之一是電機驅動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機驅動系統(tǒng)中的關鍵元件,其性能直接影響到電動汽車的動力性能和...

關鍵字: 電動汽車 新能源 驅動電源

在現代城市建設中,街道及停車場照明作為基礎設施的重要組成部分,其質量和效率直接關系到城市的公共安全、居民生活質量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關鍵字: 發(fā)光二極管 驅動電源 LED

LED通用照明設計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數校正(PFC)、空間受限和可靠性等。

關鍵字: LED 驅動電源 功率因數校正

在LED照明技術日益普及的今天,LED驅動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關鍵字: LED照明技術 電磁干擾 驅動電源

開關電源具有效率高的特性,而且開關電源的變壓器體積比串聯穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現在的LED驅動電源

關鍵字: LED 驅動電源 開關電源

LED驅動電源是把電源供應轉換為特定的電壓電流以驅動LED發(fā)光的電壓轉換器,通常情況下:LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關鍵字: LED 隧道燈 驅動電源
關閉