Control.cpp 24.5 KB
Newer Older
1
/**************************************************************************
con's avatar
con committed
2 3 4
**
** This file is part of Qt Creator
**
hjk's avatar
hjk committed
5
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
con's avatar
con committed
6
**
7
** Contact: Nokia Corporation (qt-info@nokia.com)
con's avatar
con committed
8
**
9
** Commercial Usage
10
**
11 12 13 14
** 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.
15
**
16
** GNU Lesser General Public License Usage
17
**
18 19 20 21 22 23
** 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.
24
**
25
** If you are unsure which license is appropriate for your use, please
hjk's avatar
hjk committed
26
** contact the sales department at http://qt.nokia.com/contact.
con's avatar
con committed
27
**
28
**************************************************************************/
con's avatar
con committed
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
// Copyright (c) 2008 Roberto Raggi <roberto.raggi@gmail.com>
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// 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
// AUTHORS OR COPYRIGHT HOLDERS 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.

#include "Control.h"
#include "Literals.h"
#include "LiteralTable.h"
#include "TranslationUnit.h"
#include "CoreTypes.h"
#include "Symbols.h"
#include "Names.h"
56
#include "TypeMatcher.h"
Roberto Raggi's avatar
Roberto Raggi committed
57 58
#include <map>
#include <set>
59
#include <algorithm>
con's avatar
con committed
60

Roberto Raggi's avatar
Roberto Raggi committed
61
using namespace CPlusPlus;
con's avatar
con committed
62

Roberto Raggi's avatar
Roberto Raggi committed
63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
namespace {

template <typename _Tp>
struct Compare;

template <> struct Compare<IntegerType>
{
    bool operator()(const IntegerType &ty, const IntegerType &otherTy) const
    { return ty.kind() < otherTy.kind(); }
};

template <> struct Compare<FloatType>
{
    bool operator()(const FloatType &ty, const FloatType &otherTy) const
    { return ty.kind() < otherTy.kind(); }
};

template <> struct Compare<PointerToMemberType>
{
    bool operator()(const PointerToMemberType &ty, const PointerToMemberType &otherTy) const
    {
        if (ty.memberName() < otherTy.memberName())
            return true;

        else if (ty.memberName() == otherTy.memberName())
            return ty.elementType() < otherTy.elementType();

        return false;
    }
};

template <> struct Compare<PointerType>
{
    bool operator()(const PointerType &ty, const PointerType &otherTy) const
    {
        return ty.elementType() < otherTy.elementType();
    }
};

template <> struct Compare<ReferenceType>
{
    bool operator()(const ReferenceType &ty, const ReferenceType &otherTy) const
    {
        return ty.elementType() < otherTy.elementType();
    }
};

template <> struct Compare<NamedType>
{
    bool operator()(const NamedType &ty, const NamedType &otherTy) const
    {
        return ty.name() < otherTy.name();
    }
};

template <> struct Compare<ArrayType>
{
    bool operator()(const ArrayType &ty, const ArrayType &otherTy) const
    {
        if (ty.size() < otherTy.size())
            return true;

        else if (ty.size() == otherTy.size())
            return ty.elementType() < otherTy.elementType();

        return false;
    }
};

Roberto Raggi's avatar
Roberto Raggi committed
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172
template <> struct Compare<DestructorNameId>
{
    bool operator()(const DestructorNameId &name, const DestructorNameId &otherName) const
    {
        return name.identifier() < otherName.identifier();
    }
};

template <> struct Compare<OperatorNameId>
{
    bool operator()(const OperatorNameId &name, const OperatorNameId &otherName) const
    {
        return name.kind() < otherName.kind();
    }
};

template <> struct Compare<ConversionNameId>
{
    bool operator()(const ConversionNameId &name, const ConversionNameId &otherName) const
    {
        return name.type() < otherName.type();
    }
};
template <> struct Compare<TemplateNameId>
{
    bool operator()(const TemplateNameId &name, const TemplateNameId &otherName) const
    {
        const Identifier *id = name.identifier();
        const Identifier *otherId = otherName.identifier();

        if (id == otherId)
            return std::lexicographical_compare(name.firstTemplateArgument(), name.lastTemplateArgument(),
                                                otherName.firstTemplateArgument(), otherName.lastTemplateArgument());

        return id < otherId;
    }
};
template <> struct Compare<QualifiedNameId>
{
    bool operator()(const QualifiedNameId &name, const QualifiedNameId &otherName) const
    {
173 174
        if (name.base() == otherName.base())
            return name.name() < otherName.name();
Roberto Raggi's avatar
Roberto Raggi committed
175

176
        return name.base() < otherName.base();
Roberto Raggi's avatar
Roberto Raggi committed
177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192
    }
};

template <> struct Compare<SelectorNameId>
{
    bool operator()(const SelectorNameId &name, const SelectorNameId &otherName) const
    {
        if (name.hasArguments() == otherName.hasArguments())
            return std::lexicographical_compare(name.firstName(), name.lastName(),
                                                otherName.firstName(), otherName.lastName());

        return name.hasArguments() < otherName.hasArguments();
    }
};


Roberto Raggi's avatar
Roberto Raggi committed
193 194 195
template <typename _Tp>
class Table: public std::set<_Tp, Compare<_Tp> >
{
196
    typedef std::set<_Tp, Compare<_Tp> > _Base;
Roberto Raggi's avatar
Roberto Raggi committed
197 198
public:
    _Tp *intern(const _Tp &element)
199
    { return const_cast<_Tp *>(&*_Base::insert(element).first); }
Roberto Raggi's avatar
Roberto Raggi committed
200 201 202 203
};

} // end of anonymous namespace

