2014-02-24 16:23:49 +04:00
/*
* Unix SMB / CIFS implementation .
* Samba internal messaging functions
* Copyright ( C ) 2013 by Volker Lendecke
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 3 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "lib/util/data_blob.h"
# include "lib/util/debug.h"
# include "lib/unix_msg/unix_msg.h"
# include "system/filesys.h"
2014-07-17 15:57:51 +04:00
# include "lib/messages_dgm.h"
2014-02-24 16:23:49 +04:00
# include "lib/param/param.h"
# include "poll_funcs/poll_funcs_tevent.h"
# include "unix_msg/unix_msg.h"
struct messaging_dgm_context {
2014-07-25 15:03:11 +04:00
pid_t pid ;
2014-05-05 10:45:52 +04:00
struct poll_funcs * msg_callbacks ;
void * tevent_handle ;
2014-02-24 16:23:49 +04:00
struct unix_msg_ctx * dgm_ctx ;
char * cache_dir ;
int lockfile_fd ;
2014-07-17 13:44:41 +04:00
2014-07-27 14:29:26 +04:00
void ( * recv_cb ) ( const uint8_t * msg ,
size_t msg_len ,
2014-07-17 13:44:41 +04:00
void * private_data ) ;
void * recv_cb_private_data ;
2014-07-20 16:51:47 +04:00
bool * have_dgm_context ;
2014-02-24 16:23:49 +04:00
} ;
static void messaging_dgm_recv ( struct unix_msg_ctx * ctx ,
uint8_t * msg , size_t msg_len ,
void * private_data ) ;
static int messaging_dgm_context_destructor ( struct messaging_dgm_context * c ) ;
2014-06-03 01:01:46 +04:00
static int messaging_dgm_lockfile_create ( TALLOC_CTX * tmp_ctx ,
2014-07-17 15:47:32 +04:00
const char * cache_dir ,
uid_t dir_owner , pid_t pid ,
2014-02-24 16:23:49 +04:00
int * plockfile_fd , uint64_t unique )
{
2014-06-03 00:29:44 +04:00
fstring buf ;
char * dir ;
2014-02-24 16:23:49 +04:00
char * lockfile_name ;
int lockfile_fd ;
2014-07-25 14:42:19 +04:00
struct flock lck ;
2014-02-24 16:23:49 +04:00
int unique_len , ret ;
ssize_t written ;
bool ok ;
2014-06-03 01:01:46 +04:00
dir = talloc_asprintf ( tmp_ctx , " %s/lck " , cache_dir ) ;
2014-06-03 00:29:44 +04:00
if ( dir = = NULL ) {
2014-02-24 16:23:49 +04:00
return ENOMEM ;
}
2014-07-17 15:47:32 +04:00
ok = directory_create_or_exist_strict ( dir , dir_owner , 0755 ) ;
2014-02-24 16:23:49 +04:00
if ( ! ok ) {
ret = errno ;
DEBUG ( 1 , ( " %s: Could not create lock directory: %s \n " ,
__func__ , strerror ( ret ) ) ) ;
2014-06-03 00:29:44 +04:00
TALLOC_FREE ( dir ) ;
2014-02-24 16:23:49 +04:00
return ret ;
}
2014-06-03 01:01:46 +04:00
lockfile_name = talloc_asprintf ( tmp_ctx , " %s/%u " , dir ,
2014-02-24 16:23:49 +04:00
( unsigned ) pid ) ;
2014-06-03 00:29:44 +04:00
TALLOC_FREE ( dir ) ;
2014-02-24 16:23:49 +04:00
if ( lockfile_name = = NULL ) {
DEBUG ( 1 , ( " %s: talloc_asprintf failed \n " , __func__ ) ) ;
return ENOMEM ;
}
/* no O_EXCL, existence check is via the fcntl lock */
lockfile_fd = open ( lockfile_name , O_NONBLOCK | O_CREAT | O_WRONLY , 0644 ) ;
if ( lockfile_fd = = - 1 ) {
ret = errno ;
DEBUG ( 1 , ( " %s: open failed: %s \n " , __func__ , strerror ( errno ) ) ) ;
goto fail_free ;
}
2014-07-25 14:42:19 +04:00
lck = ( struct flock ) {
. l_type = F_WRLCK ,
. l_whence = SEEK_SET
} ;
2014-02-24 16:23:49 +04:00
ret = fcntl ( lockfile_fd , F_SETLK , & lck ) ;
if ( ret = = - 1 ) {
ret = errno ;
DEBUG ( 1 , ( " %s: fcntl failed: %s \n " , __func__ , strerror ( ret ) ) ) ;
goto fail_close ;
}
2014-08-02 15:57:43 +04:00
unique_len = snprintf ( buf , sizeof ( buf ) , " %ju \n " , ( uintmax_t ) unique ) ;
2014-02-24 16:23:49 +04:00
/* shorten a potentially preexisting file */
ret = ftruncate ( lockfile_fd , unique_len ) ;
if ( ret = = - 1 ) {
ret = errno ;
DEBUG ( 1 , ( " %s: ftruncate failed: %s \n " , __func__ ,
strerror ( ret ) ) ) ;
goto fail_unlink ;
}
written = write ( lockfile_fd , buf , unique_len ) ;
if ( written ! = unique_len ) {
ret = errno ;
DEBUG ( 1 , ( " %s: write failed: %s \n " , __func__ , strerror ( ret ) ) ) ;
goto fail_unlink ;
}
2014-06-03 00:48:56 +04:00
TALLOC_FREE ( lockfile_name ) ;
2014-02-24 16:23:49 +04:00
* plockfile_fd = lockfile_fd ;
return 0 ;
fail_unlink :
unlink ( lockfile_name ) ;
fail_close :
close ( lockfile_fd ) ;
fail_free :
TALLOC_FREE ( lockfile_name ) ;
return ret ;
}
2014-05-30 19:31:33 +04:00
static int messaging_dgm_lockfile_remove ( TALLOC_CTX * tmp_ctx ,
const char * cache_dir , pid_t pid )
2014-02-24 16:23:49 +04:00
{
2014-05-30 19:31:33 +04:00
char * lockfile_name ;
2014-02-24 16:23:49 +04:00
int ret ;
2014-05-30 19:31:33 +04:00
lockfile_name = talloc_asprintf (
tmp_ctx , " %s/lck/%u " , cache_dir , ( unsigned ) pid ) ;
if ( lockfile_name = = NULL ) {
2014-02-24 16:23:49 +04:00
return ENOMEM ;
}
ret = unlink ( lockfile_name ) ;
if ( ret = = - 1 ) {
ret = errno ;
2014-05-30 19:31:33 +04:00
DEBUG ( 10 , ( " %s: unlink(%s) failed: %s \n " , __func__ ,
lockfile_name , strerror ( ret ) ) ) ;
2014-02-24 16:23:49 +04:00
}
2014-05-30 19:31:33 +04:00
TALLOC_FREE ( lockfile_name ) ;
2014-02-24 16:23:49 +04:00
return ret ;
}
2014-07-17 13:58:50 +04:00
int messaging_dgm_init ( TALLOC_CTX * mem_ctx ,
2014-07-17 13:52:28 +04:00
struct tevent_context * ev ,
2014-07-17 13:56:39 +04:00
struct server_id pid ,
2014-07-17 15:38:36 +04:00
const char * cache_dir ,
2014-07-17 15:47:32 +04:00
uid_t dir_owner ,
2014-07-27 14:29:26 +04:00
void ( * recv_cb ) ( const uint8_t * msg ,
2014-07-17 13:44:41 +04:00
size_t msg_len ,
void * private_data ) ,
2014-07-17 15:23:46 +04:00
void * recv_cb_private_data ,
struct messaging_dgm_context * * pctx )
2014-02-24 16:23:49 +04:00
{
struct messaging_dgm_context * ctx ;
int ret ;
bool ok ;
2014-06-01 22:57:21 +04:00
char * socket_dir ;
struct sockaddr_un socket_address ;
size_t sockname_len ;
2014-02-24 16:23:49 +04:00
uint64_t cookie ;
2014-07-20 16:51:47 +04:00
static bool have_dgm_context = false ;
if ( have_dgm_context ) {
return EEXIST ;
}
2014-02-24 16:23:49 +04:00
2014-07-17 15:23:46 +04:00
ctx = talloc_zero ( mem_ctx , struct messaging_dgm_context ) ;
2014-02-24 16:23:49 +04:00
if ( ctx = = NULL ) {
goto fail_nomem ;
}
2014-07-25 15:03:11 +04:00
ctx - > pid = pid . pid ;
2014-07-17 13:44:41 +04:00
ctx - > recv_cb = recv_cb ;
ctx - > recv_cb_private_data = recv_cb_private_data ;
2014-02-24 16:23:49 +04:00
ctx - > cache_dir = talloc_strdup ( ctx , cache_dir ) ;
if ( ctx - > cache_dir = = NULL ) {
goto fail_nomem ;
}
socket_dir = talloc_asprintf ( ctx , " %s/msg " , cache_dir ) ;
if ( socket_dir = = NULL ) {
goto fail_nomem ;
}
2014-06-01 22:57:21 +04:00
socket_address = ( struct sockaddr_un ) { . sun_family = AF_UNIX } ;
sockname_len = snprintf ( socket_address . sun_path ,
sizeof ( socket_address . sun_path ) ,
" %s/%u " , socket_dir , ( unsigned ) pid . pid ) ;
if ( sockname_len > = sizeof ( socket_address . sun_path ) ) {
2014-07-17 15:23:46 +04:00
TALLOC_FREE ( ctx ) ;
2014-06-10 19:21:10 +04:00
return ENAMETOOLONG ;
2014-02-24 16:23:49 +04:00
}
2014-07-17 15:47:32 +04:00
ret = messaging_dgm_lockfile_create ( ctx , cache_dir , dir_owner , pid . pid ,
2014-02-24 16:23:49 +04:00
& ctx - > lockfile_fd , pid . unique_id ) ;
if ( ret ! = 0 ) {
DEBUG ( 1 , ( " %s: messaging_dgm_create_lockfile failed: %s \n " ,
__func__ , strerror ( ret ) ) ) ;
2014-07-17 15:23:46 +04:00
TALLOC_FREE ( ctx ) ;
2014-06-10 19:21:10 +04:00
return ret ;
2014-02-24 16:23:49 +04:00
}
2014-05-05 10:45:52 +04:00
ctx - > msg_callbacks = poll_funcs_init_tevent ( ctx ) ;
if ( ctx - > msg_callbacks = = NULL ) {
2014-06-10 18:57:05 +04:00
goto fail_nomem ;
2014-05-05 10:45:52 +04:00
}
ctx - > tevent_handle = poll_funcs_tevent_register (
2014-07-17 13:52:28 +04:00
ctx , ctx - > msg_callbacks , ev ) ;
2014-05-05 10:45:52 +04:00
if ( ctx - > tevent_handle = = NULL ) {
2014-06-10 18:57:05 +04:00
goto fail_nomem ;
2014-05-05 10:45:52 +04:00
}
2014-02-24 16:23:49 +04:00
2014-07-17 15:47:32 +04:00
ok = directory_create_or_exist_strict ( socket_dir , dir_owner , 0700 ) ;
2014-02-24 16:23:49 +04:00
if ( ! ok ) {
DEBUG ( 1 , ( " Could not create socket directory \n " ) ) ;
2014-07-17 15:23:46 +04:00
TALLOC_FREE ( ctx ) ;
2014-06-10 19:21:10 +04:00
return EACCES ;
2014-02-24 16:23:49 +04:00
}
TALLOC_FREE ( socket_dir ) ;
2014-06-01 22:57:21 +04:00
unlink ( socket_address . sun_path ) ;
2014-02-24 16:23:49 +04:00
generate_random_buffer ( ( uint8_t * ) & cookie , sizeof ( cookie ) ) ;
2014-06-01 22:57:21 +04:00
ret = unix_msg_init ( & socket_address , ctx - > msg_callbacks , 1024 , cookie ,
2014-02-24 16:23:49 +04:00
messaging_dgm_recv , ctx , & ctx - > dgm_ctx ) ;
if ( ret ! = 0 ) {
DEBUG ( 1 , ( " unix_msg_init failed: %s \n " , strerror ( ret ) ) ) ;
2014-07-17 15:23:46 +04:00
TALLOC_FREE ( ctx ) ;
2014-06-10 19:21:10 +04:00
return ret ;
2014-02-24 16:23:49 +04:00
}
talloc_set_destructor ( ctx , messaging_dgm_context_destructor ) ;
2014-07-20 16:51:47 +04:00
ctx - > have_dgm_context = & have_dgm_context ;
2014-07-17 15:23:46 +04:00
* pctx = ctx ;
2014-06-10 19:21:10 +04:00
return 0 ;
2014-02-24 16:23:49 +04:00
fail_nomem :
2014-07-17 15:23:46 +04:00
TALLOC_FREE ( ctx ) ;
2014-06-10 19:21:10 +04:00
return ENOMEM ;
2014-02-24 16:23:49 +04:00
}
static int messaging_dgm_context_destructor ( struct messaging_dgm_context * c )
{
/*
* First delete the socket to avoid races . The lockfile is the
* indicator that we ' re still around .
*/
unix_msg_free ( c - > dgm_ctx ) ;
2014-07-25 15:03:11 +04:00
if ( getpid ( ) = = c - > pid ) {
( void ) messaging_dgm_lockfile_remove ( c , c - > cache_dir , c - > pid ) ;
2014-02-24 16:23:49 +04:00
}
close ( c - > lockfile_fd ) ;
2014-07-20 16:51:47 +04:00
if ( c - > have_dgm_context ! = NULL ) {
* c - > have_dgm_context = false ;
}
2014-02-24 16:23:49 +04:00
return 0 ;
}
2014-07-27 14:29:26 +04:00
int messaging_dgm_send ( struct messaging_dgm_context * ctx , pid_t pid ,
const struct iovec * iov , int iovlen )
2014-02-24 16:23:49 +04:00
{
2014-06-01 22:57:21 +04:00
struct sockaddr_un dst ;
ssize_t dst_pathlen ;
2014-02-24 16:23:49 +04:00
int ret ;
2014-06-01 22:57:21 +04:00
dst = ( struct sockaddr_un ) { . sun_family = AF_UNIX } ;
2014-02-24 16:23:49 +04:00
2014-06-01 22:57:21 +04:00
dst_pathlen = snprintf ( dst . sun_path , sizeof ( dst . sun_path ) ,
2014-07-27 14:29:26 +04:00
" %s/msg/%u " , ctx - > cache_dir , ( unsigned ) pid ) ;
2014-06-01 22:57:21 +04:00
if ( dst_pathlen > = sizeof ( dst . sun_path ) ) {
2014-06-04 18:36:57 +04:00
return ENAMETOOLONG ;
2014-02-24 16:23:49 +04:00
}
2014-07-27 14:29:26 +04:00
DEBUG ( 10 , ( " %s: Sending message to %u \n " , __func__ , ( unsigned ) pid ) ) ;
2014-02-24 16:23:49 +04:00
2014-07-27 14:29:26 +04:00
ret = unix_msg_send ( ctx - > dgm_ctx , & dst , iov , iovlen ) ;
2014-02-24 16:23:49 +04:00
2014-06-04 18:36:57 +04:00
return ret ;
2014-02-24 16:23:49 +04:00
}
static void messaging_dgm_recv ( struct unix_msg_ctx * ctx ,
uint8_t * msg , size_t msg_len ,
void * private_data )
{
struct messaging_dgm_context * dgm_ctx = talloc_get_type_abort (
private_data , struct messaging_dgm_context ) ;
2014-07-27 14:29:26 +04:00
dgm_ctx - > recv_cb ( msg , msg_len , dgm_ctx - > recv_cb_private_data ) ;
2014-02-24 16:23:49 +04:00
}
2014-07-17 15:23:46 +04:00
int messaging_dgm_cleanup ( struct messaging_dgm_context * ctx , pid_t pid )
2014-02-24 16:23:49 +04:00
{
char * lockfile_name , * socket_name ;
int fd , ret ;
struct flock lck = { } ;
lockfile_name = talloc_asprintf ( talloc_tos ( ) , " %s/lck/%u " ,
ctx - > cache_dir , ( unsigned ) pid ) ;
if ( lockfile_name = = NULL ) {
2014-06-04 18:42:46 +04:00
return ENOMEM ;
2014-02-24 16:23:49 +04:00
}
socket_name = talloc_asprintf ( lockfile_name , " %s/msg/%u " ,
ctx - > cache_dir , ( unsigned ) pid ) ;
if ( socket_name = = NULL ) {
TALLOC_FREE ( lockfile_name ) ;
2014-06-04 18:42:46 +04:00
return ENOMEM ;
2014-02-24 16:23:49 +04:00
}
fd = open ( lockfile_name , O_NONBLOCK | O_WRONLY , 0 ) ;
if ( fd = = - 1 ) {
2014-06-04 18:42:46 +04:00
ret = errno ;
2014-08-18 15:58:05 +04:00
if ( ret ! = ENOENT ) {
DEBUG ( 10 , ( " %s: open(%s) failed: %s \n " , __func__ ,
lockfile_name , strerror ( ret ) ) ) ;
}
2014-08-09 20:51:05 +04:00
TALLOC_FREE ( lockfile_name ) ;
2014-06-04 18:42:46 +04:00
return ret ;
2014-02-24 16:23:49 +04:00
}
lck . l_type = F_WRLCK ;
lck . l_whence = SEEK_SET ;
lck . l_start = 0 ;
lck . l_len = 0 ;
ret = fcntl ( fd , F_SETLK , & lck ) ;
if ( ret ! = 0 ) {
2014-06-04 18:42:46 +04:00
ret = errno ;
2014-02-24 16:23:49 +04:00
DEBUG ( 10 , ( " %s: Could not get lock: %s \n " , __func__ ,
2014-06-04 18:42:46 +04:00
strerror ( ret ) ) ) ;
2014-02-24 16:23:49 +04:00
TALLOC_FREE ( lockfile_name ) ;
close ( fd ) ;
2014-06-04 18:42:46 +04:00
return ret ;
2014-02-24 16:23:49 +04:00
}
( void ) unlink ( socket_name ) ;
( void ) unlink ( lockfile_name ) ;
( void ) close ( fd ) ;
TALLOC_FREE ( lockfile_name ) ;
2014-06-04 18:42:46 +04:00
return 0 ;
2014-02-24 16:23:49 +04:00
}
2014-04-11 00:07:11 +04:00
2014-07-17 15:23:46 +04:00
int messaging_dgm_wipe ( struct messaging_dgm_context * ctx )
2014-04-11 00:07:11 +04:00
{
char * msgdir_name ;
DIR * msgdir ;
struct dirent * dp ;
pid_t our_pid = getpid ( ) ;
2014-06-04 18:47:05 +04:00
int ret ;
2014-04-11 00:07:11 +04:00
/*
* We scan the socket directory and not the lock directory . Otherwise
* we would race against messaging_dgm_lockfile_create ' s open ( O_CREAT )
* and fcntl ( SETLK ) .
*/
msgdir_name = talloc_asprintf ( talloc_tos ( ) , " %s/msg " , ctx - > cache_dir ) ;
if ( msgdir_name = = NULL ) {
2014-06-04 18:47:05 +04:00
return ENOMEM ;
2014-04-11 00:07:11 +04:00
}
msgdir = opendir ( msgdir_name ) ;
if ( msgdir = = NULL ) {
2014-06-04 18:47:05 +04:00
ret = errno ;
TALLOC_FREE ( msgdir_name ) ;
return ret ;
2014-04-11 00:07:11 +04:00
}
2014-06-04 18:47:05 +04:00
TALLOC_FREE ( msgdir_name ) ;
2014-04-11 00:07:11 +04:00
while ( ( dp = readdir ( msgdir ) ) ! = NULL ) {
unsigned long pid ;
pid = strtoul ( dp - > d_name , NULL , 10 ) ;
if ( pid = = 0 ) {
/*
* . and . . and other malformed entries
*/
continue ;
}
if ( pid = = our_pid ) {
/*
* fcntl ( F_GETLK ) will succeed for ourselves , we hold
* that lock ourselves .
*/
continue ;
}
2014-07-17 15:23:46 +04:00
ret = messaging_dgm_cleanup ( ctx , pid ) ;
2014-04-11 00:07:11 +04:00
DEBUG ( 10 , ( " messaging_dgm_cleanup(%lu) returned %s \n " ,
2014-06-04 18:42:46 +04:00
pid , ret ? strerror ( ret ) : " ok " ) ) ;
2014-04-11 00:07:11 +04:00
}
closedir ( msgdir ) ;
2014-06-04 18:47:05 +04:00
return 0 ;
2014-04-11 00:07:11 +04:00
}
2014-05-06 11:11:17 +04:00
void * messaging_dgm_register_tevent_context ( TALLOC_CTX * mem_ctx ,
2014-07-17 15:23:46 +04:00
struct messaging_dgm_context * ctx ,
2014-05-06 11:11:17 +04:00
struct tevent_context * ev )
{
return poll_funcs_tevent_register ( mem_ctx , ctx - > msg_callbacks , ev ) ;
}