windows 如何在驱动程序中显示弹出消息框(内核模式)?

bn31dyow  于 2023-11-21  发布在  Windows
关注(0)|答案(3)|浏览(288)

我正在写一个驱动程序,它需要立即弹出一个对话框来通知用户一个事件。
(Kind类似于Windows的 “Corrupt file” 通知,但这不是与文件系统相关的驱动程序。)
我知道ExRaiseHardErrorIoRaiseInformationalHardError应该能够做到这一点,但它们似乎不起作用--它们“成功地”返回,而实际上没有做任何事情。
我该怎么做(不创建用户模式程序)?
下面是一个
用户模式**版本的代码(它 * 工作正常 *)。
在内核模式版本中,我调用ExRaiseHardError而不是NtRaiseHardError,但方式完全相同。

  1. #include <windows.h>
  2. #pragma comment(lib, "ntdll.lib") // Needs ntdll.lib from Windows Driver Kit
  3. typedef enum HardErrorResponseType {
  4. ResponseTypeAbortRetryIgnore,
  5. ResponseTypeOK,
  6. ResponseTypeOKCancel,
  7. ResponseTypeRetryCancel,
  8. ResponseTypeYesNo,
  9. ResponseTypeYesNoCancel,
  10. ResponseTypeShutdownSystem,
  11. ResponseTypeTrayNotify,
  12. ResponseTypeCancelTryAgainContinue
  13. } HardErrorResponseType;
  14. typedef enum HardErrorResponse {
  15. ResponseReturnToCaller,
  16. ResponseNotHandled,
  17. ResponseAbort, ResponseCancel,
  18. ResponseIgnore,
  19. ResponseNo,
  20. ResponseOk,
  21. ResponseRetry,
  22. ResponseYes
  23. } HardErrorResponse;
  24. typedef enum HardErrorResponseButton {
  25. ResponseButtonOK,
  26. ResponseButtonOKCancel,
  27. ResponseButtonAbortRetryIgnore,
  28. ResponseButtonYesNoCancel,
  29. ResponseButtonYesNo,
  30. ResponseButtonRetryCancel,
  31. ResponseButtonCancelTryAgainContinue
  32. } HardErrorResponseButton;
  33. typedef enum HardErrorResponseDefaultButton {
  34. DefaultButton1 = 0,
  35. DefaultButton2 = 0x100,
  36. DefaultButton3 = 0x200
  37. } HardErrorResponseDefaultButton;
  38. typedef enum HardErrorResponseIcon {
  39. IconAsterisk = 0x40,
  40. IconError = 0x10,
  41. IconExclamation = 0x30,
  42. IconHand = 0x10,
  43. IconInformation = 0x40,
  44. IconNone = 0,
  45. IconQuestion = 0x20,
  46. IconStop = 0x10,
  47. IconWarning = 0x30,
  48. IconUserIcon = 0x80
  49. } HardErrorResponseIcon;
  50. typedef enum HardErrorResponseOptions {
  51. ResponseOptionNone = 0,
  52. ResponseOptionDefaultDesktopOnly = 0x20000,
  53. ResponseOptionHelp = 0x4000,
  54. ResponseOptionRightAlign = 0x80000,
  55. ResponseOptionRightToLeftReading = 0x100000,
  56. ResponseOptionTopMost = 0x40000,
  57. ResponseOptionServiceNotification = 0x00200000,
  58. ResponseOptionServiceNotificationNT3X = 0x00040000,
  59. ResponseOptionSetForeground = 0x10000,
  60. ResponseOptionSystemModal = 0x1000,
  61. ResponseOptionTaskModal = 0x2000,
  62. ResponseOptionNoFocus = 0x00008000
  63. } HardErrorResponseOptions;
  64. typedef LONG NTSTATUS;
  65. typedef struct _UNICODE_STRING {
  66. USHORT Length;
  67. USHORT MaximumLength;
  68. PWSTR Buffer;
  69. } UNICODE_STRING, *PUNICODE_STRING;
  70. EXTERN_C DECLSPEC_IMPORT NTSTATUS NTAPI NtRaiseHardError(
  71. IN NTSTATUS ErrorStatus, IN ULONG NumberOfParameters,
  72. IN ULONG UnicodeStringParameterMask, IN PULONG_PTR Parameters,
  73. IN ULONG ValidResponseOptions,
  74. OUT HardErrorResponse *Response);
  75. EXTERN_C DECLSPEC_IMPORT VOID NTAPI RtlInitUnicodeString(
  76. IN OUT PUNICODE_STRING DestinationString, IN PCWSTR SourceString);
  77. #define STATUS_ACCESS_DENIED ((NTSTATUS)0xC0000022L)
  78. #define STATUS_SERVICE_NOTIFICATION ((NTSTATUS)0x50000018L)
  79. int main(void)
  80. {
  81. HardErrorResponse r;
  82. // To display a standard NTSTATUS value:
  83. NtRaiseHardError(STATUS_ACCESS_DENIED, 0, 0, NULL, ResponseTypeOK, &r);
  84. // To display a custom string:
  85. UNICODE_STRING wTitle, wText;
  86. RtlInitUnicodeString(&wTitle, L"Title");
  87. RtlInitUnicodeString(&wText, L"Text");
  88. ULONG_PTR params[4] = {
  89. (ULONG_PTR)&wText,
  90. (ULONG_PTR)&wTitle,
  91. (
  92. (ULONG)ResponseButtonOK |
  93. (ULONG)IconInformation |
  94. (ULONG)ResponseOptionNone |
  95. (ULONG)DefaultButton1
  96. ),
  97. INFINITE
  98. };
  99. NtRaiseHardError(STATUS_SERVICE_NOTIFICATION, 4, 0x3, params, 0, &r);
  100. return 0;
  101. }

字符串

b4lqfgs4

b4lqfgs41#

内核驱动不能显示MessageBox。如果你想这样做,那么你必须通过一个用户态应用程序提供一个与内核驱动的通信功能,然后从你的用户态应用程序显示MessageBox。
所有关于[Zw/Nt]RaiseHardError的讨论都是无关紧要的。如果你拆解MessageBox,你会注意到这个API最终会被调用。

ws51t4hk

ws51t4hk2#

我不认为没有一个应用程序(或服务)在用户模式下运行,你可以加载一个驱动程序在内存中(即使它将是一个过滤器驱动程序)。

hzbexzde

hzbexzde3#

这完全可以从你的内核代码中实现,但只能使用Map的驱动程序。
https://vollragm.github.io/posts/kernel-message-box/

  1. extern "C" NTSTATUS NTAPI ExRaiseHardError(
  2. NTSTATUS ErrorStatus, ULONG NumberOfParameters,
  3. ULONG UnicodeStringParameterMask, PULONG_PTR Parameters,
  4. ULONG ValidResponseOptions, PULONG Response);
  5. ULONG KeMessageBox(PCWSTR title, PCWSTR text, ULONG_PTR type)
  6. {
  7. UNICODE_STRING uTitle = { 0 };
  8. UNICODE_STRING uText = { 0 };
  9. RtlInitUnicodeString(&uTitle, title);
  10. RtlInitUnicodeString(&uText, text);
  11. ULONG_PTR args[] = { (ULONG_PTR)&uText, (ULONG_PTR)&uTitle, type };
  12. ULONG response = 0;
  13. ExRaiseHardError(STATUS_SERVICE_NOTIFICATION, 3, 3, args, 2, &response);
  14. return response;
  15. }

字符串

展开查看全部

相关问题