更新日期: 2022/06/01 来源: https://gitee.com/weharmony/kernel_liteos_a_note
shm.c 文件参考

浏览源代码.

结构体

struct  ipc_perm
 
struct  shmid_ds
 
struct  shminfo
 
struct  shmIDSource
 

函数

UINT32 ShmInit (VOID)
 
 LOS_MODULE_INIT (ShmInit, LOS_INIT_LEVEL_VM_COMPLETE)
 
UINT32 ShmDeinit (VOID)
 
STATIC VOID ShmSetSharedFlag (struct shmIDSource *seg)
 给共享段中所有物理页框贴上共享标签 更多...
 
STATIC VOID ShmClearSharedFlag (struct shmIDSource *seg)
 给共享段中所有物理页框撕掉共享标签 更多...
 
STATIC VOID ShmPagesRefDec (struct shmIDSource *seg)
 seg下所有共享页引用减少 更多...
 
STATIC INT32 ShmAllocSeg (key_t key, size_t size, INT32 shmflg)
 为共享段分配物理内存 例如:参数size = 4097, LOS_Align(size, PAGE_SIZE) = 8192 分配页数 size >> PAGE_SHIFT = 2页 更多...
 
STATIC INLINE VOID ShmFreeSeg (struct shmIDSource *seg)
 释放seg->node 所占物理页框,seg本身重置 更多...
 
STATIC INT32 ShmFindSegByKey (key_t key)
 通过key查找 shmId 更多...
 
STATIC INT32 ShmSegValidCheck (INT32 segNum, size_t size, INT32 shmFlg)
 共享内存段有效性检查 更多...
 
STATIC struct shmIDSourceShmFindSeg (int shmid)
 通过ID找到共享内存资源 更多...
 
STATIC VOID ShmVmmMapping (LosVmSpace *space, LOS_DL_LIST *pageList, VADDR_T vaddr, UINT32 regionFlags)
 共享内存映射 更多...
 
VOID OsShmFork (LosVmSpace *space, LosVmMapRegion *oldRegion, LosVmMapRegion *newRegion)
 fork 一个共享线性区 更多...
 
VOID OsShmRegionFree (LosVmSpace *space, LosVmMapRegion *region)
 释放共享线性区 更多...
 
BOOL OsIsShmRegion (LosVmMapRegion *region)
 是否为共享线性区,是否有标签? 更多...
 
STATIC INT32 ShmSegUsedCount (VOID)
 获取共享内存池中已被使用的段数量 更多...
 
STATIC INT32 ShmPermCheck (struct shmIDSource *seg, mode_t mode)
 对共享内存段权限检查 更多...
 
INT32 ShmGet (key_t key, size_t size, INT32 shmflg)
 ShmGet
得到一个共享内存标识符或创建一个共享内存对象 更多...
 
INT32 ShmatParamCheck (const VOID *shmaddr, INT32 shmflg)
 
LosVmMapRegionShmatVmmAlloc (struct shmIDSource *seg, const VOID *shmaddr, INT32 shmflg, UINT32 prot)
 分配一个共享线性区并映射好 更多...
 
VOID * ShmAt (INT32 shmid, const VOID *shmaddr, INT32 shmflg)
 ShmAt
用来启动对该共享内存的访问,并把共享内存连接到当前进程的地址空间。 更多...
 
INT32 ShmCtl (INT32 shmid, INT32 cmd, struct shmid_ds *buf)
 ShmCtl
此函数可以对shmid指定的共享存储进行多种操作(删除、取信息、加锁、解锁等) 更多...
 
INT32 ShmDt (const VOID *shmaddr)
 当对共享存储的操作已经结束时,则调用shmdt与该存储段分离 如果shmat成功执行,那么内核将使与该共享存储相关的shmid_ds结构中的shm_nattch计数器值减1 更多...
 
STATIC VOID OsShmInfoCmd (VOID)
 
STATIC VOID OsShmDeleteCmd (INT32 shmid)
 
STATIC VOID OsShmCmdUsage (VOID)
 
UINT32 OsShellCmdShm (INT32 argc, const CHAR *argv[])
 共享内存 更多...
 
 SHELLCMD_ENTRY (shm_shellcmd, CMD_TYPE_SHOW, "shm", 2,(CmdCallBackFunc) OsShellCmdShm)
 

变量

STATIC LosMux g_sysvShmMux
 
STATIC struct shminfo g_shmInfo
 
STATIC struct shmIDSourceg_shmSegs = NULL
 
STATIC UINT32 g_shmUsedPageCount
 

详细描述

 什么是共享内存
 顾名思义,共享内存就是允许两个不相关的进程访问同一个物理内存。共享内存是在两个正在运行的进程之间
 共享和传递数据的一种非常有效的方式。不同进程之间共享的内存通常安排为同一段物理内存。进程可以将同
 一段共享内存连接到它们自己的地址空间中,所有进程都可以访问共享内存中的地址,就好像它们是由用C语言
 函数malloc()分配的内存一样。而如果某个进程向共享内存写入数据,所做的改动将立即影响到可以访问同一段
 共享内存的任何其他进程。

 特别提醒:共享内存并未提供同步机制,也就是说,在第一个进程结束对共享内存的写操作之前,并无自动机制
 可以阻止第二个进程开始对它进行读取。所以我们通常需要用其他的机制来同步对共享内存的访问

 共享线性区可以由任意的进程创建,每个使用共享线性区都必须经过映射.
版本
作者
weharmonyos.com | 鸿蒙研究站 | 每天死磕一点点
日期
2021-12-24

在文件 shm.c 中定义.

函数说明

◆ LOS_MODULE_INIT()

LOS_MODULE_INIT ( ShmInit  ,
LOS_INIT_LEVEL_VM_COMPLETE   
)

◆ OsIsShmRegion()

BOOL OsIsShmRegion ( LosVmMapRegion region)

是否为共享线性区,是否有标签?

在文件 shm.c435 行定义.

436{
437 return (region->regionFlags & VM_MAP_REGION_FLAG_SHM) ? TRUE : FALSE;
438}
UINT32 regionFlags
Definition: los_vm_map.h:125
这是这个函数的调用关系图:

◆ OsShellCmdShm()

UINT32 OsShellCmdShm ( INT32  argc,
const CHAR argv[] 
)

共享内存

在文件 shm.c937 行定义.

938{
939 INT32 shmid;
940 CHAR *endPtr = NULL;
941
942 if (argc == 0) {
943 OsShmInfoCmd();
944 } else if (argc == 1) {
945 if ((strcmp(argv[0], "-h") != 0) && (strcmp(argv[0], "--help") != 0)) {
946 PRINTK("Invalid option: %s\n", argv[0]);
947 }
949 } else if (argc == 2) { /* 2: two parameter */
950 if (strcmp(argv[0], "-r") != 0) {
951 PRINTK("Invalid option: %s\n", argv[0]);
952 goto DONE;
953 }
954 shmid = strtoul((CHAR *)argv[1], &endPtr, 0);
955 if ((endPtr == NULL) || (*endPtr != 0)) {
956 PRINTK("check shmid %s(us) invalid\n", argv[1]);
957 goto DONE;
958 }
959 /* try to delete shm about shmid */
960 OsShmDeleteCmd(shmid);
961 }
962 return 0;
963DONE:
965 return -1;
966}
signed int INT32
Definition: los_typedef.h:60
char CHAR
Definition: los_typedef.h:63
STATIC VOID OsShmDeleteCmd(INT32 shmid)
Definition: shm.c:908
STATIC VOID OsShmInfoCmd(VOID)
Definition: shm.c:888
STATIC VOID OsShmCmdUsage(VOID)
Definition: shm.c:930
函数调用图:

