blob: 6cda1f279e8d03cf17d8c4dc0a377d43001af98b [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
2 * Copyright (c) 2015 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15#include "ssvm.h"
16
Dave Barach8a7fb0c2016-07-08 14:44:23 -040017int
18ssvm_master_init (ssvm_private_t * ssvm, u32 master_index)
Ed Warnickecb9cada2015-12-08 15:45:58 -070019{
20 int ssvm_fd;
Dave Barach8a7fb0c2016-07-08 14:44:23 -040021 u8 *ssvm_filename;
Ed Warnickecb9cada2015-12-08 15:45:58 -070022 u8 junk = 0;
23 int flags;
Dave Barach8a7fb0c2016-07-08 14:44:23 -040024 ssvm_shared_header_t *sh;
25 u64 ticks = clib_cpu_time_now ();
Ed Warnickecb9cada2015-12-08 15:45:58 -070026 u64 randomize_baseva;
Dave Barach8a7fb0c2016-07-08 14:44:23 -040027 void *oldheap;
Ed Warnickecb9cada2015-12-08 15:45:58 -070028
29 if (ssvm->ssvm_size == 0)
30 return SSVM_API_ERROR_NO_SIZE;
31
32 ssvm_filename = format (0, "/dev/shm/%s%c", ssvm->name, 0);
33
34 unlink ((char *) ssvm_filename);
35
Dave Barach8a7fb0c2016-07-08 14:44:23 -040036 vec_free (ssvm_filename);
Ed Warnickecb9cada2015-12-08 15:45:58 -070037
Dave Barach8a7fb0c2016-07-08 14:44:23 -040038 ssvm_fd = shm_open ((char *) ssvm->name, O_RDWR | O_CREAT | O_EXCL, 0777);
Ed Warnickecb9cada2015-12-08 15:45:58 -070039
40 if (ssvm_fd < 0)
41 {
42 clib_unix_warning ("create segment '%s'", ssvm->name);
43 return SSVM_API_ERROR_CREATE_FAILURE;
44 }
45
Dave Barach56faee82016-08-04 18:58:05 -040046 if (lseek (ssvm_fd, ssvm->ssvm_size, SEEK_SET) < 0)
47 {
48 clib_unix_warning ("lseek");
49 close (ssvm_fd);
50 return SSVM_API_ERROR_SET_SIZE;
51 }
Ed Warnicke853e7202016-08-12 11:42:26 -070052
Dave Barach8a7fb0c2016-07-08 14:44:23 -040053 if (write (ssvm_fd, &junk, 1) != 1)
Ed Warnickecb9cada2015-12-08 15:45:58 -070054 {
55 clib_unix_warning ("set ssvm size");
Dave Barach8a7fb0c2016-07-08 14:44:23 -040056 close (ssvm_fd);
Ed Warnickecb9cada2015-12-08 15:45:58 -070057 return SSVM_API_ERROR_SET_SIZE;
58 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -040059
Ed Warnickecb9cada2015-12-08 15:45:58 -070060 flags = MAP_SHARED;
61 if (ssvm->requested_va)
62 flags |= MAP_FIXED;
63
64 randomize_baseva = (ticks & 15) * MMAP_PAGESIZE;
65
66 if (ssvm->requested_va)
67 ssvm->requested_va += randomize_baseva;
Dave Barach8a7fb0c2016-07-08 14:44:23 -040068
69 sh = ssvm->sh =
70 (ssvm_shared_header_t *) mmap ((void *) ssvm->requested_va,
71 ssvm->ssvm_size, PROT_READ | PROT_WRITE,
72 flags, ssvm_fd, 0);
Ed Warnickecb9cada2015-12-08 15:45:58 -070073
Christophe Fontainefef15b42016-04-09 12:38:49 +090074 if (ssvm->sh == MAP_FAILED)
Ed Warnickecb9cada2015-12-08 15:45:58 -070075 {
76 clib_unix_warning ("mmap");
Dave Barach8a7fb0c2016-07-08 14:44:23 -040077 close (ssvm_fd);
Ed Warnickecb9cada2015-12-08 15:45:58 -070078 return SSVM_API_ERROR_MMAP;
79 }
80
Dave Barach8a7fb0c2016-07-08 14:44:23 -040081 close (ssvm_fd);
Ed Warnickecb9cada2015-12-08 15:45:58 -070082
Dave Barach8a7fb0c2016-07-08 14:44:23 -040083 ssvm->my_pid = getpid ();
Ed Warnickecb9cada2015-12-08 15:45:58 -070084 sh->master_pid = ssvm->my_pid;
85 sh->ssvm_size = ssvm->ssvm_size;
Dave Barach8a7fb0c2016-07-08 14:44:23 -040086 sh->heap = mheap_alloc_with_flags
87 (((u8 *) sh) + MMAP_PAGESIZE, ssvm->ssvm_size - MMAP_PAGESIZE,
Ed Warnickecb9cada2015-12-08 15:45:58 -070088 MHEAP_FLAG_DISABLE_VM | MHEAP_FLAG_THREAD_SAFE);
89
Dave Barach8a7fb0c2016-07-08 14:44:23 -040090 sh->ssvm_va = pointer_to_uword (sh);
Ed Warnickecb9cada2015-12-08 15:45:58 -070091 sh->master_index = master_index;
92
93 oldheap = ssvm_push_heap (sh);
94 sh->name = format (0, "%s%c", ssvm->name, 0);
95 ssvm_pop_heap (oldheap);
96
97 ssvm->i_am_master = 1;
98
99 /* The application has to set set sh->ready... */
100 return 0;
101}
102
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400103int
104ssvm_slave_init (ssvm_private_t * ssvm, int timeout_in_seconds)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700105{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400106 struct stat stat;
107 int ssvm_fd = -1;
108 ssvm_shared_header_t *sh;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700109
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400110 ssvm->i_am_master = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700111
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400112 while (timeout_in_seconds-- > 0)
113 {
114 if (ssvm_fd < 0)
115 ssvm_fd = shm_open ((char *) ssvm->name, O_RDWR, 0777);
116 if (ssvm_fd < 0)
117 {
118 sleep (1);
119 continue;
120 }
121 if (fstat (ssvm_fd, &stat) < 0)
122 {
123 sleep (1);
124 continue;
125 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700126
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400127 if (stat.st_size > 0)
128 goto map_it;
129 }
130 clib_warning ("slave timeout");
131 return SSVM_API_ERROR_SLAVE_TIMEOUT;
132
133map_it:
134 sh = (void *) mmap (0, MMAP_PAGESIZE, PROT_READ | PROT_WRITE, MAP_SHARED,
135 ssvm_fd, 0);
136 if (sh == MAP_FAILED)
137 {
138 clib_unix_warning ("slave research mmap");
139 close (ssvm_fd);
140 return SSVM_API_ERROR_MMAP;
141 }
142
143 while (timeout_in_seconds-- > 0)
144 {
145 if (sh->ready)
146 goto re_map_it;
147 }
148 close (ssvm_fd);
149 munmap (sh, MMAP_PAGESIZE);
150 clib_warning ("slave timeout 2");
151 return SSVM_API_ERROR_SLAVE_TIMEOUT;
152
153re_map_it:
154 ssvm->requested_va = (u64) sh->ssvm_va;
155 ssvm->ssvm_size = sh->ssvm_size;
156 munmap (sh, MMAP_PAGESIZE);
157
158 sh = ssvm->sh = (void *) mmap ((void *) ssvm->requested_va, ssvm->ssvm_size,
159 PROT_READ | PROT_WRITE,
160 MAP_SHARED | MAP_FIXED, ssvm_fd, 0);
161
162 if (sh == MAP_FAILED)
163 {
164 clib_unix_warning ("slave final mmap");
165 close (ssvm_fd);
166 return SSVM_API_ERROR_MMAP;
167 }
168 sh->slave_pid = getpid ();
169 return 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700170}
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400171
Dave Barach68b0fb02017-02-28 15:15:56 -0500172void
173ssvm_delete (ssvm_private_t * ssvm)
174{
175 u8 *fn;
176
177 fn = format (0, "/dev/shm/%s%c", ssvm->name, 0);
178
179 /* Throw away the backing file */
180 if (unlink ((char *) fn) < 0)
181 clib_unix_warning ("unlink segment '%s'", ssvm->name);
182
183 munmap ((void *) ssvm->requested_va, ssvm->ssvm_size);
184 vec_free (fn);
185}
186
187
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400188/*
189 * fd.io coding-style-patch-verification: ON
190 *
191 * Local Variables:
192 * eval: (c-set-style "gnu")
193 * End:
194 */