Kernel version:2.6.14
CPU architecture:ARM920T
Author:ce123(http://blog.csdn.net/ce123)
1.close函数
上图说明晰close(fd)的履行进程,首要包含两部分:开释文件描述符fd,封闭文件file。
[plain]view plaincopy
print?
- //fs/open.c
- asmlinkagelongsys_close(unsignedintfd)
- {
- structfile*filp;
- structfiles_struct*files=current->files;//取得当时进程的files结构
- structfdtable*fdt;
- spin_lock(&files->file_lock);
- fdt=files_fdtable(files);//经过进程的翻开文件列表取得文件描述符位图结构
- if(fd>=fdt->max_fds)
- gotoout_unlock;
- filp=fdt->fd[fd];
- if(!filp)
- gotoout_unlock;
- rcu_assign_pointer(fdt->fd[fd],NULL);
- FD_CLR(fd,fdt->close_on_exec);
- __put_unused_fd(files,fd);//开释文件描述符
- spin_unlock(&files->file_lock);
- returnfilp_close(filp,files);//封闭文件
- out_unlock:
- spin_unlock(&files->file_lock);
- return-EBADF;
- }
2.开释文件描述符__put_unused_fd
[plain]view plaincopy
print?
- staticinlinevoid__put_unused_fd(structfiles_struct*files,unsignedintfd)
- {
- structfdtable*fdt=files_fdtable(files);
- __FD_CLR(fd,fdt->open_fds);//铲除位图中的相应符号
- if(fd
next_fd) - fdt->next_fd=fd;//假如开释的fd小于next_fd,则next_fd=fd,下次分配从next_fd开端。
- //因而开释一个fd后,再翻开或创立一个文件放回的或许仍是刚开释的fd
- }
3.封闭文件filp_close
[plain]view plaincopy
print?
- intfilp_close(structfile*filp,fl_owner_tid)
- {
- intretval=0;
- if(!file_count(filp)){
- printk(KERN_ERR”VFS:Close:filecountis0\n”);
- return0;
- }
- if(filp->f_op&&filp->f_op->flush)
- retval=filp->f_op->flush(filp);
- dnotify_flush(filp,id);
- locks_remove_posix(filp,id);
- fput(filp);
- returnretval;
- }
filp_close函数调用fput,在fput中调用release函数。
[plain]view plaincopy
print?
- //fs/file_table.c
- voidfastcallfput(structfile*file)
- {
- if(rcuref_dec_and_test(&file->f_count))
- __fput(file);
- }
- voidfastcall__fput(structfile*file)
- {
- structdentry*dentry=file->f_dentry;
- structvfsmount*mnt=file->f_vfsmnt;
- structinode*inode=dentry->d_inode;
- might_sleep();
- fsnotify_close(file);
- /*
- *Thefunctioneventpoll_release()shouldbethefirstcalled
- *inthefilecleanupchain.
- */
- eventpoll_release(file);
- locks_remove_flock(file);
- if(file->f_op&&file->f_op->release)
- file->f_op->release(inode,file);//在这里调用release函数。在socket中即socket_close函数
- security_file_free(file);
- if(unlikely(inode->i_cdev!=NULL))
- cdev_put(inode->i_cdev);
- fops_put(file->f_op);
- if(file->f_mode&FMODE_WRITE)
- put_write_access(inode);
- file_kill(file);
- file->f_dentry=NULL;
- file->f_vfsmnt=NULL;
- file_free(file);
- dput(dentry);
- mntput(mnt);
- }