百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Linux系列:聊一聊 SystemV 下的进程间共享内存

gudong366 2025-06-10 14:20 8 浏览

一:背景

1. 讲故事

昨天在分析一个 linux 的 dump 时,看到了这么一话警告,参考如下:


0:000> !eeheap -gc
*** WARNING: Unable to verify timestamp for SYSV10cf21d1 (deleted)

对,就是上面的 SYSV10cf21d1,拆分一下为 System V + 10cf21d1 ,前者的System V表示共享内存机制,后面的 10cf21d1 表示共享内存中用到的唯一键key,所以这表示当前的 .net 程序直接或者间接的使用了 System V的进程间共享内存,我对 Linux 不是特别熟悉,所以稍微研究了下就有了这篇文章。

二:System V 研究

1. 什么是进程间通信

其实在 Linux 中有很多中方式进行 IPC(进程间通信),我用大模型帮我做了一下汇总,截图如下:

现如今Linux使用最多的还是 POSIX 标准,而 System V 相对来说比较老,为了研究我们写一个小例子观察下基本实现。

2. System V 的一个小例子

为了能够实现进程间通信,开启两个进程(writer,reader)端,一个是往共享内存写入,一个从共享内存中读取,画个简图如下:

接下来在内存段的首位置设置控制flag,后面跟着传输的 content 内容,然后创建一个key与申请的内存段进行绑定,参考代码如下:

1)writer.c


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>

#define SHM_SIZE 1024 // 共享内存段大小

int main()
{
key_t key;
int shmid;
char *shm_ptr;

// 生成key值 - 使用当前目录和项目ID
if ((key = ftok(".", 'x')) == -1)
{
perror("ftok");
exit(1);
}

// 创建共享内存段
if ((shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0666)) == -1)
{
perror("shmget");
exit(1);
}

// 附加到共享内存
if ((shm_ptr = shmat(shmid, , 0)) == (void *)-1)
{
perror("shmat");
exit(1);
}

printf("Writer: 连接到共享内存段 %d\n", shmid);

// 第一个字节作为标志位,其余部分存储数据
char *flag_ptr = shm_ptr;
char *data_ptr = shm_ptr + 1;

// 初始化标志位
*flag_ptr = 0;

// 写入数据到共享内存
char message[] = "Hello from writer process!";
strncpy(data_ptr, message, sizeof(message));

// 设置标志位表示数据已准备好
*flag_ptr = 1;

printf("Writer: 已写入消息: \"%s\"\n", message);

// 等待读取进程完成
printf("Writer: 等待读取进程确认...\n");
while (*flag_ptr != 2)
{
sleep(1);
}

// 分离共享内存
if (shmdt(shm_ptr) == -1)
{
perror("shmdt");
exit(1);
}

// 删除共享内存段
if (shmctl(shmid, IPC_RMID, ) == -1)
{
perror("shmctl");
exit(1);
}

printf("Writer: 完成\n");

return0;
}

接下来就是 gcc 编译并运行,参考如下:


root@ubuntu2404:/data2# gcc -g writer.c -o writer
root@ubuntu2404:/data2# ls
writer writer.c
root@ubuntu2404:/data2# ./writer
Writer: 连接到共享内存段 2
Writer: 已写入消息: "Hello from writer process!"
Writer: 等待读取进程确认...

从输出看已经将 "Hello from writer process!" 写到了共享内存,接下来可以用 ipcs -m 观察共享内存段列表,以及虚拟地址段。


root@ubuntu2404:/proc# ipcs -m

------ Shared Memory Segments --------
key shmid owner perms bytes nattch status
0x780300023 root 666 1024 1

root@ubuntu2404:/proc# ps -ef | grep writer
root 7711 7593010:41 pts/1 00:00:00 ./writer
root 7714 7618010:41 pts/2 00:00:00 grep --color=auto writer

