/* *------------------------------------------------------------------ * svmtool.c * * Copyright (c) 2009 Cisco and/or its affiliates. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. *------------------------------------------------------------------ */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "svm.h" /* * format_all_svm_regions * Maps / unmaps regions. Do NOT call from client code! */ u8 *format_all_svm_regions (u8 *s, va_list * args) { int verbose = va_arg (*args, int); svm_region_t *root_rp = svm_get_root_rp(); svm_main_region_t *mp; svm_subregion_t *subp; svm_region_t *rp; svm_map_region_args_t *a = 0; u8 ** svm_names=0; u8 *name=0; int i; ASSERT(root_rp); pthread_mutex_lock (&root_rp->mutex); s = format (s, "%U", format_svm_region, root_rp, verbose); mp = root_rp->data_base; /* * Snapshoot names, can't hold root rp mutex across * find_or_create. */ pool_foreach (subp, mp->subregions, ({ name = vec_dup (subp->subregion_name); vec_add1(svm_names, name); })); pthread_mutex_unlock (&root_rp->mutex); for (i = 0; i < vec_len(svm_names); i++) { vec_validate(a, 0); a->name = (char *) svm_names[i]; rp = svm_region_find_or_create (a); if (rp) { pthread_mutex_lock (&rp->mutex); s = format (s, "%U", format_svm_region, rp, verbose); pthread_mutex_unlock (&rp->mutex); svm_region_unmap (rp); vec_free(svm_names[i]); } vec_free (a); } vec_free(svm_names); return (s); } void show (char *chroot_path, int verbose) { svm_map_region_args_t *a = 0; vec_validate (a, 0); svm_region_init_chroot(chroot_path); fformat(stdout, "My pid is %d\n", getpid()); fformat(stdout, "%U", format_all_svm_regions, verbose); svm_region_exit (); vec_free (a); } static void *svm_map_region_nolock (svm_map_region_args_t *a) { int svm_fd; svm_region_t *rp; int deadman=0; u8 *shm_name; ASSERT((a->size & ~(MMAP_PAGESIZE-1)) == a->size); shm_name = shm_name_from_svm_map_region_args (a); svm_fd = shm_open((char *)shm_name, O_RDWR, 0777); if (svm_fd < 0) { perror("svm_region_map(mmap open)"); return (0); } vec_free (shm_name); rp = mmap(0, MMAP_PAGESIZE, PROT_READ | PROT_WRITE, MAP_SHARED, svm_fd, 0); if (rp == (svm_region_t *) MAP_FAILED) { close(svm_fd); clib_warning("mmap"); return (0); } /* * We lost the footrace to create this region; make sure * the winner has crossed the finish line. */ while (rp->version == 0 && deadman++ < 5) { sleep(1); } /* * -ed? */ if (rp->version == 0) { clib_warning("rp->version %d not %d", rp->version, SVM_VERSION); return (0); } /* Remap now that the region has been placed */ a->baseva = rp->virtual_base; a->size = rp->virtual_size; munmap(rp, MMAP_PAGESIZE); rp = (void *) mmap ((void *)a->baseva, a->size, PROT_READ | PROT_WRITE, MAP_SHARED | MAP_FIXED, svm_fd, 0); if ((uword)rp == (uword)MAP_FAILED) { clib_unix_warning ("mmap"); return (0); } if ((uword) rp != rp->virtual_base) { clib_warning("mmap botch"); } if (pthread_mutex_trylock(&rp->mutex)) { clib_warning ("rp->mutex LOCKED by pid %d, tag %d, cleared...", rp->mutex_owner_pid, rp->mutex_owner_tag); memset(&rp->mutex, 0, sizeof (rp->mutex)); } else { clib_warning ("mutex OK...\n"); pthread_mutex_unlock(&rp->mutex); } return ((void *) rp); } /* * rnd_pagesize * Round to a pagesize multiple, presumably 4k works */ static unsigned int rnd_pagesize(unsigned int size) { unsigned int rv; rv = (size + (MMAP_PAGESIZE-1)) & ~(MMAP_PAGESIZE-1); return(rv); } #define MUTEX_DEBUG always_inline void region_lock(svm_region_t *rp, int tag) { pthread_mutex_lock(&rp->mutex); #ifdef MUTEX_DEBUG rp->mutex_owner_pid = getpid(); rp->mutex_owner_tag = tag; #endif } always_inline void region_unlock(svm_region_t *rp) { #ifdef MUTEX_DEBUG rp->mutex_owner_pid = 0; rp->mutex_owner_tag = 0; #endif pthread_mutex_unlock(&rp->mutex); } static void *svm_existing_region_map_nolock (void *root_arg, svm_map_region_args_t *a) { svm_region_t *root_rp = root_arg; svm_main_region_t *mp; svm_region_t *rp; void *oldheap; uword *p; a->size += MMAP_PAGESIZE + SVM_PVT_MHEAP_SIZE; a->size = rnd_pagesize(a->size); region_lock (root_rp, 4); oldheap = svm_push_pvt_heap(root_rp); mp = root_rp->data_base; ASSERT(mp); p = hash_get_mem (mp->name_hash, a->name); if (p) { rp = svm_map_region_nolock (a); region_unlock(root_rp); svm_pop_heap (oldheap);