• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  *
3  * Copyright (c) 2009 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::add_path(const fs::path& p)
10  *    void bcp_implementation::add_directory(const fs::path& p)
11  *    void bcp_implementation::add_file(const fs::path& p)
12  *    void bcp_implementation::add_dependent_lib(const std::string& libname, const fs::path& p, const fileview& view)
13  */
14 
15 #include "bcp_imp.hpp"
16 #include "fileview.hpp"
17 #include <boost/regex.hpp>
18 #include <boost/filesystem/operations.hpp>
19 #include <boost/filesystem/exception.hpp>
20 #include <iostream>
21 
22 //
23 // This file contains the code required to work out whether the source/header file being scanned
24 // is actually dependent upon some library's source code or not.
25 //
26 
27 static std::map<std::string, boost::regex> scanner;
28 
29 static std::map<std::string, std::set<std::string> > free_function_names;
30 static std::map<std::string, std::set<std::string> > class_names;
31 static std::map<std::string, std::set<std::string> > variable_names;
32 
init_library_scanner(const fs::path & p,bool cvs_mode,const std::string & libname,bool recurse=false)33 static void init_library_scanner(const fs::path& p, bool cvs_mode, const std::string& libname, bool recurse = false)
34 {
35    /*
36    if(free_function_names.count(libname) == 0)
37    {
38       free_function_names[libname] = "[\\x0]";
39       class_names[libname] = "[\\x0]";
40       variable_names[libname] = "[\\x0]";
41    }
42    */
43    //
44    // Don't add files created by build system:
45    //
46    if((p.leaf() == "bin") || (p.leaf() == "bin-stage"))
47       return;
48    //
49    // Don't add version control directories:
50    //
51    if((p.leaf() == "CVS") || (p.leaf() == ".svn"))
52       return;
53    //
54    // don't add directories not under version control:
55    //
56    if(cvs_mode && !fs::exists(p / "CVS/Entries"))
57       return;
58    if(cvs_mode && !fs::exists(p / ".svn/entries"))
59       return;
60    //
61    // Enumerate files and directories:
62    //
63    fs::directory_iterator i(p);
64    fs::directory_iterator j;
65    while(i != j)
66    {
67       if(fs::is_directory(*i))
68          init_library_scanner(*i, cvs_mode, libname, true);
69       if(bcp_implementation::is_source_file(*i))
70       {
71          static boost::regex function_scanner(
72             "(?|"                       // Branch reset group
73                "(?:\\<\\w+\\>[^>;{},:]*)"  // Return type
74                "(?:"
75                   "(\\<\\w+\\>)"           // Maybe class name
76                   "\\s*"
77                   "(?:<[^>;{]*>)?"         // Maybe template specialisation
78                   "::\\s*)?"
79                "(\\<(?!throw|if|while|for|catch)\\w+\\>)" // function name
80                "\\s*"
81                "\\("
82                   "[^\\(\\);{}]*"          // argument list
83                "\\)"
84                "\\s*(?:BOOST[_A-Z]+\\s*)?"
85                "\\{"                       // start of definition
86             "|"
87                "(\\<\\w+\\>)"              // Maybe class name
88                "\\s*"
89                "(?:<[^>;{]*>)?"            // Maybe template specialisation
90                "::\\s*"
91                "~?\\1"                     // function name, same as class name
92                "\\s*"
93                "\\("
94                   "[^\\(\\);{}]*"          // argument list
95                "\\)"
96                "\\s*(?:BOOST[_A-Z]+\\s*)?"
97                "\\{"                       // start of definition
98             ")"                            // end branch reset
99             );
100          fileview view(*i);
101          boost::regex_iterator<const char*> a(view.begin(), view.end(), function_scanner);
102          boost::regex_iterator<const char*> b;
103          while(a != b)
104          {
105             if((*a)[1].matched)
106             {
107                std::string n = a->str(1);
108                class_names[libname].insert(n);
109             }
110             else
111             {
112                std::string n = a->str(2);
113                free_function_names[libname].insert(n);
114             }
115             ++a;
116          }
117       }
118       ++i;
119    }
120 
121    if(recurse == false)
122    {
123       //
124       // Build the regular expressions:
125       //
126       const char* e1 =
127          "^(?>[[:blank:]]*)(?!#)[^;{}\\r\\n]*"
128          "(?|"
129          "(?:class|struct)[^:;{}#]*"
130          "(";
131       // list of class names goes here...
132       const char* e2 =
133          ")\\s*(?:<[^;{>]*>\\s*)?(?::[^;{]*)?\\{"
134          "|"
135          "\\<(?!return)\\w+\\>[^:;{}#=<>!~%.\\w]*(";
136          // List of function names goes here...
137       const char* e3 =
138          ")\\s*\\([^;()]*\\)\\s*(?:BOOST[_A-Z]+\\s*)?;)";
139 
140       std::string class_name_list;
141       std::set<std::string>::const_iterator i = class_names[libname].begin(), j = class_names[libname].end();
142       if(i != j)
143       {
144          class_name_list = *i;
145          ++i;
146          while(i != j)
147          {
148             class_name_list += "|" + *i;
149             ++i;
150          }
151       }
152       else
153       {
154          class_name_list = "[\\x0]";
155       }
156       std::string function_name_list;
157       i = free_function_names[libname].begin();
158       j = free_function_names[libname].end();
159       if(i != j)
160       {
161          function_name_list = *i;
162          ++i;
163          while(i != j)
164          {
165             function_name_list += "|" + *i;
166             ++i;
167          }
168       }
169       else
170       {
171          function_name_list = "[\\x0]";
172       }
173 
174       scanner[libname] = boost::regex(e1 + class_name_list + e2 + function_name_list + e3);
175    }
176 }
177 
add_dependent_lib(const std::string & libname,const fs::path & p,const fileview & view)178 void bcp_implementation::add_dependent_lib(const std::string& libname, const fs::path& p, const fileview& view)
179 {
180    //
181    // if the boost library libname has source associated with it
182    // then add the source to our list:
183    //
184    if(fs::exists(m_boost_path / "libs" / libname / "src"))
185    {
186       if(!m_dependencies.count(fs::path("libs") / libname / "src"))
187       {
188          if(scanner.count(libname) == 0)
189             init_library_scanner(m_boost_path / "libs" / libname / "src", m_cvs_mode, libname);
190          boost::cmatch what;
191          if(regex_search(view.begin(), view.end(), what, scanner[libname]))
192          {
193             std::cout << "INFO: tracking source dependencies of library " << libname
194                << " due to presence of \"" << what << "\" in file " << p << std::endl;
195             //std::cout << "Full text match was: " << what << std::endl;
196             m_dependencies[fs::path("libs") / libname / "src"] = p; // set up dependency tree
197             add_path(fs::path("libs") / libname / "src");
198 
199             if(fs::exists(m_boost_path / "libs" / libname / "build"))
200             {
201                if(!m_dependencies.count(fs::path("libs") / libname / "build"))
202                {
203                   m_dependencies[fs::path("libs") / libname / "build"] = p; // set up dependency tree
204                   add_path(fs::path("libs") / libname / "build");
205                   //m_dependencies[fs::path("boost-build.jam")] = p;
206                   //add_path(fs::path("boost-build.jam"));
207                   m_dependencies[fs::path("Jamroot")] = p;
208                   add_path(fs::path("Jamroot"));
209                   //m_dependencies[fs::path("tools/build")] = p;
210                   //add_path(fs::path("tools/build"));
211                }
212             }
213             if(fs::exists(m_boost_path / "libs" / libname / "config"))
214             {
215                if(!m_dependencies.count(fs::path("libs") / libname / "config"))
216                {
217                   m_dependencies[fs::path("libs") / libname / "config"] = p; // set up dependency tree
218                   add_path(fs::path("libs") / libname / "config");
219                   //m_dependencies[fs::path("boost-build.jam")] = p;
220                   //add_path(fs::path("boost-build.jam"));
221                   m_dependencies[fs::path("Jamroot")] = p;
222                   add_path(fs::path("Jamroot"));
223                   //m_dependencies[fs::path("tools/build")] = p;
224                   //add_path(fs::path("tools/build"));
225                }
226             }
227          }
228       }
229    }
230 }
231