204
#ifdef Q_OS_SYMBIAN
Wolfgang Beck's avatar
Wolfgang Beck committed
205 206 207 208 209 210 211 212 213
//Symbian compiler has some difficulties to understand the templates.
static void delete_array_entries(std::vector<Symbol *> vt)
{
    std::vector<Symbol *>::iterator it;
    for (it = vt.begin(); it != vt.end(); ++it) {
        delete *it;
    }
}
#else
214 215 216 217 218 219 220
template <typename _Iterator>
static void delete_array_entries(_Iterator first, _Iterator last)
{
    for (; first != last; ++first)
        delete *first;
}

con's avatar
con committed
221 222
template <typename _Array>
static void delete_array_entries(const _Array &a)
223
{ delete_array_entries(a.begin(), a.end()); }
Wolfgang Beck's avatar
Wolfgang Beck committed
224
#endif
225

con's avatar
con committed
226 227 228 229
class Control::Data
{
public:
    Data(Control *control)
Erik Verbruggen's avatar
Erik Verbruggen committed
230 231 232 233 234 235 236 237 238 239 240 241 242
        : control(control)
        , translationUnit(0)
        , diagnosticClient(0)
        , deprecatedId(0)
        , unavailableId(0)
        , objcGetterId(0)
        , objcSetterId(0)
        , objcReadwriteId(0)
        , objcReadonlyId(0)
        , objcAssignId(0)
        , objcRetainId(0)
        , objcCopyId(0)
        , objcNonatomicId(0)
243
    {}
con's avatar
con committed
244 245 246 247

    ~Data()
    {
        // symbols
248
        delete_array_entries(symbols);
con's avatar
con committed
249 250
    }

Roberto Raggi's avatar
Roberto Raggi committed
251 252
    template <typename _Iterator>
    const TemplateNameId *findOrInsertTemplateNameId(const Identifier *id, _Iterator first, _Iterator last)
con's avatar
con committed
253
    {
Roberto Raggi's avatar
Roberto Raggi committed
254
        return templateNameIds.intern(TemplateNameId(id, first, last));
con's avatar
con committed
255 256
    }

Roberto Raggi's avatar
Roberto Raggi committed
257
    const DestructorNameId *findOrInsertDestructorNameId(const Identifier *id)
con's avatar
con committed
258
    {
Roberto Raggi's avatar
Roberto Raggi committed
259
        return destructorNameIds.intern(DestructorNameId(id));
con's avatar
con committed
260 261
    }

Roberto Raggi's avatar
Roberto Raggi committed
262
    const OperatorNameId *findOrInsertOperatorNameId(int kind)
con's avatar
con committed
263
    {
Roberto Raggi's avatar
Roberto Raggi committed
264
        return operatorNameIds.intern(OperatorNameId(kind));
con's avatar
con committed
265 266
    }

Roberto Raggi's avatar
Roberto Raggi committed
267
    const ConversionNameId *findOrInsertConversionNameId(const FullySpecifiedType &type)
con's avatar
con committed
268
    {
Roberto Raggi's avatar
Roberto Raggi committed
269
        return conversionNameIds.intern(ConversionNameId(type));
con's avatar
con committed
270 271
    }

272
    const QualifiedNameId *findOrInsertQualifiedNameId(const Name *base, const Name *name)
con's avatar
con committed
273
    {
274
        return qualifiedNameIds.intern(QualifiedNameId(base, name));
con's avatar
con committed
275 276
    }

Roberto Raggi's avatar
Roberto Raggi committed
277 278
    template <typename _Iterator>
    const SelectorNameId *findOrInsertSelectorNameId(_Iterator first, _Iterator last, bool hasArguments)
279
    {
Roberto Raggi's avatar
Roberto Raggi committed
280
        return selectorNameIds.intern(SelectorNameId(first, last, hasArguments));
281 282
    }

con's avatar
con committed
283 284
    IntegerType *findOrInsertIntegerType(int kind)
    {
Roberto Raggi's avatar
Roberto Raggi committed
285
        return integerTypes.intern(IntegerType(kind));
con's avatar
con committed
286 287 288 289
    }

    FloatType *findOrInsertFloatType(int kind)
    {
Roberto Raggi's avatar
Roberto Raggi committed
290
        return floatTypes.intern(FloatType(kind));
con's avatar
con committed
291 292
    }

Roberto Raggi's avatar
Roberto Raggi committed
293
    PointerToMemberType *findOrInsertPointerToMemberType(const Name *memberName, const FullySpecifiedType &elementType)
con's avatar
con committed
294
    {
Roberto Raggi's avatar
Roberto Raggi committed
295
        return pointerToMemberTypes.intern(PointerToMemberType(memberName, elementType));
con's avatar
con committed
296 297
    }

298
    PointerType *findOrInsertPointerType(const FullySpecifiedType &elementType)
con's avatar
con committed
299
    {
Roberto Raggi's avatar
Roberto Raggi committed
300
        return pointerTypes.intern(PointerType(elementType));
con's avatar
con committed
301 302
    }

303
    ReferenceType *findOrInsertReferenceType(const FullySpecifiedType &elementType, bool rvalueRef)
con's avatar
con committed
304
    {
305
        return referenceTypes.intern(ReferenceType(elementType, rvalueRef));
con's avatar
con committed
306 307
    }

308
    ArrayType *findOrInsertArrayType(const FullySpecifiedType &elementType, unsigned size)
con's avatar
con committed
309
    {
Roberto Raggi's avatar
Roberto Raggi committed
310
        return arrayTypes.intern(ArrayType(elementType, size));
con's avatar
con committed
311 312
    }

Roberto Raggi's avatar
Roberto Raggi committed
313
    NamedType *findOrInsertNamedType(const Name *name)
con's avatar
con committed
314
    {
Roberto Raggi's avatar
Roberto Raggi committed
315
        return namedTypes.intern(NamedType(name));
con's avatar
con committed
316 317
    }

Roberto Raggi's avatar
Roberto Raggi committed
318
    Declaration *newDeclaration(unsigned sourceLocation, const Name *name)
con's avatar
con committed
319
    {
320
        Declaration *declaration = new Declaration(translationUnit, sourceLocation, name);
321
        symbols.push_back(declaration);
con's avatar
con committed
322 323 324
        return declaration;
    }

Roberto Raggi's avatar
Roberto Raggi committed
325
    Argument *newArgument(unsigned sourceLocation, const Name *name)
con's avatar
con committed
326
    {
327
        Argument *argument = new Argument(translationUnit, sourceLocation, name);
328
        symbols.push_back(argument);
con's avatar
con committed
329 330 331
        return argument;
    }

Roberto Raggi's avatar
Roberto Raggi committed
332 333
    TypenameArgument *newTypenameArgument(unsigned sourceLocation, const Name *name)
    {
334
        TypenameArgument *argument = new TypenameArgument(translationUnit, sourceLocation, name);
Roberto Raggi's avatar
Roberto Raggi committed
335 336 337 338
        symbols.push_back(argument);
        return argument;
    }

Roberto Raggi's avatar
Roberto Raggi committed
339
    Function *newFunction(unsigned sourceLocation, const Name *name)
con's avatar
con committed
340
    {
341
        Function *function = new Function(translationUnit, sourceLocation, name);
342
        symbols.push_back(function);
con's avatar
con committed
343 344 345
        return function;
    }

Roberto Raggi's avatar
Roberto Raggi committed
346
    BaseClass *newBaseClass(unsigned sourceLocation, const Name *name)
con's avatar
con committed
347
    {
348
        BaseClass *baseClass = new BaseClass(translationUnit, sourceLocation, name);
349
        symbols.push_back(baseClass);
con's avatar
con committed
350 351 352 353 354 355
        return baseClass;
    }

    Block *newBlock(unsigned sourceLocation)
    {
        Block *block = new Block(translationUnit, sourceLocation);
356
        symbols.push_back(block);
con's avatar
con committed
357 358 359
        return block;
    }

Roberto Raggi's avatar
Roberto Raggi committed
360
    Class *newClass(unsigned sourceLocation, const Name *name)
con's avatar
con committed
361
    {
362
        Class *klass = new Class(translationUnit, sourceLocation, name);
363
        symbols.push_back(klass);
con's avatar
con committed
364 365 366
        return klass;
    }

Roberto Raggi's avatar
Roberto Raggi committed
367
    Namespace *newNamespace(unsigned sourceLocation, const Name *name)
con's avatar
con committed
368
    {
369 370 371 372 373 374 375 376
        Namespace *ns = new Namespace(translationUnit, sourceLocation, name);
        symbols.push_back(ns);
        return ns;
    }

    Template *newTemplate(unsigned sourceLocation, const Name *name)
    {
        Template *ns = new Template(translationUnit, sourceLocation, name);
377
        symbols.push_back(ns);
con's avatar
con committed
378 379 380
        return ns;
    }

381 382
    NamespaceAlias *newNamespaceAlias(unsigned sourceLocation, const Name *name)
    {
383
        NamespaceAlias *ns = new NamespaceAlias(translationUnit, sourceLocation, name);
384 385 386 387
        symbols.push_back(ns);
        return ns;
    }

Roberto Raggi's avatar
Roberto Raggi committed
388
    UsingNamespaceDirective *newUsingNamespaceDirective(unsigned sourceLocation, const Name *name)
con's avatar
con committed
389
    {
390
        UsingNamespaceDirective *u = new UsingNamespaceDirective(translationUnit, sourceLocation, name);
391
        symbols.push_back(u);
con's avatar
con committed
392 393 394
        return u;
    }

Roberto Raggi's avatar
Roberto Raggi committed
395
    ForwardClassDeclaration *newForwardClassDeclaration(unsigned sourceLocation, const Name *name)
396
    {
397
        ForwardClassDeclaration *c = new ForwardClassDeclaration(translationUnit, sourceLocation, name);
398
        symbols.push_back(c);
399 400 401
        return c;
    }

Roberto Raggi's avatar
Roberto Raggi committed
402
    ObjCBaseClass *newObjCBaseClass(unsigned sourceLocation, const Name *name)
403 404
    {
        ObjCBaseClass *c = new ObjCBaseClass(translationUnit, sourceLocation, name);
405
        symbols.push_back(c);
406 407 408
        return c;
    }

Roberto Raggi's avatar
Roberto Raggi committed
409
    ObjCBaseProtocol *newObjCBaseProtocol(unsigned sourceLocation, const Name *name)
410 411
    {
        ObjCBaseProtocol *p = new ObjCBaseProtocol(translationUnit, sourceLocation, name);
412
        symbols.push_back(p);
413 414 415
        return p;
    }

Roberto Raggi's avatar
Roberto Raggi committed
416
    ObjCClass *newObjCClass(unsigned sourceLocation, const Name *name)
417 418
    {
        ObjCClass *c = new ObjCClass(translationUnit, sourceLocation, name);
419
        symbols.push_back(c);
420 421 422
        return c;
    }

Roberto Raggi's avatar
Roberto Raggi committed
423
    ObjCForwardClassDeclaration *newObjCForwardClassDeclaration(unsigned sourceLocation, const Name *name)
424 425
    {
        ObjCForwardClassDeclaration *fwd = new ObjCForwardClassDeclaration(translationUnit, sourceLocation, name);
426
        symbols.push_back(fwd);
427 428 429
        return fwd;
    }

Roberto Raggi's avatar
Roberto Raggi committed
430
    ObjCProtocol *newObjCProtocol(unsigned sourceLocation, const Name *name)
431 432
    {
        ObjCProtocol *p = new ObjCProtocol(translationUnit, sourceLocation, name);
433
        symbols.push_back(p);
434 435 436
        return p;
    }

Roberto Raggi's avatar
Roberto Raggi committed
437
    ObjCForwardProtocolDeclaration *newObjCForwardProtocolDeclaration(unsigned sourceLocation, const Name *name)
438 439
    {
        ObjCForwardProtocolDeclaration *fwd = new ObjCForwardProtocolDeclaration(translationUnit, sourceLocation, name);
440
        symbols.push_back(fwd);
441 442 443
        return fwd;
    }

Roberto Raggi's avatar
Roberto Raggi committed
444
    ObjCMethod *newObjCMethod(unsigned sourceLocation, const Name *name)
445 446
    {
        ObjCMethod *method = new ObjCMethod(translationUnit, sourceLocation, name);
447
        symbols.push_back(method);
448 449 450
        return method;
    }

Roberto Raggi's avatar
Roberto Raggi committed
451
    ObjCPropertyDeclaration *newObjCPropertyDeclaration(unsigned sourceLocation, const Name *name)
452 453
    {
        ObjCPropertyDeclaration *decl = new ObjCPropertyDeclaration(translationUnit, sourceLocation, name);
454
        symbols.push_back(decl);
455 456 457
        return decl;
    }

Roberto Raggi's avatar
Roberto Raggi committed
458
    Enum *newEnum(unsigned sourceLocation, const Name *name)
con's avatar
con committed
459
    {
460
        Enum *e = new Enum(translationUnit, sourceLocation, name);
461
        symbols.push_back(e);
con's avatar
con committed
462 463 464
        return e;
    }

Roberto Raggi's avatar
Roberto Raggi committed
465
    UsingDeclaration *newUsingDeclaration(unsigned sourceLocation, const Name *name)
con's avatar
con committed
466
    {
467
        UsingDeclaration *u = new UsingDeclaration(translationUnit, sourceLocation, name);
468
        symbols.push_back(u);
con's avatar
con committed
469 470 471 472 473 474
        return u;
    }