root@ubuntu2404:/proc# cat /proc/7711/maps
5b412c9bc000-5b412c9bd000 r--p 0000000008:031966088 /data2/writer
5b412c9bd000-5b412c9be000 r-xp 0000100008:031966088 /data2/writer
5b412c9be000-5b412c9bf000 r--p 0000200008:031966088 /data2/writer
5b412c9bf000-5b412c9c0000 r--p 0000200008:031966088 /data2/writer
5b412c9c0000-5b412c9c1000 rw-p 0000300008:031966088 /data2/writer
5b415ad13000-5b415ad34000 rw-p 0000000000:000 [heap]
...
7c755ce80000-7c755ce81000 rw-s 0000000000:013 /SYSV78030002 (deleted)
...
ffffffffff600000-ffffffffff601000 --xp 0000000000:000 [vsyscall]
root@ubuntu2404:/proc#

上面输出的 /SYSV78030002 (deleted) 便是,哈哈,现在回头看这句 WARNING: Unable to verify timestamp for SYSV10cf21d1 (deleted) 是不是豁然开朗啦。。。

接下来继续聊,另一个进程要想读取共享内存,需要通过同名的key寻找,即下面的 shmget 方法。

2)reader.c


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>

#define SHM_SIZE 1024 // 共享内存段大小

int main()
{
key_t key;
int shmid;
char *shm_ptr;

// 生成相同的key值
if ((key = ftok(".", 'x')) == -1)
{
perror("ftok");
exit(1);
}

// 获取共享内存段
if ((shmid = shmget(key, SHM_SIZE, 0666)) == -1)
{
perror("shmget");
exit(1);
}

// 附加到共享内存
if ((shm_ptr = shmat(shmid, , 0)) == (void *)-1)
{
perror("shmat");
exit(1);
}

printf("Reader: 连接到共享内存段 %d\n", shmid);

// 第一个字节是标志位,其余是数据
char *flag_ptr = shm_ptr;
char *data_ptr = shm_ptr + 1;

// 等待数据准备好
printf("Reader: 等待数据...\n");
while (*flag_ptr != 1)
{
sleep(1);
}

// 读取数据
printf("Reader: 接收到消息: \"%s\"\n", data_ptr);

// 通知写入进程已完成读取
*flag_ptr = 2;

// 分离共享内存
if (shmdt(shm_ptr) == -1)
{
perror("shmdt");
exit(1);
}

printf("Reader: 完成\n");

return0;
}

如果有朋友对绑定逻辑(shmget)的底层感兴趣,可以观察 Linux 中的 ipcget_public 方法,其中的 rhashtable_lookup_fast 便是。


static int ipcget_public(struct ipc_namespace *ns, struct ipc_ids *ids,
const struct ipc_ops *ops, struct ipc_params *params)

{
struct kern_ipc_perm *ipcp;
int flg = params->flg;
int err;

/*
* Take the lock as a writer since we are potentially going to add
* a new entry + read locks are not "upgradable"
*/

down_write(&ids->rwsem);
ipcp = ipc_findkey(ids, params->key);
...
}

static struct kern_ipc_perm *ipc_findkey(struct ipc_ids *ids, key_t key)
{
struct kern_ipc_perm *ipcp;

ipcp = rhashtable_lookup_fast(&ids->key_ht, &key,
ipc_kht_params);
if (!ipcp)
return;

rcu_read_lock();
ipc_lock_object(ipcp);
return ipcp;
}

最后就是相同方式的编译运行,截一张图如下:

三:总结

哈哈,dump分析之旅就是这样,在分析中不断的学习新知识,再用新知识指导dump分析,就这样的不断的螺旋迭代,乐此不疲。


相关推荐

linux sed系列 第四篇:sed工业实战——日志处理与数据清洗

“掌握了sed的编程能力后,我们如同装备精良的工匠,终于可以踏入真实的工业战场。本篇将聚焦sed在日志分析、数据合规化、多文件批处理等场景中的应用,看它如何在海量数据中游刃有余,展现文本处理的...

