| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | /**************************************************************************
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-06-17 00:01:27 +10:00
										 |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** Commercial Usage | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** 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. | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** GNU Lesser General Public License Usage | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** 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.
 | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** If you are unsure which license is appropriate for your use, please | 
					
						
							| 
									
										
										
										
											2009-08-14 09:30:56 +02:00
										 |  |  | ** contact the sales department at http://qt.nokia.com/contact.
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | **************************************************************************/ | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |   Copyright 2005 Roberto Raggi <roberto@kdevelop.org> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Permission to use, copy, modify, distribute, and sell this software and its | 
					
						
							|  |  |  |   documentation for any purpose is hereby granted without fee, provided that | 
					
						
							|  |  |  |   the above copyright notice appear in all copies and that both that | 
					
						
							|  |  |  |   copyright notice and this permission notice appear in supporting | 
					
						
							|  |  |  |   documentation. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |   all copies or substantial portions of the Software. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |   IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE | 
					
						
							|  |  |  |   KDEVELOP TEAM BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN | 
					
						
							|  |  |  |   AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | 
					
						
							|  |  |  |   CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-22 14:10:47 +01:00
										 |  |  | #include "PreprocessorEnvironment.h"
 | 
					
						
							|  |  |  | #include "Macro.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | #include <cstring>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 12:17:39 +01:00
										 |  |  | using namespace CPlusPlus; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-09 15:25:01 +01:00
										 |  |  | Environment::Environment() | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     : currentLine(0), | 
					
						
							| 
									
										
										
										
											2008-12-22 14:10:47 +01:00
										 |  |  |       hideNext(false), | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |       _macros(0), | 
					
						
							|  |  |  |       _allocated_macros(0), | 
					
						
							|  |  |  |       _macro_count(-1), | 
					
						
							|  |  |  |       _hash(0), | 
					
						
							|  |  |  |       _hash_count(401) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-09 15:25:01 +01:00
										 |  |  | Environment::~Environment() | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (_macros) { | 
					
						
							|  |  |  |         qDeleteAll(firstMacro(), lastMacro()); | 
					
						
							|  |  |  |         free(_macros); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (_hash) | 
					
						
							|  |  |  |         free(_hash); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-09 15:25:01 +01:00
										 |  |  | unsigned Environment::macroCount() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _macro_count + 1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-09 15:25:01 +01:00
										 |  |  | Macro *Environment::macroAt(unsigned index) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _macros[index]; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | Macro *Environment::bind(const Macro &__macro) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-12-22 13:55:42 +01:00
										 |  |  |     Q_ASSERT(! __macro.name().isEmpty()); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Macro *m = new Macro (__macro); | 
					
						
							| 
									
										
										
										
											2008-12-22 14:10:47 +01:00
										 |  |  |     m->_hashcode = hashCode(m->name()); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (++_macro_count == _allocated_macros) { | 
					
						
							|  |  |  |         if (! _allocated_macros) | 
					
						
							|  |  |  |             _allocated_macros = 401; | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             _allocated_macros <<= 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-02 18:08:43 +01:00
										 |  |  |         _macros = reinterpret_cast<Macro **>(realloc(_macros, sizeof(Macro *) * _allocated_macros)); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _macros[_macro_count] = m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (! _hash || _macro_count > (_hash_count >> 1)) { | 
					
						
							|  |  |  |         rehash(); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2008-12-22 13:55:42 +01:00
										 |  |  |         const unsigned h = m->_hashcode % _hash_count; | 
					
						
							|  |  |  |         m->_next = _hash[h]; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         _hash[h] = m; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return m; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-23 12:49:56 +01:00
										 |  |  | void Environment::addMacros(const QList<Macro> ¯os) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     foreach (const Macro ¯o, macros) { | 
					
						
							|  |  |  |         bind(macro); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-09 16:19:56 +01:00
										 |  |  | Macro *Environment::remove(const QByteArray &name) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     Macro macro; | 
					
						
							| 
									
										
										
										
											2008-12-22 13:55:42 +01:00
										 |  |  |     macro.setName(name); | 
					
						
							|  |  |  |     macro.setHidden(true); | 
					
						
							|  |  |  |     macro.setFileName(currentFile); | 
					
						
							|  |  |  |     macro.setLine(currentLine); | 
					
						
							| 
									
										
										
										
											2008-12-08 12:59:33 +01:00
										 |  |  |     return bind(macro); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-23 12:49:56 +01:00
										 |  |  | void Environment::reset() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (_macros) { | 
					
						
							|  |  |  |         qDeleteAll(firstMacro(), lastMacro()); | 
					
						
							|  |  |  |         free(_macros); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (_hash) | 
					
						
							|  |  |  |         free(_hash); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _macros = 0; | 
					
						
							|  |  |  |     _allocated_macros = 0; | 
					
						
							|  |  |  |     _macro_count = -1; | 
					
						
							|  |  |  |     _hash = 0; | 
					
						
							|  |  |  |     _hash_count = 401; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-02 18:08:43 +01:00
										 |  |  | bool Environment::isBuiltinMacro(const QByteArray &s) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (s.length() != 8) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s[0] == '_') { | 
					
						
							|  |  |  |         if (s[1] == '_') { | 
					
						
							|  |  |  |             if (s[2] == 'D') { | 
					
						
							|  |  |  |                 if (s[3] == 'A') { | 
					
						
							|  |  |  |                     if (s[4] == 'T') { | 
					
						
							|  |  |  |                         if (s[5] == 'E') { | 
					
						
							|  |  |  |                             if (s[6] == '_') { | 
					
						
							|  |  |  |                                 if (s[7] == '_') { | 
					
						
							|  |  |  |                                     return true; | 
					
						
							|  |  |  |                                 } | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (s[2] == 'F') { | 
					
						
							|  |  |  |                 if (s[3] == 'I') { | 
					
						
							|  |  |  |                     if (s[4] == 'L') { | 
					
						
							|  |  |  |                         if (s[5] == 'E') { | 
					
						
							|  |  |  |                             if (s[6] == '_') { | 
					
						
							|  |  |  |                                 if (s[7] == '_') { | 
					
						
							|  |  |  |                                     return true; | 
					
						
							|  |  |  |                                 } | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (s[2] == 'L') { | 
					
						
							|  |  |  |                 if (s[3] == 'I') { | 
					
						
							|  |  |  |                     if (s[4] == 'N') { | 
					
						
							|  |  |  |                         if (s[5] == 'E') { | 
					
						
							|  |  |  |                             if (s[6] == '_') { | 
					
						
							|  |  |  |                                 if (s[7] == '_') { | 
					
						
							|  |  |  |                                     return true; | 
					
						
							|  |  |  |                                 } | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (s[2] == 'T') { | 
					
						
							|  |  |  |                 if (s[3] == 'I') { | 
					
						
							|  |  |  |                     if (s[4] == 'M') { | 
					
						
							|  |  |  |                         if (s[5] == 'E') { | 
					
						
							|  |  |  |                             if (s[6] == '_') { | 
					
						
							|  |  |  |                                 if (s[7] == '_') { | 
					
						
							|  |  |  |                                     return true; | 
					
						
							|  |  |  |                                 } | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-02 18:08:43 +01:00
										 |  |  | Environment::iterator Environment::firstMacro() const | 
					
						
							|  |  |  | { return _macros; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Environment::iterator Environment::lastMacro() const | 
					
						
							|  |  |  | { return _macros + _macro_count + 1; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-22 14:10:47 +01:00
										 |  |  | Macro *Environment::resolve(const QByteArray &name) const | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! _macros) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-22 14:10:47 +01:00
										 |  |  |     Macro *it = _hash[hashCode(name) % _hash_count]; | 
					
						
							| 
									
										
										
										
											2008-12-22 13:55:42 +01:00
										 |  |  |     for (; it; it = it->_next) { | 
					
						
							|  |  |  |         if (it->name() != name) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2008-12-22 13:55:42 +01:00
										 |  |  |         else if (it->isHidden()) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         else break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return it; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-22 14:10:47 +01:00
										 |  |  | unsigned Environment::hashCode(const QByteArray &s) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned hash_value = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = 0; i < s.size (); ++i) | 
					
						
							|  |  |  |         hash_value = (hash_value << 5) - hash_value + s.at (i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return hash_value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Environment::rehash() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (_hash) { | 
					
						
							|  |  |  |         free(_hash); | 
					
						
							|  |  |  |         _hash_count <<= 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-02 18:08:43 +01:00
										 |  |  |     _hash = reinterpret_cast<Macro **>(calloc(_hash_count, sizeof(Macro *))); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-02 18:08:43 +01:00
										 |  |  |     for (iterator it = firstMacro(); it != lastMacro(); ++it) { | 
					
						
							|  |  |  |         Macro *m = *it; | 
					
						
							| 
									
										
										
										
											2008-12-22 13:55:42 +01:00
										 |  |  |         const unsigned h = m->_hashcode % _hash_count; | 
					
						
							|  |  |  |         m->_next = _hash[h]; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         _hash[h] = m; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |