- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
如果WriteFile函数同步完成并成功,则是否通过lpOverlapped
参数传递了通过事件传递的信号?如果事件同步失败,是否会向事件发出信号?我已经打开了带有FILE_FLAG_OVERLAPPED
标志的文件的句柄。我无法从文档中弄清楚这一点,也无法在代码中轻松地复制这种情况。
最佳答案
首先,这个问题不仅与WriteFile
有关,而且与任何异步I/O函数有关-几乎所有获得 OVERLAPPED
结构指针的函数。因为为所有这些功能分配了IRP
(I/O请求数据包)(在wdm.h中查看它的定义)。 hEvent
中的OVERLAPPED
句柄转换为对象指针,并存储在PKEVENT UserEvent;
的IRP
成员中。该事件是在IRP
例程中完成IopCompleteRequest
时设置(或未设置)的。 IRP
完成函数对于所有I/O api都是通用的,因此,规则和规则(当完成触发时)与所有人都相关。不幸的是,这是非常糟糕的记录。 win32层(比较NT层)在此处添加了额外的,非常薄的问题。
基于wrk src code,我们可以看到异步io的I/O Manager触发完成(事件,apc和iocp(互斥)三种)是!NT_ERROR( irp->IoStatus.Status )
或irp->PendingReturned
。
如果我们使用 native api,则直接返回NTSTATUS
-当(ULONG)status < 0xc0000000
时。但是在不清楚的情况下,这里的0x80000000 <= status < 0xc0000000
或NT_WARNING(status)
范围非常有问题-将会设置完成(甚至设置,apc或iocp队列的数据包)。这是因为在分配IRP
I/O管理器之前,请进行一些基本检查,并可以从此处返回错误。通常,I/O管理器从NT_ERROR(status)
返回错误,该错误意味着不会完成(不会设置事件),但存在且很少异常(exception)。例如对于 ReadDirectoryChangesW
(或ZwNotifyChangeDirectoryFile
),lpBuffer指针必须是DWORD对齐的(与 FILE_NOTIFY_INFORMATION
完全对齐),否则I/O Manager从STATUS_DATATYPE_MISALIGNMENT
范围返回NT_WARNING
(0x80000002)。但在这种情况下将不会完成操作(事件集),因为在分配IRP
之前,函数失败。在另一种情况下,如果我们在缓冲区不足的情况下调用 FSCTL_FILESYSTEM_GET_STATISTICS
,则文件系统驱动程序(不是I/O Manager)将返回STATUS_BUFFER_OVERFLOW
(0x80000005)。但是由于此时IRP
已经分配并且代码不在NT_ERROR
范围内,因此将被设置为事件。
因此,如果来自I/O管理器的错误(在分配IRP
之前)将无法完成。否则,如果函数返回IRP
导致驱动程序错误(传递了!NT_ERROR(status)
)出错,则完成。结果,如果我们得到:
NT_SUCCESS(status)
(STATUS_PENDING
(0x103
)是其中的一部分)-将NT_ERROR(status)
将不会完成NT_WARNING(status)
-不清楚,取决于I/O管理器的此错误NT_WARNING(status)
-大多数win32 api将此解释为错误-返回false并设置最后一个错误(从状态转换)。但是某些API(例如 ReadDirectoryChangesW
将此解释为成功代码)返回true,但未设置上一个错误。结果是,如果我们使用对齐错误的缓冲区(但有效的其他参数)调用 ReadDirectoryChangesW
,则返回-。 true 且未设置任何错误。但是api调用确实失败了。 ZwNotifyChangeDirectoryFile
内部在这里返回STATUS_DATATYPE_MISALIGNMENT
。DeviceIoControl
的FSCTL_FILESYSTEM_GET_STATISTICS
失败(返回false),代码为ERROR_MORE_DATA
(从STATUS_BUFFER_OVERFLOW
转换),则将设置事件(完成)。NT_ERROR
还是NT_WARNING
代码-转换( RtlNtStatusToDosError
)状态到win32错误丢失了此信息NT_WARNING(status)
,则从vista开始的FILE_SKIP_COMPLETION_PORT_ON_SUCCESS
范围问题可以解决,在这种情况下,当且仅当返回STATUS_PENDING
时,I/O管理器将完成条目排队到端口通过本地api调用。对于win32层,这通常意味着api返回false,最后一个错误是ERROR_IO_PENDING
。异常(exception)-WriteFileEx
和ReadFileEx
,在此处返回true。但是无论如何ReadDirectoryChangesW
都无济于事(我假设这是Windows错误)FILE_SKIP_SET_EVENT_ON_HANDLE
部分-这是隐式的说明,如果在异步功能的情况下设置了显式事件(重叠),则请求将返回成功代码,或者返回的错误是ERROR_IO_PENDING
。但是这里的问题是-成功代码是什么?是否由win32 api返回true?并非总是如此,从FSCTL_FILESYSTEM_GET_STATISTICS
可见-ERROR_MORE_DATA
(STATUS_BUFFER_OVERFLOW
)也是成功代码。或 STATUS_NO_MORE_FILES
返回的NtQueryDirectoryFile
以及成功代码-事件(apc或iocp完成)将被设置。但是当NtQueryDirectoryFile
对齐错误时,相同的 STATUS_DATATYPE_MISALIGNMENT
可以返回FileInformation
-这是失败代码,因为在分配IRP
之前从I/O Manager返回NT_WARNING
状态为成功代码(将完成),但是在大多数情况下,win32层将其解释为失败代码(返回false)。ULONG BOOL_TO_ERROR(BOOL fOk)
{
return fOk ? NOERROR : GetLastError();
}
void CheckEventState(HANDLE hEvent, ULONG err, NTSTATUS status = RtlGetLastNtStatus())
{
DbgPrint("error = %u(%x)", err, err ? status : STATUS_SUCCESS);
switch (WaitForSingleObject(hEvent, 0))
{
case WAIT_OBJECT_0:
DbgPrint("Signaled\n");
break;
case WAIT_TIMEOUT:
DbgPrint("NON signaled\n");
break;
default:
DbgPrint("error=%u\n", GetLastError());
}
#if 1
EVENT_BASIC_INFORMATION ebi;
if (0 <= ZwQueryEvent(hEvent, EventBasicInformation, &ebi, sizeof(ebi), 0))
{
DbgPrint("EventState = %x\n", ebi.EventState);
}
#endif
}
void demoIoEvent()
{
WCHAR sz[MAX_PATH];
GetSystemDirectoryW(sz, RTL_NUMBER_OF(sz));
HANDLE hFile = CreateFileW(sz, 0, FILE_SHARE_VALID_FLAGS, 0,
OPEN_EXISTING, FILE_FLAG_OVERLAPPED|FILE_FLAG_BACKUP_SEMANTICS, 0);
if (hFile != INVALID_HANDLE_VALUE)
{
FILESYSTEM_STATISTICS fs;
OVERLAPPED ov = {};
if (ov.hEvent = CreateEvent(0, TRUE, FALSE, 0))
{
FILE_NOTIFY_INFORMATION fni;
IO_STATUS_BLOCK iosb;
// STATUS_DATATYPE_MISALIGNMENT from I/O manager
// event will be not set
NTSTATUS status = ZwNotifyChangeDirectoryFile(hFile, ov.hEvent, 0, 0, &iosb,
(FILE_NOTIFY_INFORMATION*)(1 + (PBYTE)&fni), 1, FILE_NOTIFY_VALID_MASK, FALSE);
CheckEventState(ov.hEvent, ERROR_NOACCESS, status);
// windows bug ! ReadDirectoryChangesW return .. true and no last error
// but really api fail. event will be not set and no notifications
ULONG err = BOOL_TO_ERROR(ReadDirectoryChangesW(hFile,
(FILE_NOTIFY_INFORMATION*)(1 + (PBYTE)&fni), 1, 0, FILE_NOTIFY_VALID_MASK, 0, &ov, 0));
CheckEventState(ov.hEvent, err);
// fail with ERROR_INSUFFICIENT_BUFFER (STATUS_BUFFER_TOO_SMALL)
// NT_ERROR(c0000023) - event will be not set
err = BOOL_TO_ERROR(DeviceIoControl(hFile,
FSCTL_FILESYSTEM_GET_STATISTICS, 0, 0, 0, 0, 0, &ov));
CheckEventState(ov.hEvent, err);
// ERROR_MORE_DATA (STATUS_BUFFER_OVERFLOW)
// !NT_ERROR(80000005) - event will be set
// note - win 32 api return false and error != ERROR_IO_PENDING
err = BOOL_TO_ERROR(DeviceIoControl(hFile,
FSCTL_FILESYSTEM_GET_STATISTICS, 0, 0, &fs, sizeof(fs), 0, &ov));
CheckEventState(ov.hEvent, err);
if (err == ERROR_MORE_DATA)
{
SYSTEM_INFO si;
GetSystemInfo(&si);
ULONG cb = si.dwNumberOfProcessors * fs.SizeOfCompleteStructure;
union {
PVOID pv;
PBYTE pb;
PFILESYSTEM_STATISTICS pfs;
};
pv = alloca(cb);
// must be NOERROR(0) here
// !NT_ERROR(0) - event will be set
err = BOOL_TO_ERROR(DeviceIoControl(hFile, FSCTL_FILESYSTEM_GET_STATISTICS, 0, 0,
pv, cb, 0, &ov));
CheckEventState(ov.hEvent, err);
if (!err && GetOverlappedResult(hFile, &ov, &cb, FALSE))
{
do
{
// use pfs here
} while (pb += fs.SizeOfCompleteStructure, --si.dwNumberOfProcessors);
}
}
CloseHandle(ov.hEvent);
}
CloseHandle(hFile);
}
}
error = 998(80000002)NON signaled
EventState = 0
error = 0(0)NON signaled
EventState = 0
error = 122(c0000023)NON signaled
EventState = 0
error = 234(80000005)Signaled
EventState = 1
error = 0(0)Signaled
EventState = 1
关于c - 如果WriteFile同步完成,是否发出信号通知事件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49959547/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!