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-17 13:56:39 +04:00
struct server_id 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
void ( * recv_cb ) ( int msg_type ,
struct server_id src , struct server_id dst ,
const uint8_t * msg , size_t msg_len ,
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
} ;
struct messaging_dgm_hdr {
2014-07-17 15:51:13 +04:00
int msg_type ;
2014-02-24 16:23:49 +04:00
struct server_id dst ;
struct server_id src ;
} ;
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 ;
struct flock lck = { } ;
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 ;
}
lck . l_type = F_WRLCK ;
lck . l_whence = SEEK_SET ;
lck . l_start = 0 ;
lck . l_len = 0 ;
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 ;
}
unique_len = snprintf ( buf , sizeof ( buf ) , " % " PRIu64 , unique ) ;
/* 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-17 13:44:41 +04:00
void ( * recv_cb ) ( int msg_type ,
struct server_id src ,
struct server_id dst ,
const uint8_t * msg ,
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-17 13:56:39 +04:00
ctx - > 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 )
{
2014-07-17 13:56:39 +04:00
struct server_id pid = c - > pid ;
2014-02-24 16:23:49 +04:00
/*
* First delete the socket to avoid races . The lockfile is the
* indicator that we ' re still around .
*/
unix_msg_free ( c - > dgm_ctx ) ;
if ( getpid ( ) = = pid . pid ) {
2014-05-30 19:31:33 +04:00
( void ) messaging_dgm_lockfile_remove ( c , c - > cache_dir , pid . 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-17 15:23:46 +04:00
int messaging_dgm_send ( struct messaging_dgm_context * ctx ,
struct server_id src , struct server_id pid ,
int msg_type , const struct iovec * iov , int iovlen )
2014-02-24 16:23:49 +04:00
{
struct messaging_dgm_hdr hdr ;
2014-05-13 13:55:37 +04:00
struct iovec iov2 [ iovlen + 1 ] ;
2014-06-03 17:11:05 +04:00
struct server_id_buf idbuf ;
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 ) ,
" %s/msg/%u " , ctx - > cache_dir , ( unsigned ) pid . pid ) ;
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-17 15:56:18 +04:00
hdr . msg_type = msg_type ;
2014-02-24 16:23:49 +04:00
hdr . dst = pid ;
2014-05-29 16:51:37 +04:00
hdr . src = src ;
2014-02-24 16:23:49 +04:00
2014-05-13 13:55:37 +04:00
DEBUG ( 10 , ( " %s: Sending message 0x%x to %s \n " , __func__ ,
( unsigned ) hdr . msg_type ,
2014-06-03 17:11:05 +04:00
server_id_str_buf ( pid , & idbuf ) ) ) ;
2014-02-24 16:23:49 +04:00
2014-05-13 13:55:37 +04:00
iov2 [ 0 ] . iov_base = & hdr ;
iov2 [ 0 ] . iov_len = sizeof ( hdr ) ;
memcpy ( iov2 + 1 , iov , iovlen * sizeof ( struct iovec ) ) ;
2014-02-24 16:23:49 +04:00
2014-06-01 22:57:21 +04:00
ret = unix_msg_send ( ctx - > dgm_ctx , & dst , iov2 , iovlen + 1 ) ;
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 ) ;
struct messaging_dgm_hdr * hdr ;
2014-06-03 17:11:05 +04:00
struct server_id_buf idbuf ;
2014-02-24 16:23:49 +04:00
if ( msg_len < sizeof ( * hdr ) ) {
DEBUG ( 1 , ( " message too short: %u \n " , ( unsigned ) msg_len ) ) ;
return ;
}
/*
* unix_msg guarantees alignment , so we can cast here
*/
hdr = ( struct messaging_dgm_hdr * ) msg ;
DEBUG ( 10 , ( " %s: Received message 0x%x len %u from %s \n " , __func__ ,
2014-07-17 13:44:41 +04:00
( unsigned ) hdr - > msg_type , ( unsigned ) ( msg_len - sizeof ( * hdr ) ) ,
server_id_str_buf ( hdr - > src , & idbuf ) ) ) ;
2014-02-24 16:23:49 +04:00
2014-07-17 13:44:41 +04:00
dgm_ctx - > recv_cb ( hdr - > msg_type , hdr - > src , hdr - > dst ,
msg + sizeof ( * hdr ) , msg_len - sizeof ( * hdr ) ,
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-02-24 16:23:49 +04:00
DEBUG ( 10 , ( " %s: open(%s) failed: %s \n " , __func__ ,
2014-06-04 18:42:46 +04:00
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 ) ;
}