Windows系统下的PostgreSQL进程fork

本篇内容介绍了“Windows系统下的PostgreSQL进程fork”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

成都创新互联公司致力于互联网品牌建设与网络营销,包括做网站、成都网站制作、SEO优化、网络推广、整站优化营销策划推广、电子商务、移动互联网营销等。成都创新互联公司为不同类型的客户提供良好的互联网应用定制及解决方案,成都创新互联公司核心团队十年专注互联网开发,积累了丰富的网站经验,为广大企业客户提供一站式企业网站建设服务,在网站建设行业内树立了良好口碑。

原稿用Markdown编写,拷贝进微信平台时段落有时发生反转,段落1、2、3变成段落3、2、1,发布前可能没有检查出来,遇到别扭的地方试试反过来读。

Windows系统API我并不熟悉,所以本篇大致点出过程,更多细节还请阅读代码和详查微软文档。后边还会有一篇讲Windows下信号处理的模拟,内容跟这篇是关联的。

这篇文章假定读者已经了解*nix的fork,如果不了解,请自行阅读相关资料。

1、*nix下PG后端(backend)进程的发起:

static int
BackendStartup(Port *port)
{
...
#ifdef EXEC_BACKEND
	pid = backend_forkexec(port);
#else							/* !EXEC_BACKEND */
	pid = fork_process();
	if (pid == 0)				/* child */
	{
...

当然这里还有一些其他逻辑,不细表,有兴趣可以自己瞅瞅,都不复杂。

2、函数 fork_process

代码位于 src/backend/postmaster/fork_process.c, 没有什么很复杂的逻辑:

#ifndef WIN32
...
pid_t
fork_process(void)
{
...
	result = fork();
	if (result == 0)
	{

注意:这里有preprocessor,上边这段代码是 #ifndef WIN32 控制,也就是只在 *nix下有效。

3、Windows下的后端进程创建

编译Windows版的方法,可以自行阅读,它不一样的地方之一是启用预处理符 EXEC_BACKEND。这个开关在*nix一样有效,有兴趣可以自己试试启用它编译Linux版,个人感觉进程fork效率一定要低很多。

函数 backend_forkexec 中:

	av[ac++] = "postgres";
	av[ac++] = "--forkbackend";

这里增加两个参数,一个是程序启动的文件名,一个是参数forkbackend,指定这是要启动一个fork后端的进程,启动过程中会有一些影响:

	if (strcmp(argv[1], "--forkbackend") == 0 ||
	...
		PGSharedMemoryReAttach();

4、参数传递

fork()不需要考虑变量传递的问题,EXEC_BACKEND 时保存到临时文件里:

paramHandle = CreateFileMapping(INVALID_HANDLE_VALUE,
...
param = MapViewOfFile(paramHandle, ...
...
if (!save_backend_variables(param, port, pi.hProcess, pi.dwProcessId))
...

这部分代码在 EXEC_BACKEND 函数实现 internal_forkexec 里,可以结合Windows文档去理解。

5、为新进程准备共享内存

if (!pgwin32_ReserveSharedMemoryRegion(pi.hProcess))

以后有机会写共享内存时再写,感觉尤其是Windows下挺有必要。

6、进程创建

在调用 save_backend_variables 之前:

	if (!CreateProcess(NULL, cmdLine, NULL, NULL, TRUE, CREATE_SUSPENDED,
					   NULL, NULL, &si, &pi))
	{

这里 cmdLine安装路径\postgres.exe --forkbackend nnn,最后一个参数是参数处理句柄,第一个参数的计算前几天写过。

从微软文档看,并没有说新创建的进程与postmaster之间是父子关系:

Creates a new process and its primary thread. The new process runs in the security context of the calling process.

7、后端进程启动不一样的处理分支

Windows下调用 SubPostmasterMain

...
#ifdef EXEC_BACKEND
	if (argc > 1 && strncmp(argv[1], "--fork", 6) == 0)
		SubPostmasterMain(argc, argv);	/* does not return */
#endif
...

此函数位于:src/backend/postmaster/postmaster.c

SubPostmasterMain(int argc, char *argv[])

8、参数读入

在Windows下,读取由 CreateFileMapping 创建的文件映射句柄:

#ifdef _WIN64
	paramHandle = (HANDLE) _atoi64(id);
#else
	paramHandle = (HANDLE) atol(id);
#endif
	paramp = MapViewOfFile(paramHandle, FILE_MAP_READ, 0, 0, 0);

9、共享内存ReAttach

	if (strcmp(argv[1], "--forkbackend") == 0 ||
		strcmp(argv[1], "--forkavlauncher") == 0 ||
		strcmp(argv[1], "--forkavworker") == 0 ||
		strcmp(argv[1], "--forkboot") == 0 ||
		strncmp(argv[1], "--forkbgworker=", 15) == 0)
		PGSharedMemoryReAttach();

“Windows系统下的PostgreSQL进程fork”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注创新互联网站,小编将为大家输出更多高质量的实用文章!


标题名称:Windows系统下的PostgreSQL进程fork
分享地址:http://hbruida.cn/article/jgeedp.html