mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
sharedMemSegment.cpp
Go to the documentation of this file.
1/** \file sharedMemSegment.cpp
2 * \author Jared R. Males (jaredmales@gmail.com)
3 * \brief Definitions for the mxlib shared memory facility
4 * \ingroup IPC_sharedmem
5 * \ingroup IPC
6 *
7 */
8
9//***********************************************************************//
10// Copyright 2021 Jared R. Males (jaredmales@gmail.com)
11//
12// This file is part of mxlib.
13//
14// mxlib is free software: you can redistribute it and/or modify
15// it under the terms of the GNU General Public License as published by
16// the Free Software Foundation, either version 3 of the License, or
17// (at your option) any later version.
18//
19// mxlib is distributed in the hope that it will be useful,
20// but WITHOUT ANY WARRANTY; without even the implied warranty of
21// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22// GNU General Public License for more details.
23//
24// You should have received a copy of the GNU General Public License
25// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
26//***********************************************************************//
27
29
30namespace mx
31{
32namespace ipc
33{
34
36{
37 key_id = 0;
38 key = 0;
39 shmemid = 0;
40 addr = 0;
41 size = 0;
42
43 attached = 0;
44}
45
46key_t sharedMemSegment::setKey( const char *path, const int id )
47{
48 if( path != 0 )
49 {
50 strncpy( key_path, path, MX_IPC_KEYLEN );
51 key_id = id;
52
53 key = ftok( key_path, key_id );
54 }
55 else
56 {
57 key_path[0] = 0;
58 key_id = id;
59
60 key = id;
61 }
62
63 return key;
64}
65
67{
68 if( ( shmemid = shmget( key, sz + 1 * sizeof( uintptr_t ), IPC_CREAT | 0666 ) ) < 0 )
69 {
70 // If it failed, try to remove the shmem block and recreate it.
71 shmemid = shmget( key, 1, 0666 );
72 if( shmctl( shmemid, IPC_RMID, 0 ) < 0 )
73 {
74 fprintf( stderr, "Could not remove shared memory with key %i\n", key );
75 return -1;
76 }
77
78 // removal successful, now try to create again
79 if( ( shmemid = shmget( key, sz + 1 * sizeof( uintptr_t ), IPC_CREAT | 0666 ) ) < 0 )
80 {
81 fprintf( stderr, "Could not create shared memory with key %i\n", key );
82 return -1;
83 }
84 }
85
86 attach( true );
87
88 // Since we created this segment, we set the address field.
89 *( (uintptr_t *)addr ) = (uintptr_t)addr;
90
91 return 0;
92}
93
94int sharedMemSegment::attach( bool donot_set_addr )
95{
96 struct shmid_ds shmstats;
97 void *new_addr;
98
99 if( shmemid == 0 )
100 {
101 if( ( shmemid = shmget( key, 0, 0666 ) ) < 0 )
102 {
103 fprintf( stderr, "Could not remove shared memory with key %i\n", key );
104 return -1;
105 }
106 }
107
108 if( ( new_addr = shmat( shmemid, 0, 0 ) ) == (char *)-1 )
109 {
110 fprintf( stderr, "Could not attach to shared memory with key %i\n", key );
111 return -1;
112 }
113
114 attached = 1;
115
116 if( shmctl( shmemid, IPC_STAT, &shmstats ) < 0 )
117 {
118 fprintf( stderr, "Could not get shared memory stats with key %i\n", key );
119 return -1;
120 }
121
122 size = shmstats.shm_segsz;
123
124 // Here we first read in the address from the first unitptr_t size block
125 // then detach, then re-attach specifying the address.
126 if( !donot_set_addr )
127 {
128 addr = (void *)*( (uintptr_t *)new_addr ); // read the address from the segment itself
129
130 // now detach
131 if( shmdt( new_addr ) != 0 )
132 {
133 fprintf( stderr, "Unable to detach from shared memory\n" );
134 return -1;
135 }
136
137 attached = 0;
138
139 // and then re-attach, but now specifying an address
140 if( ( new_addr = shmat( shmemid, addr, 0 ) ) == (char *)-1 )
141 {
142 fprintf( stderr, "Could not re-attach shared memory with key %i\n", key );
143 return -1;
144 }
145 }
146 else
147 {
148 addr = new_addr;
149 }
150
151 attached = 1;
152
153 return 0;
154}
155
157{
158
159 if( attached )
160 return 0;
161
162 if( addr == 0 )
163 return 0;
164
165 // now detach
166 if( shmdt( addr ) != 0 )
167 {
168 fprintf( stderr, "Unable to detach from shared memory\n" );
169 return -1;
170 }
171
172 return 0;
173}
174
175} // namespace ipc
176} // namespace mx
void initialize()
Initialize the class.
int shmemid
The shared memory id associated with the key.
int create(size_t sz)
Create and attach to the segment.
int detach()
Detach from the segment.
int key_id
The id to use for key creation.
size_t size
The size of the segment.
int attach(bool donot_set_addr=false)
Attach to a segment without creating it.
void * addr
The base address of the segment.
key_t key
The shared memory key.
int attached
Flag indicating whether or not the segment is attached.
char key_path[MX_IPC_KEYLEN]
The path to use for key creation.
key_t setKey(const char *path, const int id)
Set the key.
The mxlib c++ namespace.
Definition mxError.hpp:106
Declarations for the mxlib shared memory facility.