| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-03-05 11:25:49 +01:00
										 |  |  | ** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Commercial Usage | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Licensees holding valid Qt Commercial licenses may use this file in | 
					
						
							|  |  |  | ** accordance with the Qt Commercial License Agreement provided with the | 
					
						
							|  |  |  | ** Software or, alternatively, in accordance with the terms contained in | 
					
						
							|  |  |  | ** a written agreement between you and Nokia. | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Alternatively, this file may be used under the terms of the GNU Lesser | 
					
						
							|  |  |  | ** General Public License version 2.1 as published by the Free Software | 
					
						
							|  |  |  | ** Foundation and appearing in the file LICENSE.LGPL included in the | 
					
						
							|  |  |  | ** packaging of this file.  Please review the following information to | 
					
						
							|  |  |  | ** ensure the GNU Lesser General Public License version 2.1 requirements | 
					
						
							|  |  |  | ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** If you are unsure which license is appropriate for your use, please | 
					
						
							|  |  |  | ** contact the sales department at http://qt.nokia.com/contact.
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:49:59 +01:00
										 |  |  | #include "Preprocessor.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | #include <AST.h>
 | 
					
						
							|  |  |  | #include <ASTVisitor.h>
 | 
					
						
							|  |  |  | #include <Control.h>
 | 
					
						
							|  |  |  | #include <Scope.h>
 | 
					
						
							|  |  |  | #include <Semantic.h>
 | 
					
						
							|  |  |  | #include <TranslationUnit.h>
 | 
					
						
							|  |  |  | #include <Literals.h>
 | 
					
						
							|  |  |  | #include <Symbols.h>
 | 
					
						
							|  |  |  | #include <Names.h>
 | 
					
						
							|  |  |  | #include <CoreTypes.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <cstdlib>
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:38:33 +01:00
										 |  |  | #include <sstream>
 | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | using namespace CPlusPlus; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:49:59 +01:00
										 |  |  | enum { BLOCK_SIZE = 4 * 1024}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void parse(const char *fileName, const char *source, unsigned size); | 
					
						
							|  |  |  | int runWithSystemPreprocessor(int argc, char *argv[]); | 
					
						
							|  |  |  | int runWithNewPreprocessor(int argc, char *argv[]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | int main(int argc, char *argv[]) | 
					
						
							| 
									
										
										
										
											2009-12-15 16:49:59 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (getenv("CPLUSPLUS_WITH_NEW_PREPROCESSOR")) | 
					
						
							|  |  |  |         return runWithNewPreprocessor(argc, argv); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return runWithSystemPreprocessor(argc, argv); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int runWithSystemPreprocessor(int argc, char *argv[]) | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     std::string cmdline; | 
					
						
							|  |  |  |     cmdline += "gcc -E -xc++ -U__BLOCKS__"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = 1; i < argc; ++i) { | 
					
						
							|  |  |  |         cmdline += ' '; | 
					
						
							|  |  |  |         cmdline += argv[i]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     char block[BLOCK_SIZE]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:49:59 +01:00
										 |  |  |     std::string preprocessedCode; | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (FILE *fp = popen(cmdline.c_str(), "r")) { | 
					
						
							|  |  |  |         while (size_t sz = fread(block, 1, BLOCK_SIZE, fp)) | 
					
						
							| 
									
										
										
										
											2009-12-15 16:49:59 +01:00
										 |  |  |             preprocessedCode.append(block, sz); | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         pclose(fp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         fprintf(stderr, "c++: No such file or directory\n"); | 
					
						
							|  |  |  |         return EXIT_FAILURE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:49:59 +01:00
										 |  |  |     parse("<stdin>", preprocessedCode.c_str(), preprocessedCode.size()); | 
					
						
							|  |  |  |     return EXIT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int runWithNewPreprocessor(int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (argc == 1) { | 
					
						
							|  |  |  |         fprintf(stderr, "c++: No such file or directory\n"); | 
					
						
							|  |  |  |         return EXIT_FAILURE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     char block[BLOCK_SIZE]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string source; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (FILE *fp = fopen(argv[1], "r")) { | 
					
						
							|  |  |  |         while (size_t sz = fread(block, 1, BLOCK_SIZE, fp)) | 
					
						
							|  |  |  |             source.append(block, sz); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fclose(fp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         fprintf(stderr, "c++: No such file or directory\n"); | 
					
						
							|  |  |  |         return EXIT_FAILURE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::ostringstream out; | 
					
						
							|  |  |  |     Preprocessor pp(out); | 
					
						
							|  |  |  |     pp(source.c_str(), source.size(), StringRef(argv[1])); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const std::string preprocessedCode = out.str(); | 
					
						
							|  |  |  |     parse(argv[1], preprocessedCode.c_str(), preprocessedCode.size()); | 
					
						
							|  |  |  |     return EXIT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void parse(const char *fileName, const char *source, unsigned size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  |     Control control; | 
					
						
							| 
									
										
										
										
											2009-12-15 16:49:59 +01:00
										 |  |  |     TranslationUnit unit(&control, control.findOrInsertStringLiteral(fileName)); | 
					
						
							|  |  |  |     unit.setSource(source, size); | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  |     unit.parse(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-15 16:38:33 +01:00
										 |  |  |     if (TranslationUnitAST *ast = unit.ast()->asTranslationUnit()) { | 
					
						
							|  |  |  |         Semantic sem(&unit); | 
					
						
							|  |  |  |         Namespace *globalNamespace = control.newNamespace(0); | 
					
						
							|  |  |  |         for (List<DeclarationAST *> *it = ast->declaration_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2010-08-11 13:46:32 +02:00
										 |  |  |             sem.check(it->value, globalNamespace); | 
					
						
							| 
									
										
										
										
											2009-12-15 16:38:33 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-23 12:58:39 +02:00
										 |  |  | } |