1*a9fa9459Szrj /* filemode.c -- make a string describing file modes
2*a9fa9459Szrj    Copyright (C) 1985-2016 Free Software Foundation, Inc.
3*a9fa9459Szrj 
4*a9fa9459Szrj    This program is free software; you can redistribute it and/or modify
5*a9fa9459Szrj    it under the terms of the GNU General Public License as published by
6*a9fa9459Szrj    the Free Software Foundation; either version 3, or (at your option)
7*a9fa9459Szrj    any later version.
8*a9fa9459Szrj 
9*a9fa9459Szrj    This program is distributed in the hope that it will be useful,
10*a9fa9459Szrj    but WITHOUT ANY WARRANTY; without even the implied warranty of
11*a9fa9459Szrj    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12*a9fa9459Szrj    GNU General Public License for more details.
13*a9fa9459Szrj 
14*a9fa9459Szrj    You should have received a copy of the GNU General Public License
15*a9fa9459Szrj    along with this program; if not, write to the Free Software
16*a9fa9459Szrj    Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA
17*a9fa9459Szrj    02110-1301, USA.  */
18*a9fa9459Szrj 
19*a9fa9459Szrj #include "sysdep.h"
20*a9fa9459Szrj #include "bfd.h"
21*a9fa9459Szrj #include "bucomm.h"
22*a9fa9459Szrj 
23*a9fa9459Szrj static char ftypelet (unsigned long);
24*a9fa9459Szrj static void setst (unsigned long, char *);
25*a9fa9459Szrj 
26*a9fa9459Szrj /* filemodestring - fill in string STR with an ls-style ASCII
27*a9fa9459Szrj    representation of the st_mode field of file stats block STATP.
28*a9fa9459Szrj    10 characters are stored in STR; no terminating null is added.
29*a9fa9459Szrj    The characters stored in STR are:
30*a9fa9459Szrj 
31*a9fa9459Szrj    0	File type.  'd' for directory, 'c' for character
32*a9fa9459Szrj 	special, 'b' for block special, 'm' for multiplex,
33*a9fa9459Szrj 	'l' for symbolic link, 's' for socket, 'p' for fifo,
34*a9fa9459Szrj 	'-' for any other file type
35*a9fa9459Szrj 
36*a9fa9459Szrj    1	'r' if the owner may read, '-' otherwise.
37*a9fa9459Szrj 
38*a9fa9459Szrj    2	'w' if the owner may write, '-' otherwise.
39*a9fa9459Szrj 
40*a9fa9459Szrj    3	'x' if the owner may execute, 's' if the file is
41*a9fa9459Szrj 	set-user-id, '-' otherwise.
42*a9fa9459Szrj 	'S' if the file is set-user-id, but the execute
43*a9fa9459Szrj 	bit isn't set.
44*a9fa9459Szrj 
45*a9fa9459Szrj    4	'r' if group members may read, '-' otherwise.
46*a9fa9459Szrj 
47*a9fa9459Szrj    5	'w' if group members may write, '-' otherwise.
48*a9fa9459Szrj 
49*a9fa9459Szrj    6	'x' if group members may execute, 's' if the file is
50*a9fa9459Szrj 	set-group-id, '-' otherwise.
51*a9fa9459Szrj 	'S' if it is set-group-id but not executable.
52*a9fa9459Szrj 
53*a9fa9459Szrj    7	'r' if any user may read, '-' otherwise.
54*a9fa9459Szrj 
55*a9fa9459Szrj    8	'w' if any user may write, '-' otherwise.
56*a9fa9459Szrj 
57*a9fa9459Szrj    9	'x' if any user may execute, 't' if the file is "sticky"
58*a9fa9459Szrj 	(will be retained in swap space after execution), '-'
59*a9fa9459Szrj 	otherwise.
60*a9fa9459Szrj 	'T' if the file is sticky but not executable.  */
61*a9fa9459Szrj 
62*a9fa9459Szrj /* Get definitions for the file permission bits.  */
63*a9fa9459Szrj 
64*a9fa9459Szrj #ifndef S_IRWXU
65*a9fa9459Szrj #define S_IRWXU 0700
66*a9fa9459Szrj #endif
67*a9fa9459Szrj #ifndef S_IRUSR
68*a9fa9459Szrj #define S_IRUSR 0400
69*a9fa9459Szrj #endif
70*a9fa9459Szrj #ifndef S_IWUSR
71*a9fa9459Szrj #define S_IWUSR 0200
72*a9fa9459Szrj #endif
73*a9fa9459Szrj #ifndef S_IXUSR
74*a9fa9459Szrj #define S_IXUSR 0100
75*a9fa9459Szrj #endif
76*a9fa9459Szrj 
77*a9fa9459Szrj #ifndef S_IRWXG
78*a9fa9459Szrj #define S_IRWXG 0070
79*a9fa9459Szrj #endif
80*a9fa9459Szrj #ifndef S_IRGRP
81*a9fa9459Szrj #define S_IRGRP 0040
82*a9fa9459Szrj #endif
83*a9fa9459Szrj #ifndef S_IWGRP
84*a9fa9459Szrj #define S_IWGRP 0020
85*a9fa9459Szrj #endif
86*a9fa9459Szrj #ifndef S_IXGRP
87*a9fa9459Szrj #define S_IXGRP 0010
88*a9fa9459Szrj #endif
89*a9fa9459Szrj 
90*a9fa9459Szrj #ifndef S_IRWXO
91*a9fa9459Szrj #define S_IRWXO 0007
92*a9fa9459Szrj #endif
93*a9fa9459Szrj #ifndef S_IROTH
94*a9fa9459Szrj #define S_IROTH 0004
95*a9fa9459Szrj #endif
96*a9fa9459Szrj #ifndef S_IWOTH
97*a9fa9459Szrj #define S_IWOTH 0002
98*a9fa9459Szrj #endif
99*a9fa9459Szrj #ifndef S_IXOTH
100*a9fa9459Szrj #define S_IXOTH 0001
101*a9fa9459Szrj #endif
102*a9fa9459Szrj 
103*a9fa9459Szrj /* Like filemodestring, but only the relevant part of the `struct stat'
104*a9fa9459Szrj    is given as an argument.  */
105*a9fa9459Szrj 
106*a9fa9459Szrj void
mode_string(unsigned long mode,char * str)107*a9fa9459Szrj mode_string (unsigned long mode, char *str)
108*a9fa9459Szrj {
109*a9fa9459Szrj   str[0] = ftypelet ((unsigned long) mode);
110*a9fa9459Szrj   str[1] = (mode & S_IRUSR) != 0 ? 'r' : '-';
111*a9fa9459Szrj   str[2] = (mode & S_IWUSR) != 0 ? 'w' : '-';
112*a9fa9459Szrj   str[3] = (mode & S_IXUSR) != 0 ? 'x' : '-';
113*a9fa9459Szrj   str[4] = (mode & S_IRGRP) != 0 ? 'r' : '-';
114*a9fa9459Szrj   str[5] = (mode & S_IWGRP) != 0 ? 'w' : '-';
115*a9fa9459Szrj   str[6] = (mode & S_IXGRP) != 0 ? 'x' : '-';
116*a9fa9459Szrj   str[7] = (mode & S_IROTH) != 0 ? 'r' : '-';
117*a9fa9459Szrj   str[8] = (mode & S_IWOTH) != 0 ? 'w' : '-';
118*a9fa9459Szrj   str[9] = (mode & S_IXOTH) != 0 ? 'x' : '-';
119*a9fa9459Szrj   setst ((unsigned long) mode, str);
120*a9fa9459Szrj }
121*a9fa9459Szrj 
122*a9fa9459Szrj /* Return a character indicating the type of file described by
123*a9fa9459Szrj    file mode BITS:
124*a9fa9459Szrj    'd' for directories
125*a9fa9459Szrj    'b' for block special files
126*a9fa9459Szrj    'c' for character special files
127*a9fa9459Szrj    'm' for multiplexer files
128*a9fa9459Szrj    'l' for symbolic links
129*a9fa9459Szrj    's' for sockets
130*a9fa9459Szrj    'p' for fifos
131*a9fa9459Szrj    '-' for any other file type.  */
132*a9fa9459Szrj 
133*a9fa9459Szrj #ifndef S_ISDIR
134*a9fa9459Szrj #ifdef S_IFDIR
135*a9fa9459Szrj #define S_ISDIR(i) (((i) & S_IFMT) == S_IFDIR)
136*a9fa9459Szrj #else /* ! defined (S_IFDIR) */
137*a9fa9459Szrj #define S_ISDIR(i) (((i) & 0170000) == 040000)
138*a9fa9459Szrj #endif /* ! defined (S_IFDIR) */
139*a9fa9459Szrj #endif /* ! defined (S_ISDIR) */
140*a9fa9459Szrj 
141*a9fa9459Szrj #ifndef S_ISBLK
142*a9fa9459Szrj #ifdef S_IFBLK
143*a9fa9459Szrj #define S_ISBLK(i) (((i) & S_IFMT) == S_IFBLK)
144*a9fa9459Szrj #else /* ! defined (S_IFBLK) */
145*a9fa9459Szrj #define S_ISBLK(i) 0
146*a9fa9459Szrj #endif /* ! defined (S_IFBLK) */
147*a9fa9459Szrj #endif /* ! defined (S_ISBLK) */
148*a9fa9459Szrj 
149*a9fa9459Szrj #ifndef S_ISCHR
150*a9fa9459Szrj #ifdef S_IFCHR
151*a9fa9459Szrj #define S_ISCHR(i) (((i) & S_IFMT) == S_IFCHR)
152*a9fa9459Szrj #else /* ! defined (S_IFCHR) */
153*a9fa9459Szrj #define S_ISCHR(i) 0
154*a9fa9459Szrj #endif /* ! defined (S_IFCHR) */
155*a9fa9459Szrj #endif /* ! defined (S_ISCHR) */
156*a9fa9459Szrj 
157*a9fa9459Szrj #ifndef S_ISFIFO
158*a9fa9459Szrj #ifdef S_IFIFO
159*a9fa9459Szrj #define S_ISFIFO(i) (((i) & S_IFMT) == S_IFIFO)
160*a9fa9459Szrj #else /* ! defined (S_IFIFO) */
161*a9fa9459Szrj #define S_ISFIFO(i) 0
162*a9fa9459Szrj #endif /* ! defined (S_IFIFO) */
163*a9fa9459Szrj #endif /* ! defined (S_ISFIFO) */
164*a9fa9459Szrj 
165*a9fa9459Szrj #ifndef S_ISSOCK
166*a9fa9459Szrj #ifdef S_IFSOCK
167*a9fa9459Szrj #define S_ISSOCK(i) (((i) & S_IFMT) == S_IFSOCK)
168*a9fa9459Szrj #else /* ! defined (S_IFSOCK) */
169*a9fa9459Szrj #define S_ISSOCK(i) 0
170*a9fa9459Szrj #endif /* ! defined (S_IFSOCK) */
171*a9fa9459Szrj #endif /* ! defined (S_ISSOCK) */
172*a9fa9459Szrj 
173*a9fa9459Szrj #ifndef S_ISLNK
174*a9fa9459Szrj #ifdef S_IFLNK
175*a9fa9459Szrj #define S_ISLNK(i) (((i) & S_IFMT) == S_IFLNK)
176*a9fa9459Szrj #else /* ! defined (S_IFLNK) */
177*a9fa9459Szrj #define S_ISLNK(i) 0
178*a9fa9459Szrj #endif /* ! defined (S_IFLNK) */
179*a9fa9459Szrj #endif /* ! defined (S_ISLNK) */
180*a9fa9459Szrj 
181*a9fa9459Szrj static char
ftypelet(unsigned long bits)182*a9fa9459Szrj ftypelet (unsigned long bits)
183*a9fa9459Szrj {
184*a9fa9459Szrj   if (S_ISDIR (bits))
185*a9fa9459Szrj     return 'd';
186*a9fa9459Szrj   if (S_ISLNK (bits))
187*a9fa9459Szrj     return 'l';
188*a9fa9459Szrj   if (S_ISBLK (bits))
189*a9fa9459Szrj     return 'b';
190*a9fa9459Szrj   if (S_ISCHR (bits))
191*a9fa9459Szrj     return 'c';
192*a9fa9459Szrj   if (S_ISSOCK (bits))
193*a9fa9459Szrj     return 's';
194*a9fa9459Szrj   if (S_ISFIFO (bits))
195*a9fa9459Szrj     return 'p';
196*a9fa9459Szrj 
197*a9fa9459Szrj #ifdef S_IFMT
198*a9fa9459Szrj #ifdef S_IFMPC
199*a9fa9459Szrj   if ((bits & S_IFMT) == S_IFMPC
200*a9fa9459Szrj       || (bits & S_IFMT) == S_IFMPB)
201*a9fa9459Szrj     return 'm';
202*a9fa9459Szrj #endif
203*a9fa9459Szrj #ifdef S_IFNWK
204*a9fa9459Szrj   if ((bits & S_IFMT) == S_IFNWK)
205*a9fa9459Szrj     return 'n';
206*a9fa9459Szrj #endif
207*a9fa9459Szrj #endif
208*a9fa9459Szrj 
209*a9fa9459Szrj   return '-';
210*a9fa9459Szrj }
211*a9fa9459Szrj 
212*a9fa9459Szrj /* Set the 's' and 't' flags in file attributes string CHARS,
213*a9fa9459Szrj    according to the file mode BITS.  */
214*a9fa9459Szrj 
215*a9fa9459Szrj static void
setst(unsigned long bits ATTRIBUTE_UNUSED,char * chars ATTRIBUTE_UNUSED)216*a9fa9459Szrj setst (unsigned long bits ATTRIBUTE_UNUSED, char *chars ATTRIBUTE_UNUSED)
217*a9fa9459Szrj {
218*a9fa9459Szrj #ifdef S_ISUID
219*a9fa9459Szrj   if (bits & S_ISUID)
220*a9fa9459Szrj     {
221*a9fa9459Szrj       if (chars[3] != 'x')
222*a9fa9459Szrj 	/* Set-uid, but not executable by owner.  */
223*a9fa9459Szrj 	chars[3] = 'S';
224*a9fa9459Szrj       else
225*a9fa9459Szrj 	chars[3] = 's';
226*a9fa9459Szrj     }
227*a9fa9459Szrj #endif
228*a9fa9459Szrj #ifdef S_ISGID
229*a9fa9459Szrj   if (bits & S_ISGID)
230*a9fa9459Szrj     {
231*a9fa9459Szrj       if (chars[6] != 'x')
232*a9fa9459Szrj 	/* Set-gid, but not executable by group.  */
233*a9fa9459Szrj 	chars[6] = 'S';
234*a9fa9459Szrj       else
235*a9fa9459Szrj 	chars[6] = 's';
236*a9fa9459Szrj     }
237*a9fa9459Szrj #endif
238*a9fa9459Szrj #ifdef S_ISVTX
239*a9fa9459Szrj   if (bits & S_ISVTX)
240*a9fa9459Szrj     {
241*a9fa9459Szrj       if (chars[9] != 'x')
242*a9fa9459Szrj 	/* Sticky, but not executable by others.  */
243*a9fa9459Szrj 	chars[9] = 'T';
244*a9fa9459Szrj       else
245*a9fa9459Szrj 	chars[9] = 't';
246*a9fa9459Szrj     }
247*a9fa9459Szrj #endif
248*a9fa9459Szrj }
249