2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2005-04-16 15:20:36 -07:00
/*
* linux / net / sunrpc / sunrpc_syms . c
*
* Symbols exported by the sunrpc module .
*
* Copyright ( C ) 1997 Olaf Kirch < okir @ monad . swb . de >
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/uio.h>
# include <linux/unistd.h>
# include <linux/init.h>
# include <linux/sunrpc/sched.h>
# include <linux/sunrpc/clnt.h>
# include <linux/sunrpc/svc.h>
# include <linux/sunrpc/svcsock.h>
# include <linux/sunrpc/auth.h>
# include <linux/workqueue.h>
# include <linux/sunrpc/rpc_pipe_fs.h>
2007-09-10 13:47:31 -04:00
# include <linux/sunrpc/xprtsock.h>
2005-04-16 15:20:36 -07:00
2020-05-22 12:01:32 +10:00
# include "sunrpc.h"
2021-06-08 15:59:10 -04:00
# include "sysfs.h"
2010-09-27 14:01:27 +04:00
# include "netns.h"
netns: make struct pernet_operations::id unsigned int
Make struct pernet_operations::id unsigned.
There are 2 reasons to do so:
1)
This field is really an index into an zero based array and
thus is unsigned entity. Using negative value is out-of-bound
access by definition.
2)
On x86_64 unsigned 32-bit data which are mixed with pointers
via array indexing or offsets added or subtracted to pointers
are preffered to signed 32-bit data.
"int" being used as an array index needs to be sign-extended
to 64-bit before being used.
void f(long *p, int i)
{
g(p[i]);
}
roughly translates to
movsx rsi, esi
mov rdi, [rsi+...]
call g
MOVSX is 3 byte instruction which isn't necessary if the variable is
unsigned because x86_64 is zero extending by default.
Now, there is net_generic() function which, you guessed it right, uses
"int" as an array index:
static inline void *net_generic(const struct net *net, int id)
{
...
ptr = ng->ptr[id - 1];
...
}
And this function is used a lot, so those sign extensions add up.
Patch snipes ~1730 bytes on allyesconfig kernel (without all junk
messing with code generation):
add/remove: 0/0 grow/shrink: 70/598 up/down: 396/-2126 (-1730)
Unfortunately some functions actually grow bigger.
This is a semmingly random artefact of code generation with register
allocator being used differently. gcc decides that some variable
needs to live in new r8+ registers and every access now requires REX
prefix. Or it is shifted into r12, so [r12+0] addressing mode has to be
used which is longer than [r8]
However, overall balance is in negative direction:
add/remove: 0/0 grow/shrink: 70/598 up/down: 396/-2126 (-1730)
function old new delta
nfsd4_lock 3886 3959 +73
tipc_link_build_proto_msg 1096 1140 +44
mac80211_hwsim_new_radio 2776 2808 +32
tipc_mon_rcv 1032 1058 +26
svcauth_gss_legacy_init 1413 1429 +16
tipc_bcbase_select_primary 379 392 +13
nfsd4_exchange_id 1247 1260 +13
nfsd4_setclientid_confirm 782 793 +11
...
put_client_renew_locked 494 480 -14
ip_set_sockfn_get 730 716 -14
geneve_sock_add 829 813 -16
nfsd4_sequence_done 721 703 -18
nlmclnt_lookup_host 708 686 -22
nfsd4_lockt 1085 1063 -22
nfs_get_client 1077 1050 -27
tcf_bpf_init 1106 1076 -30
nfsd4_encode_fattr 5997 5930 -67
Total: Before=154856051, After=154854321, chg -0.00%
Signed-off-by: Alexey Dobriyan <adobriyan@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2016-11-17 04:58:21 +03:00
unsigned int sunrpc_net_id ;
2012-01-19 21:42:37 +04:00
EXPORT_SYMBOL_GPL ( sunrpc_net_id ) ;
2010-09-27 14:01:27 +04:00
static __net_init int sunrpc_init_net ( struct net * net )
{
2010-09-27 14:01:58 +04:00
int err ;
2012-01-12 22:07:51 +04:00
struct sunrpc_net * sn = net_generic ( net , sunrpc_net_id ) ;
2010-09-27 14:01:58 +04:00
err = rpc_proc_init ( net ) ;
if ( err )
goto err_proc ;
2010-09-27 14:02:29 +04:00
err = ip_map_cache_create ( net ) ;
if ( err )
goto err_ipmap ;
2012-01-19 21:42:29 +04:00
err = unix_gid_cache_create ( net ) ;
if ( err )
goto err_unixgid ;
2013-11-14 07:25:17 -05:00
err = rpc_pipefs_init_net ( net ) ;
if ( err )
goto err_pipefs ;
2012-01-12 22:07:51 +04:00
INIT_LIST_HEAD ( & sn - > all_clients ) ;
spin_lock_init ( & sn - > rpc_client_lock ) ;
2012-02-16 17:42:12 +04:00
spin_lock_init ( & sn - > rpcb_clnt_lock ) ;
2010-09-27 14:01:27 +04:00
return 0 ;
2010-09-27 14:01:58 +04:00
2013-11-14 07:25:17 -05:00
err_pipefs :
unix_gid_cache_destroy ( net ) ;
2012-01-19 21:42:29 +04:00
err_unixgid :
ip_map_cache_destroy ( net ) ;
2010-09-27 14:02:29 +04:00
err_ipmap :
rpc_proc_exit ( net ) ;
2010-09-27 14:01:58 +04:00
err_proc :
return err ;
2010-09-27 14:01:27 +04:00
}
static __net_exit void sunrpc_exit_net ( struct net * net )
{
2017-11-12 11:48:43 +03:00
struct sunrpc_net * sn = net_generic ( net , sunrpc_net_id ) ;
2013-11-14 07:25:17 -05:00
rpc_pipefs_exit_net ( net ) ;
2012-01-19 21:42:29 +04:00
unix_gid_cache_destroy ( net ) ;
2010-09-27 14:02:29 +04:00
ip_map_cache_destroy ( net ) ;
2010-09-27 14:01:58 +04:00
rpc_proc_exit ( net ) ;
2017-11-12 11:48:43 +03:00
WARN_ON_ONCE ( ! list_empty ( & sn - > all_clients ) ) ;
2010-09-27 14:01:27 +04:00
}
static struct pernet_operations sunrpc_net_ops = {
. init = sunrpc_init_net ,
. exit = sunrpc_exit_net ,
. id = & sunrpc_net_id ,
. size = sizeof ( struct sunrpc_net ) ,
} ;
2005-04-16 15:20:36 -07:00
static int __init
init_sunrpc ( void )
{
2012-04-05 21:04:37 +04:00
int err = rpc_init_mempool ( ) ;
2005-04-16 15:20:36 -07:00
if ( err )
goto out ;
2010-07-31 14:29:07 -04:00
err = rpcauth_init_module ( ) ;
if ( err )
2012-04-05 21:04:37 +04:00
goto out2 ;
2010-09-27 14:01:27 +04:00
cache_initialize ( ) ;
err = register_pernet_subsys ( & sunrpc_net_ops ) ;
2012-04-05 21:04:37 +04:00
if ( err )
goto out3 ;
err = register_rpc_pipefs ( ) ;
2010-09-27 14:01:27 +04:00
if ( err )
goto out4 ;
2014-11-26 14:44:43 -05:00
2021-06-08 15:59:10 -04:00
err = rpc_sysfs_init ( ) ;
if ( err )
goto out5 ;
2015-03-31 12:03:28 -04:00
sunrpc_debugfs_init ( ) ;
2014-11-17 16:58:04 -05:00
# if IS_ENABLED(CONFIG_SUNRPC_DEBUG)
2005-04-16 15:20:36 -07:00
rpc_register_sysctl ( ) ;
# endif
2007-12-30 21:07:17 -06:00
svc_init_xprt_sock ( ) ; /* svc sock transport */
init_socket_xprt ( ) ; /* clnt sock transport */
2010-07-31 14:29:07 -04:00
return 0 ;
2010-09-27 14:01:27 +04:00
2021-06-08 15:59:10 -04:00
out5 :
unregister_rpc_pipefs ( ) ;
2010-09-27 14:01:27 +04:00
out4 :
2012-04-05 21:04:37 +04:00
unregister_pernet_subsys ( & sunrpc_net_ops ) ;
2010-07-31 14:29:07 -04:00
out3 :
2012-04-05 21:04:37 +04:00
rpcauth_remove_module ( ) ;
2010-07-31 14:29:07 -04:00
out2 :
2012-04-05 21:04:37 +04:00
rpc_destroy_mempool ( ) ;
2005-04-16 15:20:36 -07:00
out :
return err ;
}
static void __exit
cleanup_sunrpc ( void )
{
2021-06-08 15:59:10 -04:00
rpc_sysfs_exit ( ) ;
2017-01-20 16:48:39 +08:00
rpc_cleanup_clids ( ) ;
2021-06-08 15:59:13 -04:00
xprt_cleanup_ids ( ) ;
2021-06-08 15:59:14 -04:00
xprt_multipath_cleanup_ids ( ) ;
2007-06-25 17:11:20 -04:00
rpcauth_remove_module ( ) ;
2006-12-05 16:35:51 -05:00
cleanup_socket_xprt ( ) ;
2007-12-30 21:07:17 -06:00
svc_cleanup_xprt_sock ( ) ;
2014-11-26 14:44:43 -05:00
sunrpc_debugfs_exit ( ) ;
2005-04-16 15:20:36 -07:00
unregister_rpc_pipefs ( ) ;
rpc_destroy_mempool ( ) ;
2010-09-27 14:01:27 +04:00
unregister_pernet_subsys ( & sunrpc_net_ops ) ;
2020-05-22 12:01:32 +10:00
auth_domain_cleanup ( ) ;
2014-11-17 16:58:04 -05:00
# if IS_ENABLED(CONFIG_SUNRPC_DEBUG)
2005-04-16 15:20:36 -07:00
rpc_unregister_sysctl ( ) ;
# endif
2009-06-26 10:45:58 +00:00
rcu_barrier ( ) ; /* Wait for completion of call_rcu()'s */
2005-04-16 15:20:36 -07:00
}
MODULE_LICENSE ( " GPL " ) ;
2009-08-21 08:17:56 -04:00
fs_initcall ( init_sunrpc ) ; /* Ensure we're initialised before nfs */
2005-04-16 15:20:36 -07:00
module_exit ( cleanup_sunrpc ) ;