    Control *control;
    TranslationUnit *translationUnit;
    DiagnosticClient *diagnosticClient;
475 476 477

    TypeMatcher matcher;

con's avatar
con committed
478 479 480 481 482 483 484
    LiteralTable<Identifier> identifiers;
    LiteralTable<StringLiteral> stringLiterals;
    LiteralTable<NumericLiteral> numericLiterals;

    // ### replace std::map with lookup tables. ASAP!

    // names
Roberto Raggi's avatar
Roberto Raggi committed
485 486 487 488 489 490
    Table<DestructorNameId> destructorNameIds;
    Table<OperatorNameId> operatorNameIds;
    Table<ConversionNameId> conversionNameIds;
    Table<TemplateNameId> templateNameIds;
    Table<QualifiedNameId> qualifiedNameIds;
    Table<SelectorNameId> selectorNameIds;
con's avatar
con committed
491 492 493

    // types
    VoidType voidType;
Roberto Raggi's avatar
Roberto Raggi committed
494 495 496 497 498 499 500
    Table<IntegerType> integerTypes;
    Table<FloatType> floatTypes;
    Table<PointerToMemberType> pointerToMemberTypes;
    Table<PointerType> pointerTypes;
    Table<ReferenceType> referenceTypes;
    Table<ArrayType> arrayTypes;
    Table<NamedType> namedTypes;
con's avatar
con committed
501 502

    // symbols
503
    std::vector<Symbol *> symbols;
504

Erik Verbruggen's avatar
Erik Verbruggen committed
505
    const Identifier *deprecatedId;
506
    const Identifier *unavailableId;
507
    // ObjC context keywords:
Roberto Raggi's avatar
Roberto Raggi committed
508 509 510 511 512 513 514 515
    const Identifier *objcGetterId;
    const Identifier *objcSetterId;
    const Identifier *objcReadwriteId;
    const Identifier *objcReadonlyId;
    const Identifier *objcAssignId;
    const Identifier *objcRetainId;
    const Identifier *objcCopyId;
    const Identifier *objcNonatomicId;
con's avatar
con committed
516 517 518
};

