Add headers for OpenSolaris extensions that glibc implements
[kopensolaris-gnu/glibc.git] / sysdeps / unix / sysv / solaris2 / kopensolaris-gnu / priv_str.c
1 /* Copyright (C) 2008 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3    Contributed by David Bartley <dtbartle@csclub.uwaterloo.ca>, 2008.
4
5    The GNU C Library is free software; you can redistribute it and/or
6    modify it under the terms of the GNU Lesser General Public
7    License as published by the Free Software Foundation; either
8    version 2.1 of the License, or (at your option) any later version.
9
10    The GNU C Library is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13    Lesser General Public License for more details.
14
15    You should have received a copy of the GNU Lesser General Public
16    License along with the GNU C Library; if not, write to the Free
17    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
18    02111-1307 USA.  */
19
20 #include <privP.h>
21 #include <priv.h>
22 #include <zone.h>
23 #include <string.h>
24 #include <stdlib.h>
25 #include <errno.h>
26
27 /* Docs: http://docs.sun.com/app/docs/doc/816-5168/priv-str-to-set-3c  */
28
29 priv_set_t *priv_str_to_set (const char *buf, const char *sep,
30       const char **endptr)
31 {
32   /* Take a copy of the string since strtok_r will modify it.  */
33   char *str = strdup (buf);
34   if (!str)
35     return NULL;
36
37   priv_set_t *set = priv_allocset ();
38   if (!set)
39     {
40       free (str);
41       return NULL;
42     }
43   priv_emptyset (set);
44   const priv_data_t *data = __priv_parse_data_cached ();
45   if (!data)
46     return NULL;
47   priv_set_t *basic = data->pd_basicprivs;
48
49   char *saveptr;
50   char *priv = strtok_r (str, sep, &saveptr);
51   if (!priv)
52     return set;
53   do
54     {
55       if (strcmp (priv, "basic") == 0 && basic)
56           priv_union (basic, set);
57       else if (strcmp (priv, "all") == 0)
58           priv_fillset (set);
59       else if (strcmp (priv, "none") == 0)
60           priv_emptyset (set);
61       else if (strcmp (priv, "zone") == 0)
62         {
63           priv_set_t *zone = priv_allocset ();
64           if (!zone)
65             goto inval;
66           if (zone_getattr (getzoneid (), ZONE_ATTR_PRIVSET,
67                 zone, __PRIVSETSIZE) == 0)
68             priv_union (zone, set);
69           priv_freeset (zone);
70         }
71       else
72         {
73           int negate = *str == '-' || *str == '!';
74           if (negate)
75             str++;
76
77           int res;
78           if (negate)
79             res = priv_delset (set, str);
80           else
81             res = priv_addset (set, str);
82           if (res == -1)
83             goto inval;
84         }
85     }
86   while ((priv = strtok_r (NULL, sep, &saveptr))) ;
87
88   free (str);
89   return set;
90
91 inval:
92
93    priv_freeset (set);
94    free (str);
95    __set_errno (EINVAL);
96   return NULL;
97 }