Date: Thu, 16 Aug 2007 00:01:18 -0700 (PDT) From: Jeff Roberson <jroberson@chesapeake.net> To: arch@freebsd.org Subject: file locking. Message-ID: <20070815233852.X568@10.0.0.1>
next in thread | raw e-mail | index | archive | help
I have been looking at file locking for 8.0 and have come up with a way to completely eliminate the file lock, reduce the size of struct file by ~4 pointers (22%), remove the global list of files and associated lock, and restrict the scope of unp_gc while removing several race conditions. The whole thing hinges on reducing the complexity and scope of unp_gc to remove several fields from struct file. The remaining parts can be protected by atomics or are already protected by other locks. f_count and f_type are now completely updated using atomics. The ref counting with atomics results in significantly fewer atomics and cheaper fhold/fdrop. Protecting f_type was only complicated in cases where there were compound operations done on it, which are now implemented with atomic_cmpset_int loops. The unix domain socket garbage collection was changed to scan the list of unp sockets rather than the list of all files. This code is only responsible for finding dead cycles of unp sockets which reference each other. Evaluating other descriptors is not necessary. This allowed me to move f_gcflag and f_msgcount into unpcb. This also removed a use of the global filelist allowing me to remove two pointers from struct file. The only negative part of the new algorithm is that a back-pointer to the referencing struct file must be stored in any unix domain socket that is referenced via another in a rights message. It is a slight layering violation but there is only ever 1 file for each unix domain socket so it is correct. This is required because the garbage collection algorithm needs to know about the external references via the file. The patch is available at: http://people.freebsd.org/~jeff/fd.diff pho and kris have both tested this and found it to be stable. Kris has done some performance measurement and found it to be a win on microbenchmarks. I can't imagine a case that would actually be slower, except perhaps endless loops of sysctl kern.file. This also removes several sources of contention for multithreaded applications in particular, but also a global lock on allocating/freeing files. This also resolves several cases where f_flag was not protected when it should be, as well as removing race conditions in the garbage collector code due to dropping locks, and fixing unprotected variables in the garbage collection code. I intend to commit this soon after the 7.0 branch is made. I also have the final revision of my improved select locking ready. Thanks, Jeff
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?20070815233852.X568>