成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Linux系列:聊一聊 SystemV 下的進(jìn)程間共享內(nèi)存

系統(tǒng) Linux
前者的System V表示共享內(nèi)存機(jī)制,后面的?10cf21d1?表示共享內(nèi)存中用到的唯一鍵key,所以這表示當(dāng)前的 .net 程序直接或者間接的使用了?System V的進(jìn)程間共享內(nèi)存,我對(duì) Linux 不是特別熟悉,所以稍微研究了下就有了這篇文章。

一、背景

1. 講故事

昨天在分析一個(gè) linux 的 dump 時(shí),看到了這么一話警告,參考如下:

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

對(duì),就是上面的 SYSV10cf21d1,拆分一下為 System V + 10cf21d1 ,前者的System V表示共享內(nèi)存機(jī)制,后面的 10cf21d1 表示共享內(nèi)存中用到的唯一鍵key,所以這表示當(dāng)前的 .net 程序直接或者間接的使用了 System V的進(jìn)程間共享內(nèi)存,我對(duì) Linux 不是特別熟悉,所以稍微研究了下就有了這篇文章。

二、System V 研究

1. 什么是進(jìn)程間通信

其實(shí)在 Linux 中有很多中方式進(jìn)行 IPC(進(jìn)程間通信),我用大模型幫我做了一下匯總,截圖如下:

圖片圖片

現(xiàn)如今Linux使用最多的還是 POSIX 標(biāo)準(zhǔn),而 System V 相對(duì)來說比較老,為了研究我們寫一個(gè)小例子觀察下基本實(shí)現(xiàn)。

2. System V 的一個(gè)小例子

為了能夠?qū)崿F(xiàn)進(jìn)程間通信,開啟兩個(gè)進(jìn)程(writer,reader)端,一個(gè)是往共享內(nèi)存寫入,一個(gè)從共享內(nèi)存中讀取,畫個(gè)簡圖如下:

圖片圖片

接下來在內(nèi)存段的首位置設(shè)置控制flag,后面跟著傳輸?shù)?nbsp;content 內(nèi)容,然后創(chuàng)建一個(gè)key與申請的內(nèi)存段進(jìn)行綁定,參考代碼如下:

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 // 共享內(nèi)存段大小

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

    // 生成key值 - 使用當(dāng)前目錄和項(xiàng)目ID
    if ((key = ftok(".", 'x')) == -1)
    {
        perror("ftok");
        exit(1);
    }

    // 創(chuàng)建共享內(nèi)存段
    if ((shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0666)) == -1)
    {
        perror("shmget");
        exit(1);
    }

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

    printf("Writer: 連接到共享內(nèi)存段 %d\n", shmid);

    // 第一個(gè)字節(jié)作為標(biāo)志位,其余部分存儲(chǔ)數(shù)據(jù)
    char *flag_ptr = shm_ptr;
    char *data_ptr = shm_ptr + 1;

    // 初始化標(biāo)志位
    *flag_ptr = 0;

    // 寫入數(shù)據(jù)到共享內(nèi)存
    char message[] = "Hello from writer process!";
    strncpy(data_ptr, message, sizeof(message));

    // 設(shè)置標(biāo)志位表示數(shù)據(jù)已準(zhǔn)備好
    *flag_ptr = 1;

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

    // 等待讀取進(jìn)程完成
    printf("Writer: 等待讀取進(jìn)程確認(rèn)...\n");
    while (*flag_ptr != 2)
    {
        sleep(1);
    }

    // 分離共享內(nèi)存
    if (shmdt(shm_ptr) == -1)
    {
        perror("shmdt");
        exit(1);
    }

    // 刪除共享內(nèi)存段
    if (shmctl(shmid, IPC_RMID, NULL) == -1)
    {
        perror("shmctl");
        exit(1);
    }

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

    return0;
}

接下來就是 gcc 編譯并運(yùn)行,參考如下:

root@ubuntu2404:/data2# gcc -g writer.c -o writer
root@ubuntu2404:/data2# ls   
writer  writer.c
root@ubuntu2404:/data2# ./writer
Writer: 連接到共享內(nèi)存段 2
Writer: 已寫入消息: "Hello from writer process!"
Writer: 等待讀取進(jìn)程確認(rèn)...

從輸出看已經(jīng)將 "Hello from writer process!" 寫到了共享內(nèi)存,接下來可以用 ipcs -m 觀察共享內(nèi)存段列表,以及虛擬地址段。

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) 便是,哈哈,現(xiàn)在回頭看這句 WARNING: Unable to verify timestamp for SYSV10cf21d1 (deleted) 是不是豁然開朗啦。。。

