_GLIBCXX_MOF_REF noexcept(_Noex)>
     : __polyfunc::_Cpy_base
     {
+      static_assert(
+       (std::__is_complete_or_unbounded(__type_identity<_ArgTypes>()) && ...),
+       "each parameter type must be a complete class");
+
       using _Base = __polyfunc::_Cpy_base;
       using _Invoker = __polyfunc::_Invoker<_Noex, _Res, _ArgTypes...>;
       using _Signature = _Invoker::_Signature;
 
     class function_ref<_Res(_ArgTypes...) _GLIBCXX_MOF_CV
                       noexcept(_Noex)>
     {
+      static_assert(
+       (std::__is_complete_or_unbounded(__type_identity<_ArgTypes>()) && ...),
+       "each parameter type must be a complete class");
+
       using _Invoker = __polyfunc::_Invoker<_Noex, _Res, _ArgTypes...>;
       using _Signature = _Invoker::_Signature;
 
 
      };
 
    template<typename _Tp>
-     using __param_t = __conditional_t<is_scalar_v<_Tp>, _Tp, _Tp&&>;
+     consteval bool
+     __pass_by_value()
+     {
+       // n.b. sizeof(Incomplete&) is ill-formed for incomplete types,
+       // so we check is_reference_v first.
+       if constexpr (is_reference_v<_Tp> || is_scalar_v<_Tp>)
+        return true;
+       else
+        // n.b. we already asserted that types are complete in wrappers,
+        // avoid triggering additional errors from this function.
+        if constexpr (std::__is_complete_or_unbounded(__type_identity<_Tp>()))
+          if constexpr (sizeof(_Tp) <= 2 * sizeof(void*))
+            return is_trivially_move_constructible_v<_Tp>
+                   && is_trivially_destructible_v<_Tp>;
+       return false;
+     }
+
+   template<typename _Tp>
+     using __param_t = __conditional_t<__pass_by_value<_Tp>(), _Tp, _Tp&&>;
 
    template<bool _Noex, typename _Ret, typename... _Args>
      using _Invoker = _Base_invoker<_Noex, remove_cv_t<_Ret>, __param_t<_Args>...>;
 
                               _GLIBCXX_MOF_REF noexcept(_Noex)>
     : __polyfunc::_Mo_base
     {
+      static_assert(
+       (std::__is_complete_or_unbounded(__type_identity<_ArgTypes>()) && ...),
+       "each parameter type must be a complete class");
+
       using _Base = __polyfunc::_Mo_base;
       using _Invoker = __polyfunc::_Invoker<_Noex, _Res, _ArgTypes...>;
       using _Signature = _Invoker::_Signature;
 
 }
 
 struct Incomplete;
+enum CompleteEnum : int;
 
 void
 test_params()
 {
-  std::copyable_function<void(Incomplete)> f1;
-  std::copyable_function<void(Incomplete&)> f2;
-  std::copyable_function<void(Incomplete&&)> f3;
+  std::copyable_function<void(Incomplete&)> f1;
+  std::copyable_function<void(Incomplete&&)> f2;
+  std::copyable_function<void(CompleteEnum)> f4;
 }
 
 int main()
 
 // { dg-require-effective-target hosted }
 
 #include <functional>
+#include <string_view>
 #include <testsuite_hooks.h>
 
 using std::copyable_function;
 static_assert( !std::is_constructible_v<std::copyable_function<void() const>,
                                        std::copyable_function<void()>> );
 
+using FuncType = int(int);
+
+// Top level const qualifiers are ignored and decay is performed in parameters
+// of function_types.
+static_assert( std::is_same_v<std::copyable_function<void(int const)>,
+                             std::copyable_function<void(int)>> );
+static_assert( std::is_same_v<std::copyable_function<void(int[2])>,
+                             std::copyable_function<void(int*)>>);
+static_assert( std::is_same_v<std::copyable_function<void(int[])>,
+                             std::copyable_function<void(int*)>>);
+static_assert( std::is_same_v<std::copyable_function<void(int const[5])>,
+                             std::copyable_function<void(int const*)>>);
+static_assert( std::is_same_v<std::copyable_function<void(FuncType)>,
+                             std::copyable_function<void(FuncType*)>>);
+
 // Non-trivial args, guarantess that type is not passed by copy
 struct CountedArg
 {
   VERIFY( f2(c) == 2 );
 }
 
+void
+test07()
+{
+  // Scalar types and small trivially move constructible types are passed
+  // by value to invoker. So int&& signature is not compatible for such types.
+  auto fi = [](CountedArg const& arg, int) noexcept { return arg.counter; };
+  std::copyable_function<int(CountedArg, int) const noexcept> ci1(fi);
+  VERIFY( ci1(c, 0) == 1 );
+  std::copyable_function<int(CountedArg, int&&) const noexcept> ci2(ci1);
+  VERIFY( ci2(c, 0) == 2 );
+
+  auto fs = [](CountedArg const& arg, std::string_view) noexcept { return arg.counter; };
+  std::copyable_function<int(CountedArg, std::string_view) const noexcept> cs1(fs);
+  VERIFY( cs1(c, "") == 1 );
+  std::copyable_function<int(CountedArg, std::string_view&&) const noexcept> cs2(cs1);
+  VERIFY( cs2(c, "") == 2 );
+}
+
 int main()
 {
   test01();
   test04();
   test05();
   test06();
+  test07();
 }
 
--- /dev/null
+// { dg-do compile { target c++26 } }
+
+#include <functional>
+
+struct IncompleteClass;
+
+using T1 = std::copyable_function<int(IncompleteClass)>::result_type; // { dg-error "here" }
+using T2 = std::copyable_function<int(int, IncompleteClass)>::result_type; // { dg-error "here" }
+
+enum Enum {
+  x = [] {
+    // Enum enumeration is incomplete here
+    using T3 = std::copyable_function<int(Enum)>::result_type; // { dg-error "here" }
+    return T3(1);
+  }()
+};
+
+// { dg-error "static assertion failed" "" { target *-*-* } 0 }
 
 }
 
 struct Incomplete;
