子进程从其父进程继承了一个句柄但未能写入其中

鳗鱼

我想在同一个文件中写入 2 个进程。我在第一个进程(父进程)中创建了一个文件并调用了第二个进程(子进程)。我使文件可继承,第二个进程将句柄作为命令行参数继承。

虽然句柄不等于 INVALID_HANDLE_VALUE,但第二个进程不能对文件做任何事情。WriteFile() 以错误 6 结束(句柄无效)。LockFile(),Unlockfile() 也是如此。

代码有什么问题,我该如何解决?我想让 2 个进程使用同一个文件。

流程一:

#include <Windows.h>
#include <process.h>

int _tmain(int argc, _TCHAR* argv[])
{
    _SECURITY_ATTRIBUTES sec_attr;
    sec_attr.nLength = sizeof(_SECURITY_ATTRIBUTES);
    sec_attr.bInheritHandle = true;
    sec_attr.lpSecurityDescriptor = 0;
    //argv[1] - the name of the file I want to create
    HANDLE h = CreateFile(argv[1],GENERIC_READ | GENERIC_WRITE,FILE_SHARE_WRITE | 
             FILE_SHARE_READ,&sec_attr,CREATE_ALWAYS,FILE_ATTRIBUTE_NORMAL,0);

    STARTUPINFO sinfo;      
    ZeroMemory(&sinfo,sizeof(sinfo));
    sinfo.cb = sizeof(sinfo);

    PROCESS_INFORMATION pinfo;

    char cmd[300] = "PATH to process 2";
    CreateProcess(0,cmd,0,0,true,NORMAL_PRIORITY_CLASS,0,0,&sinfo,&pinfo);
    char c = '1';
    DWORD w;
    WriteFile(h,&c,1,&w,0); //works fine
    ...
}

过程2:

#include <Windows.h>
#include <process.h>

int _tmain(int argc, _TCHAR* argv[]) 
{
    HANDLE h = argv[1];
    if(h == INVALID_HANDLE_VALUE) { 
        //it's okay, h is NOT equal to INVALID_HANDLE_VALUE
    }
    char c = '2';
    DWORD w;
    WriteFile(h,&c,1,&w,0);  //error 6
    ...
}

PS 第一个进程成功写入文件。到那时,第二个过程已经失败。

雷米勒博

您可能正在使用可继承的文件句柄创建子进程,但实际上并没有告诉子进程该句柄的值是什么!孩子正在 中寻找该值argv[1],但您显示的代码未在该参数中传递任何值!即使有,您也没有正确检索值。

命令行参数始终是字符串,argv[1]指向以空字符结尾的字符串的指针也是如此,而不是句柄。当你把它放在命令行上时,你必须将句柄指针转换为字符串表示,然后在处理命令行时必须将该字符串解析回指针

尝试更像这样的事情:

流程一:

#include <Windows.h>
#include <process.h>
#include <stdio.h>
#include <tchar.h>

