shared_memory_mapping_test.cpp 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. //////////////////////////////////////////////////////////////////////////////
  2. //
  3. // (C) Copyright Ion Gaztanaga 2004-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. #include <boost/interprocess/detail/config_begin.hpp>
  11. #include <fstream>
  12. #include <iostream>
  13. #include <boost/interprocess/shared_memory_object.hpp>
  14. #include <boost/interprocess/mapped_region.hpp>
  15. #include <boost/interprocess/anonymous_shared_memory.hpp>
  16. #include <string>
  17. #include "get_process_id_name.hpp"
  18. using namespace boost::interprocess;
  19. shared_memory_object get_shared_memory_mapping()
  20. {
  21. shared_memory_object sh;
  22. return shared_memory_object(boost::move(sh));
  23. }
  24. int main ()
  25. {
  26. std::string process_id = test::get_process_id_name();
  27. std::string process_id2(process_id);
  28. process_id2 += "_2";
  29. try{
  30. const std::size_t FileSize = 99999*4;
  31. {
  32. //Remove shared memory
  33. shared_memory_object::remove(process_id.c_str());
  34. shared_memory_object::remove(process_id2.c_str());
  35. //Create shared memory and file mapping
  36. shared_memory_object mapping(create_only, process_id.c_str(), read_write);
  37. mapping.truncate(FileSize);
  38. }
  39. {
  40. //Create a file mapping
  41. shared_memory_object mapping(open_only, process_id.c_str(), read_write);
  42. //Create two mapped regions, one half of the file each
  43. mapped_region region (mapping
  44. ,read_write
  45. ,0
  46. ,FileSize/2
  47. ,0);
  48. mapped_region region2(mapping
  49. ,read_write
  50. ,FileSize/2
  51. ,FileSize - FileSize/2
  52. ,0);
  53. //Fill two regions with a pattern
  54. unsigned char *filler = static_cast<unsigned char*>(region.get_address());
  55. for(std::size_t i = 0
  56. ;i < FileSize/2
  57. ;++i){
  58. *filler++ = static_cast<unsigned char>(i);
  59. }
  60. filler = static_cast<unsigned char*>(region2.get_address());
  61. for(std::size_t i = FileSize/2
  62. ;i < FileSize
  63. ;++i){
  64. *filler++ = static_cast<unsigned char>(i);
  65. }
  66. if(!region.flush(0, 0, false)){
  67. return 1;
  68. }
  69. if(!region2.flush(0, 0, true)){
  70. return 1;
  71. }
  72. }
  73. //See if the pattern is correct in the file using two mapped regions
  74. {
  75. //Create a file mapping
  76. shared_memory_object mapping(open_only, process_id.c_str(), read_write);
  77. mapped_region region(mapping, read_write, 0, FileSize/2, 0);
  78. mapped_region region2(mapping, read_write, FileSize/2, FileSize - FileSize/2, 0);
  79. unsigned char *checker = static_cast<unsigned char*>(region.get_address());
  80. //Check pattern
  81. for(std::size_t i = 0
  82. ;i < FileSize/2
  83. ;++i){
  84. if(*checker++ != static_cast<unsigned char>(i)){
  85. return 1;
  86. }
  87. }
  88. //Check second half
  89. checker = static_cast<unsigned char *>(region2.get_address());
  90. //Check pattern
  91. for(std::size_t i = FileSize/2
  92. ;i < FileSize
  93. ;++i){
  94. if(*checker++ != static_cast<unsigned char>(i)){
  95. return 1;
  96. }
  97. }
  98. }
  99. //Now check the pattern mapping a single read only mapped_region
  100. {
  101. //Create a file mapping
  102. shared_memory_object mapping(open_only, process_id.c_str(), read_only);
  103. //Create a single regions, mapping all the file
  104. mapped_region region (mapping, read_only);
  105. //Check pattern
  106. unsigned char *pattern = static_cast<unsigned char*>(region.get_address());
  107. for(std::size_t i = 0
  108. ;i < FileSize
  109. ;++i, ++pattern){
  110. if(*pattern != static_cast<unsigned char>(i)){
  111. return 1;
  112. }
  113. }
  114. //Now shrink
  115. const std::size_t original_region_size = region.get_size();
  116. if(!region.shrink_by(region.get_size()/2, false) || region.get_size() != original_region_size/2){
  117. return 1;
  118. }
  119. const std::size_t shrunk_region_size = region.get_size();
  120. if(!region.shrink_by(region.get_size()/2, true) || region.get_size() != shrunk_region_size/2){
  121. return 1;
  122. }
  123. //Now advise
  124. #if defined(POSIX_MADV_NORMAL) || defined(MADV_NORMAL)
  125. std::cout << "Advice normal" << std::endl;
  126. if(!region.advise(mapped_region::advice_normal)){
  127. return 1;
  128. }
  129. #endif
  130. #if defined(POSIX_MADV_SEQUENTIAL) || defined(MADV_SEQUENTIAL)
  131. std::cout << "Advice sequential" << std::endl;
  132. if(!region.advise(mapped_region::advice_sequential)){
  133. return 1;
  134. }
  135. #endif
  136. #if defined(POSIX_MADV_RANDOM) || defined(MADV_RANDOM)
  137. std::cout << "Advice random" << std::endl;
  138. if(!region.advise(mapped_region::advice_random)){
  139. return 1;
  140. }
  141. #endif
  142. #if defined(POSIX_MADV_WILLNEED) || defined(MADV_WILLNEED)
  143. std::cout << "Advice 'will need'" << std::endl;
  144. if(!region.advise(mapped_region::advice_willneed)){
  145. return 1;
  146. }
  147. #endif
  148. #if defined(POSIX_MADV_DONTNEED) || (defined(MADV_DONTNEED) && defined(BOOST_INTERPROCESS_MADV_DONTNEED_HAS_NONDESTRUCTIVE_SEMANTICS))
  149. std::cout << "Advice 'dont't need'" << std::endl;
  150. if(!region.advise(mapped_region::advice_dontneed)){
  151. return 1;
  152. }
  153. #endif
  154. }
  155. {
  156. //Check for busy address space
  157. shared_memory_object mapping(open_only, process_id.c_str(), read_only);
  158. mapped_region region (mapping, read_only);
  159. shared_memory_object mapping2(create_only, process_id2.c_str(), read_write);
  160. mapping2.truncate(FileSize);
  161. try{
  162. mapped_region region2 (mapping2, read_only, 0, FileSize, region.get_address());
  163. }
  164. catch(interprocess_exception &e){
  165. shared_memory_object::remove(process_id2.c_str());
  166. if(e.get_error_code() != busy_error){
  167. throw e;
  168. }
  169. }
  170. catch(std::exception &){
  171. shared_memory_object::remove(process_id2.c_str());
  172. throw;
  173. }
  174. shared_memory_object::remove(process_id2.c_str());
  175. }
  176. {
  177. //Now check anonymous mapping
  178. mapped_region region(anonymous_shared_memory(FileSize));
  179. //Write pattern
  180. unsigned char *pattern = static_cast<unsigned char*>(region.get_address());
  181. for(std::size_t i = 0
  182. ;i < FileSize
  183. ;++i, ++pattern){
  184. *pattern = static_cast<unsigned char>(i);
  185. }
  186. //Check pattern
  187. pattern = static_cast<unsigned char*>(region.get_address());
  188. for(std::size_t i = 0
  189. ;i < FileSize
  190. ;++i, ++pattern){
  191. if(*pattern != static_cast<unsigned char>(i)){
  192. return 1;
  193. }
  194. }
  195. }
  196. {
  197. //Now test move semantics
  198. shared_memory_object mapping(open_only, process_id.c_str(), read_write);
  199. shared_memory_object move_ctor(boost::move(mapping));
  200. shared_memory_object move_assign;
  201. move_assign = boost::move(move_ctor);
  202. shared_memory_object ret(get_shared_memory_mapping());
  203. }
  204. }
  205. catch(std::exception &exc){
  206. shared_memory_object::remove(process_id.c_str());
  207. shared_memory_object::remove(process_id2.c_str());
  208. std::cout << "Unhandled exception: " << exc.what() << std::endl;
  209. return 1;
  210. }
  211. shared_memory_object::remove(process_id.c_str());
  212. return 0;
  213. }