1 /*
2 *
3 * Copyright (c) 2003 Dr John Maddock
4 * Use, modification and distribution is subject to the
5 * Boost Software License, Version 1.0. (See accompanying file
6 * LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
7 *
8 * This file implements the following:
9 * void bcp_implementation::copy_path(const fs::path& p)
10 * void bcp_implementation::create_path(const fs::path& p)
11 */
12
13 #include "bcp_imp.hpp"
14 #include "fileview.hpp"
15 #include <boost/filesystem/operations.hpp>
16 #include <boost/filesystem/fstream.hpp>
17 #include <boost/regex.hpp>
18 #include <boost/assert.hpp>
19 #include <fstream>
20 #include <iterator>
21 #include <algorithm>
22 #include <iostream>
23
24 struct get_new_library_name
25 {
get_new_library_nameget_new_library_name26 get_new_library_name(const std::string& n) : m_new_name(n) {}
27 template <class I>
operator ()get_new_library_name28 std::string operator()(const boost::match_results<I>& what)
29 {
30 std::string s = what[0];
31 std::string::size_type n = s.find("boost");
32 if(n == std::string::npos)
33 {
34 s.insert(0, m_new_name);
35 }
36 else
37 {
38 s.replace(n, 5, m_new_name);
39 }
40 return s;
41 }
42 private:
43 std::string m_new_name;
44 };
45
copy_path(const fs::path & p)46 void bcp_implementation::copy_path(const fs::path& p)
47 {
48 BOOST_ASSERT(!fs::is_directory(m_boost_path / p));
49 if(fs::exists(m_dest_path / p))
50 {
51 std::cout << "Copying (and overwriting) file: " << p.string() << "\n";
52 fs::remove(m_dest_path / p);
53 }
54 else
55 std::cout << "Copying file: " << p.string() << "\n";
56 //
57 // create the path to the new file if it doesn't already exist:
58 //
59 create_path(p.branch_path());
60 //
61 // do text based copy if requested:
62 //
63 if((p.leaf() == "Jamroot") && m_namespace_name.size())
64 {
65 static std::vector<char> v1, v2;
66 v1.clear();
67 v2.clear();
68 boost::filesystem::ifstream is((m_boost_path / p));
69 std::copy(std::istreambuf_iterator<char>(is), std::istreambuf_iterator<char>(), std::back_inserter(v1));
70
71 static boost::regex libname_matcher;
72 if(libname_matcher.empty())
73 {
74 libname_matcher.assign("boost_");
75 }
76
77 regex_replace(std::back_inserter(v2), v1.begin(), v1.end(), libname_matcher, m_namespace_name + "_");
78 std::swap(v1, v2);
79 v2.clear();
80
81 boost::filesystem::ofstream os;
82 if(m_unix_lines)
83 os.open((m_dest_path / p), std::ios_base::binary | std::ios_base::out);
84 else
85 os.open((m_dest_path / p), std::ios_base::out);
86 os.write(&*v1.begin(), v1.size());
87 os.close();
88 }
89 else if(m_namespace_name.size() && m_lib_names.size() && is_jam_file(p))
90 {
91 static std::vector<char> v1, v2;
92 v1.clear();
93 v2.clear();
94 boost::filesystem::ifstream is((m_boost_path / p));
95 std::copy(std::istreambuf_iterator<char>(is), std::istreambuf_iterator<char>(), std::back_inserter(v1));
96
97 static boost::regex libname_matcher;
98 if(libname_matcher.empty())
99 {
100 std::string re = "\\<";
101 re += *m_lib_names.begin();
102 for(std::set<std::string>::const_iterator i = ++m_lib_names.begin(); i != m_lib_names.end(); ++i)
103 {
104 re += "|" + *i;
105 }
106 re += "\\>";
107 libname_matcher.assign(re);
108 }
109
110 regex_replace(std::back_inserter(v2), v1.begin(), v1.end(), libname_matcher, get_new_library_name(m_namespace_name));
111 std::swap(v1, v2);
112 v2.clear();
113
114 boost::filesystem::ofstream os;
115 if(m_unix_lines)
116 os.open((m_dest_path / p), std::ios_base::binary | std::ios_base::out);
117 else
118 os.open((m_dest_path / p), std::ios_base::out);
119 os.write(&*v1.begin(), v1.size());
120 os.close();
121 }
122 else if(m_namespace_name.size() && is_source_file(p))
123 {
124 //
125 // v1 hold the current content, v2 is temp buffer.
126 // Each time we do a search and replace the new content
127 // ends up in v2: we then swap v1 and v2, and clear v2.
128 //
129 static std::vector<char> v1, v2;
130 v1.clear();
131 v2.clear();
132 boost::filesystem::ifstream is((m_boost_path / p));
133 std::copy(std::istreambuf_iterator<char>(is), std::istreambuf_iterator<char>(), std::back_inserter(v1));
134
135 static const boost::regex namespace_matcher(
136 "(?|"
137 "(namespace\\s+)boost(_\\w+)?(?:(\\s*::\\s*)phoenix)?"
138 "|"
139 "(namespace\\s+(?:detail::)?)(adstl|phoenix|rapidxml)\\>"
140 "|"
141 "()\\<boost((?:_(?!intrusive_tags)\\w+)?\\s*(?:::))(?:(\\s*)phoenix)?"
142 "|"
143 "()\\<((?:adstl|phoenix|rapidxml)\\s*(?:::))"
144 "|"
145 "(namespace\\s+\\w+\\s*=\\s*(?:::\\s*)?)boost(_\\w+)?(?:(\\s*::\\s*)phoenix)?"
146 "|"
147 "(namespace\\s+\\w+\\s*=\\s*(?:::\\s*)?(?:\\w+\\s*::\\s*)?)(adstl|phoenix|rapidxml)\\>"
148 "|"
149 "(^\\s*#\\s*define\\s+\\w+\\s+)boost((?:_\\w+)?\\s*)$"
150 "|"
151 "(^\\s*#\\s*define[^\\n]+)((?:adstl|phoenix|rapidxml)\\s*)$"
152 "|"
153 "()boost(_asio_detail_posix_thread_function|_regex_free_static_mutex)"
154 "|"
155 "()\\<(lw_thread_routine|at_thread_exit|on_process_enter|on_process_exit|on_thread_enter|on_thread_exit|tss_cleanup_implemented)\\>"
156 "|"
157 "(BOOST_CLASS_REQUIRE4?[^;]*)boost((?:_\\w+)?\\s*,)"
158 "|"
159 "(::tr1::|TR1_DECL\\s+)boost(_\\w+\\s*)" // math tr1
160 "|"
161 "(\\(\\s*)boost(\\s*\\))\\s*(\\(\\s*)phoenix(\\s*\\))"
162 "|"
163 "(\\(\\s*)boost(\\s*\\))"
164 "|"
165 "(BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE[^\\)]*)boost(\\))"
166 ")"
167 );
168
169 regex_replace(std::back_inserter(v2), v1.begin(), v1.end(), namespace_matcher, "$1" + m_namespace_name + "$2(?3$3" + m_namespace_name + "phoenix?4$4)", boost::regex_constants::format_all);
170 std::swap(v1, v2);
171 v2.clear();
172
173 if(m_namespace_alias)
174 {
175 static const boost::regex namespace_alias(
176 /*
177 "namespace\\s+" + m_namespace_name +
178 "\\s*"
179 "("
180 "\\{"
181 "(?:"
182 "(?>[^\\{\\}/]+)"
183 "(?>"
184 "(?:"
185 "(?1)"
186 "|//[^\\n]+$"
187 "|/[^/]"
188 "|(?:^\\s*#[^\\n]*"
189 "(?:(?<=\\\\)\\n[^\\n]*)*)"
190 ")"
191 "[^\\{\\}]+"
192 ")*"
193 ")*"
194 "\\}"
195 ")"
196 */
197 /*
198 "(namespace\\s+" + m_namespace_name +
199 "\\s*\\{.*"
200 "\\})([^\\{\\};]*)\\z"
201 */
202 "(namespace)(\\s+)(" + m_namespace_name + ")"
203 "(adstl|phoenix|rapidxml)?(\\s*\\{)"
204 );
205 regex_replace(std::back_inserter(v2), v1.begin(), v1.end(), namespace_alias,
206 "$1 $3$4 {} $1 (?4$4:boost) = $3$4; $1$2$3$4$5", boost::regex_constants::format_all);
207 std::swap(v1, v2);
208 v2.clear();
209 }
210
211 boost::filesystem::ofstream os;
212 if(m_unix_lines)
213 os.open((m_dest_path / p), std::ios_base::binary | std::ios_base::out);
214 else
215 os.open((m_dest_path / p), std::ios_base::out);
216 if(v1.size())
217 os.write(&*v1.begin(), v1.size());
218 os.close();
219 }
220 else if(m_unix_lines && !is_binary_file(p))
221 {
222 boost::filesystem::ifstream is((m_boost_path / p));
223 std::istreambuf_iterator<char> isi(is);
224 std::istreambuf_iterator<char> end;
225
226 boost::filesystem::ofstream os((m_dest_path / p), std::ios_base::binary | std::ios_base::out);
227 std::ostreambuf_iterator<char> osi(os);
228
229 std::copy(isi, end, osi);
230 }
231 else
232 {
233 // binary copy:
234 fs::copy_file(m_boost_path / p, m_dest_path / p);
235 }
236 }
237
create_path(const fs::path & p)238 void bcp_implementation::create_path(const fs::path& p)
239 {
240 if(!fs::exists(m_dest_path / p))
241 {
242 // recurse then create the path:
243 create_path(p.branch_path());
244 fs::create_directory(m_dest_path / p);
245 }
246 }
247
248
249