linux内核中的文件描述符(六)--fd的分配--expand_files
时间:11-22
来源:互联网
点击:
Kernel version:2.6.14
CPU architecture:ARM920T
Author:ce123(http://blog.csdn.net/ce123)
我们先贴出expand_files函数的源码:
[plain]view plaincopyprint?
- intexpand_files(structfiles_struct*files,intnr)
- {
- interr,expand=0;
- structfdtable*fdt;
- fdt=files_fdtable(files);
- if(nr>=fdt->max_fdset||nr>=fdt->max_fds){//我们在前面的文章中已经分析过,初始时max_fdset=1024,max_fds=32
- if(fdt->max_fdset>=NR_OPEN||//#defineNR_OPEN(1024*1024)/*Absoluteupperlimitonfdnum*/
- fdt->max_fds>=NR_OPEN||nr>=NR_OPEN){
- err=-EMFILE;//max_fdset和max_fds都不能大于NR_OPEN,否则返回-EMFILE,即打开太多的文件
- gotoout;
- }
- expand=1;
- if((err=expand_fdtable(files,nr)))//真正进行扩展
- gotoout;
- }
- err=expand;
- out:
- returnerr;
- }
expand_files函数进行一些检查后调用expand_fdtable进行文件描述符表的扩展,下面分析expand_fdtable函数。
[plain]view plaincopyprint?
- staticintexpand_fdtable(structfiles_struct*files,intnr)
- __releases(files->file_lock)
- __acquires(files->file_lock)
- {
- interror=0;
- structfdtable*fdt;
- structfdtable*nfdt=NULL;
- spin_unlock(&files->file_lock);
- nfdt=alloc_fdtable(nr);//根据nr重新创建一个新的fdtable
- if(!nfdt){
- error=-ENOMEM;
- spin_lock(&files->file_lock);
- gotoout;
- }
- spin_lock(&files->file_lock);
- fdt=files_fdtable(files);
- /*
- *Checkagainsinceanothertaskmayhaveexpandedthe
- *fdtablewhilewedroppedthelock
- */
- if(nr>=fdt->max_fds||nr>=fdt->max_fdset){//nr值必须大于max_fds和max_fdset值,这里再次进行检查是防止另一个进程进行了expand
- copy_fdtable(nfdt,fdt);//将旧的fdtable中的内容拷贝至新的fdtable
- }else{
- /*Somebodyexpandedwhilewedroppedfile_lock*/
- spin_unlock(&files->file_lock);
- __free_fdtable(nfdt);
- spin_lock(&files->file_lock);
- gotoout;
- }
- rcu_assign_pointer(files->fdt,nfdt);//用新的fdtable替换旧的fdtable
- free_fdtable(fdt);//释放旧的fdtable
- out:
- returnerror;
- }
[plain]view plaincopyprint?
- staticstructfdtable*alloc_fdtable(intnr)
- {
- structfdtable*fdt=NULL;
- intnfds=0;
- fd_set*new_openset=NULL,*new_execset=NULL;
- structfile**new_fds;
- fdt=kmalloc(sizeof(*fdt),GFP_KERNEL);
- if(!fdt)
- gotoout;
- memset(fdt,0,sizeof(*fdt));
- nfds=__FD_SETSIZE;//#define__FD_SETSIZE1024
- //#definePAGE_SHIFT12
- //#definePAGE_SIZE(1UL
- /*Expandtothemaxineasysteps*/
- do{
- if(nfds<(PAGE_SIZE*8))//dfds=1024
- nfds=PAGE_SIZE*8;
- else{
- nfds=nfds*2;
- if(nfds>NR_OPEN)
- nfds=NR_OPEN;
- }
- }while(nfds<=nr);//第一次expand时,nr应该等于32
- new_openset=alloc_fdset(nfds);//分配打开文件位图
- new_execset=alloc_fdset(nfds);
- if(!new_openset||!new_execset)
- gotoout;
- fdt->open_fds=new_openset;
- fdt->close_on_exec=new_execset;
- fdt->max_fdset=nfds;//更新max_fdset值,此时这个值为32k
- nfds=NR_OPEN_DEFAULT;//nfds=32
- /*
- *Expandtothemaxineasysteps,andkeepexpandingituntil
- *wehaveenoughfortherequestedfdarraysize.
- */
- do{
- #ifNR_OPEN_DEFAULT<256
- if(nfds<256)
- nfds=256;//nfds=256(32->256->1024)
- //无法超过1024,因为在最开始的就进行了检查,一定要小于current->signal->rlim[RLIMIT_NOFILE].rlim_cur)
- else
- #endif
- if(nfds<(PAGE_SIZE/sizeof(structfile*)))
- nfds=PAGE_SIZE/sizeof(structfile*);
- else{
- nfds=nfds*2;
- if(nfds>NR_OPEN)
- nfds=NR_OPEN;
- }
- }while(nfds<=nr);
- new_fds=alloc_fd_array(nfds);//分配文件描述符数组
- if(!new_fds)
- gotoout;
- fdt->fd=new_fds;
- fdt->max_fds=nfds;//更新max_fds
- fdt->free_files=NULL;
- returnfdt;
- out:
- if(new_openset)
- free_fdset(new_openset,nfds);
- if(new_execset)
- free_fdset(new_execset,nfds);
- kfree(fdt);
- returnNULL;
- }
linux内核中文件描述 相关文章:
- Windows CE 进程、线程和内存管理(11-09)
- RedHatLinux新手入门教程(5)(11-12)
- uClinux介绍(11-09)
- openwebmailV1.60安装教学(11-12)
- Linux嵌入式系统开发平台选型探讨(11-09)
- Windows CE 进程、线程和内存管理(二)(11-09)