ftok产生的key冲突了

pinyue 2014-08-29 11:29:41
最近遇到一个问题,在用户b下创建共享内存失败,跟踪代码发现shmget的errno为17(要创建的这个共享内存已经存在了),可是在该用户下通过ipcs查看确实没有共享内存啊,后来发现,用户a下已经创建的共享内存与用户b要创建的共享内存冲突了,准确的说是key值冲突了,key值是通过ftok函数生成的。

上网查询了一个,ftok是根据文件i节点和调用ftok时的id值产生的,而且还给出了例子加以说明,如下:
在一般的UNIX实现中,是将文件的索引节点号取出,前面加上子序号得到key_t的返回值。如指定文件的索引节点号为65538,
换算成16进制为 0x010002,而你指定的ID值为38,换算成16进制为0x26,则最后的key_t返回值为0x26010002。

b用户下有目录trnlog,通过ls -i命令查看tt的i节点为 2228246
ftok 得到的key值为 16908310 <==> 0x1020016

a用户下有文件comm,通过ls -i命令查看xx的i节点为 2097174
ftok 得到的key值为 16908310 <==> 0x1020016

问题来了:
1)两个用户下的文件i节点不同,通过ftok函数得到的key值是相同的,为什么呢?
2)用户a创建的共享内存对其他用户有影响吗?也就是说进程创建共享内存的时候系统会全系统检查该key值有没有创建过共享内存?
小弟恳请大神点拨说明,感谢万分!
...全文
467 4 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
4 条回复
切换为时间正序
请发表友善的回复…
发表回复
pinyue 2014-08-29
  • 打赏
  • 举报
回复
用户a和b 文件i节点及程序调试查看ftok产生的key值如下 用户a:
a-/home/a/etc/ipckey>ls -li
2097174 -rwxrw-r--. 1 a a    0 Sep 22  2013 comm
程序调试:
64		sprintf(strPath,"%s/etc/ipckey/comm",getenv("HOME"));
(gdb) p strPath
$1 = "/home/a/etc/ipckey/comm", '\000' <repeats 91 times>
(gdb) n
67		if ((ilShareKey = ftok(strPath,1)) == (key_t)-1)
(gdb) p ilShareKey
$2 = 16908310
用户b:
b-/home/b/etc/ipckey>ls -li
2228246 drwxr-xr-x. 2 b b 4096 Sep 22  2013 trnlog
程序调试:
64		sprintf(strPath,"%s/etc/ipckey/trnlog",getenv("HOME"));
(gdb) p strPath
$1 = "/home/b/etc/ipckey/trnlog", '\000' <repeats 89 times>
(gdb) n
67		if ((ilShareKey = ftok(strPath,1)) == (key_t)-1)
(gdb) p ilShareKey
$2 = 16908310
雨焰 2014-08-29
  • 打赏
  • 举报
回复
ftok就是通过文件的一些属性加上id值得到的key值,的确存在不同文件同一id值生成相同key值的可以,比如我遇到的这种情况。
雨焰 2014-08-29
  • 打赏
  • 举报
回复
亲,恭喜,恭喜啊!问题解决了! 赞一个!恩,不错!继续努力!
pinyue 2014-08-29
  • 打赏
  • 举报
回复
好吧,没人解答我来说说吧! 翻看了ftok的实现源码,得知ftok就是通过文件的一些属性加上id值得到的key值,的确存在不同文件同一id值生成相同key值的可以,比如我遇到的这种情况。 先列出ftok的实现,精简如下:
key = ((st.st_ino & 0xffff) | ((st.st_dev & 0xff) << 16) | ((proj_id & 0xff) << 24));
用户a:
a-/home/a/etc/ipckey>stat -c "%i %d" comm
2097174 64770
用户b:
b-/home/b/etc/ipckey>stat -c "%i %d" trnlog
2228246 64770
按照ftok的实现经过计算,的确得出相同的结果。 所以说,ftok的使用不只是大家所说的,如果生成key值的文件在使用,不允许删除重建,否则会出现意想不到的问题;而且,相同的机器,我们也需要注意我遇到的这种情况。 通过以上,我明白了ftok产生的问题,让我纠结了几天的问题。 ~~~结贴~~~
共享内存是系统出于多个进程之间通讯的考虑,而预留的的一块内存区。在/proc/sys/kernel/目录下,记录着共享内存的一些限制,如一个共享内存区的最大字节数shmmax,系统范围内最大共享内存区标识符数shmmni等,可以手工对其调整,但不推荐这样做。 一、应用 共享内存的使用,主要有以下几个API:ftok()、shmget()、shmat()、shmdt()及shmctl()。 1)用ftok()函数获得一个ID号. 应用说明: 在IPC中,我们经常用用key_t的值来创建或者打开信号量,共享内存和消息队列。 函数原型: key_t ftok(const char *pathname, int proj_id); Keys: 1)pathname一定要在系统中存在并且进程能够访问的 3)proj_id是一个1-255之间的一个整数值,典型的值是一个ASCII值。 当成功执行的时候,一个key_t值将会被返回,否则-1被返回。我们可以使用strerror(errno)来确定具体的错误信息。 考虑到应用系统可能在不同的主机上应用,可以直接定义一个key,而不用ftok获得: #define IPCKEY 0x344378 2)shmget()用来开辟/指向一块共享内存的函数 应用说明: shmget()用来获得共享内存区域的ID,如果不存在指定的共享区域就创建相应的区域。 函数原型: int shmget(key_t key, size_t size, int shmflg); key_t key 是这块共享内存的标识符。如果是父子关系的进程间通信的话,这个标识符用IPC_PRIVATE来代替。如果两个进程没有任何关系,所以就用ftok()算出来一个标识符(或者自己定义一个)使用了。 int size 是这块内存的大小. int flag 是这块内存的模式(mode)以及权限标识。 模式可取如下值: IPC_CREAT 新建(如果已创建则返回目前共享内存的id) IPC_EXCL 与IPC_CREAT结合使用,如果已创建则则返回错误 然后将“模式” 和“权限标识”进行“或”运算,做为第三个参数。 如: IPC_CREAT | IPC_EXCL | 0640 例子中的0666为权限标识,4/2/1 分别表示读/写/执行3种权限,第一个0是UID,第一个6(4+2)表示拥有者的权限,第二个4表示同组权限,第3个0表示他人的权限。 这个函数成功时返回共享内存的ID,失败时返回-1。 关于这个函数,要多说两句。 创建共享内存时,shmflg参数至少需要 IPC_CREAT | 权限标识,如果只有IPC_CREAT 则申请的地址都是k=0xffffffff,不能使用; 获取已创建的共享内存时,shmflg不要用IPC_CREAT(只能用创建共享内存时的权限标识,如0640),否则在某些情况下,比如用ipcrm删除共享内存后,用该函数并用IPC_CREAT参数获取一次共享内存(当然,获取失败),则即使再次创建共享内存也不能成功,此时必须更改key来重建共享内存。 3) shmat()将这个内存区映射到本进程的虚拟地址空间。 函数原型: void *shmat( int shmid , char *shmaddr , int shmflag ); shmat()是用来允许本进程访问一块共享内存的函数。 int shmid是那块共享内存的ID。 char *shmaddr是共享内存的起始地址,如果shmaddr为0,内核会把共享内存映像到调用进程的地址空间中选定位置;如果shmaddr不为0,内核会把共享内存映像到shmaddr指定的位置。所以一般把shmaddr设为0。 int shmflag是本进程对该内存的操作模式。如果是SHM_RDONLY的话,就是只读模式。其它的是读写模式 成功时,这个函数返回共享内存的起始地址。失败时返回-1。 4) shmdt()函数删除本进程对这块内存的使用,shmdt()与shmat()相反,是用来禁止本进程访问一块共享内存的函数。 函数原型: int shmdt( char *shmaddr ); 参数char *shmaddr是那块共享内存的起始地址。 成功时返回0。失败时返回-1。 5) shmctl() 控制对这块共享内存的使用 函数原型: int shmctl( int shmid , int cmd , struct shmid_ds *buf ); int shmid是共享内存的ID。 int cmd是控制命令,可取值如下: IPC_STAT 得到共享内存的状态 IPC_SET 改变共享内存的状态 IPC_RMID 删除共享内存 struct shmid_ds *buf是一个结构体指针。IPC_STAT的时候,取得的状态放在这个结构体中。如果要改变共享内存的状态,用这个结构体指定。 返回值: 成功:0 失败:-1

70,020

社区成员

发帖
与我相关
我的任务
社区描述
C语言相关问题讨论
社区管理员
  • C语言
  • 花神庙码农
  • 架构师李肯
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