Skip to content

Commit 24d8218

Browse files
peffgitster
authored andcommitted
tempfile: use list.h for linked list
The tempfile API keeps to-be-cleaned tempfiles in a singly-linked list and never removes items from the list. A future patch would like to start removing items, but removal from a singly linked list is O(n), as we have to walk the list to find the predecessor element. This means that a process which takes "n" simultaneous lockfiles (for example, an atomic transaction on "n" refs) may end up quadratic in "n". Before we start allowing items to be removed, it would be nice to have a way to cover this case in linear time. The simplest solution is to make an assumption about the order in which tempfiles are added and removed from the list. If both operations iterate over the tempfiles in the same order, then by putting new items at the end of the list our removal search will always find its items at the beginning of the list. And indeed, that would work for the case of refs. But it creates a hidden dependency between unrelated parts of the code. If anybody changes the ref code (or if we add a new caller that opens multiple simultaneous tempfiles) they may unknowingly introduce a performance regression. Another solution is to use a better data structure. A doubly-linked list works fine, and we already have an implementation in list.h. But there's one snag: the elements of "struct tempfile" are all marked as "volatile", since a signal handler may interrupt us and iterate over the list at any moment (even if we were in the middle of adding a new entry). We can declare a "volatile struct list_head", but we can't actually use it with the normal list functions. The compiler complains about passing a pointer-to-volatile via a regular pointer argument. And rightfully so, as the sub-function would potentially need different code to deal with the volatile case. That leaves us with a few options: 1. Drop the "volatile" modifier for the list items. This is probably a bad idea. I checked the assembly output from "gcc -O2", and the "volatile" really does impact the order in which it updates memory. 2. Use macros instead of inline functions. The irony here is that list.h is entirely implemented as trivial inline functions. So we basically are already generating custom code for each call. But sadly there's no way in C to declare the inline function to take a more generic type. We could do so by switching the inline functions to macros, but it does make the end result harder to read. And it doesn't fully solve the problem (for instance, the declaration of list_head needs to change so that its "prev" and "next" pointers point to other volatile structs). 3. Don't use list.h, and just make our own ad-hoc doubly-linked list. It's not that much code to implement the basics that we need here. But if we're going to do so, why not add the few extra lines required to model it after the actual list.h interface? We can even reuse a few of the macro helpers. So this patch takes option 3, but actually implements a parallel "volatile list" interface in list.h, where it could potentially be reused by other code. This implements just enough for tempfile.c's use, though we could easily port other functions later if need be. Signed-off-by: Jeff King <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 102cf7a commit 24d8218

File tree

3 files changed

+48
-7
lines changed

3 files changed

+48
-7
lines changed

list.h

+38
Original file line numberDiff line numberDiff line change
@@ -163,4 +163,42 @@ static inline void list_replace_init(struct list_head *old,
163163
INIT_LIST_HEAD(old);
164164
}
165165

166+
/*
167+
* This is exactly the same as a normal list_head, except that it can be
168+
* declared volatile (e.g., if you have a list that may be accessed from signal
169+
* handlers).
170+
*/
171+
struct volatile_list_head {
172+
volatile struct volatile_list_head *next, *prev;
173+
};
174+
175+
#define VOLATILE_LIST_HEAD(name) \
176+
volatile struct volatile_list_head name = { &(name), &(name) }
177+
178+
static inline void __volatile_list_del(volatile struct volatile_list_head *prev,
179+
volatile struct volatile_list_head *next)
180+
{
181+
next->prev = prev;
182+
prev->next = next;
183+
}
184+
185+
static inline void volatile_list_del(volatile struct volatile_list_head *elem)
186+
{
187+
__volatile_list_del(elem->prev, elem->next);
188+
}
189+
190+
static inline int volatile_list_empty(volatile struct volatile_list_head *head)
191+
{
192+
return head == head->next;
193+
}
194+
195+
static inline void volatile_list_add(volatile struct volatile_list_head *newp,
196+
volatile struct volatile_list_head *head)
197+
{
198+
head->next->prev = newp;
199+
newp->next = head->next;
200+
newp->prev = head;
201+
head->next = newp;
202+
}
203+
166204
#endif /* LIST_H */

tempfile.c

+7-6
Original file line numberDiff line numberDiff line change
@@ -55,14 +55,16 @@
5555
#include "tempfile.h"
5656
#include "sigchain.h"
5757

58-
static struct tempfile *volatile tempfile_list;
58+
static VOLATILE_LIST_HEAD(tempfile_list);
5959

6060
static void remove_tempfiles(int in_signal_handler)
6161
{
6262
pid_t me = getpid();
63-
struct tempfile *volatile p;
63+
volatile struct volatile_list_head *pos;
64+
65+
list_for_each(pos, &tempfile_list) {
66+
struct tempfile *p = list_entry(pos, struct tempfile, list);
6467

65-
for (p = tempfile_list; p; p = p->next) {
6668
if (!is_tempfile_active(p) || p->owner != me)
6769
continue;
6870

@@ -95,7 +97,7 @@ static void remove_tempfiles_on_signal(int signo)
9597
*/
9698
static void prepare_tempfile_object(struct tempfile *tempfile)
9799
{
98-
if (!tempfile_list) {
100+
if (volatile_list_empty(&tempfile_list)) {
99101
/* One-time initialization */
100102
sigchain_push_common(remove_tempfiles_on_signal);
101103
atexit(remove_tempfiles_on_exit);
@@ -110,8 +112,7 @@ static void prepare_tempfile_object(struct tempfile *tempfile)
110112
tempfile->active = 0;
111113
tempfile->owner = 0;
112114
strbuf_init(&tempfile->filename, 0);
113-
tempfile->next = tempfile_list;
114-
tempfile_list = tempfile;
115+
volatile_list_add(&tempfile->list, &tempfile_list);
115116
tempfile->on_list = 1;
116117
} else if (tempfile->filename.len) {
117118
/* This shouldn't happen, but better safe than sorry. */

tempfile.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
#ifndef TEMPFILE_H
22
#define TEMPFILE_H
33

4+
#include "list.h"
5+
46
/*
57
* Handle temporary files.
68
*
@@ -81,7 +83,7 @@
8183
*/
8284

8385
struct tempfile {
84-
struct tempfile *volatile next;
86+
volatile struct volatile_list_head list;
8587
volatile sig_atomic_t active;
8688
volatile int fd;
8789
FILE *volatile fp;

0 commit comments

Comments
 (0)