xref: /freebsd/contrib/llvm-project/clang/lib/AST/SelectorLocationsKind.cpp (revision b64c5a0ace59af62eff52bfe110a521dc73c937b)
1 //===--- SelectorLocationsKind.cpp - Kind of selector locations -*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // Describes whether the identifier locations for a selector are "standard"
10 // or not.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #include "clang/AST/SelectorLocationsKind.h"
15 #include "clang/AST/Expr.h"
16 
17 using namespace clang;
18 
19 static SourceLocation getStandardSelLoc(unsigned Index,
20                                         Selector Sel,
21                                         bool WithArgSpace,
22                                         SourceLocation ArgLoc,
23                                         SourceLocation EndLoc) {
24   unsigned NumSelArgs = Sel.getNumArgs();
25   if (NumSelArgs == 0) {
26     assert(Index == 0);
27     if (EndLoc.isInvalid())
28       return SourceLocation();
29     const IdentifierInfo *II = Sel.getIdentifierInfoForSlot(0);
30     unsigned Len = II ? II->getLength() : 0;
31     return EndLoc.getLocWithOffset(-Len);
32   }
33 
34   assert(Index < NumSelArgs);
35   if (ArgLoc.isInvalid())
36     return SourceLocation();
37   const IdentifierInfo *II = Sel.getIdentifierInfoForSlot(Index);
38   unsigned Len = /* selector id */ (II ? II->getLength() : 0) + /* ':' */ 1;
39   if (WithArgSpace)
40     ++Len;
41   return ArgLoc.getLocWithOffset(-Len);
42 }
43 
44 namespace {
45 
46 template <typename T>
47 SourceLocation getArgLoc(T* Arg);
48 
49 template <>
50 SourceLocation getArgLoc<Expr>(Expr *Arg) {
51   return Arg->getBeginLoc();
52 }
53 
54 template <>
55 SourceLocation getArgLoc<ParmVarDecl>(ParmVarDecl *Arg) {
56   SourceLocation Loc = Arg->getBeginLoc();
57   if (Loc.isInvalid())
58     return Loc;
59   // -1 to point to left paren of the method parameter's type.
60   return Loc.getLocWithOffset(-1);
61 }
62 
63 template <typename T>
64 SourceLocation getArgLoc(unsigned Index, ArrayRef<T*> Args) {
65   return Index < Args.size() ? getArgLoc(Args[Index]) : SourceLocation();
66 }
67 
68 template <typename T>
69 SelectorLocationsKind hasStandardSelLocs(Selector Sel,
70                                          ArrayRef<SourceLocation> SelLocs,
71                                          ArrayRef<T *> Args,
72                                          SourceLocation EndLoc) {
73   // Are selector locations in standard position with no space between args ?
74   unsigned i;
75   for (i = 0; i != SelLocs.size(); ++i) {
76     if (SelLocs[i] != getStandardSelectorLoc(i, Sel, /*WithArgSpace=*/false,
77                                              Args, EndLoc))
78       break;
79   }
80   if (i == SelLocs.size())
81     return SelLoc_StandardNoSpace;
82 
83   // Are selector locations in standard position with space between args ?
84   for (i = 0; i != SelLocs.size(); ++i) {
85     if (SelLocs[i] != getStandardSelectorLoc(i, Sel, /*WithArgSpace=*/true,
86                                              Args, EndLoc))
87       return SelLoc_NonStandard;
88   }
89 
90   return SelLoc_StandardWithSpace;
91 }
92 
93 } // anonymous namespace
94 
95 SelectorLocationsKind
96 clang::hasStandardSelectorLocs(Selector Sel,
97                                ArrayRef<SourceLocation> SelLocs,
98                                ArrayRef<Expr *> Args,
99                                SourceLocation EndLoc) {
100   return hasStandardSelLocs(Sel, SelLocs, Args, EndLoc);
101 }
102 
103 SourceLocation clang::getStandardSelectorLoc(unsigned Index,
104                                              Selector Sel,
105                                              bool WithArgSpace,
106                                              ArrayRef<Expr *> Args,
107                                              SourceLocation EndLoc) {
108   return getStandardSelLoc(Index, Sel, WithArgSpace,
109                            getArgLoc(Index, Args), EndLoc);
110 }
111 
112 SelectorLocationsKind
113 clang::hasStandardSelectorLocs(Selector Sel,
114                                ArrayRef<SourceLocation> SelLocs,
115                                ArrayRef<ParmVarDecl *> Args,
116                                SourceLocation EndLoc) {
117   return hasStandardSelLocs(Sel, SelLocs, Args, EndLoc);
118 }
119 
120 SourceLocation clang::getStandardSelectorLoc(unsigned Index,
121                                              Selector Sel,
122                                              bool WithArgSpace,
123                                              ArrayRef<ParmVarDecl *> Args,
124                                              SourceLocation EndLoc) {
125   return getStandardSelLoc(Index, Sel, WithArgSpace,
126                            getArgLoc(Index, Args), EndLoc);
127 }
128