int _tmain(int argc, _TCHAR* argv[])
{
    SECURITY_ATTRIBUTES sec_attr = {};
    sec_attr.nLength = sizeof(sec_attr);
    sec_attr.bInheritHandle = TRUE;
    sec_attr.lpSecurityDescriptor = 0;

    //argv[1] - the name of the file I want to create
    HANDLE h = CreateFile(argv[1], GENERIC_READ | GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_READ, &sec_attr, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
    if (h == INVALID_HANDLE_VALUE)
    {
        // handle the error ...
        return 1;
    }

    STARTUPINFO sinfo;      
    ZeroMemory(&sinfo, sizeof(sinfo));
    sinfo.cb = sizeof(sinfo);

    PROCESS_INFORMATION pinfo;
    ZeroMemory(&pinfo, sizeof(pinfo));

    _TCHAR cmd[MAX_PATH + 16] = {};
    _tsprintf(cmd, _T("\"%s\" %p"), _T("PATH to process 2"), h);

    // TODO: consider using STARTUPINFOEX instead so the child inherits
    // ONLY the file handle and not ALL inheritable handles it doesn't
    // care about:
    //
    // Programmatically controlling which handles are inherited by new processes in Win32
    // https://blogs.msdn.microsoft.com/oldnewthing/20111216-00/?p=8873
    //
    if (!CreateProcess(0, cmd, 0, 0, TRUE, NORMAL_PRIORITY_CLASS, 0, 0, &sinfo, &pinfo))
    {
        // handle the error ...
        return 1;
    }

    CloseHandle(pinfo.hThread);
    CloseHandle(pinfo.hProcess);

    char c = '1';
    DWORD w;
    WriteFile(h, &c, 1, &w, 0);
    ...

    CloseHandle(h);
    return 0;
}

过程2:

#include <Windows.h>
#include <process.h>
#include <stdio.h>
#include <tchar.h>

int _tmain(int argc, _TCHAR* argv[]) 
{
    if (argc < 2)
    { 
        // ...
        return 1;
    }

    //argv[1] - string representation of a handle to an open file
    HANDLE h = INVALID_HANDLE_VALUE;
    _stscanf(argv[1], _T("%p"), &h);

    char c = '2';
    DWORD w;
    WriteFile(h, &c, 1, &w, 0);
    ...

    CloseHandle(h);
    return 0;
}

话虽如此,拥有两个线程,更不用说两个进程,共享一个文件句柄并不是一个好主意。给定的文件句柄中只有 1 个读/写位置,因此共享该句柄会冒着线程/进程相互交错的风险,除非您在它们之间提供某种同步,以便只有一个人可以同时访问该文件时间。

考虑将目标文件名传递给子进程,并让它打开自己唯一的文件句柄,例如:

流程一:

#include <Windows.h>
#include <process.h>
#include <stdio.h>
#include <tchar.h>

int _tmain(int argc, _TCHAR* argv[])
{
    if (argc < 2)
    { 
        // ...
        return 1;
    }

    //argv[1] - the name of the file I want to create
    HANDLE h = CreateFile(argv[1], GENERIC_READ | GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_READ, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
    if (h == INVALID_HANDLE_VALUE)
    {
        // handle the error ...
        return 1;
    }

    STARTUPINFO sinfo;      
    ZeroMemory(&sinfo, sizeof(sinfo));
    sinfo.cb = sizeof(sinfo);

    PROCESS_INFORMATION pinfo;
    ZeroMemory(&pinfo, sizeof(pinfo));

    _TCHAR cmd[(MAX_PATH *2) + 10] = {};
    _tsprintf(cmd, _T("\"%s\" \"%s\""), _T("PATH to process 2"), argv[1]);

    if (!CreateProcess(0, cmd, 0, 0, TRUE, NORMAL_PRIORITY_CLASS, 0, 0, &sinfo, &pinfo))
    {
        // handle the error ...
        return 1;
    }

    char c = '1';
    DWORD w;
    WriteFile(h, &c, 1, &w, 0);
    ...

    CloseHandle(h);
    return 0;
}

过程2:

#include <Windows.h>
#include <process.h>
#include <stdio.h>
#include <tchar.h>

int _tmain(int argc, _TCHAR* argv[]) 
{
    if (argc < 2)
    { 
        // ...
        return 1;
    }

    //argv[1] - the name of the file I want to open
    HANDLE h = CreateFile(argv[1], GENERIC_READ | GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
    if (h == INVALID_HANDLE_VALUE)
    {
        // handle the error ...
        return 1;
    }

    char c = '2';
    DWORD w;
    WriteFile(h, &c, 1, &w, 0);
    ...

    CloseHandle(h);
    return 0;
}

否则,请考虑让进程 1 复制文件句柄,然后让进程 2 继承副本而不是原始文件。

本文收集自互联网,转载请注明来源。

如有侵权,请联系[email protected] 删除。

编辑于
0

我来说两句

0条评论
登录后参与评论

相关文章

来自分类Dev

我有一个执行流程问题,其中有nodejs子进程

来自分类Dev

Process.Start而不创建子进程(端口句柄继承)?

来自分类Dev

父进程有多个子进程,而两个子进程有一个子进程

来自分类Dev

另一个 Web 进程未能在启动后 60 秒内绑定到 $PORT

来自分类Dev

在父进程中仅将一个写入流用于多个子进程

来自分类Dev

当另一个子进程完成时,如何终止子进程?

来自分类Dev

使用python multiprocessing子进程如何终止另一个子进程?

来自分类Dev

如何区分一个子进程与其他子进程

来自分类Dev

如何关闭SYSTEM_HANDLE(或在另一个进程中打开文件句柄)

来自分类Dev

从另一个进程的线程ID获取线程句柄

来自分类Dev

我可以知道我的窗口句柄是否由另一个进程请求吗?

来自分类Dev

如何将 DXGI 共享资源的句柄传递给另一个进程?

来自分类Dev

在另一个进程将行写入文件之前,应该如何阻止一个进程?

来自分类Dev

在Electron中产生一个子进程

来自分类Dev

采用另一个进程的子窗口

来自分类Dev

开始后分离一个生成子进程

来自分类Dev

子进程:在后台启动进程并在一次调用中启动另一个进程

来自分类Dev

如何减少Linux上一个进程的所有线程(和子进程)?

来自分类Dev

如何减少Linux上一个进程的所有线程(和子进程)?

来自分类Dev

子进程中的Ctrl + c会在脚本的前面杀死一个nohup'ed进程

来自分类Dev

如果已知父进程,如何获取一个非常短的子进程的ID?

来自分类Dev

多人纸牌游戏-启动子进程或在一个进程中处理

来自分类Dev

Shell函数来检查是否有一个挂起的进程是该Shell的子进程?

来自分类Dev

如何使用htop挂起一个进程及其所有子进程?

来自分类Dev

子进程是否与其父进程一起死亡,是否有任何UNIX变体?

来自分类Dev

python:子进程调用一个脚本,该脚本运行一个挂起的后台进程

来自分类Dev

Python杀死一个子进程(启动另一个进程)并再次启动

来自分类Dev

如何强制2个Python子进程一个接一个地运行

来自分类Dev

如何强制2个Python子进程一个接一个地运行

Related 相关文章

  1. 1

    我有一个执行流程问题,其中有nodejs子进程

  2. 2

    Process.Start而不创建子进程(端口句柄继承)?

  3. 3

    父进程有多个子进程,而两个子进程有一个子进程

  4. 4

    另一个 Web 进程未能在启动后 60 秒内绑定到 $PORT

  5. 5

    在父进程中仅将一个写入流用于多个子进程

  6. 6

    当另一个子进程完成时,如何终止子进程?

  7. 7

    使用python multiprocessing子进程如何终止另一个子进程?

  8. 8

    如何区分一个子进程与其他子进程

  9. 9

    如何关闭SYSTEM_HANDLE(或在另一个进程中打开文件句柄)

  10. 10

    从另一个进程的线程ID获取线程句柄

  11. 11

    我可以知道我的窗口句柄是否由另一个进程请求吗?

  12. 12

    如何将 DXGI 共享资源的句柄传递给另一个进程?

  13. 13

    在另一个进程将行写入文件之前,应该如何阻止一个进程?

  14. 14

    在Electron中产生一个子进程

  15. 15

    采用另一个进程的子窗口

  16. 16

    开始后分离一个生成子进程

  17. 17

    子进程:在后台启动进程并在一次调用中启动另一个进程

  18. 18

    如何减少Linux上一个进程的所有线程(和子进程)?

  19. 19

    如何减少Linux上一个进程的所有线程(和子进程)?

  20. 20

    子进程中的Ctrl + c会在脚本的前面杀死一个nohup'ed进程

  21. 21

    如果已知父进程,如何获取一个非常短的子进程的ID?

  22. 22

    多人纸牌游戏-启动子进程或在一个进程中处理

  23. 23

    Shell函数来检查是否有一个挂起的进程是该Shell的子进程?

  24. 24

    如何使用htop挂起一个进程及其所有子进程?

  25. 25

    子进程是否与其父进程一起死亡,是否有任何UNIX变体?

  26. 26

    python:子进程调用一个脚本,该脚本运行一个挂起的后台进程

  27. 27

    Python杀死一个子进程(启动另一个进程)并再次启动

  28. 28

    如何强制2个Python子进程一个接一个地运行

  29. 29

    如何强制2个Python子进程一个接一个地运行

热门标签

归档