1 | ////////////////////////////////////////////////////////////////////////////// |
2 | // |
3 | // (C) Copyright Ion Gaztanaga 2006-2012. Distributed under the Boost |
4 | // Software License, Version 1.0. (See accompanying file |
5 | // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) |
6 | // |
7 | // See http://www.boost.org/libs/interprocess for documentation. |
8 | // |
9 | ////////////////////////////////////////////////////////////////////////////// |
10 | |
11 | //[doc_managed_allocation_command |
12 | #include <boost/interprocess/managed_shared_memory.hpp> |
13 | #include <cassert> |
14 | //<- |
15 | #include "../test/get_process_id_name.hpp" |
16 | //-> |
17 | |
18 | int main() |
19 | { |
20 | using namespace boost::interprocess; |
21 | |
22 | //Remove shared memory on construction and destruction |
23 | struct shm_remove |
24 | { |
25 | //<- |
26 | #if 1 |
27 | shm_remove() { shared_memory_object::remove(filename: test::get_process_id_name()); } |
28 | ~shm_remove(){ shared_memory_object::remove(filename: test::get_process_id_name()); } |
29 | #else |
30 | //-> |
31 | shm_remove() { shared_memory_object::remove("MySharedMemory" ); } |
32 | ~shm_remove(){ shared_memory_object::remove("MySharedMemory" ); } |
33 | //<- |
34 | #endif |
35 | //-> |
36 | } remover; |
37 | //<- |
38 | (void)remover; |
39 | //-> |
40 | |
41 | //Managed memory segment that allocates portions of a shared memory |
42 | //segment with the default management algorithm |
43 | //<- |
44 | #if 1 |
45 | managed_shared_memory managed_shm(create_only, test::get_process_id_name(), 10000*sizeof(std::size_t)); |
46 | #else |
47 | //-> |
48 | managed_shared_memory managed_shm(create_only, "MySharedMemory" , 10000*sizeof(std::size_t)); |
49 | //<- |
50 | #endif |
51 | //-> |
52 | |
53 | //Allocate at least 100 bytes, 1000 bytes if possible |
54 | managed_shared_memory::size_type min_size = 100; |
55 | managed_shared_memory::size_type first_received_size = 1000; |
56 | std::size_t *hint = 0; |
57 | std::size_t *ptr = managed_shm.allocation_command<std::size_t> |
58 | (command: boost::interprocess::allocate_new, limit_size: min_size, prefer_in_recvd_out_size&: first_received_size, reuse&: hint); |
59 | |
60 | //Received size must be bigger than min_size |
61 | assert(first_received_size >= min_size); |
62 | |
63 | //Get free memory |
64 | managed_shared_memory::size_type free_memory_after_allocation = managed_shm.get_free_memory(); |
65 | //<- |
66 | (void)free_memory_after_allocation; |
67 | //-> |
68 | |
69 | //Now write the data |
70 | for(std::size_t i = 0; i < first_received_size; ++i) ptr[i] = i; |
71 | |
72 | //Now try to triplicate the buffer. We won't admit an expansion |
73 | //lower to the double of the original buffer. |
74 | //This "should" be successful since no other class is allocating |
75 | //memory from the segment |
76 | min_size = first_received_size*2; |
77 | managed_shared_memory::size_type expanded_size = first_received_size*3; |
78 | std::size_t * ret = managed_shm.allocation_command |
79 | (command: boost::interprocess::expand_fwd, limit_size: min_size, prefer_in_recvd_out_size&: expanded_size, reuse&: ptr); |
80 | //<- |
81 | (void)ret; |
82 | //-> |
83 | //Check invariants |
84 | assert(ptr != 0); |
85 | assert(ret == ptr); |
86 | assert(expanded_size >= first_received_size*2); |
87 | |
88 | //Get free memory and compare |
89 | managed_shared_memory::size_type free_memory_after_expansion = managed_shm.get_free_memory(); |
90 | assert(free_memory_after_expansion < free_memory_after_allocation); |
91 | //<- |
92 | (void)free_memory_after_expansion; |
93 | //-> |
94 | |
95 | //Write new values |
96 | for(std::size_t i = first_received_size; i < expanded_size; ++i) ptr[i] = i; |
97 | |
98 | //Try to shrink approximately to min_size, but the new size |
99 | //should be smaller than min_size*2. |
100 | //This "should" be successful since no other class is allocating |
101 | //memory from the segment |
102 | managed_shared_memory::size_type shrunk_size = min_size; |
103 | ret = managed_shm.allocation_command |
104 | (command: boost::interprocess::shrink_in_place, limit_size: min_size*2, prefer_in_recvd_out_size&: shrunk_size, reuse&: ptr); |
105 | |
106 | //Check invariants |
107 | assert(ptr != 0); |
108 | assert(ret == ptr); |
109 | assert(shrunk_size <= min_size*2); |
110 | assert(shrunk_size >= min_size); |
111 | |
112 | //Get free memory and compare |
113 | managed_shared_memory::size_type free_memory_after_shrinking = managed_shm.get_free_memory(); |
114 | assert(free_memory_after_shrinking > free_memory_after_expansion); |
115 | //<- |
116 | (void)free_memory_after_shrinking; |
117 | //-> |
118 | |
119 | //Deallocate the buffer |
120 | managed_shm.deallocate(addr: ptr); |
121 | return 0; |
122 | } |
123 | //] |
124 | |
125 | |