前段时间整理了《WinCE下调试串口的动态复用》,基本实现了调试串口与普通功能串口之间的动态切换。其中实现的方法有点欠缺,在重新烧录或者升级系统后,导致系统无法正常启动。这算是个BUG。该功能加上才几天,就陆续有好几个同事碰到。本来想着使用方便的,没想到反而增添了一些麻烦。
这个问题在实现时曾考虑到,发布版本的日志里也加了说明,如果启用了调试串口,那么在烧录或者升级系统前,须禁用调试串口。说实在的,确实有点麻烦,不小心就忘了做这个工作。而问题的根源是实现机制不太合理。调试串口的配置被分散在两处,一处是存储在NAND Flash的特定区域,另一处是注册表中。系统启动时,OAL根据Flash中保存的状态,确定是否启用调试串口,而驱动加载时又根据注册表的状态,确定是否加载调试串口的驱动。两处保存的状态不同步时,问题就出现了。问题分析清楚了,解决方法自然就有了,保证两处的状态一致即可。
调试串口的配置由存储在NAND Flash中的参数决定,系统启动时根据该值,动态修改调试串口对应的注册表配置,确保在启用调试串口时,不再加载它所对应的驱动,也就不会产生冲突,导致系统无法正常启动了。
在驱动注册表中,Flags是用于控制流驱动的加载行为的。其中DEVFLAGS_NOLOAD即表示不要加载该驱动。所以,在合适的地方添加如下代码,即可控制调试串口驱动的加载。
1
void
DisableDebugSerial(BOOL bDisable)
2 {
3 HKEY Key;
4 DWORD Status;
5 DWORD Disposition;
6 DWORD Value;
7 DWORD Flags;
8
9 Status = RegCreateKeyEx(HKEY_LOCAL_MACHINE, L " Drivers\\BuiltIn\\Serial3 " , 0 , NULL, 0 , 0 , NULL, & Key, & Disposition);
10
11 if (Status == ERROR_SUCCESS)
12 {
13 Value = bDisable ? DEVFLAGS_NONE : DEVFLAGS_NOLOAD;
14 RegSetValueEx(Key, DEVLOAD_FLAGS_VALNAME, 0 , DEVLOAD_FLAGS_VALTYPE, (PBYTE) & Value, sizeof (Value));
15 RegCloseKey(Key);
16 }
17 }
2 {
3 HKEY Key;
4 DWORD Status;
5 DWORD Disposition;
6 DWORD Value;
7 DWORD Flags;
8
9 Status = RegCreateKeyEx(HKEY_LOCAL_MACHINE, L " Drivers\\BuiltIn\\Serial3 " , 0 , NULL, 0 , 0 , NULL, & Key, & Disposition);
10
11 if (Status == ERROR_SUCCESS)
12 {
13 Value = bDisable ? DEVFLAGS_NONE : DEVFLAGS_NOLOAD;
14 RegSetValueEx(Key, DEVLOAD_FLAGS_VALNAME, 0 , DEVLOAD_FLAGS_VALTYPE, (PBYTE) & Value, sizeof (Value));
15 RegCloseKey(Key);
16 }
17 }
如果禁用了调试串口,则将Flag是设置为DEVFLAGS_NONE,设备管理器将正常加载驱动。如果没有禁用调试串口,则将Flags设置为DEVFLAGS_NOLOAD,设备管理器就不会加载该驱动。
通过以上方法的改进,调试串口的动态复用就更方便了。