Linux下sed的简单使用(linux中sed是什么意思)

1、sed简介stremeditor流编辑器,它是一项Linux指令,功能同awk类似,差别在于,sed简单,对列处理的功能要差一些,awk的功能复杂,对列处理的功能比较强大,sed编辑器是一行一...

linux基础命令之date命令(linux中的date)

date命令主要用于显示或者设置系统时间语法格式:date参数对象使用date命令时,最好先使用date--help命令查看支持哪些参数,有些小型Linux系统下的date命令,只支持一些基本参...

Ubuntu linux 常用命令(ubuntu常用的50个命令)

使用dpkg命令来安装.deb包。sudodpkg-i~/example.deb如果在安装过程中遇到依赖问题,可以使用以下命令来修复:sudoapt-getinstall-f将flut...

Linux基础命令-sed命令(linux教程:sed命令的用法)

Sed全名streameditor流编辑器,它是一个强大的文本处理工具,它可以从文件中接受输入,也可以接受来自标准输入流的输入,它擅长取行。Sed的用途非常广泛,包括:1)文本替换2)选择性的输...

linux sed系列 第二篇:sed进阶技巧——地址定位与正则表达式

“上一篇我们掌握了sed的基础替换,如同获得了第一把钥匙。现在,让我们更进一步,学习如何精准锁定目标行,如同拥有了导航地图,让每一次操作都直击要害!”地址定位的四种维度sed的强大,很大程度上源...

火狐Firefox浏览器140发布:手动Unload标签页、优化翻译体验等

IT之家6月24日消息,Mozilla在发布版本139不到一个月后,推出了最新的开源网页浏览器Firefox140。新版本增加了手动Unload标签页的功能,优化了垂直标签页的调...

Linux 基本正则表达式及扩展正则表达式功能举例

在Linux中,正则表达式(RegularExpression)是一种强大的模式匹配工具,用于在文本中查找、匹配和处理特定模式的字符串。Linux支持两种类型的正则表达式:基本正则表达式(Basic...

linux下find命令的经典26个使用示例

简介find命令是基于unix的操作系统中常用的工具之一。顾名思义,它在目录层次结构中查找文件和目录。用户可以传递不同的参数,并根据文件的名称、扩展名、类型、大小、权限、修改时间、所有者、组等搜索文件...

linux运维中特殊符号的应用与实践

路径位置类的特殊符号(1)、波浪线(~)在linux系统的命令行中,~表示用户的家目录,超级用户为/root,普通用户为/home。假设我当前目录在usr/local下[root@xrylocal]...

开源框架log4cpp实战(开源gui框架)

1.Log4cpp使用Log4cpp中主要包含Category(种类),Appender(附加器),Layout(布局),Priorty(优先级),NDC(嵌套的诊断上下文)。Category、App...

Linux find命令详解(linux find -l)

一、命令介绍Linuxfind命令是类unix操作系统中最重要和最常用的命令行实用程序之一。find命令用于根据指定的条件搜索和定位与参数匹配的文件和目录列表。find命令提供了广泛的选项,允许用户...

Linux运维:单引号与双引号的使用(linux 单引号和双引号)

1、单引号的使用单引号可以将它中间的所有任意字符还原为字面意义,实现屏蔽Shell元字符的功能。注意不可以在两个单引号中间单独插入一个单引号,单引号必须成对出现。示例1:定义一个变量,并输出变量的...

Linux技巧:find 命令用法详细说明,看完会有收获

在Linux命令中,find是比较复杂难用的命令。使用该命令搜索文件时,常常发现自己找了一些例子能用,但稍微改一下条件,就搜不到想要的结果。下面会以一些实例来说明使用find命令的关键要点和...

Linux Shell中单引号、双引号、反引号的解释

1、单引号('')单引号所见即所得,直接显示单引号里的内容。即单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的。比如下面的例子,单引号所见即所得。2、双引号("...