mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-10 15:20:10 +00:00
50304ef057
1998-07-16 10:23 Ulrich Drepper <drepper@cygnus.com> * argp/argp-fmtstream.c: Unify names of used global functions. * argp/argp-help.c: Likewise. * assert/assert-perr.c: Likewise * assert/assert.c: Likewise * dirent/scandir.c: Likewise * dirent/scandir64.c: Likewise * dirent/versionsort.c: Likewise * dirent/versionsort64.c: Likewise * gmon/bb_exit_func.c: Likewise. * gmon/gmon.c: Likewise. * grp/initgroups.c: Likewise. * iconv/gconv_conf.c: Likewise. * inet/getnameinfo.c: Likewise. * inet/getnetgrent_r.c: Likewise. * inet/inet_ntoa.c: Likewise. * inet/rcmd.c: Likewise. * inet/rexec.c: Likewise. * inet/ruserpass.c: Likewise. * io/fts.c: Likewise. * io/ftw.c: Likewise. * io/ftw64.c: Likewise. * io/getdirname.c: Likewise. * io/getwd.c: Likewise. * io/lockf.c: Likewise. * libio/iofdopen.c: Likewise. * libio/iopopen.c: Likewise. * login/utmp_daemon.c: Likewise. * malloc/mtrace.c: Likewise. * malloc/obstack.c * misc/daemon.c: Likewise. * misc/efgcvt_r.c: Likewise. * misc/err.c: Likewise. * misc/error.c: Likewise. * misc/fstab.c: Likewise. * misc/getpass.c: Likewise. * misc/getttyent.c: Likewise. * misc/mntent_r.c: Likewise. * misc/search.h: Likewise. * misc/syslog.c: Likewise. * nscd/nscd_getgr_r.c: Likewise. * nscd/nscd_getpw_r.c: Likewise. * posix/getpgrp.c: Likewise. * posix/wordexp.c: Likewise. * pwd/fgetpwent_r.c: Likewise. * pwd/getpw.c: Likewise. * resolv/herror.c: Likewise. * resolv/res_init.c: Likewise. * shadow/fgetspent_r.c: Likewise. * shadow/lckpwdf.c: Likewise. * signal/sigrelse.c: Likewise. * stdio-common/asprintf.c: Likewise. * stdio-common/dprintf.c: Likewise. * stdio-common/getw.c: Likewise. * stdio-common/putw.c: Likewise. * stdio-common/snprintf.c: Likewise. * stdio-common/sprintf.c: Likewise. * stdio-common/sscanf.c: Likewise. * stdlib/lrand48_r.c: Likewise. * stdlib/mrand48_r.c: Likewise. * string/argz-replace.c: Likewise. * string/envz.c: Likewise. * sunrpc/auth_des.c: Likewise. * sunrpc/auth_unix.c: Likewise. * sunrpc/bindrsvprt.c: Likewise. * sunrpc/clnt_gen.c: Likewise. * sunrpc/clnt_perr.c: Likewise. * sunrpc/clnt_simp.c: Likewise. * sunrpc/clnt_tcp.c: Likewise. * sunrpc/clnt_udp.c: Likewise. * sunrpc/get_myaddr.c: Likewise. * sunrpc/key_call.c: Likewise. * sunrpc/netname.c: Likewise. * sunrpc/openchild.c: Likewise. * sunrpc/pmap_rmt.c: Likewise. * sunrpc/rpc_dtable.c: Likewise. * sunrpc/rtime.c: Likewise. * sunrpc/svc_run.c: Likewise. * sunrpc/svc_simple.c: Likewise. * sunrpc/svc_tcp.c: Likewise. * sunrpc/svc_udp.c: Likewise. * sunrpc/svcauth_des.c: Likewise. * sunrpc/xdr_array.c: Likewise. * sunrpc/xdr_rec.c: Likewise. * sunrpc/xdr_ref.c: Likewise. * sunrpc/xdr_stdio.c: Likewise. * sysdeps/generic/abort.c: Likewise. * sysdeps/generic/dl-sysdep.c: Likewise. * sysdeps/generic/fstatfs64.c: Likewise. * sysdeps/generic/ftruncate64.c: Likewise. * sysdeps/generic/getrlimit64.c: Likewise. * sysdeps/generic/glob.c: Likewise. * sysdeps/generic/prof-freq.c: Likewise. * sysdeps/generic/putenv.c: Likewise. * sysdeps/generic/statfs64.c: Likewise. * sysdeps/generic/ttyname_r.c: Likewise. * sysdeps/generic/utmp_file.c: Likewise. * sysdeps/generic/vlimit.c: Likewise. * sysdeps/generic/vtimes.c: Likewise. * sysdeps/posix/cuserid.c: Likewise. * sysdeps/posix/euidaccess.c: Likewise. * sysdeps/posix/mkstemp.c: Likewise. * sysdeps/posix/mktemp.c: Likewise. * sysdeps/posix/pread.c: Likewise. * sysdeps/posix/pread64.c: Likewise. * sysdeps/posix/profil.c: Likewise. * sysdeps/posix/pwrite.c: Likewise. * sysdeps/posix/pwrite64.c: Likewise. * sysdeps/posix/sigblock.c: Likewise. * sysdeps/posix/sigpause.c: Likewise. * sysdeps/posix/ttyname.c: Likewise. * sysdeps/posix/ttyname_r.c: Likewise. * sysdeps/posix/waitid.c: Likewise. * sysdeps/unix/getlogin_r.c: Likewise. * sysdeps/unix/grantpt.c: Likewise. * sysdeps/unix/rewinddir.c: Likewise. * sysdeps/unix/sysv/linux/gethostid.c: Likewise. * sysdeps/unix/sysv/linux/getpt.c: Likewise. * sysdeps/unix/sysv/linux/if_index.c: Likewise. * sysdeps/unix/sysv/linux/ptsname.c: Likewise. * sysdeps/unix/sysv/linux/sendmsg.c: Likewise. * sysdeps/unix/sysv/linux/statvfs.c: Likewise. * sysdeps/unix/sysv/linux/ttyname.c: Likewise. * sysdeps/unix/sysv/linux/ttyname_r.c: Likewise. * sysdeps/unix/sysv/linux/ulimit.c: Likewise. * sysdeps/unix/sysv/linux/unlockpt.c: Likewise. * sysvipc/sys/shm.h: Likewise. * time/ctime_r.c: Likewise. * time/strptime.c: Likewise. * wcsmbs/mbrlen.c: Likewise. * wcsmbs/wcsdup.c: Likewise. * wcsmbs/wcsxfrm.c: Likewise. * wctype/wcfuncs.c: Likewise. * sysdeps/unix/sysv/linux/i386/socker.S: Change to honor NO_WEAK_ALIAS. * sysdeps/unix/sysv/linux/accept.S: Don't generate __ name. * sysdeps/unix/sysv/linux/bind.S: Likewise. * sysdeps/unix/sysv/linux/getsockname.S: Likewise. * sysdeps/unix/sysv/linux/listen.S: Likewise. * sysdeps/unix/sysv/linux/recvfrom.S: Likewise. * sysdeps/unix/sysv/linux/sendto.S: Likewise. * sysdeps/unix/sysv/linux/setsockopt.S: Likewise. * grp/fgetgrent_r.c: Use explicit locking of the stream. * elf/Makefile (rtld-routines): Add dl-environ. * sysdeps/generic/dl-environ.c: New file. * libio/Makefile [REENTRANT] (routines): Add iofputs_u. * libio/Versions: Add fputs_unlocked. * libio/iofputs_u.c: New file. * libio/stdio.h: Add prototype for fputs_unlocked. * sunrpc/rpc/auth.h: Use __PMT instead of __P in type definitions. * sunrpc/rpc/clnt.h: Likewise. * sunrpc/rpc/pmap_clnt.h: Likewise. * sunrpc/rpc/svc.h: Likewise. * sunrpc/rpc/xdr.h: Likewise. * sysdeps/i386/memchr.S: Correct for more strict gas. * sysdeps/i386/fpu/bits/mathinline.h: Likewise. * sysdeps/libm-i387/i686/s_fdim.S: Likewise. * sysdeps/libm-i387/i686/s_fdimf.S: Likewise. * sysdeps/libm-i387/i686/s_fdiml.S: Likewise. 1998-07-15 Andreas Jaeger <aj@arthur.rhein-neckar.de> * configure.in: Change message for binutils version from 2.8.1.0.17->2.8.1.0.23. 1998-07-15 Ulrich Drepper <drepper@cygnus.com> * sysdeps/unix/sysv/sysv4/solaris2/sparc/sysdep.h: Define LOC. Patch by John Tobey <jtobey@banta-im.com>.
455 lines
10 KiB
C
455 lines
10 KiB
C
/* Copyright (C) 1997, 1998 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
Contributed by Mark Kettenis <kettenis@phys.uva.nl>, 1997.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public License as
|
|
published by the Free Software Foundation; either version 2 of the
|
|
License, or (at your option) any later version.
|
|
|
|
The GNU C Library 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
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
License along with the GNU C Library; see the file COPYING.LIB. If not,
|
|
write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
Boston, MA 02111-1307, USA. */
|
|
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/un.h>
|
|
#include <unistd.h>
|
|
#include <utmp.h>
|
|
|
|
#include "utmp-private.h"
|
|
#include "programs/utmpd.h"
|
|
|
|
|
|
/* Descriptor for the socket. */
|
|
static int daemon_sock = -1;
|
|
|
|
|
|
/* Functions defined here. */
|
|
static int setutent_daemon (void);
|
|
static int getutent_r_daemon (struct utmp *buffer, struct utmp **result);
|
|
static int getutid_r_daemon (const struct utmp *line, struct utmp *buffer,
|
|
struct utmp **result);
|
|
static int getutline_r_daemon (const struct utmp *id, struct utmp *buffer,
|
|
struct utmp **result);
|
|
static struct utmp *pututline_daemon (const struct utmp *utmp);
|
|
static void endutent_daemon (void);
|
|
static int updwtmp_daemon (const char *file, const struct utmp *utmp);
|
|
|
|
/* Jump table for daemon functions. */
|
|
struct utfuncs __libc_utmp_daemon_functions =
|
|
{
|
|
setutent_daemon,
|
|
getutent_r_daemon,
|
|
getutid_r_daemon,
|
|
getutline_r_daemon,
|
|
pututline_daemon,
|
|
endutent_daemon,
|
|
updwtmp_daemon
|
|
};
|
|
|
|
static int do_setutent (int sock);
|
|
static int do_getutent (int sock, struct utmp *buffer);
|
|
static int do_getutid (int sock, const struct utmp *id,
|
|
struct utmp *buffer);
|
|
static int do_pututline (int sock, const struct utmp *utmp);
|
|
static int do_getutline (int sock, const struct utmp *line,
|
|
struct utmp *buffer);
|
|
static int do_pututline (int sock, const struct utmp *utmp);
|
|
static int do_endutent (int sock);
|
|
static int do_updwtmp (int sock, const char *file,
|
|
const struct utmp *utmp);
|
|
|
|
static int open_socket (const char *name);
|
|
static int send_request (int sock, const request_header *request,
|
|
reply_header *reply);
|
|
|
|
|
|
static int
|
|
setutent_daemon (void)
|
|
{
|
|
if (__access (_PATH_UTMPD_RW, F_OK) == -1
|
|
&& __access (_PATH_UTMPD_RO, F_OK) == -1)
|
|
return 0;
|
|
|
|
if (daemon_sock < 0)
|
|
{
|
|
daemon_sock = open_socket (_PATH_UTMPD_RW);
|
|
if (daemon_sock < 0)
|
|
{
|
|
/* Hhm, read-write access did not work. Try read-only. */
|
|
daemon_sock = open_socket (_PATH_UTMPD_RO);
|
|
if (daemon_sock < 0)
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* Send request to the daemon. */
|
|
if (do_setutent (daemon_sock) < 0)
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
static int
|
|
getutent_r_daemon (struct utmp *buffer, struct utmp **result)
|
|
{
|
|
assert (daemon_sock >= 0);
|
|
|
|
/* Send request to the daemon. */
|
|
if (do_getutent (daemon_sock, buffer) < 0)
|
|
{
|
|
*result = NULL;
|
|
return -1;;
|
|
}
|
|
|
|
*result = buffer;
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int
|
|
getutid_r_daemon (const struct utmp *id, struct utmp *buffer,
|
|
struct utmp **result)
|
|
{
|
|
assert (daemon_sock >= 0);
|
|
|
|
/* Send request to the daemon. */
|
|
if (do_getutid (daemon_sock, id, buffer) < 0)
|
|
{
|
|
*result = NULL;
|
|
return -1;
|
|
}
|
|
|
|
*result = buffer;
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int
|
|
getutline_r_daemon (const struct utmp *line, struct utmp *buffer,
|
|
struct utmp **result)
|
|
{
|
|
assert (daemon_sock >= 0);
|
|
|
|
/* Send request to the daemon. */
|
|
if (do_getutline (daemon_sock, line, buffer) < 0)
|
|
{
|
|
*result = NULL;
|
|
return -1;
|
|
}
|
|
|
|
*result = buffer;
|
|
return 0;
|
|
}
|
|
|
|
|
|
static struct utmp *
|
|
pututline_daemon (const struct utmp *utmp)
|
|
{
|
|
assert (daemon_sock >= 0);
|
|
|
|
/* Send request to the daemon. */
|
|
if (do_pututline (daemon_sock, utmp) < 0)
|
|
return NULL;
|
|
|
|
return (struct utmp *)utmp;
|
|
}
|
|
|
|
|
|
static void
|
|
endutent_daemon (void)
|
|
{
|
|
assert (daemon_sock >= 0);
|
|
|
|
/* Send request to the daemon. */
|
|
do_endutent (daemon_sock);
|
|
|
|
__close (daemon_sock);
|
|
daemon_sock = -1;
|
|
}
|
|
|
|
|
|
static int
|
|
updwtmp_daemon (const char *file, const struct utmp *utmp)
|
|
{
|
|
int sock;
|
|
|
|
/* Only try to open for both reading and writing. */
|
|
sock = open_socket (_PATH_UTMPD_RW);
|
|
if (sock < 0)
|
|
return -1;
|
|
|
|
/* Send request to the daemon. */
|
|
if (do_updwtmp (sock, file, utmp) < 0)
|
|
{
|
|
__close (sock);
|
|
return -1;
|
|
}
|
|
|
|
__close (sock);
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int
|
|
do_setutent (int sock)
|
|
{
|
|
setutent_request *request;
|
|
setutent_reply reply;
|
|
size_t size;
|
|
size_t name_len;
|
|
|
|
name_len = strlen (__libc_utmp_file_name) + 1;
|
|
size = sizeof (setutent_request) + name_len;
|
|
|
|
request = malloc (size);
|
|
if (request == NULL)
|
|
return -1;
|
|
|
|
request->header.version = UTMPD_VERSION;
|
|
request->header.size = size;
|
|
request->header.type = UTMPD_REQ_SETUTENT;
|
|
memcpy (request->file, __libc_utmp_file_name, name_len);
|
|
|
|
reply.header.version = UTMPD_VERSION;
|
|
reply.header.size = sizeof (setutent_reply);
|
|
reply.header.type = UTMPD_REQ_SETUTENT;
|
|
|
|
if (send_request (sock, &request->header, &reply.header) < 0)
|
|
{
|
|
free (request);
|
|
return -1;
|
|
}
|
|
|
|
if (reply.result < 0)
|
|
__set_errno (reply.errnum);
|
|
|
|
free (request);
|
|
return reply.result;
|
|
}
|
|
|
|
static int
|
|
do_getutent (int sock, struct utmp *buffer)
|
|
{
|
|
getutent_request request;
|
|
getutent_reply reply;
|
|
|
|
request.header.version = UTMPD_VERSION;
|
|
request.header.size = sizeof (getutent_request);
|
|
request.header.type = UTMPD_REQ_GETUTENT;
|
|
|
|
reply.header.version = UTMPD_VERSION;
|
|
reply.header.size = sizeof (getutent_reply);
|
|
reply.header.type = UTMPD_REQ_GETUTENT;
|
|
|
|
if (send_request (sock, &request.header, &reply.header) < 0)
|
|
return -1;
|
|
|
|
if (reply.result < 0)
|
|
__set_errno (reply.errnum);
|
|
else
|
|
memcpy (buffer, &reply.entry, sizeof (struct utmp));
|
|
|
|
return reply.result;
|
|
}
|
|
|
|
static int
|
|
do_getutid (int sock, const struct utmp *id, struct utmp *buffer)
|
|
{
|
|
getutid_request request;
|
|
getutid_reply reply;
|
|
|
|
request.header.version = UTMPD_VERSION;
|
|
request.header.size = sizeof (getutid_request);
|
|
request.header.type = UTMPD_REQ_GETUTID;
|
|
memcpy (&request.id, id, sizeof (struct utmp));
|
|
|
|
reply.header.version = UTMPD_VERSION;
|
|
reply.header.size = sizeof (getutid_reply);
|
|
reply.header.type = UTMPD_REQ_GETUTID;
|
|
|
|
if (send_request (sock, &request.header, &reply.header) < 0)
|
|
return -1;
|
|
|
|
if (reply.result < 0)
|
|
__set_errno (reply.errnum);
|
|
else
|
|
memcpy (buffer, &reply.entry, sizeof (struct utmp));
|
|
|
|
return reply.result;
|
|
}
|
|
|
|
static int
|
|
do_getutline (int sock, const struct utmp *line, struct utmp *buffer)
|
|
{
|
|
getutline_request request;
|
|
getutline_reply reply;
|
|
|
|
request.header.version = UTMPD_VERSION;
|
|
request.header.size = sizeof (getutline_request);
|
|
request.header.type = UTMPD_REQ_GETUTLINE;
|
|
memcpy (&request.line, line, sizeof (struct utmp));
|
|
|
|
reply.header.version = UTMPD_VERSION;
|
|
reply.header.size = sizeof (getutline_reply);
|
|
reply.header.type = UTMPD_REQ_GETUTLINE;
|
|
|
|
if (send_request (sock, &request.header, &reply.header) < 0)
|
|
return -1;
|
|
|
|
if (reply.result < 0)
|
|
__set_errno (reply.errnum);
|
|
else
|
|
memcpy (buffer, &reply.entry, sizeof (struct utmp));
|
|
|
|
return reply.result;
|
|
}
|
|
|
|
static int
|
|
do_pututline (int sock, const struct utmp *utmp)
|
|
{
|
|
pututline_request request;
|
|
pututline_reply reply;
|
|
|
|
request.header.version = UTMPD_VERSION;
|
|
request.header.size = sizeof (pututline_request);
|
|
request.header.type = UTMPD_REQ_PUTUTLINE;
|
|
memcpy (&request.utmp, utmp, sizeof (struct utmp));
|
|
|
|
reply.header.version = UTMPD_VERSION;
|
|
reply.header.size = sizeof (pututline_reply);
|
|
reply.header.type = UTMPD_REQ_PUTUTLINE;
|
|
|
|
if (send_request (sock, &request.header, &reply.header) < 0)
|
|
return -1;
|
|
|
|
if (reply.result < 0)
|
|
__set_errno (reply.errnum);
|
|
|
|
return reply.result;
|
|
}
|
|
|
|
static int
|
|
do_endutent (int sock)
|
|
{
|
|
endutent_request request;
|
|
endutent_reply reply;
|
|
|
|
request.header.version = UTMPD_VERSION;
|
|
request.header.size = sizeof (endutent_request);
|
|
request.header.type = UTMPD_REQ_ENDUTENT;
|
|
|
|
reply.header.version = UTMPD_VERSION;
|
|
reply.header.size = sizeof (endutent_reply);
|
|
reply.header.type = UTMPD_REQ_ENDUTENT;
|
|
|
|
if (send_request (sock, &request.header, &reply.header) < 0)
|
|
return -1;
|
|
|
|
if (reply.result < 0)
|
|
__set_errno (reply.errnum);
|
|
|
|
return reply.result;
|
|
}
|
|
|
|
static int
|
|
do_updwtmp (int sock, const char *file, const struct utmp *utmp)
|
|
{
|
|
updwtmp_request *request;
|
|
updwtmp_reply reply;
|
|
size_t size;
|
|
size_t file_len;
|
|
|
|
file_len = strlen (file) + 1;
|
|
size = sizeof (updwtmp_request) + file_len;
|
|
|
|
request = malloc (size);
|
|
if (request == NULL)
|
|
return -1;
|
|
|
|
request->header.version = UTMPD_VERSION;
|
|
request->header.size = size;
|
|
request->header.type = UTMPD_REQ_UPDWTMP;
|
|
memcpy (&request->utmp, utmp, sizeof (struct utmp));
|
|
memcpy (request->file, file, file_len);
|
|
|
|
reply.header.version = UTMPD_VERSION;
|
|
reply.header.size = sizeof (updwtmp_reply);
|
|
reply.header.type = UTMPD_REQ_UPDWTMP;
|
|
|
|
if (send_request (sock, &request->header, &reply.header) < 0)
|
|
{
|
|
free (request);
|
|
return -1;
|
|
}
|
|
|
|
if (reply.result < 0)
|
|
__set_errno (reply.errnum);
|
|
|
|
free (request);
|
|
return reply.result;
|
|
}
|
|
|
|
|
|
/* Create a socket connected to NAME. */
|
|
static int
|
|
open_socket (const char *name)
|
|
{
|
|
struct sockaddr_un addr;
|
|
int sock;
|
|
|
|
sock = __socket (PF_UNIX, SOCK_STREAM, 0);
|
|
if (sock < 0)
|
|
return -1;
|
|
|
|
addr.sun_family = AF_UNIX;
|
|
strcpy (addr.sun_path, name);
|
|
if (__connect (sock, (struct sockaddr *) &addr, sizeof (addr)) < 0)
|
|
{
|
|
__close (sock);
|
|
return -1;
|
|
}
|
|
|
|
return sock;
|
|
}
|
|
|
|
/* Send REQUEST to SOCK, and wait for reply. Returns 0 if successful,
|
|
storing the reply in REPLY, and -1 if not. */
|
|
static int
|
|
send_request (int sock, const request_header *request,
|
|
reply_header *reply)
|
|
{
|
|
reply_header header;
|
|
ssize_t nbytes;
|
|
|
|
nbytes = __write (sock, request, request->size);
|
|
if (nbytes != (ssize_t) request->size)
|
|
return -1;
|
|
|
|
nbytes = __read (sock, &header, sizeof (reply_header));
|
|
if (nbytes != sizeof (reply_header))
|
|
return -1;
|
|
|
|
if (reply->version != header.version
|
|
|| reply->size != header.size
|
|
|| reply->type != header.type)
|
|
return -1;
|
|
|
|
nbytes = __read (sock, reply + 1, reply->size - sizeof (reply_header));
|
|
if (nbytes != (ssize_t) (reply->size - sizeof (reply_header)))
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|