Win32 API函數CreateFile()除了可打開普通文件外,還可以打開設備,比如可用於打開串口,獲得串口句柄。
使用CreateFile()函數打開串口時文件共享模式應設置爲0(表示獨佔),創建參數設置爲OPEN_EXISTING,模板必須設置爲NULL。
如果爲COM1至COM9,可使用“COM1”-“COM9”作爲文件名傳遞給CreateFile()函數,函數可成功返回。但是,如果操作對象爲COM10及以上的端口,以此方式命名文件名調用CreateFile()函數會返回INVALID_HANDLE_VALUE,表示端口無法打開。
產生這種奇怪現象的原因是:微軟預定義的標準設備中含有“COM1”-“COM9”。所以,“COM1”-“COM9”作爲文件名傳遞給函數時操作系統會自動地將之解析爲相應的設備。但對於COM10及以上的串口,“COM10”之類的文件名系統只視之爲一般意義上的文件,而非串行設備。
爲了增加對COM10及以上串行端口的支持,微軟規定,如果要訪問這樣的設備,應使用這樣的文件名(以COM10爲例):\\.COM10
所以,對於COM10及以上的串口,CreateFile()的調用樣式應調整如下:
CreateFile(
"\\\\.\\COM10", // 定義串口名
fdwAccess, // 存取模式(讀寫)
0, // 共享模式:必須設置爲0,表示設備獨佔使用
NULL, // 保密性
OPEN_EXISTING, // 必須設置爲OPEN_EXISTING
0, // 文件屬性,如果是異步模式,可設置爲
NULL // 模版,串口設備必須設置爲NULL
);
char szPort[50];
if (portNo < 10)
{
sprintf_s(szPort, "COM%d", portNo);
}
else
{
sprintf_s(szPort, "\\\\.\\COM%d", portNo);
}
/** 打開指定的串口 */
m_hComm = CreateFileA(szPort, /** 設備名,COM1,COM2等 */
GENERIC_READ | GENERIC_WRITE, /** 訪問模式,可同時讀寫 */
0, /** 共享模式,0表示不共享 */
NULL, /** 安全性設置,一般使用NULL */
OPEN_EXISTING, /** 該參數表示設備必須存在,否則創建失敗 */
0,
0);