+enum CompleteEnum : int;
 
 void
 test_params()
 {
   auto f = [](auto&&) {};
-  // There is discussion if this is supported.
-  // std::function_ref<void(Incomplete)> f1(f);
-  std::function_ref<void(Incomplete&)> f2(f);
-  // See PR120259, this should be supported.
-  // std::function_ref<void(Incomplete&&)> f3(f);
+  std::function_ref<void(Incomplete&)> f1(f);
+  // See PR libstdc++/120259, this should be supported.
+  // std::function_ref<void(Incomplete&&)> f2(f);
+  std::function_ref<void(CompleteEnum)> f3(f);
 }
 
 int main()
 
 // { dg-require-effective-target hosted }
 
 #include <functional>
+#include <string_view>
 #include <testsuite_hooks.h>
 
 using std::function_ref;
 };
 CountedArg const c;
 
+using FuncType = int(int);
+
+// Top level const qualifiers are ignored in function types, and decay
+// is performed.
+static_assert( std::is_same_v<std::function_ref<void(int const)>,
+                             std::function_ref<void(int)>> );
+static_assert( std::is_same_v<std::function_ref<void(int[2])>,
+                             std::function_ref<void(int*)>>);
+static_assert( std::is_same_v<std::function_ref<void(int[])>,
+                             std::function_ref<void(int*)>>);
+static_assert( std::is_same_v<std::function_ref<void(int const[5])>,
+                             std::function_ref<void(int const*)>>);
+static_assert( std::is_same_v<std::function_ref<void(FuncType)>,
+                             std::function_ref<void(FuncType*)>>);
+
 // The C++26 [func.wrap.general] p2 does not currently cover funciton_ref,
 // so we make extra copies of arguments.
 
   return true;
 };
 
+void
+test09()
+{
+  // Scalar types and small trivially move constructible types are passed
+  // by value to invoker. So int&& signature is not compatible for such types.
+  auto fi = [](CountedArg const& arg, int) noexcept { return arg.counter; };
+  std::function_ref<int(CountedArg, int) const noexcept> ri1(fi);
+  VERIFY( ri1(c, 0) == 1 );
+  std::function_ref<int(CountedArg, int&&) const noexcept> ri2(ri1);
+  VERIFY( ri2(c, 0) == 2 );
+
+  auto fs = [](CountedArg const& arg, std::string_view) noexcept { return arg.counter; };
+  std::function_ref<int(CountedArg, std::string_view) const noexcept> rs1(fs);
+  VERIFY( rs1(c, "") == 1 );
+  std::function_ref<int(CountedArg, std::string_view&&) const noexcept> rs2(rs1);
+  VERIFY( rs2(c, "") == 2 );
+}
 
 int main()
 {
   test05();
   test06();
   test07();
+  test09();
 
   static_assert( test08() );
 }
 
