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