15a83710eSEric Fiselier //===----------------------------------------------------------------------===//
25a83710eSEric Fiselier //
357b08b09SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
457b08b09SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
557b08b09SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65a83710eSEric Fiselier //
75a83710eSEric Fiselier //===----------------------------------------------------------------------===//
85a83710eSEric Fiselier 
95a83710eSEric Fiselier // <functional>
105a83710eSEric Fiselier 
115a83710eSEric Fiselier // template<Returnable R, class T, CopyConstructible... Args>
125a83710eSEric Fiselier //   unspecified mem_fn(R (T::* pm)(Args...) volatile);
135a83710eSEric Fiselier 
145a83710eSEric Fiselier #include <functional>
155a83710eSEric Fiselier #include <cassert>
165a83710eSEric Fiselier 
17*7fc6a556SMarshall Clow #include "test_macros.h"
18*7fc6a556SMarshall Clow 
195a83710eSEric Fiselier struct A
205a83710eSEric Fiselier {
test0A215a83710eSEric Fiselier     char test0() volatile {return 'a';}
test1A225a83710eSEric Fiselier     char test1(int) volatile {return 'b';}
test2A235a83710eSEric Fiselier     char test2(int, double) volatile {return 'c';}
245a83710eSEric Fiselier };
255a83710eSEric Fiselier 
265a83710eSEric Fiselier template <class F>
275a83710eSEric Fiselier void
test0(F f)285a83710eSEric Fiselier test0(F f)
295a83710eSEric Fiselier {
305a83710eSEric Fiselier     {
315a83710eSEric Fiselier     A a;
325a83710eSEric Fiselier     assert(f(a) == 'a');
335a83710eSEric Fiselier     A* ap = &a;
345a83710eSEric Fiselier     assert(f(ap) == 'a');
355a83710eSEric Fiselier     volatile A* cap = &a;
365a83710eSEric Fiselier     assert(f(cap) == 'a');
375a83710eSEric Fiselier     const F& cf = f;
385a83710eSEric Fiselier     assert(cf(ap) == 'a');
395a83710eSEric Fiselier     }
405a83710eSEric Fiselier }
415a83710eSEric Fiselier 
425a83710eSEric Fiselier template <class F>
435a83710eSEric Fiselier void
test1(F f)445a83710eSEric Fiselier test1(F f)
455a83710eSEric Fiselier {
465a83710eSEric Fiselier     {
475a83710eSEric Fiselier     A a;
485a83710eSEric Fiselier     assert(f(a, 1) == 'b');
495a83710eSEric Fiselier     A* ap = &a;
505a83710eSEric Fiselier     assert(f(ap, 2) == 'b');
515a83710eSEric Fiselier     volatile A* cap = &a;
525a83710eSEric Fiselier     assert(f(cap, 2) == 'b');
535a83710eSEric Fiselier     const F& cf = f;
545a83710eSEric Fiselier     assert(cf(ap, 2) == 'b');
555a83710eSEric Fiselier     }
565a83710eSEric Fiselier }
575a83710eSEric Fiselier 
585a83710eSEric Fiselier template <class F>
595a83710eSEric Fiselier void
test2(F f)605a83710eSEric Fiselier test2(F f)
615a83710eSEric Fiselier {
625a83710eSEric Fiselier     {
635a83710eSEric Fiselier     A a;
645a83710eSEric Fiselier     assert(f(a, 1, 2) == 'c');
655a83710eSEric Fiselier     A* ap = &a;
665a83710eSEric Fiselier     assert(f(ap, 2, 3.5) == 'c');
675a83710eSEric Fiselier     volatile A* cap = &a;
685a83710eSEric Fiselier     assert(f(cap, 2, 3.5) == 'c');
695a83710eSEric Fiselier     const F& cf = f;
705a83710eSEric Fiselier     assert(cf(ap, 2, 3.5) == 'c');
715a83710eSEric Fiselier     }
725a83710eSEric Fiselier }
735a83710eSEric Fiselier 
main(int,char **)742df59c50SJF Bastien int main(int, char**)
755a83710eSEric Fiselier {
765a83710eSEric Fiselier     test0(std::mem_fn(&A::test0));
775a83710eSEric Fiselier     test1(std::mem_fn(&A::test1));
785a83710eSEric Fiselier     test2(std::mem_fn(&A::test2));
792df59c50SJF Bastien 
802df59c50SJF Bastien   return 0;
815a83710eSEric Fiselier }
82