X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~ianmdlvl/git?p=elogind.git;a=blobdiff_plain;f=src%2Fjournal%2Fmmap-cache.c;h=7dbbb5ef381b4c6b617441c4a6dae2c74aae9ca3;hp=7813f037544546fe8bf8e33c85c6768acce001a0;hb=03e334a1c7dc8c20c38902aa039440763acc9b17;hpb=89de694724f376a6852e879fe987e7e531327654 diff --git a/src/journal/mmap-cache.c b/src/journal/mmap-cache.c index 7813f0375..7dbbb5ef3 100644 --- a/src/journal/mmap-cache.c +++ b/src/journal/mmap-cache.c @@ -38,12 +38,12 @@ typedef struct FileDescriptor FileDescriptor; struct Window { MMapCache *cache; - bool keep_always; + unsigned keep_always; bool in_unused; + int prot; void *ptr; uint64_t offset; - int prot; size_t size; FileDescriptor *fd; @@ -70,12 +70,14 @@ struct FileDescriptor { struct MMapCache { int n_ref; + unsigned n_windows; + + unsigned n_hit, n_missed; + Hashmap *fds; Hashmap *contexts; - unsigned n_windows; - LIST_HEAD(Window, unused); Window *last_unused; }; @@ -111,13 +113,13 @@ static void window_unlink(Window *w) { munmap(w->ptr, w->size); if (w->fd) - LIST_REMOVE(Window, by_fd, w->fd->windows, w); + LIST_REMOVE(by_fd, w->fd->windows, w); if (w->in_unused) { if (w->cache->last_unused == w) w->cache->last_unused = w->unused_prev; - LIST_REMOVE(Window, unused, w->cache->unused, w); + LIST_REMOVE(unused, w->cache->unused, w); } LIST_FOREACH(by_window, c, w->contexts) { @@ -134,7 +136,7 @@ static void window_free(Window *w) { free(w); } -static bool window_matches(Window *w, int fd, int prot, uint64_t offset, size_t size) { +_pure_ static bool window_matches(Window *w, int fd, int prot, uint64_t offset, size_t size) { assert(w); assert(fd >= 0); assert(size > 0); @@ -181,11 +183,11 @@ static void context_detach_window(Context *c) { w = c->window; c->window = NULL; - LIST_REMOVE(Context, by_window, w->contexts, c); + LIST_REMOVE(by_window, w->contexts, c); - if (!w->contexts && !w->keep_always) { + if (!w->contexts && w->keep_always == 0) { /* Not used anymore? */ - LIST_PREPEND(Window, unused, c->cache->unused, w); + LIST_PREPEND(unused, c->cache->unused, w); if (!c->cache->last_unused) c->cache->last_unused = w; @@ -204,15 +206,15 @@ static void context_attach_window(Context *c, Window *w) { if (w->in_unused) { /* Used again? */ - LIST_REMOVE(Window, unused, c->cache->unused, w); - if (!c->cache->last_unused) - c->cache->last_unused = w; + LIST_REMOVE(unused, c->cache->unused, w); + if (c->cache->last_unused == w) + c->cache->last_unused = w->unused_prev; w->in_unused = false; } c->window = w; - LIST_PREPEND(Context, by_window, w->contexts, c); + LIST_PREPEND(by_window, w->contexts, c); } static Context *context_add(MMapCache *m, unsigned id) { @@ -308,9 +310,13 @@ static void mmap_cache_free(MMapCache *m) { while ((c = hashmap_first(m->contexts))) context_free(c); + hashmap_free(m->contexts); + while ((f = hashmap_first(m->fds))) fd_free(f); + hashmap_free(m->fds); + while (m->unused) window_free(m->unused); @@ -354,7 +360,6 @@ static int try_context( assert(m->n_ref > 0); assert(fd >= 0); assert(size > 0); - assert(ret); c = hashmap_get(m->contexts, UINT_TO_PTR(context+1)); if (!c) @@ -372,9 +377,10 @@ static int try_context( return 0; } - c->window->keep_always = c->window->keep_always || keep_always; + c->window->keep_always += keep_always; - *ret = (uint8_t*) c->window->ptr + (offset - c->window->offset); + if (ret) + *ret = (uint8_t*) c->window->ptr + (offset - c->window->offset); return 1; } @@ -396,7 +402,6 @@ static int find_mmap( assert(m->n_ref > 0); assert(fd >= 0); assert(size > 0); - assert(ret); f = hashmap_get(m->fds, INT_TO_PTR(fd + 1)); if (!f) @@ -416,9 +421,10 @@ static int find_mmap( return -ENOMEM; context_attach_window(c, w); - w->keep_always = w->keep_always || keep_always; + w->keep_always += keep_always; - *ret = (uint8_t*) w->ptr + (offset - w->offset); + if (ret) + *ret = (uint8_t*) w->ptr + (offset - w->offset); return 1; } @@ -444,7 +450,6 @@ static int add_mmap( assert(m->n_ref > 0); assert(fd >= 0); assert(size > 0); - assert(ret); woffset = offset & ~((uint64_t) page_size() - 1ULL); wsize = size + (offset - woffset); @@ -508,13 +513,14 @@ static int add_mmap( w->size = wsize; w->fd = f; - LIST_PREPEND(Window, by_fd, f->windows, w); + LIST_PREPEND(by_fd, f->windows, w); context_detach_window(c); c->window = w; - LIST_PREPEND(Context, by_window, w->contexts, c); + LIST_PREPEND(by_window, w->contexts, c); - *ret = (uint8_t*) w->ptr + (offset - w->offset); + if (ret) + *ret = (uint8_t*) w->ptr + (offset - w->offset); return 1; } @@ -535,22 +541,63 @@ int mmap_cache_get( assert(m->n_ref > 0); assert(fd >= 0); assert(size > 0); - assert(ret); /* Check whether the current context is the right one already */ r = try_context(m, fd, prot, context, keep_always, offset, size, ret); - if (r != 0) + if (r != 0) { + m->n_hit ++; return r; + } /* Search for a matching mmap */ r = find_mmap(m, fd, prot, context, keep_always, offset, size, ret); - if (r != 0) + if (r != 0) { + m->n_hit ++; return r; + } + + m->n_missed++; /* Create a new mmap */ return add_mmap(m, fd, prot, context, keep_always, offset, size, st, ret); } +int mmap_cache_release( + MMapCache *m, + int fd, + int prot, + unsigned context, + uint64_t offset, + size_t size) { + + FileDescriptor *f; + Window *w; + + assert(m); + assert(m->n_ref > 0); + assert(fd >= 0); + assert(size > 0); + + f = hashmap_get(m->fds, INT_TO_PTR(fd + 1)); + if (!f) + return -EBADF; + + assert(f->fd == fd); + + LIST_FOREACH(by_fd, w, f->windows) + if (window_matches(w, fd, prot, offset, size)) + break; + + if (!w) + return -ENOENT; + + if (w->keep_always == 0) + return -ENOLCK; + + w->keep_always -= 1; + return 0; +} + void mmap_cache_close_fd(MMapCache *m, int fd) { FileDescriptor *f; @@ -575,3 +622,15 @@ void mmap_cache_close_context(MMapCache *m, unsigned context) { context_free(c); } + +unsigned mmap_cache_get_hit(MMapCache *m) { + assert(m); + + return m->n_hit; +} + +unsigned mmap_cache_get_missed(MMapCache *m) { + assert(m); + + return m->n_missed; +}