Control::Control()
519 520 521
{
    d = new Data(this);

522 523 524 525 526 527 528 529 530 531 532
    d->deprecatedId = identifier("deprecated");
    d->unavailableId = identifier("unavailable");

    d->objcGetterId = identifier("getter");
    d->objcSetterId = identifier("setter");
    d->objcReadwriteId = identifier("readwrite");
    d->objcReadonlyId = identifier("readonly");
    d->objcAssignId = identifier("assign");
    d->objcRetainId = identifier("retain");
    d->objcCopyId = identifier("copy");
    d->objcNonatomicId = identifier("nonatomic");
533
}
con's avatar
con committed
534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553

Control::~Control()
{ delete d; }

TranslationUnit *Control::translationUnit() const
{ return d->translationUnit; }

TranslationUnit *Control::switchTranslationUnit(TranslationUnit *unit)
{
    TranslationUnit *previousTranslationUnit = d->translationUnit;
    d->translationUnit = unit;
    return previousTranslationUnit;
}

DiagnosticClient *Control::diagnosticClient() const
{ return d->diagnosticClient; }

void Control::setDiagnosticClient(DiagnosticClient *diagnosticClient)
{ d->diagnosticClient = diagnosticClient; }

Roberto Raggi's avatar
Roberto Raggi committed
554
const Identifier *Control::findIdentifier(const char *chars, unsigned size) const
555 556
{ return d->identifiers.findLiteral(chars, size); }

557
const Identifier *Control::identifier(const char *chars, unsigned size)
con's avatar
con committed
558 559
{ return d->identifiers.findOrInsertLiteral(chars, size); }

560
const Identifier *Control::identifier(const char *chars)
con's avatar
con committed
561
{
Roberto Raggi's avatar
Roberto Raggi committed
562
    unsigned length = std::strlen(chars);
563
    return identifier(chars, length);
con's avatar
con committed
564 565
}

566 567 568 569 570 571
Control::IdentifierIterator Control::firstIdentifier() const
{ return d->identifiers.begin(); }

Control::IdentifierIterator Control::lastIdentifier() const
{ return d->identifiers.end(); }

572 573 574 575 576 577 578 579 580 581 582 583
Control::StringLiteralIterator Control::firstStringLiteral() const
{ return d->stringLiterals.begin(); }

Control::StringLiteralIterator Control::lastStringLiteral() const
{ return d->stringLiterals.end(); }

Control::NumericLiteralIterator Control::firstNumericLiteral() const
{ return d->numericLiterals.begin(); }

Control::NumericLiteralIterator Control::lastNumericLiteral() const
{ return d->numericLiterals.end(); }

584
const StringLiteral *Control::stringLiteral(const char *chars, unsigned size)
con's avatar
con committed
585 586
{ return d->stringLiterals.findOrInsertLiteral(chars, size); }

587
const StringLiteral *Control::stringLiteral(const char *chars)
con's avatar
con committed
588
{
Roberto Raggi's avatar
Roberto Raggi committed
589
    unsigned length = std::strlen(chars);
590
    return stringLiteral(chars, length);
con's avatar
con committed
591 592
}

593
const NumericLiteral *Control::numericLiteral(const char *chars, unsigned size)
con's avatar
con committed
594 595
{ return d->numericLiterals.findOrInsertLiteral(chars, size); }

596
const NumericLiteral *Control::numericLiteral(const char *chars)
con's avatar
con committed
597
{
Roberto Raggi's avatar
Roberto Raggi committed
598
    unsigned length = std::strlen(chars);
599
    return numericLiteral(chars, length);
con's avatar
con committed
600 601
}

Roberto Raggi's avatar
Roberto Raggi committed
602 603 604
const TemplateNameId *Control::templateNameId(const Identifier *id,
                                              const FullySpecifiedType *const args,
                                              unsigned argv)
con's avatar
con committed
605
{
Roberto Raggi's avatar
Roberto Raggi committed
606
    return d->findOrInsertTemplateNameId(id, args, args + argv);
con's avatar
con committed
607 608
}

Roberto Raggi's avatar
Roberto Raggi committed
609
const DestructorNameId *Control::destructorNameId(const Identifier *id)
con's avatar
con committed
610 611
{ return d->findOrInsertDestructorNameId(id); }

