2012-05-16 02:35:53 +08:00
|
|
|
/* Copyright (c) 1998-2012 Free Software Foundation, Inc.
|
1998-10-19 06:02:47 +08:00
|
|
|
This file is part of the GNU C Library.
|
1999-09-27 08:22:04 +08:00
|
|
|
Contributed by Thorsten Kukuk <kukuk@suse.de>, 1998.
|
1998-10-19 06:02:47 +08:00
|
|
|
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
2001-07-06 12:58:11 +08:00
|
|
|
modify it under the terms of the GNU Lesser General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2.1 of the License, or (at your option) any later version.
|
1998-10-19 06:02:47 +08:00
|
|
|
|
|
|
|
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
|
2001-07-06 12:58:11 +08:00
|
|
|
Lesser General Public License for more details.
|
1998-10-19 06:02:47 +08:00
|
|
|
|
2001-07-06 12:58:11 +08:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
2012-02-10 07:18:22 +08:00
|
|
|
License along with the GNU C Library; if not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
1998-10-19 06:02:47 +08:00
|
|
|
|
|
|
|
/* This file defines everything that client code should need to
|
|
|
|
know to talk to the nscd daemon. */
|
|
|
|
|
|
|
|
#ifndef _NSCD_CLIENT_H
|
|
|
|
#define _NSCD_CLIENT_H 1
|
|
|
|
|
2004-09-09 03:49:17 +08:00
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <sys/types.h>
|
2004-09-08 23:46:42 +08:00
|
|
|
#include <atomic.h>
|
2000-09-30 06:04:20 +08:00
|
|
|
#include <nscd-types.h>
|
2005-02-23 06:51:26 +08:00
|
|
|
#include <sys/uio.h>
|
2000-09-30 06:04:20 +08:00
|
|
|
|
2004-09-09 03:49:17 +08:00
|
|
|
|
1998-10-19 06:02:47 +08:00
|
|
|
/* Version number of the daemon interface */
|
2000-09-30 06:04:20 +08:00
|
|
|
#define NSCD_VERSION 2
|
1998-10-19 06:02:47 +08:00
|
|
|
|
|
|
|
/* Path of the file where the PID of the running system is stored. */
|
2004-01-01 05:44:08 +08:00
|
|
|
#define _PATH_NSCDPID "/var/run/nscd/nscd.pid"
|
1998-10-19 06:02:47 +08:00
|
|
|
|
|
|
|
/* Path for the Unix domain socket. */
|
2004-01-01 05:44:08 +08:00
|
|
|
#define _PATH_NSCDSOCKET "/var/run/nscd/socket"
|
1998-10-19 06:02:47 +08:00
|
|
|
|
|
|
|
/* Path for the configuration file. */
|
|
|
|
#define _PATH_NSCDCONF "/etc/nscd.conf"
|
|
|
|
|
2009-05-16 12:33:43 +08:00
|
|
|
/* Maximum allowed length for the key. */
|
2007-02-01 07:16:14 +08:00
|
|
|
#define MAXKEYLEN 1024
|
|
|
|
|
1998-10-19 06:02:47 +08:00
|
|
|
|
|
|
|
/* Available services. */
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
GETPWBYNAME,
|
|
|
|
GETPWBYUID,
|
|
|
|
GETGRBYNAME,
|
|
|
|
GETGRBYGID,
|
|
|
|
GETHOSTBYNAME,
|
|
|
|
GETHOSTBYNAMEv6,
|
|
|
|
GETHOSTBYADDR,
|
|
|
|
GETHOSTBYADDRv6,
|
|
|
|
SHUTDOWN, /* Shut the server down. */
|
|
|
|
GETSTAT, /* Get the server statistic. */
|
1999-09-27 08:22:04 +08:00
|
|
|
INVALIDATE, /* Invalidate one special cache. */
|
2004-09-08 23:46:42 +08:00
|
|
|
GETFDPW,
|
|
|
|
GETFDGR,
|
|
|
|
GETFDHST,
|
2004-09-15 16:25:49 +08:00
|
|
|
GETAI,
|
2004-09-30 10:24:51 +08:00
|
|
|
INITGROUPS,
|
2007-01-14 13:26:04 +08:00
|
|
|
GETSERVBYNAME,
|
|
|
|
GETSERVBYPORT,
|
|
|
|
GETFDSERV,
|
2011-10-07 22:06:31 +08:00
|
|
|
GETNETGRENT,
|
|
|
|
INNETGR,
|
|
|
|
GETFDNETGR,
|
1999-09-27 08:22:04 +08:00
|
|
|
LASTREQ
|
1998-10-19 06:02:47 +08:00
|
|
|
} request_type;
|
|
|
|
|
|
|
|
|
|
|
|
/* Header common to all requests */
|
|
|
|
typedef struct
|
|
|
|
{
|
2000-09-30 06:04:20 +08:00
|
|
|
int32_t version; /* Version number of the daemon interface. */
|
1998-10-19 06:02:47 +08:00
|
|
|
request_type type; /* Service requested. */
|
2000-09-30 00:45:44 +08:00
|
|
|
int32_t key_len; /* Key length. */
|
1998-10-19 06:02:47 +08:00
|
|
|
} request_header;
|
|
|
|
|
|
|
|
|
|
|
|
/* Structure sent in reply to password query. Note that this struct is
|
|
|
|
sent also if the service is disabled or there is no record found. */
|
|
|
|
typedef struct
|
|
|
|
{
|
2000-09-30 00:45:44 +08:00
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
2000-09-30 06:04:20 +08:00
|
|
|
nscd_ssize_t pw_name_len;
|
|
|
|
nscd_ssize_t pw_passwd_len;
|
1998-10-19 06:02:47 +08:00
|
|
|
uid_t pw_uid;
|
|
|
|
gid_t pw_gid;
|
2000-09-30 06:04:20 +08:00
|
|
|
nscd_ssize_t pw_gecos_len;
|
|
|
|
nscd_ssize_t pw_dir_len;
|
|
|
|
nscd_ssize_t pw_shell_len;
|
1998-10-19 06:02:47 +08:00
|
|
|
} pw_response_header;
|
|
|
|
|
|
|
|
|
|
|
|
/* Structure sent in reply to group query. Note that this struct is
|
|
|
|
sent also if the service is disabled or there is no record found. */
|
|
|
|
typedef struct
|
|
|
|
{
|
2000-09-30 00:45:44 +08:00
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
2000-09-30 06:04:20 +08:00
|
|
|
nscd_ssize_t gr_name_len;
|
|
|
|
nscd_ssize_t gr_passwd_len;
|
1998-10-19 06:02:47 +08:00
|
|
|
gid_t gr_gid;
|
2000-09-30 06:04:20 +08:00
|
|
|
nscd_ssize_t gr_mem_cnt;
|
1998-10-19 06:02:47 +08:00
|
|
|
} gr_response_header;
|
|
|
|
|
|
|
|
|
|
|
|
/* Structure sent in reply to host query. Note that this struct is
|
|
|
|
sent also if the service is disabled or there is no record found. */
|
|
|
|
typedef struct
|
|
|
|
{
|
2000-09-30 00:45:44 +08:00
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
2000-09-30 06:04:20 +08:00
|
|
|
nscd_ssize_t h_name_len;
|
|
|
|
nscd_ssize_t h_aliases_cnt;
|
2000-09-30 00:45:44 +08:00
|
|
|
int32_t h_addrtype;
|
|
|
|
int32_t h_length;
|
2000-09-30 06:04:20 +08:00
|
|
|
nscd_ssize_t h_addr_list_cnt;
|
2000-09-30 00:45:44 +08:00
|
|
|
int32_t error;
|
1998-10-19 06:02:47 +08:00
|
|
|
} hst_response_header;
|
|
|
|
|
|
|
|
|
2004-09-15 16:25:49 +08:00
|
|
|
/* Structure sent in reply to addrinfo query. Note that this struct is
|
|
|
|
sent also if the service is disabled or there is no record found. */
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
|
|
|
nscd_ssize_t naddrs;
|
|
|
|
nscd_ssize_t addrslen;
|
|
|
|
nscd_ssize_t canonlen;
|
|
|
|
int32_t error;
|
|
|
|
} ai_response_header;
|
|
|
|
|
|
|
|
/* Structure filled in by __nscd_getai. */
|
|
|
|
struct nscd_ai_result
|
|
|
|
{
|
|
|
|
int naddrs;
|
|
|
|
char *canon;
|
|
|
|
uint8_t *family;
|
|
|
|
char *addrs;
|
|
|
|
};
|
|
|
|
|
2004-09-30 10:24:51 +08:00
|
|
|
/* Structure sent in reply to initgroups query. Note that this struct is
|
|
|
|
sent also if the service is disabled or there is no record found. */
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
|
|
|
nscd_ssize_t ngrps;
|
|
|
|
} initgr_response_header;
|
|
|
|
|
2004-09-15 16:25:49 +08:00
|
|
|
|
2007-01-14 13:26:04 +08:00
|
|
|
/* Structure sent in reply to services query. Note that this struct is
|
|
|
|
sent also if the service is disabled or there is no record found. */
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
|
|
|
nscd_ssize_t s_name_len;
|
|
|
|
nscd_ssize_t s_proto_len;
|
|
|
|
nscd_ssize_t s_aliases_cnt;
|
|
|
|
int32_t s_port;
|
|
|
|
} serv_response_header;
|
|
|
|
|
|
|
|
|
2011-10-07 22:06:31 +08:00
|
|
|
/* Structure send in reply to netgroup query. Note that this struct is
|
|
|
|
sent also if the service is disabled or there is no record found. */
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
|
|
|
nscd_ssize_t nresults;
|
|
|
|
nscd_ssize_t result_len;
|
|
|
|
} netgroup_response_header;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
int32_t version;
|
|
|
|
int32_t found;
|
|
|
|
int32_t result;
|
|
|
|
} innetgroup_response_header;
|
|
|
|
|
|
|
|
|
2004-09-08 23:46:42 +08:00
|
|
|
/* Type for offsets in data part of database. */
|
|
|
|
typedef uint32_t ref_t;
|
|
|
|
/* Value for invalid/no reference. */
|
|
|
|
#define ENDREF UINT32_MAX
|
|
|
|
|
2004-09-13 13:52:46 +08:00
|
|
|
/* Timestamp type. */
|
|
|
|
typedef uint64_t nscd_time_t;
|
|
|
|
|
2011-02-06 09:07:27 +08:00
|
|
|
/* Maximum timestamp. */
|
|
|
|
#define MAX_TIMEOUT_VALUE \
|
|
|
|
(sizeof (time_t) == sizeof (long int) ? LONG_MAX : INT_MAX)
|
|
|
|
|
2004-09-08 23:46:42 +08:00
|
|
|
/* Alignment requirement of the beginning of the data region. */
|
|
|
|
#define ALIGN 16
|
|
|
|
|
|
|
|
|
|
|
|
/* Head of record in data part of database. */
|
|
|
|
struct datahead
|
|
|
|
{
|
2004-09-13 13:52:46 +08:00
|
|
|
nscd_ssize_t allocsize; /* Allocated Bytes. */
|
|
|
|
nscd_ssize_t recsize; /* Size of the record. */
|
|
|
|
nscd_time_t timeout; /* Time when this entry becomes invalid. */
|
2004-09-15 16:25:49 +08:00
|
|
|
uint8_t notfound; /* Nonzero if data has not been found. */
|
2004-09-13 13:52:46 +08:00
|
|
|
uint8_t nreloads; /* Reloads without use. */
|
2004-09-15 16:25:49 +08:00
|
|
|
uint8_t usable; /* False if the entry must be ignored. */
|
2011-02-06 09:07:27 +08:00
|
|
|
uint8_t unused; /* Unused. */
|
|
|
|
uint32_t ttl; /* TTL value used. */
|
2004-09-08 23:46:42 +08:00
|
|
|
|
|
|
|
/* We need to have the following element aligned for the response
|
|
|
|
header data types and their use in the 'struct dataset' types
|
|
|
|
defined in the XXXcache.c files. */
|
|
|
|
union
|
|
|
|
{
|
|
|
|
pw_response_header pwdata;
|
|
|
|
gr_response_header grdata;
|
|
|
|
hst_response_header hstdata;
|
2004-09-15 16:25:49 +08:00
|
|
|
ai_response_header aidata;
|
2004-09-30 10:24:51 +08:00
|
|
|
initgr_response_header initgrdata;
|
2007-01-14 13:26:04 +08:00
|
|
|
serv_response_header servdata;
|
2011-10-07 22:06:31 +08:00
|
|
|
netgroup_response_header netgroupdata;
|
|
|
|
innetgroup_response_header innetgroupdata;
|
2004-09-13 13:52:46 +08:00
|
|
|
nscd_ssize_t align1;
|
|
|
|
nscd_time_t align2;
|
2004-09-08 23:46:42 +08:00
|
|
|
} data[0];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* Structure for one hash table entry. */
|
|
|
|
struct hashentry
|
|
|
|
{
|
|
|
|
request_type type:8; /* Which type of dataset. */
|
|
|
|
bool first; /* True if this was the original key. */
|
2004-09-13 13:52:46 +08:00
|
|
|
nscd_ssize_t len; /* Length of key. */
|
2004-09-08 23:46:42 +08:00
|
|
|
ref_t key; /* Pointer to key. */
|
2004-09-13 13:52:46 +08:00
|
|
|
int32_t owner; /* If secure table, this is the owner. */
|
2004-09-08 23:46:42 +08:00
|
|
|
ref_t next; /* Next entry in this hash bucket list. */
|
|
|
|
ref_t packet; /* Records for the result. */
|
|
|
|
union
|
|
|
|
{
|
|
|
|
struct hashentry *dellist; /* Next record to be deleted. This can be a
|
|
|
|
pointer since only nscd uses this field. */
|
|
|
|
ref_t *prevp; /* Pointer to field containing forward
|
|
|
|
reference. */
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* Current persistent database version. */
|
2011-10-31 13:51:16 +08:00
|
|
|
#define DB_VERSION 2
|
2004-09-08 23:46:42 +08:00
|
|
|
|
|
|
|
/* Maximum time allowed between updates of the timestamp. */
|
|
|
|
#define MAPPING_TIMEOUT (5 * 60)
|
|
|
|
|
|
|
|
|
2011-10-31 13:51:16 +08:00
|
|
|
/* Used indices for the EXTRA_DATA element of 'database_pers_head'.
|
|
|
|
Each database has its own indices. */
|
|
|
|
#define NSCD_HST_IDX_CONF_TIMESTAMP 0
|
|
|
|
|
|
|
|
|
2004-09-08 23:46:42 +08:00
|
|
|
/* Header of persistent database file. */
|
|
|
|
struct database_pers_head
|
|
|
|
{
|
2004-09-13 13:52:46 +08:00
|
|
|
int32_t version;
|
|
|
|
int32_t header_size;
|
|
|
|
volatile int32_t gc_cycle;
|
|
|
|
volatile int32_t nscd_certainly_running;
|
|
|
|
volatile nscd_time_t timestamp;
|
2011-10-31 13:51:16 +08:00
|
|
|
/* Room for extensions. */
|
|
|
|
volatile uint32_t extra_data[4];
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2004-09-13 13:52:46 +08:00
|
|
|
nscd_ssize_t module;
|
|
|
|
nscd_ssize_t data_size;
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2004-09-13 13:52:46 +08:00
|
|
|
nscd_ssize_t first_free; /* Offset of first free byte in data area. */
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2004-09-13 13:52:46 +08:00
|
|
|
nscd_ssize_t nentries;
|
|
|
|
nscd_ssize_t maxnentries;
|
|
|
|
nscd_ssize_t maxnsearched;
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2004-09-13 13:52:46 +08:00
|
|
|
uint64_t poshit;
|
|
|
|
uint64_t neghit;
|
|
|
|
uint64_t posmiss;
|
|
|
|
uint64_t negmiss;
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2004-09-13 13:52:46 +08:00
|
|
|
uint64_t rdlockdelayed;
|
|
|
|
uint64_t wrlockdelayed;
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2004-09-13 13:52:46 +08:00
|
|
|
uint64_t addfailed;
|
2004-09-08 23:46:42 +08:00
|
|
|
|
|
|
|
ref_t array[0];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* Mapped database record. */
|
|
|
|
struct mapped_database
|
|
|
|
{
|
|
|
|
const struct database_pers_head *head;
|
|
|
|
const char *data;
|
|
|
|
size_t mapsize;
|
2004-09-30 10:24:51 +08:00
|
|
|
int counter; /* > 0 indicates it is usable. */
|
2006-10-03 00:34:25 +08:00
|
|
|
size_t datasize;
|
2004-09-08 23:46:42 +08:00
|
|
|
};
|
|
|
|
#define NO_MAPPING ((struct mapped_database *) -1l)
|
|
|
|
|
|
|
|
struct locked_map_ptr
|
|
|
|
{
|
|
|
|
int lock;
|
|
|
|
struct mapped_database *mapped;
|
|
|
|
};
|
2004-11-10 15:50:03 +08:00
|
|
|
#define libc_locked_map_ptr(class, name) class struct locked_map_ptr name
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2012-05-16 02:35:53 +08:00
|
|
|
/* Try acquiring lock for mapptr, returns true if it succeeds, false
|
|
|
|
if not. */
|
|
|
|
static inline bool __nscd_acquire_maplock (volatile struct locked_map_ptr *mapptr)
|
|
|
|
{
|
|
|
|
int cnt = 0;
|
|
|
|
while (__builtin_expect (atomic_compare_and_exchange_val_acq (&mapptr->lock,
|
|
|
|
1, 0) != 0, 0))
|
|
|
|
{
|
|
|
|
// XXX Best number of rounds?
|
|
|
|
if (__builtin_expect (++cnt > 5, 0))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
atomic_delay ();
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2003-04-23 03:52:59 +08:00
|
|
|
/* Open socket connection to nscd server. */
|
2004-06-06 14:06:02 +08:00
|
|
|
extern int __nscd_open_socket (const char *key, size_t keylen,
|
|
|
|
request_type type, void *response,
|
|
|
|
size_t responselen) attribute_hidden;
|
2003-04-23 03:52:59 +08:00
|
|
|
|
2011-10-31 13:51:16 +08:00
|
|
|
/* Try to get a file descriptor for the shared meory segment
|
|
|
|
containing the database. */
|
|
|
|
extern struct mapped_database *__nscd_get_mapping (request_type type,
|
|
|
|
const char *key,
|
|
|
|
struct mapped_database **mappedp) attribute_hidden;
|
|
|
|
|
2004-09-08 23:46:42 +08:00
|
|
|
/* Get reference of mapping. */
|
|
|
|
extern struct mapped_database *__nscd_get_map_ref (request_type type,
|
|
|
|
const char *name,
|
2006-04-26 07:50:31 +08:00
|
|
|
volatile struct locked_map_ptr *mapptr,
|
2004-09-30 16:03:44 +08:00
|
|
|
int *gc_cyclep);
|
2004-09-08 23:46:42 +08:00
|
|
|
|
|
|
|
/* Unmap database. */
|
|
|
|
extern void __nscd_unmap (struct mapped_database *mapped);
|
|
|
|
|
|
|
|
/* Drop reference of mapping. */
|
|
|
|
static inline int __nscd_drop_map_ref (struct mapped_database *map,
|
2004-09-30 16:03:44 +08:00
|
|
|
int *gc_cycle)
|
2004-09-08 23:46:42 +08:00
|
|
|
{
|
|
|
|
if (map != NO_MAPPING)
|
|
|
|
{
|
2004-09-30 16:03:44 +08:00
|
|
|
int now_cycle = map->head->gc_cycle;
|
|
|
|
if (__builtin_expect (now_cycle != *gc_cycle, 0))
|
|
|
|
{
|
|
|
|
/* We might have read inconsistent data. */
|
|
|
|
*gc_cycle = now_cycle;
|
|
|
|
return -1;
|
|
|
|
}
|
2004-09-08 23:46:42 +08:00
|
|
|
|
|
|
|
if (atomic_decrement_val (&map->counter) == 0)
|
|
|
|
__nscd_unmap (map);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Search the mapped database. */
|
2007-01-31 17:14:21 +08:00
|
|
|
extern struct datahead *__nscd_cache_search (request_type type,
|
|
|
|
const char *key,
|
|
|
|
size_t keylen,
|
2009-05-16 12:17:08 +08:00
|
|
|
const struct mapped_database *mapped,
|
|
|
|
size_t datalen);
|
2004-09-08 23:46:42 +08:00
|
|
|
|
2005-02-23 06:51:26 +08:00
|
|
|
/* Wrappers around read, readv and write that only read/write less than LEN
|
|
|
|
bytes on error or EOF. */
|
|
|
|
extern ssize_t __readall (int fd, void *buf, size_t len)
|
|
|
|
attribute_hidden;
|
|
|
|
extern ssize_t __readvall (int fd, const struct iovec *iov, int iovcnt)
|
|
|
|
attribute_hidden;
|
|
|
|
extern ssize_t writeall (int fd, const void *buf, size_t len)
|
|
|
|
attribute_hidden;
|
2005-11-23 03:06:27 +08:00
|
|
|
extern ssize_t sendfileall (int tofd, int fromfd, off_t off, size_t len)
|
|
|
|
attribute_hidden;
|
2005-02-23 06:51:26 +08:00
|
|
|
|
2011-10-31 13:51:16 +08:00
|
|
|
/* Get netlink timestamp counter from mapped area or zero. */
|
|
|
|
extern uint32_t __nscd_get_nl_timestamp (void);
|
|
|
|
|
1998-10-19 06:02:47 +08:00
|
|
|
#endif /* nscd.h */
|