普通用户要用IPC_CREATE|0666才能运行代码
代码 code1.c:
1 #include <sys/ipc.h> 2 #include <sys/shm.h> 3 #include <sys/types.h> 4 #include <unistd.h> 5 #include <stdio.h> 6 #include <string.h> 7 8 9 typedef struct 10 { 11 char name[4]; 12 int age; 13 } people; 14 15 int main(int argc, char** argv) 16 { 17 int shm_id,i; 18 key_t key; 19 char temp; 20 people *p_map; 21 char* name = "/dev/shm/myshm2"; 22 key = ftok(name,0); /*调用ftok 函数,产生标准的key*/ 23 shm_id=shmget(key,4096,IPC_CREAT|0666);/*调用shmget 函数,获取共享内存区域的ID*/ 24 if(shm_id==-1) 25 { 26 perror("获取共享内存区域的ID 出错"); 27 return -1; 28 } 29 p_map=(people*)shmat(shm_id,NULL,0);/* 调用shmat 函数,映射共享内存*/ 30 temp='a'; 31 for(i = 0;i<10;i++) 32 { 33 temp+=1; 34 memcpy((*(p_map+i)).name,&temp,1); 35 (*(p_map+i)).age=20+i; 36 } 37 if(shmdt(p_map)==-1) /*调用shmdt 函数,解除进程对共享内存区域的映射*/ 38 perror("解除映射出错"); 39 return 0; 40 }
(1) 如果没有/dev/shm下没有myshm2,则需要自己创建一个myshm2文件
(2) 第23行的
shm_id=shmget(key,4096,IPC_CREAT|0666)
如果不加0666,那么普通用户运行这段代码就会报错
但是后来加上0666还是继续报错,这是为什么呢?
因为原来没有加0666的时候创建出来的shmid没有被删掉,由于ftok的两个参数和原来一样,所以后来运行代码并没有修改掉key的值,所以并没有创建新的shmid,而是用的原来那段没加0666的代码运行产生的shmid,所以仍然是没有权限的,解决办法就是用ipcs 命令先找到这个shmid,然后再用ipcrm -m shmid 删掉这个shmid,之后再次运行那段加了0666的代码即可,此时创建出来的shmid对于普通用户就有访问权限了。