Roberto Raggi's avatar
Roberto Raggi committed
612
const OperatorNameId *Control::operatorNameId(int kind)
con's avatar
con committed
613 614
{ return d->findOrInsertOperatorNameId(kind); }

Roberto Raggi's avatar
Roberto Raggi committed
615
const ConversionNameId *Control::conversionNameId(const FullySpecifiedType &type)
con's avatar
con committed
616 617
{ return d->findOrInsertConversionNameId(type); }

618
const QualifiedNameId *Control::qualifiedNameId(const Name *base, const Name *name)
con's avatar
con committed
619
{
620
    return d->findOrInsertQualifiedNameId(base, name);
con's avatar
con committed
621 622
}

Roberto Raggi's avatar
Roberto Raggi committed
623 624 625
const SelectorNameId *Control::selectorNameId(const Name *const *names,
                                              unsigned nameCount,
                                              bool hasArguments)
626
{
Roberto Raggi's avatar
Roberto Raggi committed
627
    return d->findOrInsertSelectorNameId(names, names + nameCount, hasArguments);
628 629 630
}


con's avatar
con committed
631 632 633 634 635 636 637 638 639
VoidType *Control::voidType()
{ return &d->voidType; }

IntegerType *Control::integerType(int kind)
{ return d->findOrInsertIntegerType(kind); }

FloatType *Control::floatType(int kind)
{ return d->findOrInsertFloatType(kind); }

Roberto Raggi's avatar
Roberto Raggi committed
640
PointerToMemberType *Control::pointerToMemberType(const Name *memberName, const FullySpecifiedType &elementType)
con's avatar
con committed
641 642
{ return d->findOrInsertPointerToMemberType(memberName, elementType); }

643
PointerType *Control::pointerType(const FullySpecifiedType &elementType)
con's avatar
con committed
644 645
{ return d->findOrInsertPointerType(elementType); }

646 647
ReferenceType *Control::referenceType(const FullySpecifiedType &elementType, bool rvalueRef)
{ return d->findOrInsertReferenceType(elementType, rvalueRef); }
con's avatar
con committed
648

649
ArrayType *Control::arrayType(const FullySpecifiedType &elementType, unsigned size)
con's avatar
con committed
650 651
{ return d->findOrInsertArrayType(elementType, size); }

Roberto Raggi's avatar
Roberto Raggi committed
652
NamedType *Control::namedType(const Name *name)
con's avatar
con committed
653 654
{ return d->findOrInsertNamedType(name); }

