Kernel version:2.6.14
CPU architecture:ARM920T
Author:ce123(http://blog.csdn.net/ce123)
在linux内核中首要有两个函数涉及到文件描述符的分配:get_unused_fd和locate_fd。本文首要解说get_unused_fd,将会鄙人一篇文章中介绍locate_fd。首要给出get_unused_fd的界说(fs/open.c):
[plain]view plaincopy
print?
- intget_unused_fd(void)
- {
- structfiles_struct*files=current->files;//取得当时进程的翻开文件列表files
- intfd,error;
- structfdtable*fdt;
- error=-EMFILE;
- spin_lock(&files->file_lock);
- repeat:
- fdt=files_fdtable(files);//取得文件描述符位图结构
- fd=find_next_zero_bit(fdt->open_fds->fds_bits,
- fdt->max_fdset,
- fdt->next_fd);
- //find_next_zero_bit函数在文件描述符位图fds_bits中从next_fd位开端查找下一个(包含next_fd)为0的位,也便是分配一个文教描述符
- /*
- *N.B.Forclonetaskssharingafilesstructure,thistest
- *willlimitthetotalnumberoffilesthatcanbeopened.
- */
- if(fd>=current->signal->rlim[RLIMIT_NOFILE].rlim_cur)//查看是否超越当时进程限制的最大可翻开文件数
- gotoout;
- /*Doweneedtoexpandthefdarrayorfdset?*/
- error=expand_files(files,fd);//根据需要扩展fd,稍后咱们会具体介绍该函数。返回值<0,过错;返回值>0,扩展后再次进行fd的分配
- if(error<0)
- gotoout;
- if(error){
- /*
- *Ifweneededtoexpandthefsarraywe
- *mighthaveblocked-tryagain.
- */
- error=-EMFILE;
- gotorepeat;//之前进行了扩展操作,从头进行一次闲暇fd的分配
- }
- FD_SET(fd,fdt->open_fds);//在open_fds的位图上置位
- FD_CLR(fd,fdt->close_on_exec);
- fdt->next_fd=fd+1;//next_fd加1
- #if1
- /*Sanitycheck*/
- if(fdt->fd[fd]!=NULL){
- printk(KERN_WARNING”get_unused_fd:slot%dnotNULL!\n”,fd);
- fdt->fd[fd]=NULL;
- }
- #endif
- error=fd;
- out:
- spin_unlock(&files->file_lock);
- returnerror;
- }
current->signal->rlim[RLIMIT_NOFILE].rlim_cur是一个进程能够翻开的最大文件数量。咱们首要来看RLIMIT_NOFILE,该值界说如下:
[plain]view plaincopy
print?
- #defineRLIMIT_NOFILE7/*maxnumberofopenfiles*/
在signal结构中,rlim是struct rlimit类型的数组,
[plain]view plaincopy
print?
- structsignal_struct{
- …
- structrlimitrlim[RLIM_NLIMITS];
- …
- };
struct rlimit界说如下
[plain]view plaincopy
print?
- structrlimit{
- unsignedlongrlim_cur;//当时值
- unsignedlongrlim_max;//最大值
- };
这些值时是在哪设定的呢?咱们应该知道,linux内核经过fork创立进程,第一个进程是静态界说的。因而,假如进程创立后没有修正这些值,那么这些和第一个进程中的值应该是相同的。下面是第一个进程的task_struct结构,仅列出部分数据。
[plain]view plaincopy
print?
- linux/arch/arm/kernel/init_task.c
- structtask_structinit_task=INIT_TASK(init_task);
- #defineINIT_TASK(tsk)\
- {\
- …
- .signal=&init_signals,\
- …
- }
init_signals的界说如下:
[plain]view plaincopy
print?
- #defineINIT_SIGNALS(sig){\
- .count=ATOMIC_INIT(1),\
- .wait_chldexit=__WAIT_QUEUE_HEAD_INITIALIZER(sig.wait_chldexit),\
- .shared_pending={\
- .list=LIST_HEAD_INIT(sig.shared_pending.list),\
- .signal={{0}}},\
- .posix_timers=LIST_HEAD_INIT(sig.posix_timers),\
- .cpu_timers=INIT_CPU_TIMERS(sig.cpu_timers),\
- .rlim=INIT_RLIMITS,\
- }
- include\asm-generic\resource.h
- #defineINIT_RLIMITS\
- {\
- [RLIMIT_CPU]={RLIM_INFINITY,RLIM_INFINITY},\
- [RLIMIT_FSIZE]={RLIM_INFINITY,RLIM_INFINITY},\
- [RLIMIT_DATA]={RLIM_INFINITY,RLIM_INFINITY},\
- [RLIMIT_STACK]={_STK_LIM,_STK_LIM_MAX},\
- [RLIMIT_CORE]={0,RLIM_INFINITY},\
- [RLIMIT_RSS]={RLIM_INFINITY,RLIM_INFINITY},\
- [RLIMIT_NPROC]={0,0},\
- [RLIMIT_NOFILE]={INR_OPEN,INR_OPEN},\
- [RLIMIT_MEMLOCK]={MLOCK_LIMIT,MLOCK_LIMIT},\
- [RLIMIT_AS]={RLIM_INFINITY,RLIM_INFINITY},\
- [RLIMIT_LOCKS]={RLIM_INFINITY,RLIM_INFINITY},\
- [RLIMIT_SIGPENDING]={0,0},\
- [RLIMIT_MSGQUEUE]={MQ_BYTES_MAX,MQ_BYTES_MAX},\
- [RLIMIT_NICE]={0,0},\
- [RLIMIT_RTPRIO]={0,0},\
- }
- #defineNR_OPEN(1024*1024)/*Absoluteupperlimitonfdnum*/
- #defineINR_OPEN1024/*Initialsettingfornfilerlimits*/
从上面的代码咱们能够看到rlim_cur = 1024,也便是说进程最多能够翻开1024个文件。