--- /dev/null
+// { dg-do compile { target c++26 } }
+
+#include <functional>
+
+struct IncompleteClass;
+
+int a1 = alignof(std::function_ref<int(IncompleteClass)>); // { dg-error "here" }
+int a2 = alignof(std::function_ref<int(int, IncompleteClass)>); // { dg-error "here" }
+
+enum Enum {
+  x = [] {
+    // Enum enumeration is incomplete here
+    int a3 = alignof(std::function_ref<int(Enum)>); // { dg-error "here" }
+    return 1;
+  }()
+};
+
+// { dg-error "static assertion failed" "" { target *-*-* } 0 }
 
 }
 
 struct Incomplete;
+enum CompleteEnum : int;
 
 void
 test_params()
 {
-  std::move_only_function<void(Incomplete)> f1;
-  std::move_only_function<void(Incomplete&)> f2;
-  std::move_only_function<void(Incomplete&&)> f3;
+  std::move_only_function<void(Incomplete&)> f1;
+  std::move_only_function<void(Incomplete&&)> f2;
+  std::move_only_function<void(CompleteEnum)> f4;
 }
 
 int main()
 
 // { dg-require-effective-target hosted }
 
 #include <functional>
+#include <string_view>
 #include <testsuite_hooks.h>
 
 using std::move_only_function;
 static_assert( !std::is_constructible_v<std::move_only_function<void() const>,
                                        std::move_only_function<void()>> );
 
+using FuncType = int(int);
+
+// Top level const qualifiers are ignored in function types, and decay
+// is performed.
+static_assert( std::is_same_v<std::move_only_function<void(int const)>,
+                             std::move_only_function<void(int)>> );
+static_assert( std::is_same_v<std::move_only_function<void(int[2])>,
+                             std::move_only_function<void(int*)>>);
+static_assert( std::is_same_v<std::move_only_function<void(int[])>,
+                             std::move_only_function<void(int*)>>);
+static_assert( std::is_same_v<std::move_only_function<void(int const[5])>,
+                             std::move_only_function<void(int const*)>>);
+static_assert( std::is_same_v<std::move_only_function<void(FuncType)>,
+                             std::move_only_function<void(FuncType*)>>);
+
 // Non-trivial args, guarantess that type is not passed by copy
 struct CountedArg
 {
   VERIFY( m1(c) == 2 );
 }
 
+void
+test07()
+{
+  // Scalar types and small trivially move constructible types are passed
+  // by value to invoker. So int&& signature is not compatible for such types.
+  auto fi = [](CountedArg const& arg, int) noexcept { return arg.counter; };
+  std::move_only_function<int(CountedArg, int) const noexcept> mi1(fi);
+  VERIFY( mi1(c, 0) == 1 );
+  std::move_only_function<int(CountedArg, int&&) const noexcept> mi2(std::move(mi1));
+  VERIFY( mi2(c, 0) == 2 );
+
+  auto fs = [](CountedArg const& arg, std::string_view) noexcept { return arg.counter; };
+  std::move_only_function<int(CountedArg, std::string_view) const noexcept> ms1(fs);
+  VERIFY( ms1(c, "") == 1 );
+  std::move_only_function<int(CountedArg, std::string_view&&) const noexcept> ms2(std::move(ms1));
+  VERIFY( ms2(c, "") == 2 );
+}
+
 int main()
 {
   test01();
   test04();
   test05();
   test06();
+  test07();
 }
 
--- /dev/null
+// { dg-do compile { target c++23 } }
+
+#include <functional>
+
+struct IncompleteClass;
+
+using T1 = std::move_only_function<int(IncompleteClass)>::result_type; // { dg-error "here" }
+using T2 = std::move_only_function<int(int, IncompleteClass)>::result_type; // { dg-error "here" }
+
+enum Enum {
+  x = [] {
+    // Enum enumeration is incomplete here
+    using T3 = std::move_only_function<int(Enum)>::result_type; // { dg-error "here" }
+    return T3(1);
+  }()
+};
+
+// { dg-error "static assertion failed" "" { target *-*-* } 0 }