接下來繼續(xù)聊,另一個(gè)進(jìn)程要想讀取共享內(nèi)存,需要通過同名的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 // 共享內(nèi)存段大小

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

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

    // 獲取共享內(nèi)存段
    if ((shmid = shmget(key, SHM_SIZE, 0666)) == -1)
    {
        perror("shmget");
        exit(1);
    }

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

    printf("Reader: 連接到共享內(nèi)存段 %d\n", shmid);

    // 第一個(gè)字節(jié)是標(biāo)志位,其余是數(shù)據(jù)
    char *flag_ptr = shm_ptr;
    char *data_ptr = shm_ptr + 1;

    // 等待數(shù)據(jù)準(zhǔn)備好
    printf("Reader: 等待數(shù)據(jù)...\n");
    while (*flag_ptr != 1)
    {
        sleep(1);
    }

    // 讀取數(shù)據(jù)
    printf("Reader: 接收到消息: \"%s\"\n", data_ptr);

    // 通知寫入進(jìn)程已完成讀取
    *flag_ptr = 2;

    // 分離共享內(nèi)存
    if (shmdt(shm_ptr) == -1)
    {
        perror("shmdt");
        exit(1);
    }

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

    return0;
}

如果有朋友對(duì)綁定邏輯(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)
returnNULL;

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

最后就是相同方式的編譯運(yùn)行,截一張圖如下:

圖片圖片

三、總結(jié)

哈哈,dump分析之旅就是這樣,在分析中不斷的學(xué)習(xí)新知識(shí),再用新知識(shí)指導(dǎo)dump分析,就這樣的不斷的螺旋迭代,樂此不疲。

責(zé)任編輯:武曉燕 來源: 一線碼農(nóng)聊技術(shù)
相關(guān)推薦

2020-06-28 09:30:37

Linux內(nèi)存操作系統(tǒng)

2019-03-20 14:29:46

Linux虛擬內(nèi)存

2022-05-12 23:19:15

Redis內(nèi)存碎片處理

2021-04-20 08:40:11

內(nèi)存管理Lwip

2023-03-06 21:23:23

Redis數(shù)據(jù)庫

2019-02-13 14:15:59

Linux版本Fedora

2022-02-21 15:01:45

MySQL共享鎖獨(dú)占鎖

2022-05-18 16:35:43

Redis內(nèi)存運(yùn)維

2021-01-04 08:09:07

Linux內(nèi)核Watchdog

2021-01-26 05:06:24

LinuxXargs 命令

2023-07-06 13:56:14

微軟Skype

2023-03-03 12:37:50

JavaJVM內(nèi)存溢出

2020-09-08 06:54:29

Java Gradle語言

2018-01-12 14:35:00

Linux進(jìn)程共享內(nèi)存

2023-07-03 07:27:41

進(jìn)程線程Win32

2021-01-28 22:31:33

分組密碼算法

2023-09-22 17:36:37

2020-05-22 08:16:07

PONGPONXG-PON

2019-05-08 11:10:05

Linux進(jìn)程語言

2021-08-01 09:55:57

Netty時(shí)間輪中間件
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 亚洲va欧美va天堂v国产综合 | 91私密视频| 午夜影院在线观看视频 | 亚洲欧洲精品一区 | 黄色大片在线免费观看 | 黄色免费观看网站 | 在线中文视频 | 成人一区二区在线 | 播放一级毛片 | 国产亚洲欧美日韩精品一区二区三区 | 国产在线播放av | 91精品久久久| 欧美一二三四成人免费视频 | 久久成人在线视频 | 成人福利在线 | av网址在线播放 | 国产亚洲精品久久久优势 | 在线免费亚洲视频 | 小h片免费观看久久久久 | 久久久一区二区 | 中文字幕成人 | 中文一区 | 久久久久国产精品一区二区 | 99视频在线看 | 一区二区三区高清 | 人人干人人看 | 国产精品一区二区av | 玩丰满女领导对白露脸hd | 美国黄色一级片 | 国产精品久久久久久久久久久久久久 | 天天射网站 | www日本在线 | 91高清在线| 日日艹夜夜艹 | 久久国产欧美日韩精品 | 成人免费毛片在线观看 | 4hu最新网址 | 草草视频在线免费观看 | av大片在线| 国产成人免费视频网站高清观看视频 | 国产高清精品一区二区三区 |