1 // Copyright 2016 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include <type_traits>
6 
7 namespace not_blink {
8 
function(int x)9 void function(int x) {}
10 
11 class Class {
12  public:
method()13   void method() {}
virtualMethod()14   virtual void virtualMethod() {}
15   template <typename T>
methodTemplate(T)16   void methodTemplate(T) {}
17   template <typename T>
staticMethodTemplate(T)18   static void staticMethodTemplate(T) {}
19 };
20 
21 template <typename T>
functionTemplate(T x)22 void functionTemplate(T x) {}
23 
24 template <typename T = Class>
functionTemplate2()25 void functionTemplate2() {
26   T::staticMethodTemplate(123);
27 }
28 
29 template <typename T = Class>
30 class TemplatedClass {
31  public:
anotherMethod()32   void anotherMethod() { T::staticMethodTemplate(123); }
33 };
34 
35 }  // not_blink
36 
37 namespace blink {
38 
functionNotMarkedConstexpr(int a)39 bool functionNotMarkedConstexpr(int a) {
40   return a == 4 || a == 10;
41 }
42 
43 template <typename T>
templatedFunctionNotMarkedConstexpr(T t)44 bool templatedFunctionNotMarkedConstexpr(T t) {
45   return !!t;
46 }
47 
48 int g_globalNumber;
49 
50 template <typename T, int number>
F()51 void F() {
52   // These are const but hacker_case so we leave them alone.
53   const int maybe_a_const = sizeof(T);
54   const int is_a_const = number;
55   // These are const expressions so they get a k prefix.
56   const int maybeAConstToo = sizeof(T);
57   const int isAConstToo = number;
58   // These are built from calls to functions which produces inconsistent
59   // results so they should not be considered const to be safe.
60   const bool fromAMethod = functionNotMarkedConstexpr(number);
61   const bool fromATemplatedMethod = templatedFunctionNotMarkedConstexpr(number);
62   // A complex statement of const things is const.
63   const bool complexConst = number || (number + 1);
64   // A complex statement with a non-const thing is not const.
65   const bool complexNotConst = number || (g_globalNumber + 1);
66   // A const built from other consts is a const.
67   const bool constFromAConst = complexConst || number;
68 }
69 
70 template <int number, typename... T>
F()71 void F() {
72   // These are const but hacker_case so we leave them alone.
73   const int maybe_a_const = sizeof...(T);
74   const int is_a_const = number;
75   // These are const expressions so they get a k prefix.
76   const int maybeAConstToo = sizeof...(T);
77   const int isAConstToo = number;
78 }
79 
80 namespace test_member_in_template {
81 
82 template <typename T>
83 class HasAMember {
84  public:
HasAMember()85   HasAMember() {}
HasAMember(const T &)86   HasAMember(const T&) {}
87 
usesMember()88   void usesMember() { const int notConst = m_i; }
89   void alsoUsesMember();
90 
91  private:
92   int m_i;
93 };
94 
95 template <typename T>
alsoUsesMember()96 void HasAMember<T>::alsoUsesMember() {
97   const int notConst = m_i;
98 }
99 
100 template <typename T>
basedOnSubType(const HasAMember<T> & t)101 static void basedOnSubType(const HasAMember<T>& t) {
102   const HasAMember<T> problematicNotConst(t);
103 }
104 
Run()105 void Run() {
106   HasAMember<int>().usesMember();
107 
108   basedOnSubType<int>(HasAMember<int>());
109   enum E { A };
110   basedOnSubType<E>(HasAMember<E>());
111 }
112 }
113 
114 namespace test_template_arg_is_function {
115 
f(int x)116 void f(int x) {}
117 
118 template <typename T, void g(T)>
h(T x)119 void h(T x) {
120   g(x);
121 }
122 
test()123 void test() {
124   // f should be rewritten.
125   h<int, f>(0);
126   // Non-Blink should stay the same.
127   h<int, not_blink::function>(1);
128 
129   // The int one makes the methods called from F() considered as constexpr, and
130   // can be collapsed to not have template arguments before it reaches the AST.
131   F<int, 10>();
132   // The enum one makes them not constexpr, as it doesn't collapse away the
133   // template stuff as much. This can lead to conflicting decisions about
134   // the names inside F() vs the above instantiation.
135   enum E { A };
136   F<E, 11>();
137 }
138 
139 }  // namespace test_template_arg_is_function
140 
141 namespace test_template_arg_is_method {
142 
143 class Class {
144  public:
method()145   void method() {}
146 };
147 
148 template <typename T, void (T::*g)()>
h(T && x)149 void h(T&& x) {
150   (x.*g)();
151 }
152 
test()153 void test() {
154   // method should be rewritten.
155   h<Class, &Class::method>(Class());
156   // Non-Blink should stay the same.
157   h<not_blink::Class, &not_blink::Class::method>(not_blink::Class());
158 }
159 
160 }  // namespace test_template_arg_is_method
161 
162 namespace test_template_arg_is_function_template {
163 
164 namespace nested {
165 template <typename T>
f(T)166 void f(T) {}
167 }
168 
169 template <typename T, void g(T)>
h(T x)170 void h(T x) {
171   g(x);
172 }
173 
test()174 void test() {
175   // f should be rewritten.
176   h<int, nested::f>(0);
177   // Non-Blink should stay the same.
178   h<int, not_blink::functionTemplate>(1);
179 }
180 
181 }  // namespace test_template_arg_is_function_template
182 
183 namespace test_template_arg_is_method_template_in_non_member_context {
184 
185 struct Class {
186   template <typename T>
fblink::test_template_arg_is_method_template_in_non_member_context::Class187   static void f(T) {}
188 };
189 
190 template <typename T, void g(T)>
h(T x)191 void h(T x) {
192   g(x);
193 }
194 
test()195 void test() {
196   // f should be rewritten.
197   h<int, Class::f>(0);
198   // Non-Blink should stay the same.
199   h<int, not_blink::Class::staticMethodTemplate>(1);
200 }
201 
202 }  // test_template_arg_is_method_template_in_non_member_context
203 
204 namespace test_inherited_field {
205 
206 template <typename T>
207 class BaseClass {
208  public:
209   unsigned long m_size;
210 };
211 
212 template <typename T>
213 class DerivedClass : protected BaseClass<T> {
214  private:
215   using Base = BaseClass<T>;
216   // https://crbug.com/640016: Need to rewrite |m_size| into |size_|.
217   using Base::m_size;
method()218   void method() { m_size = 123; }
219 };
220 
221 }  // namespace test_inherited_field
222 
223 namespace test_template_arg_is_method_template_in_member_context {
224 
225 struct Class {
226   template <typename T>
fblink::test_template_arg_is_method_template_in_member_context::Class227   static void f(T) {}
228 };
229 
230 struct Class2 {
231   template <typename T>
fblink::test_template_arg_is_method_template_in_member_context::Class2232   void f(T x) {
233     // f should be rewritten.
234     Class c;
235     c.f(x);
236     // Non-Blink should stay the same.
237     not_blink::Class c2;
238     c2.method(x);
239   }
240 };
241 
242 }  // namespace test_template_arg_is_method_template_in_member_context
243 
244 namespace test_unnamed_arg {
245 
246 template <typename T>
247 class Class {
248  public:
249   // Test for https://crbug.com/598141 - shouldn't rewrite
250   //    ...int);
251   // into
252   //    ...intdata_size;
253   void f(int);
254 };
255 
256 template <typename T>
f(int dataSize)257 void Class<T>::f(int dataSize){};
258 
foo()259 void foo() {
260   Class<char>().f(123);
261 };
262 
263 }  // namespace test_unnamed_arg
264 
265 namespace cxx_dependent_scope_member_expr_testing {
266 
267 class PartitionAllocator {
268  public:
method()269   static void method() {}
270 };
271 
272 template <typename Allocator = PartitionAllocator>
273 class Vector {
274  public:
275   // https://crbug.com/582315: |Allocator::method| is a
276   // CXXDependentScopeMemberExpr.
anotherMethod()277   void anotherMethod() {
278     if (std::is_class<Allocator>::value)  // Shouldn't rename |value|
279       Allocator::method();                // Should rename |method| -> |Method|.
280   }
281 };
282 
283 template <typename Allocator = PartitionAllocator>
test()284 void test() {
285   // https://crbug.com/582315: |Allocator::method| is a
286   // DependentScopeDeclRefExpr.
287   if (std::is_class<Allocator>::value)  // Shouldn't rename |value|.
288     Allocator::method();                // Should rename |method|.
289 }
290 
291 class InterceptingCanvasBase : public ::not_blink::Class {
292  public:
virtualMethodInBlink()293   virtual void virtualMethodInBlink(){};
294 };
295 
296 template <typename DerivedCanvas>
297 class InterceptingCanvas : public InterceptingCanvasBase {
298  public:
virtualMethod()299   void virtualMethod() override {
300     this->Class::virtualMethod();  // https://crbug.com/582315#c19
301     this->InterceptingCanvasBase::virtualMethodInBlink();
302   }
303 };
304 
305 template <typename T>
306 class ThreadSpecific {
307  public:
308   T* operator->();
309   operator T*();
310 };
311 
312 template <typename T>
operator T*()313 inline ThreadSpecific<T>::operator T*() {
314   return nullptr;
315 }
316 
317 template <typename T>
operator ->()318 inline T* ThreadSpecific<T>::operator->() {
319   return operator T*();
320 }
321 
322 class Class {
323  public:
virtualMethodInBlink()324   virtual void virtualMethodInBlink() {}
325 };
326 
327 }  // namespace cxx_dependent_scope_member_expr_testing
328 
329 namespace blacklisting_of_renaming_of_begin_method {
330 
331 template <typename T>
332 class IntrusiveHeap {
333  public:
334   // https://crbug.com/672353: |begin| shouldn't be rewritten to |Begin|.
begin() const335   const T* begin() const { return nullptr; }
336 };
337 
338 }  // namespace blacklisting_of_renaming_of_begin_method
339 
340 namespace specializations {
341 
342 template <unsigned long sizeOfValue>
343 int toV8SignedIntegerInternal(long value);
344 
345 template <>
toV8SignedIntegerInternal(long value)346 int toV8SignedIntegerInternal<4>(long value) {
347   return 123 + value;
348 }
349 
350 template <>
toV8SignedIntegerInternal(long value)351 int toV8SignedIntegerInternal<8>(long value) {
352   return 456 + value;
353 }
354 
toV8(int value)355 int toV8(int value) {
356   return toV8SignedIntegerInternal<sizeof value>(value);
357 }
358 
toV8(long value)359 int toV8(long value) {
360   return toV8SignedIntegerInternal<sizeof value>(value);
361 }
362 
363 }  // namespace specializations
364 
365 }  // namespace blink
366 
367 namespace not_blink {
368 
369 namespace cxx_dependent_scope_member_expr_testing {
370 
371 class Base : public ::blink::cxx_dependent_scope_member_expr_testing::Class {
372  public:
virtualMethod()373   virtual void virtualMethod() {}
374 };
375 
376 template <typename T>
377 class Derived : public Base {
378  public:
virtualMethod()379   void virtualMethod() override {
380     this->Class::virtualMethodInBlink();
381     this->Base::virtualMethod();
382   }
383 };
384 
385 }  // namespace cxx_dependent_scope_member_expr_testing
386 
387 namespace blink_methods_called_from_mojo_traits_are_not_rewritten {
388 
389 template <typename V>
390 struct MapTraits;
391 
392 template <typename V>
393 struct MapTraits<blink::test_unnamed_arg::Class<V>> {
SetToEmptynot_blink::blink_methods_called_from_mojo_traits_are_not_rewritten::MapTraits394   static void SetToEmpty(blink::test_unnamed_arg::Class<V>* output) {
395     // Need to rewrite |f| to |F| below (because this method name
396     // does get rewritten when processing blink::test_unnamed_arg::Class).
397     // See also https://crbug.com/670434.
398     output->f(123);
399   }
400 };
401 
402 }  // namespace blink_methods_called_from_mojo_traits_are_not_rewritten
403 
404 }  // namespace not_blink
405