summaryrefslogtreecommitdiff
path: root/src/libnsss/nsss_switch_grp_getlist.c
blob: f637c0dd79587b6c7720b4091ee3885c0ef69070 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
/* ISC license. */

#include <string.h>
#include <sys/uio.h>
#include <errno.h>

#include <skalibs/posixishard.h>
#include <skalibs/uint32.h>
#include <skalibs/uint64.h>
#include <skalibs/types.h>
#include <skalibs/buffer.h>
#include <skalibs/stralloc.h>
#include <skalibs/unix-timed.h>

#include <nsss/nsss-switch.h>
#include "nsss-switch-internal.h"

 /*
    Expects:
      8 bytes: n
      8 bytes: r
      min(n, r) * sizeof(gid_t) bytes: the list of gids
  */

static int nsss_switch_grouplist_read (buffer *b, size_t n, size_t *r, gid_t *gids, stralloc *sa, tain const *deadline, tain *stamp)
{
  size_t rr ;
  uint64_t res ;
  char buf[8] ;
  if (!buffer_timed_get(b, buf, 8, deadline, stamp)) return 0 ;
  uint64_unpack_big(buf, &res) ;
  if (res != n) return (errno = EPROTO, 0) ;
  if (!buffer_timed_get(b, buf, 8, deadline, stamp)) return 0 ;
  uint64_unpack_big(buf, &res) ;
  rr = res ;
  if (res > n) res = n ;
  if (res)
  {
    int sawasnull = !sa->s ;
    size_t sabase = sa->len ;
    if (!stralloc_readyplus(sa, res * sizeof(gid_t))) return 0 ;
    if (!buffer_timed_get(b, sa->s + sabase, res * sizeof(gid_t), deadline, stamp))
    {
      if (sawasnull) stralloc_free(sa) ; else sa->len = sabase ;
      return 0 ;
    }
    for (size_t i = 0 ; i < res ; i++)
      gid_unpack_big(sa->s + sabase + i * sizeof(gid_t), gids + i) ;
    if (sawasnull) stralloc_free(sa) ; else sa->len = sabase ;  
  }
  *r = rr ;
  return 1 ;
}

int nsss_switch_grp_getlist (nsss_switch_t *a, char const *user, gid_t *gids, size_t n, size_t *r, stralloc *sa, tain const *deadline, tain *stamp)
{
  char buf[13] = { NSSS_SWITCH_GRP_GETLIST } ;
  size_t len = strlen(user) ;
  struct iovec v[2] = { { .iov_base = buf, .iov_len = 13 }, { .iov_base = (char *)user, .iov_len = len + 1 } } ;
  if (len > NSSS_SWITCH_NAME_MAXLEN - 1) return (errno = EINVAL, 0) ;
  uint64_pack_big(buf + 1, n) ;
  uint32_pack_big(buf + 9, len + 1) ;
  if (!nsss_switch_sendv(a, v, 2, deadline, stamp)) return 0 ;
  if (!buffer_timed_get(&a->b, &buf[0], 1, deadline, stamp)) return 0 ;
  if (buf[0]) return (errno = (unsigned char)buf[0], 0) ;
  return nsss_switch_grouplist_read(&a->b, n, r, gids, sa, deadline, stamp) ;
}