◆ OsShmCmdUsage()

STATIC VOID OsShmCmdUsage ( VOID  )

在文件 shm.c930 行定义.

931{
932 PRINTK("\tnone option, print shm usage info\n"
933 "\t-r [shmid], Recycle the specified shared memory about shmid\n"
934 "\t-h | --help, print shm command usage\n");
935}
这是这个函数的调用关系图:

◆ OsShmDeleteCmd()

STATIC VOID OsShmDeleteCmd ( INT32  shmid)

在文件 shm.c908 行定义.

909{
910 struct shmIDSource *seg = NULL;
911
912 if ((shmid < 0) || (shmid >= g_shmInfo.shmmni)) {
913 PRINT_ERR("shmid is invalid: %d\n", shmid);
914 return;
915 }
916
917 SYSV_SHM_LOCK();
918 seg = ShmFindSeg(shmid);
919 if (seg == NULL) {
920 SYSV_SHM_UNLOCK();
921 return;
922 }
923
924 if (seg->ds.shm_nattch <= 0) {
925 ShmFreeSeg(seg);
926 }
927 SYSV_SHM_UNLOCK();
928}
STATIC INLINE VOID ShmFreeSeg(struct shmIDSource *seg)
释放seg->node 所占物理页框,seg本身重置
Definition: shm.c:310
STATIC struct shminfo g_shmInfo
Definition: shm.c:155
STATIC struct shmIDSource * ShmFindSeg(int shmid)
通过ID找到共享内存资源
Definition: shm.c:357
struct shmid_ds ds
Definition: shm.c:146
unsigned long shm_nattch
当前使用该共享内存段的进程数量
Definition: shm.c:133
unsigned long shmmni
Definition: shm.c:139
函数调用图:
这是这个函数的调用关系图:

◆ OsShmFork()

VOID OsShmFork ( LosVmSpace space,
LosVmMapRegion oldRegion,
LosVmMapRegion newRegion 
)

fork 一个共享线性区

在文件 shm.c393 行定义.

394{
395 struct shmIDSource *seg = NULL;
396
397 SYSV_SHM_LOCK();
398 seg = ShmFindSeg(oldRegion->shmid);//通过老区ID获取对应的共享资源ID结构体
399 if (seg == NULL) {
400 SYSV_SHM_UNLOCK();
401 VM_ERR("shm fork failed!");
402 return;
403 }
404
405 newRegion->shmid = oldRegion->shmid;//一样的共享区ID
406 newRegion->forkFlags = oldRegion->forkFlags;//forkFlags也一样了
407 ShmVmmMapping(space, &seg->node, newRegion->range.base, newRegion->regionFlags);//新线性区与共享内存进行映射
408 seg->ds.shm_nattch++;//附在共享线性区上的进程数++
409 SYSV_SHM_UNLOCK();
410}
STATIC VOID ShmVmmMapping(LosVmSpace *space, LOS_DL_LIST *pageList, VADDR_T vaddr, UINT32 regionFlags)
共享内存映射
Definition: shm.c:375
VADDR_T base
Definition: los_vm_map.h:84
UINT32 shmid
Definition: los_vm_map.h:126
UINT8 forkFlags
Definition: los_vm_map.h:127
LosVmMapRange range
Definition: los_vm_map.h:123
LOS_DL_LIST node
Definition: shm.c:148
函数调用图:
这是这个函数的调用关系图:

◆ OsShmInfoCmd()

STATIC VOID OsShmInfoCmd ( VOID  )

在文件 shm.c888 行定义.

889{
890 INT32 i;
891 struct shmIDSource *seg = NULL;
892
893 PRINTK("\r\n------- Shared Memory Segments -------\n");
894 PRINTK("key shmid perms bytes nattch status owner\n");
895 SYSV_SHM_LOCK();
896 for (i = 0; i < g_shmInfo.shmmni; i++) {
897 seg = &g_shmSegs[i];
898 if (!(seg->status & SHM_SEG_USED)) {
899 continue;
900 }
901 PRINTK("%08x %-8d %-10o %-10u %-10u %-10x %s\n", seg->ds.shm_perm.key,
902 i, seg->ds.shm_perm.mode, seg->ds.shm_segsz, seg->ds.shm_nattch,
903 seg->status, seg->ownerName);
904
905 }
906 SYSV_SHM_UNLOCK();
907}
STATIC struct shmIDSource * g_shmSegs
Definition: shm.c:163
mode_t mode
Definition: shm.c:119
CHAR ownerName[OS_PCB_NAME_LEN]
Definition: shm.c:150
UINT32 status
Definition: shm.c:147
size_t shm_segsz
共享内存段的大小,单位为字节
Definition: shm.c:127
struct ipc_perm shm_perm
操作许可,里面包含共享内存的用户ID、组ID等信息
Definition: shm.c:126
这是这个函数的调用关系图:

◆ OsShmRegionFree()

VOID OsShmRegionFree ( LosVmSpace space,
LosVmMapRegion region 
)

释放共享线性区

在文件 shm.c412 行定义.

413{
414 struct shmIDSource *seg = NULL;
415
416 SYSV_SHM_LOCK();
417 seg = ShmFindSeg(region->shmid);//通过线性区ID获取对应的共享资源ID结构体
418 if (seg == NULL) {
419 SYSV_SHM_UNLOCK();
420 return;
421 }
422
423 LOS_ArchMmuUnmap(&space->archMmu, region->range.base, region->range.size >> PAGE_SHIFT);//解除线性区的映射
424 ShmPagesRefDec(seg);//ref --
425 seg->ds.shm_nattch--;//附在共享线性区上的进程数--
426 if (seg->ds.shm_nattch <= 0 && (seg->status & SHM_SEG_REMOVE)) {
427 ShmFreeSeg(seg);//就释放掉物理内存!注意是:物理内存
428 } else {
429 seg->ds.shm_dtime = time(NULL);
430 seg->ds.shm_lpid = LOS_GetCurrProcessID(); /* may not be the space's PID. */
431 }
432 SYSV_SHM_UNLOCK();
433}
STATUS_T LOS_ArchMmuUnmap(LosArchMmu *archMmu, VADDR_T vaddr, size_t count)
LOS_ArchMmuUnmap 解除进程空间虚拟地址区间与物理地址区间的映射关系
Definition: los_arch_mmu.c:619
LITE_OS_SEC_TEXT UINT32 LOS_GetCurrProcessID(VOID)
获取当前进程的进程ID
Definition: los_process.c:2161
STATIC VOID ShmPagesRefDec(struct shmIDSource *seg)
seg下所有共享页引用减少
Definition: shm.c:233
UINT32 size
Definition: los_vm_map.h:85
LosArchMmu archMmu
Definition: los_vm_map.h:157
time_t shm_dtime
最后一个进程离开共享内存的时间
Definition: shm.c:129
pid_t shm_lpid
最后操作共享内存的进程ID
Definition: shm.c:132
time_t time(time_t *t)
Definition: time.c:1224
函数调用图:
这是这个函数的调用关系图:

◆ SHELLCMD_ENTRY()

SHELLCMD_ENTRY ( shm_shellcmd  ,
CMD_TYPE_SHOW  ,
"shm"  ,
,
(CmdCallBackFunc OsShellCmdShm 
)

◆ ShmAllocSeg()

STATIC INT32 ShmAllocSeg ( key_t  key,
size_t  size,
INT32  shmflg 
)

为共享段分配物理内存 例如:参数size = 4097, LOS_Align(size, PAGE_SIZE) = 8192 分配页数 size >> PAGE_SHIFT = 2页

参数
key
size
shmflg
返回
STATIC

在文件 shm.c251 行定义.

252{
253 INT32 i;
254 INT32 segNum = -1;
255 struct shmIDSource *seg = NULL;
256 size_t count;
257
258 if ((size == 0) || (size < g_shmInfo.shmmin) ||
259 (size > g_shmInfo.shmmax)) {
260 return -EINVAL;
261 }
262 size = LOS_Align(size, PAGE_SIZE);//必须对齐
263 if ((g_shmUsedPageCount + (size >> PAGE_SHIFT)) > g_shmInfo.shmall) {
264 return -ENOMEM;
265 }
266
267 for (i = 0; i < g_shmInfo.shmmni; i++) {//试图找到一个空闲段与参数key绑定
268 if (g_shmSegs[i].status & SHM_SEG_FREE) {//找到空闲段
269 g_shmSegs[i].status &= ~SHM_SEG_FREE;//变成非空闲状态
270 segNum = i;//标号
271 break;
272 }
273 }
274
275 if (segNum < 0) {
276 return -ENOSPC;
277 }
278
279 seg = &g_shmSegs[segNum];
280 count = LOS_PhysPagesAlloc(size >> PAGE_SHIFT, &seg->node);//分配共享页面,函数内部把node都挂好了.
281 if (count != (size >> PAGE_SHIFT)) {//当未分配到足够的内存时,处理方式是:不稀罕给那么点,舍弃!
282 (VOID)LOS_PhysPagesFree(&seg->node);//释放节点上的物理页框
283 seg->status = SHM_SEG_FREE;//共享段变回空闲状态
284 return -ENOMEM;
285 }
286 ShmSetSharedFlag(seg);//将node的每个页面设置为共享页
287 g_shmUsedPageCount += size >> PAGE_SHIFT;
288
289 seg->status |= SHM_SEG_USED; //共享段贴上已在使用的标签
290 seg->ds.shm_perm.mode = (UINT32)shmflg & ACCESSPERMS;
291 seg->ds.shm_perm.key = key;//保存参数key,如此 key 和 共享ID绑定在一块
292 seg->ds.shm_segsz = size; //共享段的大小
293 seg->ds.shm_perm.cuid = LOS_GetUserID(); //设置用户ID
294 seg->ds.shm_perm.uid = LOS_GetUserID(); //设置用户ID
295 seg->ds.shm_perm.cgid = LOS_GetGroupID(); //设置组ID
296 seg->ds.shm_perm.gid = LOS_GetGroupID(); //设置组ID
297 seg->ds.shm_lpid = 0; //最后一个操作的进程
298 seg->ds.shm_nattch = 0; //绑定进程的数量
299 seg->ds.shm_cpid = LOS_GetCurrProcessID(); //获取进程ID
300 seg->ds.shm_atime = 0; //访问时间
301 seg->ds.shm_dtime = 0; //detach 分离时间 共享内存使用完之后,需要将它从进程地址空间中分离出来;将共享内存分离并不是删除它,只是使该共享内存对当前的进程不再可用
302 seg->ds.shm_ctime = time(NULL);//创建时间
303#ifdef LOSCFG_SHELL
304 (VOID)memcpy_s(seg->ownerName, OS_PCB_NAME_LEN, OsCurrProcessGet()->processName, OS_PCB_NAME_LEN);
305#endif
306
307 return segNum;
308}
LITE_OS_SEC_TEXT UINTPTR LOS_Align(UINTPTR addr, UINT32 boundary)
Align the value (addr) by some bytes (boundary) you specify.
Definition: los_misc.c:35
LITE_OS_SEC_TEXT INT32 LOS_GetUserID(VOID)
Definition: los_process.c:826
LITE_OS_SEC_TEXT INT32 LOS_GetGroupID(VOID)
Definition: los_process.c:842
STATIC INLINE LosProcessCB * OsCurrProcessGet(VOID)
unsigned int UINT32
Definition: los_typedef.h:57
size_t LOS_PhysPagesAlloc(size_t nPages, LOS_DL_LIST *list)
LOS_PhysPagesAlloc 分配nPages页个物理页框,并将页框挂入list 返回已分配的页面大小,不负责一定能分配到nPages的页框
Definition: los_vm_phys.c:581
size_t LOS_PhysPagesFree(LOS_DL_LIST *list)
释放双链表中的所有节点内存,本质是回归到伙伴orderlist中
Definition: los_vm_phys.c:661
STATIC UINT32 g_shmUsedPageCount
Definition: shm.c:164
STATIC VOID ShmSetSharedFlag(struct shmIDSource *seg)
给共享段中所有物理页框贴上共享标签
Definition: shm.c:215
CHAR processName[OS_PCB_NAME_LEN]
gid_t cgid
Definition: shm.c:118
uid_t cuid
Definition: shm.c:117
gid_t gid
Definition: shm.c:116
uid_t uid
Definition: shm.c:115
time_t shm_ctime
创建时间
Definition: shm.c:130
time_t shm_atime
最后一个进程访问共享内存的时间
Definition: shm.c:128
pid_t shm_cpid
创建共享内存的进程ID
Definition: shm.c:131
unsigned long shmmax
Definition: shm.c:139
unsigned long shmall
Definition: shm.c:139
unsigned long shmmin
Definition: shm.c:139
函数调用图:
这是这个函数的调用关系图:

◆ ShmAt()

VOID * ShmAt ( INT32  shmid,
const VOID *  shmaddr,
INT32  shmflg 
)

ShmAt
用来启动对该共享内存的访问,并把共享内存连接到当前进程的地址空间。

参数
shm_flg是一组标志位,通常为0。
shmaddr指定共享内存连接到当前进程中的地址位置,通常为空,表示让系统来选择共享内存的地址。
shmid是shmget()函数返回的共享内存标识符
返回
如果shmat成功执行,那么内核将使与该共享存储相关的shmid_ds结构中的shm_nattch计数器值加1 shmid 就是个索引,就跟进程和线程的ID一样 g_shmSegs[shmid] shmid > 192个
参见

在文件 shm.c634 行定义.

635{
636 INT32 ret;
637 UINT32 prot = PROT_READ;
638 mode_t acc_mode = SHM_S_IRUGO;
639 struct shmIDSource *seg = NULL;
640 LosVmMapRegion *r = NULL;
641
642 ret = ShmatParamCheck(shmaddr, shmflg);//参数检查
643 if (ret != 0) {
644 set_errno(ret);
645 return (VOID *)-1;
646 }
647
648 if ((UINT32)shmflg & SHM_EXEC) {//flag 转换
649 prot |= PROT_EXEC;
650 acc_mode |= SHM_S_IXUGO;
651 } else if (((UINT32)shmflg & SHM_RDONLY) == 0) {
652 prot |= PROT_WRITE;
653 acc_mode |= SHM_S_IWUGO;
654 }
655
656 SYSV_SHM_LOCK();
657 seg = ShmFindSeg(shmid);//找到段
658 if (seg == NULL) {
659 SYSV_SHM_UNLOCK();
660 return (VOID *)-1;
661 }
662
663 ret = ShmPermCheck(seg, acc_mode);
664 if (ret != 0) {
665 goto ERROR;
666 }
667
668 seg->ds.shm_nattch++;//ds上记录有一个进程绑定上来
669 r = ShmatVmmAlloc(seg, shmaddr, shmflg, prot);//在当前进程空间分配一个线性区并映射到共享内存
670 if (r == NULL) {
671 seg->ds.shm_nattch--;
672 SYSV_SHM_UNLOCK();
673 return (VOID *)-1;
674 }
675
676 r->shmid = shmid;//把ID给线性区的shmid
677 r->regionFlags |= VM_MAP_REGION_FLAG_SHM;//这是一个共享线性区
678 seg->ds.shm_atime = time(NULL);//访问时间
679 seg->ds.shm_lpid = LOS_GetCurrProcessID();//进程ID
680 SYSV_SHM_UNLOCK();
681
682 return (VOID *)(UINTPTR)r->range.base;
683ERROR:
684 set_errno(ret);
685 SYSV_SHM_UNLOCK();
686 PRINT_DEBUG("%s %d, ret = %d\n", __FUNCTION__, __LINE__, ret);
687 return (VOID *)-1;
688}
unsigned long UINTPTR
Definition: los_typedef.h:68
LosVmMapRegion * ShmatVmmAlloc(struct shmIDSource *seg, const VOID *shmaddr, INT32 shmflg, UINT32 prot)
分配一个共享线性区并映射好
Definition: shm.c:576
INT32 ShmatParamCheck(const VOID *shmaddr, INT32 shmflg)
Definition: shm.c:562
STATIC INT32 ShmPermCheck(struct shmIDSource *seg, mode_t mode)
对共享内存段权限检查
Definition: shm.c:455
ARG_NUM_3 ARG_NUM_1 ARG_NUM_2 ARG_NUM_2 ARG_NUM_3 ARG_NUM_1 ARG_NUM_4 ARG_NUM_2 ARG_NUM_2 ARG_NUM_5 ARG_NUM_2 ARG_NUM_0 ARG_NUM_2 ARG_NUM_1 ARG_NUM_2 ARG_NUM_3 ARG_NUM_7 ARG_NUM_2 ARG_NUM_3 ARG_NUM_2 ARG_NUM_4 ARG_NUM_5 ARG_NUM_6 ARG_NUM_3 ARG_NUM_5 ARG_NUM_7 ARG_NUM_1 ARG_NUM_4 ARG_NUM_5 ARG_NUM_4 ARG_NUM_7 ARG_NUM_2 ARG_NUM_3 ARG_NUM_7 ARG_NUM_7 ARG_NUM_3 ARG_NUM_3 ARG_NUM_3 ARG_NUM_7 ARG_NUM_3 ARG_NUM_2 char ARG_NUM_2 ARG_NUM_1 ARG_NUM_0 ARG_NUM_0 ARG_NUM_3 void ARG_NUM_1 ARG_NUM_0 unsigned ARG_NUM_0 ARG_NUM_2 ARG_NUM_3 ARG_NUM_2 ARG_NUM_5 ARG_NUM_3 ARG_NUM_3 ARG_NUM_4 ARG_NUM_1 ARG_NUM_1 ARG_NUM_3 ARG_NUM_2 mode_t
函数调用图:
这是这个函数的调用关系图:

◆ ShmatParamCheck()

INT32 ShmatParamCheck ( const VOID *  shmaddr,
INT32  shmflg 
)

在文件 shm.c562 行定义.

563{
564 if (((UINT32)shmflg & SHM_REMAP) && (shmaddr == NULL)) {
565 return EINVAL;
566 }
567
568 if ((shmaddr != NULL) && !IS_PAGE_ALIGNED(shmaddr) &&
569 (((UINT32)shmflg & SHM_RND) == 0)) {
570 return EINVAL;
571 }
572
573 return 0;
574}
这是这个函数的调用关系图:

◆ ShmatVmmAlloc()

LosVmMapRegion * ShmatVmmAlloc ( struct shmIDSource seg,
const VOID *  shmaddr,
INT32  shmflg,
UINT32  prot 
)

分配一个共享线性区并映射好

在文件 shm.c576 行定义.

578{
580 LosVmMapRegion *region = NULL;
581 UINT32 flags = MAP_ANONYMOUS | MAP_SHARED;//本线性区为共享+匿名标签
582 UINT32 mapFlags = flags | MAP_FIXED;
583 VADDR_T vaddr;
584 UINT32 regionFlags;
585 INT32 ret;
586
587 if (shmaddr != NULL) {
588 flags |= MAP_FIXED_NOREPLACE;
589 }
590 regionFlags = OsCvtProtFlagsToRegionFlags(prot, flags);
591 (VOID)LOS_MuxAcquire(&space->regionMux);
592 if (shmaddr == NULL) {//未指定了共享内存连接到当前进程中的地址位置
593 region = LOS_RegionAlloc(space, 0, seg->ds.shm_segsz, regionFlags, 0);//分配线性区
594 } else {//指定时,就需要先找地址所在的线性区
595 if ((UINT32)shmflg & SHM_RND) {
596 vaddr = ROUNDDOWN((VADDR_T)(UINTPTR)shmaddr, SHMLBA);
597 } else {
598 vaddr = (VADDR_T)(UINTPTR)shmaddr;
599 }//找到线性区并重新映射,当指定地址时需贴上重新映射的标签
600 if (!((UINT32)shmflg & SHM_REMAP) && (LOS_RegionFind(space, vaddr) ||
601 LOS_RegionFind(space, vaddr + seg->ds.shm_segsz - 1) ||
602 LOS_RegionRangeFind(space, vaddr, seg->ds.shm_segsz - 1))) {
603 ret = EINVAL;
604 goto ERROR;
605 }
606 vaddr = (VADDR_T)LOS_MMap(vaddr, seg->ds.shm_segsz, prot, mapFlags, -1, 0);//做好映射
607 region = LOS_RegionFind(space, vaddr);//重新查找线性区,用于返回.
608 }
609
610 if (region == NULL) {
611 ret = ENOMEM;
612 goto ERROR;
613 }
614 ShmVmmMapping(space, &seg->node, region->range.base, regionFlags);//共享内存映射
615 (VOID)LOS_MuxRelease(&space->regionMux);
616 return region;
617ERROR:
618 set_errno(ret);
619 (VOID)LOS_MuxRelease(&space->regionMux);
620 return NULL;
621}
unsigned long VADDR_T
Definition: los_typedef.h:208
STATIC INLINE STATUS_T LOS_MuxAcquire(LosMux *m)
Definition: los_vm_lock.h:48
STATIC INLINE STATUS_T LOS_MuxRelease(LosMux *m)
Definition: los_vm_lock.h:53
STATIC INLINE UINT32 OsCvtProtFlagsToRegionFlags(unsigned long prot, unsigned long flags)
从外部权限标签转化为线性区权限标签
Definition: los_vm_map.h:197
LosVmMapRegion * LOS_RegionFind(LosVmSpace *vmSpace, VADDR_T addr)
查找线性区 根据起始地址在进程空间内查找是否存在
Definition: los_vm_map.c:414
LosVmMapRegion * LOS_RegionAlloc(LosVmSpace *vmSpace, VADDR_T vaddr, size_t len, UINT32 regionFlags, VM_OFFSET_T pgoff)
Definition: los_vm_map.c:581
LosVmMapRegion * LOS_RegionRangeFind(LosVmSpace *vmSpace, VADDR_T addr, size_t len)
查找线性区 根据地址区间在进程空间内查找是否存在
Definition: los_vm_map.c:425
VADDR_T LOS_MMap(VADDR_T vaddr, size_t len, unsigned prot, unsigned long flags, int fd, unsigned long pgoff)
LosVmSpace * vmSpace
虚拟空间,每个进程都有一个属于自己的虚拟内存地址空间
Definition: los_vm_map.h:146
LosMux regionMux
Definition: los_vm_map.h:149
函数调用图:
这是这个函数的调用关系图:

◆ ShmClearSharedFlag()

STATIC VOID ShmClearSharedFlag ( struct shmIDSource seg)
inline

给共享段中所有物理页框撕掉共享标签

在文件 shm.c224 行定义.

225{
226 LosVmPage *page = NULL;
227
228 LOS_DL_LIST_FOR_EACH_ENTRY(page, &seg->node, LosVmPage, node) {
229 OsCleanPageShared(page);
230 }
231}
STATIC INLINE VOID OsCleanPageShared(LosVmPage *page)
给页面撕掉共享页标签
物理页框描述符 虚拟内存体现的是程序对内存资源的需求,而物理内存是对该请求的供应。 伙伴算法的思想是:把内存中连续的空闲页框空间看成是空闲页框块,并按照它们的大小(连续页框的数目)分组
Definition: los_vm_page.h:53
函数调用图:
这是这个函数的调用关系图:

◆ ShmCtl()

INT32 ShmCtl ( INT32  shmid,
INT32  cmd,
struct shmid_ds buf 
)

ShmCtl
此函数可以对shmid指定的共享存储进行多种操作(删除、取信息、加锁、解锁等)

参数
buf是一个结构指针,它指向共享内存模式和访问权限的结构。
cmdcommand是要采取的操作,它可以取下面的三个值 : IPC_STAT:把shmid_ds结构中的数据设置为共享内存的当前关联值,即用共享内存的当前关联值覆盖shmid_ds的值。 IPC_SET:如果进程有足够的权限,就把共享内存的当前关联值设置为shmid_ds结构中给出的值 IPC_RMID:删除共享内存段
shmid是shmget()函数返回的共享内存标识符
返回
参见

在文件 shm.c703 行定义.

704{
705 struct shmIDSource *seg = NULL;
706 INT32 ret = 0;
707 struct shm_info shmInfo;
708 struct ipc_perm shm_perm;
709
710 cmd = ((UINT32)cmd & ~IPC_64);
711
712 SYSV_SHM_LOCK();
713
714 if ((cmd != IPC_INFO) && (cmd != SHM_INFO)) {
715 seg = ShmFindSeg(shmid);//通过索引ID找到seg
716 if (seg == NULL) {
717 SYSV_SHM_UNLOCK();
718 return -1;
719 }
720 }
721
722 if ((buf == NULL) && (cmd != IPC_RMID)) {
723 ret = EINVAL;
724 goto ERROR;
725 }
726
727 switch (cmd) {
728 case IPC_STAT:
729 case SHM_STAT://取段结构
730 ret = ShmPermCheck(seg, SHM_S_IRUGO);
731 if (ret != 0) {
732 goto ERROR;
733 }
734
735 ret = LOS_ArchCopyToUser(buf, &seg->ds, sizeof(struct shmid_ds));//把内核空间的共享页数据拷贝到用户空间
736 if (ret != 0) {
737 ret = EFAULT;
738 goto ERROR;
739 }
740 if (cmd == SHM_STAT) {
741 ret = (unsigned int)((unsigned int)seg->ds.shm_perm.seq << 16) | (unsigned int)((unsigned int)shmid & 0xffff); /* 16: use the seq as the upper 16 bits */
742 }
743 break;
744 case IPC_SET://重置共享段
745 ret = ShmPermCheck(seg, SHM_M);
746 if (ret != 0) {
747 ret = EPERM;
748 goto ERROR;
749 }
750 //从用户空间拷贝数据到内核空间
751 ret = LOS_ArchCopyFromUser(&shm_perm, &buf->shm_perm, sizeof(struct ipc_perm));
752 if (ret != 0) {
753 ret = EFAULT;
754 goto ERROR;
755 }
756 seg->ds.shm_perm.uid = shm_perm.uid;
757 seg->ds.shm_perm.gid = shm_perm.gid;
758 seg->ds.shm_perm.mode = (seg->ds.shm_perm.mode & ~ACCESSPERMS) |
759 (shm_perm.mode & ACCESSPERMS);//可访问
760 seg->ds.shm_ctime = time(NULL);
761#ifdef LOSCFG_SHELL
762 (VOID)memcpy_s(seg->ownerName, OS_PCB_NAME_LEN, OS_PCB_FROM_PID(shm_perm.uid)->processName,
763 OS_PCB_NAME_LEN);
764#endif
765 break;
766 case IPC_RMID://删除共享段
767 ret = ShmPermCheck(seg, SHM_M);
768 if (ret != 0) {
769 ret = EPERM;
770 goto ERROR;
771 }
772
773 seg->status |= SHM_SEG_REMOVE;
774 if (seg->ds.shm_nattch <= 0) {//没有任何进程在使用了
775 ShmFreeSeg(seg);//释放 归还内存
776 }
777 break;
778 case IPC_INFO://把内核空间的共享页数据拷贝到用户空间
779 ret = LOS_ArchCopyToUser(buf, &g_shmInfo, sizeof(struct shminfo));
780 if (ret != 0) {
781 ret = EFAULT;
782 goto ERROR;
783 }
784 ret = g_shmInfo.shmmni;
785 break;
786 case SHM_INFO:
787 shmInfo.shm_rss = 0;
788 shmInfo.shm_swp = 0;
789 shmInfo.shm_tot = 0;
790 shmInfo.swap_attempts = 0;
791 shmInfo.swap_successes = 0;
792 shmInfo.used_ids = ShmSegUsedCount();//在使用的seg数
793 ret = LOS_ArchCopyToUser(buf, &shmInfo, sizeof(struct shm_info));//把内核空间的共享页数据拷贝到用户空间
794 if (ret != 0) {
795 ret = EFAULT;
796 goto ERROR;
797 }
798 ret = g_shmInfo.shmmni;
799 break;
800 default:
801 VM_ERR("the cmd(%d) is not supported!", cmd);
802 ret = EINVAL;
803 goto ERROR;
804 }
805
806 SYSV_SHM_UNLOCK();
807 return ret;
808
809ERROR:
810 set_errno(ret);
811 SYSV_SHM_UNLOCK();
812 PRINT_DEBUG("%s %d, ret = %d\n", __FUNCTION__, __LINE__, ret);
813 return -1;
814}
STATIC INT32 ShmSegUsedCount(VOID)
获取共享内存池中已被使用的段数量
Definition: shm.c:440
Definition: shm.c:113
Definition: shm.c:125
Definition: shm.c:138
ARG_NUM_3 int
size_t LOS_ArchCopyToUser(void *dst, const void *src, size_t len)
从内核空间拷贝到用户空间
Definition: user_copy.c:79
size_t LOS_ArchCopyFromUser(void *dst, const void *src, size_t len)
Definition: user_copy.c:58
函数调用图:
这是这个函数的调用关系图:

◆ ShmDeinit()

UINT32 ShmDeinit ( VOID  )

在文件 shm.c200 行定义.

201{
202 UINT32 ret;
203
204 (VOID)LOS_MemFree((VOID *)OS_SYS_MEM_ADDR, g_shmSegs);//归还内存池
205 g_shmSegs = NULL;
206
207 ret = LOS_MuxDestroy(&g_sysvShmMux);//销毁互斥量
208 if (ret != LOS_OK) {
209 return -1;
210 }
211
212 return 0;
213}
UINT32 LOS_MemFree(VOID *pool, VOID *ptr)
释放从指定动态内存中申请的内存
Definition: los_memory.c:1369
LITE_OS_SEC_TEXT UINT32 LOS_MuxDestroy(LosMux *mutex)
销毁互斥锁
Definition: los_mux.c:289
STATIC LosMux g_sysvShmMux
Definition: shm.c:76
函数调用图:

◆ ShmDt()

INT32 ShmDt ( const VOID *  shmaddr)

当对共享存储的操作已经结束时,则调用shmdt与该存储段分离 如果shmat成功执行,那么内核将使与该共享存储相关的shmid_ds结构中的shm_nattch计数器值减1

注意
注意:这并不从系统中删除共享存储的标识符以及其相关的数据结构。共享存储的仍然存在, 直至某个进程带IPC_RMID命令的调用shmctl特地删除共享存储为止
参数
shmaddr
返回
INT32

在文件 shm.c824 行定义.

825{
826 LosVmSpace *space = OsCurrProcessGet()->vmSpace;//获取进程空间
827 struct shmIDSource *seg = NULL;
828 LosVmMapRegion *region = NULL;
829 INT32 shmid;
830 INT32 ret;
831
832 if (IS_PAGE_ALIGNED(shmaddr) == 0) {//地址是否对齐
833 ret = EINVAL;
834 goto ERROR;
835 }
836
837 (VOID)LOS_MuxAcquire(&space->regionMux);
838 region = LOS_RegionFind(space, (VADDR_T)(UINTPTR)shmaddr);//找到线性区
839 if (region == NULL) {
840 ret = EINVAL;
841 goto ERROR_WITH_LOCK;
842 }
843 shmid = region->shmid;//线性区共享ID
844
845 if (region->range.base != (VADDR_T)(UINTPTR)shmaddr) {//这是用户空间和内核空间的一次解绑
846 ret = EINVAL; //shmaddr 必须要等于region->range.base
847 goto ERROR_WITH_LOCK;
848 }
849
850 /* remove it from aspace */
851 LOS_RbDelNode(&space->regionRbTree, &region->rbNode);//从红黑树和链表中摘除节点
852 LOS_ArchMmuUnmap(&space->archMmu, region->range.base, region->range.size >> PAGE_SHIFT);//解除线性区的映射
853 (VOID)LOS_MuxRelease(&space->regionMux);
854 /* free it */
855 free(region);//释放线性区所占内存池中的内存
856
857 SYSV_SHM_LOCK();
858 seg = ShmFindSeg(shmid);//找到seg,线性区和共享段的关系是 1:N 的关系,其他空间的线性区也会绑在共享段上
859 if (seg == NULL) {
860 ret = EINVAL;
861 SYSV_SHM_UNLOCK();
862 goto ERROR;
863 }
864
865 ShmPagesRefDec(seg);//页面引用数 --
866 seg->ds.shm_nattch--;//使用共享内存的进程数少了一个
867 if ((seg->ds.shm_nattch <= 0) && //无任何进程使用共享内存
868 (seg->status & SHM_SEG_REMOVE)) {//状态为删除时需要释放物理页内存了,否则其他进程还要继续使用共享内存
869 ShmFreeSeg(seg);//释放seg 页框链表中的页框内存,再重置seg状态
870 } else {
871
872 seg->ds.shm_dtime = time(NULL);//记录分离的时间
873 seg->ds.shm_lpid = LOS_GetCurrProcessID();//记录操作进程ID
874 }
875 SYSV_SHM_UNLOCK();
876
877 return 0;
878
879ERROR_WITH_LOCK:
880 (VOID)LOS_MuxRelease(&space->regionMux);
881ERROR:
882 set_errno(ret);
883 PRINT_DEBUG("%s %d, ret = %d\n", __FUNCTION__, __LINE__, ret);
884 return -1;
885}
VOID LOS_RbDelNode(LosRbTree *pstTree, LosRbNode *pstNode)
Definition: los_rbtree.c:700
void free(void *ptr)
释放ptr所指向的内存空间
Definition: malloc.c:66
LosRbNode rbNode
Definition: los_vm_map.h:120
LosRbTree regionRbTree
Definition: los_vm_map.h:148
函数调用图:
这是这个函数的调用关系图:

◆ ShmFindSeg()

STATIC struct shmIDSource * ShmFindSeg ( int  shmid)

通过ID找到共享内存资源

在文件 shm.c357 行定义.

358{
359 struct shmIDSource *seg = NULL;
360
361 if ((shmid < 0) || (shmid >= g_shmInfo.shmmni)) {
362 set_errno(EINVAL);
363 return NULL;
364 }
365
366 seg = &g_shmSegs[shmid];
367 if ((seg->status & SHM_SEG_FREE) || ((seg->ds.shm_nattch == 0) && (seg->status & SHM_SEG_REMOVE))) {
368 set_errno(EIDRM);
369 return NULL;
370 }
371
372 return seg;
373}
这是这个函数的调用关系图:

◆ ShmFindSegByKey()

STATIC INT32 ShmFindSegByKey ( key_t  key)

通过key查找 shmId

在文件 shm.c325 行定义.

326{
327 INT32 i;
328 struct shmIDSource *seg = NULL;
329
330 for (i = 0; i < g_shmInfo.shmmni; i++) {//遍历共享段池,找到与key绑定的共享ID
331 seg = &g_shmSegs[i];
332 if ((seg->status & SHM_SEG_USED) &&
333 (seg->ds.shm_perm.key == key)) {//满足两个条件,找到后返回
334 return i;
335 }
336 }
337
338 return -1;
339}
这是这个函数的调用关系图:

◆ ShmFreeSeg()

STATIC INLINE VOID ShmFreeSeg ( struct shmIDSource seg)

释放seg->node 所占物理页框,seg本身重置

在文件 shm.c310 行定义.

311{
312 UINT32 count;
313
314 ShmClearSharedFlag(seg);//先撕掉 seg->node 中vmpage的共享标签
315 count = LOS_PhysPagesFree(&seg->node);//再挨个删除物理页框
316 if (count != (seg->ds.shm_segsz >> PAGE_SHIFT)) {//异常,必须要一样
317 VM_ERR("free physical pages failed, count = %d, size = %d", count, seg->ds.shm_segsz >> PAGE_SHIFT);
318 return;
319 }
320 g_shmUsedPageCount -= seg->ds.shm_segsz >> PAGE_SHIFT;
321 seg->status = SHM_SEG_FREE;//seg恢复自由之身
322 LOS_ListInit(&seg->node);//重置node
323}
LITE_OS_SEC_ALW_INLINE STATIC INLINE VOID LOS_ListInit(LOS_DL_LIST *list)
Definition: los_list.h:104
STATIC VOID ShmClearSharedFlag(struct shmIDSource *seg)
给共享段中所有物理页框撕掉共享标签
Definition: shm.c:224
函数调用图:
这是这个函数的调用关系图:

◆ ShmGet()

INT32 ShmGet ( key_t  key,
size_t  size,
INT32  shmflg 
)

ShmGet
得到一个共享内存标识符或创建一个共享内存对象

参数
key建立新共享内存对象 标识符是IPC对象的内部名。为使多个合作进程能够在同一IPC对象上汇聚,需要提供一个外部命名方案。 为此,每个IPC对象都与一个键(key)相关联,这个键作为该对象的外部名,无论何时创建IPC结构(通过msgget、semget、shmget创建), 都应给IPC指定一个键, key_t由ftok创建,ftok当然在本工程里找不到,所以要写这么多.
shmflgIPC_CREAT IPC_EXCL IPC_CREAT: 在创建新的IPC时,如果key参数是IPC_PRIVATE或者和当前某种类型的IPC结构无关,则需要指明flag参数的IPC_CREAT标志位, 则用来创建一个新的IPC结构。(如果IPC结构已存在,并且指定了IPC_CREAT,则IPC_CREAT什么都不做,函数也不出错) IPC_EXCL: 此参数一般与IPC_CREAT配合使用来创建一个新的IPC结构。如果创建的IPC结构已存在函数就出错返回, 返回EEXIST(这与open函数指定O_CREAT和O_EXCL标志原理相同)
size新建的共享内存大小,以字节为单位
返回
参见

在文件 shm.c515 行定义.

516{
517 INT32 ret;
518 INT32 shmid;
519
520 SYSV_SHM_LOCK();
521
522 if (key == IPC_PRIVATE) {
523 ret = ShmAllocSeg(key, size, shmflg);
524 } else {
525 ret = ShmFindSegByKey(key);//通过key查找资源ID
526 if (ret < 0) {
527 if (((UINT32)shmflg & IPC_CREAT) == 0) {//
528 ret = -ENOENT;
529 goto ERROR;
530 } else {
531 ret = ShmAllocSeg(key, size, shmflg);//分配一个共享内存
532 }
533 } else {
534 shmid = ret;
535 if (((UINT32)shmflg & IPC_CREAT) &&
536 ((UINT32)shmflg & IPC_EXCL)) {
537 ret = -EEXIST;
538 goto ERROR;
539 }
540 ret = ShmPermCheck(ShmFindSeg(shmid), (UINT32)shmflg & ACCESSPERMS);//对共享内存权限检查
541 if (ret != 0) {
542 ret = -ret;
543 goto ERROR;
544 }
545 ret = ShmSegValidCheck(shmid, size, shmflg);
546 }
547 }
548 if (ret < 0) {
549 goto ERROR;
550 }
551
552 SYSV_SHM_UNLOCK();
553
554 return ret;
555ERROR:
556 set_errno(-ret);
557 SYSV_SHM_UNLOCK();
558 PRINT_DEBUG("%s %d, ret = %d\n", __FUNCTION__, __LINE__, ret);
559 return -1;
560}
STATIC INT32 ShmFindSegByKey(key_t key)
通过key查找 shmId
Definition: shm.c:325
STATIC INT32 ShmAllocSeg(key_t key, size_t size, INT32 shmflg)
为共享段分配物理内存 例如:参数size = 4097, LOS_Align(size, PAGE_SIZE) = 8192 分配页数 size >> PAGE_SHIFT = 2页
Definition: shm.c:251
STATIC INT32 ShmSegValidCheck(INT32 segNum, size_t size, INT32 shmFlg)
共享内存段有效性检查
Definition: shm.c:341
函数调用图:
这是这个函数的调用关系图:

◆ ShmInit()

UINT32 ShmInit ( VOID  )

在文件 shm.c166 行定义.

167{
168 UINT32 ret;
169 UINT32 i;
170
171 ret = LOS_MuxInit(&g_sysvShmMux, NULL);//初始化互斥锁
172 if (ret != LOS_OK) {
173 goto ERROR;
174 }
175
176 g_shmSegs = LOS_MemAlloc((VOID *)OS_SYS_MEM_ADDR, sizeof(struct shmIDSource) * g_shmInfo.shmmni);//分配shm段数组
177 if (g_shmSegs == NULL) {
179 goto ERROR;
180 }
181 (VOID)memset_s(g_shmSegs, (sizeof(struct shmIDSource) * g_shmInfo.shmmni),
182 0, (sizeof(struct shmIDSource) * g_shmInfo.shmmni));//数组清零
183
184 for (i = 0; i < g_shmInfo.shmmni; i++) {
185 g_shmSegs[i].status = SHM_SEG_FREE;//节点初始状态为空闲
186 g_shmSegs[i].ds.shm_perm.seq = i + 1;//struct ipc_perm shm_perm;系统为每一个IPC对象保存一个ipc_perm结构体,结构说明了IPC对象的权限和所有者
187 LOS_ListInit(&g_shmSegs[i].node);//初始化节点
188 }
190
191 return LOS_OK;
192
193ERROR:
194 VM_ERR("ShmInit fail\n");
195 return LOS_NOK;
196}
VOID * LOS_MemAlloc(VOID *pool, UINT32 size)
从指定内存池中申请size长度的内存,注意这可不是从内核堆空间中申请内存
Definition: los_memory.c:1123
LITE_OS_SEC_TEXT UINT32 LOS_MuxInit(LosMux *mutex, const LosMuxAttr *attr)
初始化互斥锁
Definition: los_mux.c:262
函数调用图:

◆ ShmPagesRefDec()

STATIC VOID ShmPagesRefDec ( struct shmIDSource seg)

seg下所有共享页引用减少

在文件 shm.c233 行定义.

234{
235 LosVmPage *page = NULL;
236
237 LOS_DL_LIST_FOR_EACH_ENTRY(page, &seg->node, LosVmPage, node) {
238 LOS_AtomicDec(&page->refCounts);
239 }
240}
STATIC INLINE VOID LOS_AtomicDec(Atomic *v)
Atomic auto-decrement. | 对32bit原子数据做减1
Definition: los_atomic.h:323
Atomic refCounts
Definition: los_vm_page.h:57
函数调用图:
这是这个函数的调用关系图:

◆ ShmPermCheck()

STATIC INT32 ShmPermCheck ( struct shmIDSource seg,
mode_t  mode 
)

对共享内存段权限检查

在文件 shm.c455 行定义.

456{
457 INT32 uid = LOS_GetUserID();//当前进程的用户ID
458 UINT32 tmpMode = 0;
459 mode_t privMode = seg->ds.shm_perm.mode;
460 mode_t accMode;
461
462 if ((uid == seg->ds.shm_perm.uid) || (uid == seg->ds.shm_perm.cuid)) {
463 tmpMode |= SHM_M;
464 accMode = mode & S_IRWXU;
465 } else if (LOS_CheckInGroups(seg->ds.shm_perm.gid) ||
467 privMode <<= SHM_GROUPE_TO_USER;
468 accMode = (mode & S_IRWXG) << SHM_GROUPE_TO_USER;
469 } else {
470 privMode <<= SHM_OTHER_TO_USER;
471 accMode = (mode & S_IRWXO) << SHM_OTHER_TO_USER;
472 }
473
474 if (privMode & SHM_R) {
475 tmpMode |= SHM_R;
476 }
477
478 if (privMode & SHM_W) {
479 tmpMode |= SHM_W;
480 }
481
482 if (privMode & SHM_X) {
483 tmpMode |= SHM_X;
484 }
485
486 if ((mode == SHM_M) && (tmpMode & SHM_M)) {
487 return 0;
488 } else if (mode == SHM_M) {
489 return EACCES;
490 }
491
492 if ((tmpMode & accMode) == accMode) {
493 return 0;
494 } else {
495 return EACCES;
496 }
497}
LITE_OS_SEC_TEXT BOOL LOS_CheckInGroups(UINT32 gid)
Definition: los_process.c:805
函数调用图:
这是这个函数的调用关系图:

◆ ShmSegUsedCount()

STATIC INT32 ShmSegUsedCount ( VOID  )

获取共享内存池中已被使用的段数量

在文件 shm.c440 行定义.

441{
442 INT32 i;
443 INT32 count = 0;
444 struct shmIDSource *seg = NULL;
445
446 for (i = 0; i < g_shmInfo.shmmni; i++) {
447 seg = &g_shmSegs[i];
448 if (seg->status & SHM_SEG_USED) {//找到一个
449 count++;
450 }
451 }
452 return count;
453}
这是这个函数的调用关系图:

◆ ShmSegValidCheck()

STATIC INT32 ShmSegValidCheck ( INT32  segNum,
size_t  size,
INT32  shmFlg 
)

共享内存段有效性检查

在文件 shm.c341 行定义.

342{
343 struct shmIDSource *seg = &g_shmSegs[segNum];//拿到shmID
344
345 if (size > seg->ds.shm_segsz) {//段长
346 return -EINVAL;
347 }
348
349 if (((UINT32)shmFlg & (IPC_CREAT | IPC_EXCL)) ==
350 (IPC_CREAT | IPC_EXCL)) {
351 return -EEXIST;
352 }
353
354 return segNum;
355}
这是这个函数的调用关系图:

◆ ShmSetSharedFlag()

STATIC VOID ShmSetSharedFlag ( struct shmIDSource seg)
inline

给共享段中所有物理页框贴上共享标签

在文件 shm.c215 行定义.

216{
217 LosVmPage *page = NULL;
218
219 LOS_DL_LIST_FOR_EACH_ENTRY(page, &seg->node, LosVmPage, node) {
220 OsSetPageShared(page);
221 }
222}
STATIC INLINE VOID OsSetPageShared(LosVmPage *page)
函数调用图:
这是这个函数的调用关系图:

◆ ShmVmmMapping()

STATIC VOID ShmVmmMapping ( LosVmSpace space,
LOS_DL_LIST pageList,
VADDR_T  vaddr,
UINT32  regionFlags 
)

共享内存映射

在文件 shm.c375 行定义.

376{
377 LosVmPage *vmPage = NULL;
378 VADDR_T va = vaddr;
379 PADDR_T pa;
380 STATUS_T ret;
381
382 LOS_DL_LIST_FOR_EACH_ENTRY(vmPage, pageList, LosVmPage, node) {//遍历一页一页映射
383 pa = VM_PAGE_TO_PHYS(vmPage);//拿到物理地址
384 LOS_AtomicInc(&vmPage->refCounts);//自增
385 ret = LOS_ArchMmuMap(&space->archMmu, va, pa, 1, regionFlags);//虚实映射
386 if (ret != 1) {
387 VM_ERR("LOS_ArchMmuMap failed, ret = %d", ret);
388 }
389 va += PAGE_SIZE;
390 }
391}
STATIC INLINE VOID LOS_AtomicInc(Atomic *v)
Atomic addSelf.
Definition: los_atomic.h:253
STATUS_T LOS_ArchMmuMap(LosArchMmu *archMmu, VADDR_T vaddr, PADDR_T paddr, size_t count, UINT32 flags)
LOS_ArchMmuMap 映射进程空间虚拟地址区间与物理地址区间 所谓的map就是生成L1,L2页表项的过程
Definition: los_arch_mmu.c:891
unsigned long PADDR_T
Definition: los_typedef.h:207
int STATUS_T
Definition: los_typedef.h:215
函数调用图:
这是这个函数的调用关系图:

变量说明

◆ g_shmInfo

STATIC struct shminfo g_shmInfo
初始值:
= {
.shmmax = SHM_MAX,
.shmmin = SHM_MIN,
.shmmni = SHM_MNI,
.shmseg = SHM_SEG,
.shmall = SHM_ALL,
}

在文件 shm.c155 行定义.

◆ g_shmSegs

STATIC struct shmIDSource* g_shmSegs = NULL

在文件 shm.c163 行定义.

◆ g_shmUsedPageCount

STATIC UINT32 g_shmUsedPageCount

在文件 shm.c164 行定义.

◆ g_sysvShmMux

STATIC LosMux g_sysvShmMux

在文件 shm.c76 行定义.