From: Roland McGrath Date: Thu, 19 Mar 2015 22:25:23 +0000 (-0700) Subject: Rewritten. X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=5c1b33d5d0d58031068f652d7950c2fd33cca832;p=thirdparty%2Fglibc.git Rewritten. --- diff --git a/sysdeps/nacl/dl-unmap-segments.h b/sysdeps/nacl/dl-unmap-segments.h index f1043ee23ce..02851e7dc30 100644 --- a/sysdeps/nacl/dl-unmap-segments.h +++ b/sysdeps/nacl/dl-unmap-segments.h @@ -22,36 +22,44 @@ #include #include -/* There is always a big gap between the executable segment and the - data segments. Other code segments and data pages lie in there. - So we must unmap each segment individually. */ +/* There is always a big gap between the executable segment and the data + segments. Other code segments and data pages lie in there. So we must + unmap each segment individually (except for a codeless module). */ static void __always_inline _dl_unmap_segments (struct link_map *l) { - /* Normally l_phdr points into the RODATA segment, which we will unmap in - one iteration of the loop. So we cannot use it directly throughout. */ - - struct { void *start; size_t size; } segments[l->l_phnum], *seg = segments; - - for (const ElfW(Phdr) *ph = l->l_phdr; ph < &l->l_phdr[l->l_phnum]; ++ph) - if (ph->p_type == PT_LOAD) - { - seg->start = (void *) (l->l_addr + ph->p_vaddr); - seg->size = l->l_map_end - l->l_map_start; - if (seg > segments && seg[-1].start + seg[-1].size == seg->start) - /* Coalesce two adjacent segments into one munmap call. */ - seg[-1].size += seg->size; - else - ++seg; - } - - do + if (l->l_contiguous) + /* Simple case. */ + __munmap ((void *) l->l_map_start, l->l_map_end - l->l_map_start); + else { - --seg; - __munmap (seg->start, seg->size); + /* Normally l_phdr points into the RODATA segment, which we will + unmap in one iteration of the loop. So we cannot use it directly + throughout. */ + + struct { ElfW(Addr) start, end; } segments[l->l_phnum], *seg = segments; + + for (const ElfW(Phdr) *ph = l->l_phdr; ph < &l->l_phdr[l->l_phnum]; ++ph) + if (ph->p_type == PT_LOAD) + { + seg->start = (l->l_addr + ph->p_vaddr) & -GLRO(dl_pagesize); + seg->end = (l->l_addr + ph->p_vaddr + ph->p_memsz + + GLRO(dl_pagesize) - 1) & -GLRO(dl_pagesize); + if (seg > segments && seg[-1].end == seg->start) + /* Coalesce two adjacent segments into one munmap call. */ + seg[-1].end = seg->end; + else + ++seg; + } + + do + { + --seg; + __munmap ((void *) seg->start, seg->end - seg->start); + } + while (seg > segments); } - while (seg > segments); } #endif /* dl-unmap-segments.h */