Roberto Raggi's avatar
Roberto Raggi committed
655
Argument *Control::newArgument(unsigned sourceLocation, const Name *name)
con's avatar
con committed
656 657
{ return d->newArgument(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
658 659 660
TypenameArgument *Control::newTypenameArgument(unsigned sourceLocation, const Name *name)
{ return d->newTypenameArgument(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
661
Function *Control::newFunction(unsigned sourceLocation, const Name *name)
con's avatar
con committed
662 663
{ return d->newFunction(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
664
Namespace *Control::newNamespace(unsigned sourceLocation, const Name *name)
con's avatar
con committed
665 666
{ return d->newNamespace(sourceLocation, name); }

667 668 669
Template *Control::newTemplate(unsigned sourceLocation, const Name *name)
{ return d->newTemplate(sourceLocation, name); }

670 671 672
NamespaceAlias *Control::newNamespaceAlias(unsigned sourceLocation, const Name *name)
{ return d->newNamespaceAlias(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
673
BaseClass *Control::newBaseClass(unsigned sourceLocation, const Name *name)
con's avatar
con committed
674 675
{ return d->newBaseClass(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
676
Class *Control::newClass(unsigned sourceLocation, const Name *name)
con's avatar
con committed
677 678
{ return d->newClass(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
679
Enum *Control::newEnum(unsigned sourceLocation, const Name *name)
con's avatar
con committed
680 681 682 683 684
{ return d->newEnum(sourceLocation, name); }

Block *Control::newBlock(unsigned sourceLocation)
{ return d->newBlock(sourceLocation); }

Roberto Raggi's avatar
Roberto Raggi committed
685
Declaration *Control::newDeclaration(unsigned sourceLocation, const Name *name)
con's avatar
con committed
686 687 688
{ return d->newDeclaration(sourceLocation, name); }

UsingNamespaceDirective *Control::newUsingNamespaceDirective(unsigned sourceLocation,
Roberto Raggi's avatar
Roberto Raggi committed
689
                                                                const Name *name)
con's avatar
con committed
690 691
{ return d->newUsingNamespaceDirective(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
692
UsingDeclaration *Control::newUsingDeclaration(unsigned sourceLocation, const Name *name)
con's avatar
con committed
693 694
{ return d->newUsingDeclaration(sourceLocation, name); }

695
ForwardClassDeclaration *Control::newForwardClassDeclaration(unsigned sourceLocation,
Roberto Raggi's avatar
Roberto Raggi committed
696
                                                             const Name *name)
697 698
{ return d->newForwardClassDeclaration(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
699
ObjCBaseClass *Control::newObjCBaseClass(unsigned sourceLocation, const Name *name)
700 701
{ return d->newObjCBaseClass(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
702
ObjCBaseProtocol *Control::newObjCBaseProtocol(unsigned sourceLocation, const Name *name)
703 704
{ return d->newObjCBaseProtocol(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
705
ObjCClass *Control::newObjCClass(unsigned sourceLocation, const Name *name)
706 707
{ return d->newObjCClass(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
708
ObjCForwardClassDeclaration *Control::newObjCForwardClassDeclaration(unsigned sourceLocation, const Name *name)
709 710
{ return d->newObjCForwardClassDeclaration(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
711
ObjCProtocol *Control::newObjCProtocol(unsigned sourceLocation, const Name *name)
712 713
{ return d->newObjCProtocol(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
714
ObjCForwardProtocolDeclaration *Control::newObjCForwardProtocolDeclaration(unsigned sourceLocation, const Name *name)
715
{ return d->newObjCForwardProtocolDeclaration(sourceLocation, name); }
716

Roberto Raggi's avatar
Roberto Raggi committed
717
ObjCMethod *Control::newObjCMethod(unsigned sourceLocation, const Name *name)
718 719
{ return d->newObjCMethod(sourceLocation, name); }

Roberto Raggi's avatar
Roberto Raggi committed
720
ObjCPropertyDeclaration *Control::newObjCPropertyDeclaration(unsigned sourceLocation, const Name *name)
721 722
{ return d->newObjCPropertyDeclaration(sourceLocation, name); }

Erik Verbruggen's avatar
Erik Verbruggen committed
723 724 725
const Identifier *Control::deprecatedId() const
{ return d->deprecatedId; }

726 727 728
const Identifier *Control::unavailableId() const
{ return d->unavailableId; }

Roberto Raggi's avatar
Roberto Raggi committed
729
const Identifier *Control::objcGetterId() const
730 731
{ return d->objcGetterId; }

Roberto Raggi's avatar
Roberto Raggi committed
732
const Identifier *Control::objcSetterId() const
733 734
{ return d->objcSetterId; }

Roberto Raggi's avatar
Roberto Raggi committed
735
const Identifier *Control::objcReadwriteId() const
736 737
{ return d->objcReadwriteId; }

Roberto Raggi's avatar
Roberto Raggi committed
738
const Identifier *Control::objcReadonlyId() const
739 740
{ return d->objcReadonlyId; }

Roberto Raggi's avatar
Roberto Raggi committed
741
const Identifier *Control::objcAssignId() const
742 743
{ return d->objcAssignId; }

Roberto Raggi's avatar
Roberto Raggi committed
744
const Identifier *Control::objcRetainId() const
745 746
{ return d->objcRetainId; }

Roberto Raggi's avatar
Roberto Raggi committed
747
const Identifier *Control::objcCopyId() const
748
{ return d->objcCopyId; }
Roberto Raggi's avatar
Roberto Raggi committed
749

Roberto Raggi's avatar
Roberto Raggi committed
750
const Identifier *Control::objcNonatomicId() const
751
{ return d->objcNonatomicId; }
752 753 754 755 756

bool Control::hasSymbol(Symbol *symbol) const
{
    return std::find(d->symbols.begin(), d->symbols.end(), symbol) != d->symbols.end();
}
757 758 759 760 761

void Control::squeeze()
{
    d->numericLiterals.reset();
}