blob: bca237920591bde8e80487716bfe2ceee4b79d10 [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
2 *------------------------------------------------------------------
3 * svm.h - shared VM allocation, mmap(...MAP_FIXED...)
4 * brain police
5 *
6 * Copyright (c) 2009 Cisco and/or its affiliates.
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at:
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 *------------------------------------------------------------------
19 */
20
21#ifndef __included_svm_h__
22#define __included_svm_h__
23
24#include <pthread.h>
25#include <vppinfra/clib.h>
26#include <vppinfra/mem.h>
27
Dave Barach95bb8832015-12-12 10:37:00 -050028#define MMAP_PAGESIZE (clib_mem_get_page_size())
Ed Warnickecb9cada2015-12-08 15:45:58 -070029
30#define SVM_VERSION ((1<<16) | 1) /* set to declare region ready. */
31
32#define SVM_FLAGS_MHEAP (1<<0) /* region contains an mheap */
33#define SVM_FLAGS_FILE (1<<1) /* region backed by one or more files */
34#define SVM_FLAGS_NODATA (1<<2) /* region will be further subdivided */
35#define SVM_FLAGS_NEED_DATA_INIT (1<<3)
36
Dave Barach95bb8832015-12-12 10:37:00 -050037#define SVM_PVT_MHEAP_SIZE (128<<10) /* region's private mheap (128k) */
Ed Warnickecb9cada2015-12-08 15:45:58 -070038
39typedef struct svm_region_ {
40 volatile uword version;
41 pthread_mutex_t mutex;
42 pthread_cond_t condvar;
43 int mutex_owner_pid; /* in case of trouble */
44 int mutex_owner_tag;
45 uword flags;
46 uword virtual_base; /* base of the region object */
47 uword virtual_size;
48 void *region_heap;
49 void *data_base; /* data portion base address */
50 void *data_heap; /* data heap, if any */
51 volatile void *user_ctx; /* user context pointer */
52 /* stuff allocated in the region's heap */
53 uword bitmap_size; /* nbits in virtual alloc bitmap */
54 uword *bitmap; /* the bitmap */
55 char *region_name;
56 char *backing_file;
57 char **filenames;
58 uword *client_pids;
59 /* pad */
60
61 /* next page:
62 * (64K) clib heap for the region itself
63 *
64 * data_base -> whatever is in this region
65 */
66
67} svm_region_t;
68
69typedef struct svm_map_region_args_ {
70 char *root_path; /* NULL means use the truly global arena */
71 char *name;
72 uword baseva;
73 uword size;
74 uword flags;
75 char *backing_file;
76 uword backing_mmap_size;
Dave Barach16c75df2016-05-31 14:05:46 -040077 /* uid, gid to own the svm region(s) */
78 int uid;
79 int gid;
Ed Warnickecb9cada2015-12-08 15:45:58 -070080} svm_map_region_args_t;
81
82
83/*
84 * Memory shared across all router instances. Packet buffers, etc
85 * Base should be "out of the way," and size should be big enough to
86 * cover everything we plan to put here.
87 */
88#define SVM_GLOBAL_REGION_BASEVA 0x30000000
89#define SVM_GLOBAL_REGION_SIZE (64<<20)
90#define SVM_GLOBAL_REGION_NAME "/global_vm"
91
92/*
93 * Memory shared across individual router instances.
94 */
95#define SVM_OVERLAY_REGION_BASEVA \
96 (SVM_GLOBAL_REGION_BASEVA + SVM_GLOBAL_REGION_SIZE)
97#define SVM_OVERLAY_REGION_SIZE (1<<20)
98#define SVM_OVERLAY_REGION_BASENAME "/overlay_vm"
99
100typedef struct {
101 u8 *subregion_name;
102} svm_subregion_t;
103
104typedef struct {
105 svm_subregion_t *subregions; /* subregion pool */
106 uword *name_hash;
107 u8 *root_path;
108} svm_main_region_t;
109
110
111void *svm_region_find_or_create (svm_map_region_args_t *a);
112void svm_region_init(void);
113void svm_region_init_chroot(char *root_path);
Dave Barach16c75df2016-05-31 14:05:46 -0400114void svm_region_init_chroot_uid_gid(char *root_path, int uid, int gid);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700115void svm_region_exit (void);
116void svm_region_unmap(void *rp_arg);
117void svm_client_scan (char *root_path);
118void svm_client_scan_this_region_nolock (svm_region_t *rp);
119u8 *shm_name_from_svm_map_region_args (svm_map_region_args_t *a);
120
121static inline void *svm_mem_alloc (svm_region_t *rp, uword size)
122{
123 u8 *oldheap;
124 ASSERT(rp->flags & SVM_FLAGS_MHEAP);
125 u8 *rv;
126
127 pthread_mutex_lock(&rp->mutex);
128 oldheap = clib_mem_set_heap(rp->data_heap);
129 rv = clib_mem_alloc (size);
130 clib_mem_set_heap(oldheap);
131 pthread_mutex_unlock(&rp->mutex);
132 return (rv);
133}
134
135static inline void *svm_mem_alloc_aligned_at_offset (svm_region_t *rp,
136 uword size,
137 uword align,
138 uword offset)
139{
140 u8 *oldheap;
141 ASSERT(rp->flags & SVM_FLAGS_MHEAP);
142 u8 *rv;
143
144 pthread_mutex_lock(&rp->mutex);
145 oldheap = clib_mem_set_heap(rp->data_heap);
146 rv = clib_mem_alloc_aligned_at_offset (size, align, offset);
147 clib_mem_set_heap(oldheap);
148 pthread_mutex_unlock(&rp->mutex);
149 return (rv);
150}
151
152static inline void svm_mem_free (svm_region_t *rp, void *ptr)
153{
154 u8 *oldheap;
155 ASSERT(rp->flags & SVM_FLAGS_MHEAP);
156
157 pthread_mutex_lock(&rp->mutex);
158 oldheap = clib_mem_set_heap(rp->data_heap);
159 clib_mem_free (ptr);
160 clib_mem_set_heap(oldheap);
161 pthread_mutex_unlock(&rp->mutex);
162
163}
164
165static inline void *svm_push_pvt_heap (svm_region_t *rp)
166{
167 u8 *oldheap;
168 oldheap = clib_mem_set_heap(rp->region_heap);
169 return ((void *) oldheap);
170}
171
172static inline void *svm_push_data_heap (svm_region_t *rp)
173{
174 u8 *oldheap;
175 oldheap = clib_mem_set_heap(rp->data_heap);
176 return ((void *) oldheap);
177}
178
179static inline void svm_pop_heap (void *oldheap)
180{
181 clib_mem_set_heap(oldheap);
182}
183
184u8 * format_svm_region (u8 * s, va_list * args);
185
186svm_region_t *svm_get_root_rp (void);
187
188#endif /* __included_svm_h__ */