blob: aaf6fe5ca29cb4b6a6832225d47972e6953fd51d [file] [log] [blame]
Christian Heimesc77d9f32013-06-22 21:05:02 +02001/* stat.h interface
2 *
3 * The module defines all S_IF*, S_I*, UF_*, SF_* and ST_* constants to
4 * sensible default values as well as defines S_IS*() macros in order to keep
5 * backward compatibility with the old stat.py module.
6 *
7 * New constants and macros such as S_IFDOOR / S_ISDOOR() are always defined
8 * as int 0.
9 *
10 * NOTE: POSIX only defines the values of the S_I* permission bits.
11 *
12 */
13
14#define PY_SSIZE_T_CLEAN
15#include "Python.h"
16
17#ifdef __cplusplus
18extern "C" {
19#endif
20
21#ifdef HAVE_SYS_TYPES_H
22#include <sys/types.h>
23#endif /* HAVE_SYS_TYPES_H */
24
25#ifdef HAVE_SYS_STAT_H
26#include <sys/stat.h>
27#endif /* HAVE_SYS_STAT_H */
28
Victor Stinnerc6ebd162013-06-23 01:49:42 +020029#ifdef MS_WINDOWS
30typedef unsigned short mode_t;
31#endif
32
Christian Heimesc77d9f32013-06-22 21:05:02 +020033/* From Python's stat.py */
34#ifndef S_IMODE
35# define S_IMODE 07777
36#endif
37
38/* S_IFXXX constants (file types)
39 *
40 * Only the names are defined by POSIX but not their value. All common file
41 * types seems to have the same numeric value on all platforms, though.
42 */
43#ifndef S_IFMT
44# define S_IFMT 0170000
45#endif
46
47#ifndef S_IFDIR
48# define S_IFDIR 0040000
49#endif
50
51#ifndef S_IFCHR
52# define S_IFCHR 0020000
53#endif
54
55#ifndef S_IFBLK
56# define S_IFBLK 0060000
57#endif
58
59#ifndef S_IFREG
60# define S_IFREG 0100000
61#endif
62
63#ifndef S_IFIFO
64# define S_IFIFO 0010000
65#endif
66
67#ifndef S_IFLNK
68# define S_IFLNK 0120000
69#endif
70
71#ifndef S_IFSOCK
72# define S_IFSOCK 0140000
73#endif
74
75#ifndef S_IFDOOR
76# define S_IFDOOR 0
77#endif
78
79#ifndef S_IFPORT
80# define S_IFPORT 0
81#endif
82
83#ifndef S_IFWHT
84# define S_IFWHT 0
85#endif
86
87
88/* S_ISXXX() */
89#ifndef S_ISDIR
90# define S_ISDIR(mode) ((mode) & S_IFMT) == S_IDIR
91#endif
92
93#ifndef S_ISCHR
94# define S_ISCHR(mode) ((mode) & S_IFMT) == S_ICHR
95#endif
96
97#ifndef S_ISBLK
98# define S_ISBLK(mode) ((mode) & S_IFMT) == S_IBLK
99#endif
100
101#ifndef S_ISREG
102# define S_ISREG(mode) ((mode) & S_IFMT) == S_IREG
103#endif
104
105#ifndef S_ISFIFO
106# define S_ISFIFO(mode) ((mode) & S_IFMT) == S_IFIFO
107#endif
108
109#ifndef S_ISLNK
110# define S_ISLNK(mode) ((mode) & S_IFMT) == S_IFLNK
111#endif
112
113#ifndef S_ISSOCK
114# define S_ISSOCK(mode) ((mode) & S_IFMT) == S_IFSOCK
115#endif
116
117#ifndef S_ISDOOR
118# define S_ISDOOR(mode) 0
119#endif
120
121#ifndef S_ISPORT
122# define S_ISPORT(mode) 0
123#endif
124
125#ifndef S_ISWHT
126# define S_ISWHT(mode) 0
127#endif
128
129
130/* S_I* file permission
131 *
132 * The permission bit value are defined by POSIX standards.
133 */
134#ifndef S_ISUID
135# define S_ISUID 04000
136#endif
137
138#ifndef S_ISGID
139# define S_ISGID 02000
140#endif
141
142/* what is S_ENFMT? */
143#ifndef S_ENFMT
144# define S_ENFMT S_ISGID
145#endif
146
147#ifndef S_ISVTX
148# define S_ISVTX 01000
149#endif
150
151#ifndef S_IREAD
152# define S_IREAD 00400
153#endif
154
155#ifndef S_IWRITE
156# define S_IWRITE 00200
157#endif
158
159#ifndef S_IEXEC
160# define S_IEXEC 00100
161#endif
162
163#ifndef S_IRWXU
164# define S_IRWXU 00700
165#endif
166
167#ifndef S_IRUSR
168# define S_IRUSR 00400
169#endif
170
171#ifndef S_IWUSR
172# define S_IWUSR 00200
173#endif
174
175#ifndef S_IXUSR
176# define S_IXUSR 00100
177#endif
178
179#ifndef S_IRWXG
180# define S_IRWXG 00070
181#endif
182
183#ifndef S_IRGRP
184# define S_IRGRP 00040
185#endif
186
187#ifndef S_IWGRP
188# define S_IWGRP 00020
189#endif
190
191#ifndef S_IXGRP
192# define S_IXGRP 00010
193#endif
194
195#ifndef S_IRWXO
196# define S_IRWXO 00007
197#endif
198
199#ifndef S_IROTH
200# define S_IROTH 00004
201#endif
202
203#ifndef S_IWOTH
204# define S_IWOTH 00002
205#endif
206
207#ifndef S_IXOTH
208# define S_IXOTH 00001
209#endif
210
211
212/* Names for file flags */
213#ifndef UF_NODUMP
214# define UF_NODUMP 0x00000001
215#endif
216
217#ifndef UF_IMMUTABLE
218# define UF_IMMUTABLE 0x00000002
219#endif
220
221#ifndef UF_APPEND
222# define UF_APPEND 0x00000004
223#endif
224
225#ifndef UF_OPAQUE
226# define UF_OPAQUE 0x00000008
227#endif
228
229#ifndef UF_NOUNLINK
230# define UF_NOUNLINK 0x00000010
231#endif
232
233#ifndef UF_COMPRESSED
234# define UF_COMPRESSED 0x00000020
235#endif
236
237#ifndef UF_HIDDEN
238# define UF_HIDDEN 0x00008000
239#endif
240
241#ifndef SF_ARCHIVED
242# define SF_ARCHIVED 0x00010000
243#endif
244
245#ifndef SF_IMMUTABLE
246# define SF_IMMUTABLE 0x00020000
247#endif
248
249#ifndef SF_APPEND
250# define SF_APPEND 0x00040000
251#endif
252
253#ifndef SF_NOUNLINK
254# define SF_NOUNLINK 0x00100000
255#endif
256
257#ifndef SF_SNAPSHOT
258# define SF_SNAPSHOT 0x00200000
259#endif
260
261
262#define stat_S_ISFUNC(isfunc, doc) \
263 static PyObject * \
264 stat_ ##isfunc (PyObject *self, PyObject *omode) \
265 { \
266 unsigned long mode = PyLong_AsUnsignedLong(omode); \
267 if ((mode == (unsigned long)-1) && PyErr_Occurred()) { \
268 return NULL; \
269 } \
270 return PyBool_FromLong(isfunc(mode)); \
271 } \
272 PyDoc_STRVAR(stat_ ## isfunc ## _doc, doc)
273
274stat_S_ISFUNC(S_ISDIR,
275 "S_ISDIR(mode) -> bool\n\n"
276 "Return True if mode is from a directory.");
277
278stat_S_ISFUNC(S_ISCHR,
279 "S_ISCHR(mode) -> bool\n\n"
280 "Return True if mode is from a character special device file.");
281
282stat_S_ISFUNC(S_ISBLK,
283 "S_ISBLK(mode) -> bool\n\n"
284 "Return True if mode is from a block special device file.");
285
286stat_S_ISFUNC(S_ISREG,
287 "S_ISREG(mode) -> bool\n\n"
288 "Return True if mode is from a regular file.");
289
290stat_S_ISFUNC(S_ISFIFO,
291 "S_ISFIFO(mode) -> bool\n\n"
292 "Return True if mode is from a FIFO (named pipe).");
293
294stat_S_ISFUNC(S_ISLNK,
295 "S_ISLNK(mode) -> bool\n\n"
296 "Return True if mode is from a symbolic link.");
297
298stat_S_ISFUNC(S_ISSOCK,
299 "S_ISSOCK(mode) -> bool\n\n"
300 "Return True if mode is from a socket.");
301
302stat_S_ISFUNC(S_ISDOOR,
303 "S_ISDOOR(mode) -> bool\n\n"
304 "Return True if mode is from a door.");
305
306stat_S_ISFUNC(S_ISPORT,
307 "S_ISPORT(mode) -> bool\n\n"
308 "Return True if mode is from an event port.");
309
310stat_S_ISFUNC(S_ISWHT,
311 "S_ISWHT(mode) -> bool\n\n"
312 "Return True if mode is from a whiteout.");
313
314
315PyDoc_STRVAR(stat_S_IMODE_doc,
316"Return the portion of the file's mode that can be set by os.chmod().");
317
318static PyObject *
319stat_S_IMODE(PyObject *self, PyObject *omode)
320{
321 unsigned long mode = PyLong_AsUnsignedLong(omode);
322 if ((mode == (unsigned long)-1) && PyErr_Occurred()) {
323 return NULL;
324 }
325 return PyLong_FromUnsignedLong(mode & S_IMODE);
326}
327
328
329PyDoc_STRVAR(stat_S_IFMT_doc,
330"Return the portion of the file's mode that describes the file type.");
331
332static PyObject *
333stat_S_IFMT(PyObject *self, PyObject *omode)
334{
335 unsigned long mode = PyLong_AsUnsignedLong(omode);
336 if ((mode == (unsigned long)-1) && PyErr_Occurred()) {
337 return NULL;
338 }
339 return PyLong_FromUnsignedLong(mode & S_IFMT);
340}
341
342/* file type chars according to
343 http://en.wikibooks.org/wiki/C_Programming/POSIX_Reference/sys/stat.h */
344
345static char
346filetype(mode_t mode)
347{
348 /* common cases first */
349 if (S_ISREG(mode)) return '-';
350 if (S_ISDIR(mode)) return 'd';
351 if (S_ISLNK(mode)) return 'l';
352 /* special files */
353 if (S_ISBLK(mode)) return 'b';
354 if (S_ISCHR(mode)) return 'c';
355 if (S_ISFIFO(mode)) return 'p';
356 if (S_ISSOCK(mode)) return 's';
357 /* non-standard types */
358 if (S_ISDOOR(mode)) return 'D';
359 if (S_ISPORT(mode)) return 'P';
360 if (S_ISWHT(mode)) return 'w';
361 /* unknown */
362 return '?';
363}
364
365static void
366fileperm(mode_t mode, char *buf)
367{
368 buf[0] = mode & S_IRUSR ? 'r' : '-';
369 buf[1] = mode & S_IWUSR ? 'w' : '-';
370 if (mode & S_ISUID) {
371 buf[2] = mode & S_IXUSR ? 's' : 'S';
372 } else {
373 buf[2] = mode & S_IXUSR ? 'x' : '-';
374 }
375 buf[3] = mode & S_IRGRP ? 'r' : '-';
376 buf[4] = mode & S_IWGRP ? 'w' : '-';
377 if (mode & S_ISGID) {
378 buf[5] = mode & S_IXGRP ? 's' : 'S';
379 } else {
380 buf[5] = mode & S_IXGRP ? 'x' : '-';
381 }
382 buf[6] = mode & S_IROTH ? 'r' : '-';
383 buf[7] = mode & S_IWOTH ? 'w' : '-';
384 if (mode & S_ISVTX) {
385 buf[8] = mode & S_IXOTH ? 't' : 'T';
386 } else {
387 buf[8] = mode & S_IXOTH ? 'x' : '-';
388 }
389}
390
391PyDoc_STRVAR(stat_filemode_doc,
392"Convert a file's mode to a string of the form '-rwxrwxrwx'");
393
394static PyObject *
395stat_filemode(PyObject *self, PyObject *omode)
396{
397 char buf[10];
398 unsigned long mode;
399
400 mode = PyLong_AsUnsignedLong(omode);
401 if ((mode == (unsigned long)-1) && PyErr_Occurred()) {
402 return NULL;
403 }
404
405 buf[0] = filetype(mode);
406 fileperm(mode, &buf[1]);
407 return PyUnicode_FromStringAndSize(buf, 10);
408}
409
410
411static PyMethodDef stat_methods[] = {
412 {"S_ISDIR", stat_S_ISDIR, METH_O, stat_S_ISDIR_doc},
413 {"S_ISCHR", stat_S_ISCHR, METH_O, stat_S_ISCHR_doc},
414 {"S_ISBLK", stat_S_ISBLK, METH_O, stat_S_ISBLK_doc},
415 {"S_ISREG", stat_S_ISREG, METH_O, stat_S_ISREG_doc},
416 {"S_ISFIFO", stat_S_ISFIFO, METH_O, stat_S_ISFIFO_doc},
417 {"S_ISLNK", stat_S_ISLNK, METH_O, stat_S_ISLNK_doc},
418 {"S_ISSOCK", stat_S_ISSOCK, METH_O, stat_S_ISSOCK_doc},
419 {"S_ISDOOR", stat_S_ISDOOR, METH_O, stat_S_ISDOOR_doc},
420 {"S_ISPORT", stat_S_ISPORT, METH_O, stat_S_ISPORT_doc},
421 {"S_ISWHT", stat_S_ISWHT, METH_O, stat_S_ISWHT_doc},
422 {"S_IMODE", stat_S_IMODE, METH_O, stat_S_IMODE_doc},
423 {"S_IFMT", stat_S_IFMT, METH_O, stat_S_IFMT_doc},
424 {"filemode", stat_filemode, METH_O, stat_filemode_doc},
425 {NULL, NULL} /* sentinel */
426};
427
428
429PyDoc_STRVAR(module_doc,
430"S_IFMT_: file type bits\n\
431S_IFDIR: directory\n\
432S_IFCHR: character device\n\
433S_IFBLK: block device\n\
434S_IFREG: regular file\n\
435S_IFIFO: fifo (named pipe)\n\
436S_IFLNK: symbolic link\n\
437S_IFSOCK: socket file\n\
438S_IFDOOR: door\n\
439S_IFPORT: event port\n\
440S_IFWHT: whiteout\n\
441\n"
442
443"S_ISUID: set UID bit\n\
444S_ISGID: set GID bit\n\
445S_ENFMT: file locking enforcement\n\
446S_ISVTX: sticky bit\n\
447S_IREAD: Unix V7 synonym for S_IRUSR\n\
448S_IWRITE: Unix V7 synonym for S_IWUSR\n\
449S_IEXEC: Unix V7 synonym for S_IXUSR\n\
450S_IRWXU: mask for owner permissions\n\
451S_IRUSR: read by owner\n\
452S_IWUSR: write by owner\n\
453S_IXUSR: execute by owner\n\
454S_IRWXG: mask for group permissions\n\
455S_IRGRP: read by group\n\
456S_IWGRP: write by group\n\
457S_IXGRP: execute by group\n\
458S_IRWXO: mask for others (not in group) permissions\n\
459S_IROTH: read by others\n\
460S_IWOTH: write by others\n\
461S_IXOTH: execute by others\n\
462\n"
463
464"UF_NODUMP: do not dump file\n\
465UF_IMMUTABLE: file may not be changed\n\
466UF_APPEND: file may only be appended to\n\
467UF_OPAQUE: directory is opaque when viewed through a union stack\n\
468UF_NOUNLINK: file may not be renamed or deleted\n\
469UF_COMPRESSED: OS X: file is hfs-compressed\n\
470UF_HIDDEN: OS X: file should not be displayed\n\
471SF_ARCHIVED: file may be archived\n\
472SF_IMMUTABLE: file may not be changed\n\
473SF_APPEND: file may only be appended to\n\
474SF_NOUNLINK: file may not be renamed or deleted\n\
475SF_SNAPSHOT: file is a snapshot file\n\
476\n"
477
478"ST_MODE\n\
479ST_INO\n\
480ST_DEV\n\
481ST_NLINK\n\
482ST_UID\n\
483ST_GID\n\
484ST_SIZE\n\
485ST_ATIME\n\
486ST_MTIME\n\
487ST_CTIME\n\
488");
489
490
491static struct PyModuleDef statmodule = {
492 PyModuleDef_HEAD_INIT,
493 "_stat",
494 module_doc,
495 -1,
496 stat_methods,
497 NULL,
498 NULL,
499 NULL,
500 NULL
501};
502
503PyMODINIT_FUNC
504PyInit__stat(void)
505{
506 PyObject *m;
507 m = PyModule_Create(&statmodule);
508 if (m == NULL)
509 return NULL;
510
511 if (PyModule_AddIntMacro(m, S_IFDIR)) return NULL;
512 if (PyModule_AddIntMacro(m, S_IFCHR)) return NULL;
513 if (PyModule_AddIntMacro(m, S_IFBLK)) return NULL;
514 if (PyModule_AddIntMacro(m, S_IFREG)) return NULL;
515 if (PyModule_AddIntMacro(m, S_IFIFO)) return NULL;
516 if (PyModule_AddIntMacro(m, S_IFLNK)) return NULL;
517 if (PyModule_AddIntMacro(m, S_IFSOCK)) return NULL;
518 if (PyModule_AddIntMacro(m, S_IFDOOR)) return NULL;
519 if (PyModule_AddIntMacro(m, S_IFPORT)) return NULL;
520 if (PyModule_AddIntMacro(m, S_IFWHT)) return NULL;
521
522 if (PyModule_AddIntMacro(m, S_ISUID)) return NULL;
523 if (PyModule_AddIntMacro(m, S_ISGID)) return NULL;
524 if (PyModule_AddIntMacro(m, S_ISVTX)) return NULL;
525 if (PyModule_AddIntMacro(m, S_ENFMT)) return NULL;
526
527 if (PyModule_AddIntMacro(m, S_IREAD)) return NULL;
528 if (PyModule_AddIntMacro(m, S_IWRITE)) return NULL;
529 if (PyModule_AddIntMacro(m, S_IEXEC)) return NULL;
530
531 if (PyModule_AddIntMacro(m, S_IRWXU)) return NULL;
532 if (PyModule_AddIntMacro(m, S_IRUSR)) return NULL;
533 if (PyModule_AddIntMacro(m, S_IWUSR)) return NULL;
534 if (PyModule_AddIntMacro(m, S_IXUSR)) return NULL;
535
536 if (PyModule_AddIntMacro(m, S_IRWXG)) return NULL;
537 if (PyModule_AddIntMacro(m, S_IRGRP)) return NULL;
538 if (PyModule_AddIntMacro(m, S_IWGRP)) return NULL;
539 if (PyModule_AddIntMacro(m, S_IXGRP)) return NULL;
540
541 if (PyModule_AddIntMacro(m, S_IRWXO)) return NULL;
542 if (PyModule_AddIntMacro(m, S_IROTH)) return NULL;
543 if (PyModule_AddIntMacro(m, S_IWOTH)) return NULL;
544 if (PyModule_AddIntMacro(m, S_IXOTH)) return NULL;
545
546 if (PyModule_AddIntMacro(m, UF_NODUMP)) return NULL;
547 if (PyModule_AddIntMacro(m, UF_IMMUTABLE)) return NULL;
548 if (PyModule_AddIntMacro(m, UF_APPEND)) return NULL;
549 if (PyModule_AddIntMacro(m, UF_OPAQUE)) return NULL;
550 if (PyModule_AddIntMacro(m, UF_NOUNLINK)) return NULL;
551 if (PyModule_AddIntMacro(m, UF_COMPRESSED)) return NULL;
552 if (PyModule_AddIntMacro(m, UF_HIDDEN)) return NULL;
553 if (PyModule_AddIntMacro(m, SF_ARCHIVED)) return NULL;
554 if (PyModule_AddIntMacro(m, SF_IMMUTABLE)) return NULL;
555 if (PyModule_AddIntMacro(m, SF_APPEND)) return NULL;
556 if (PyModule_AddIntMacro(m, SF_NOUNLINK)) return NULL;
557 if (PyModule_AddIntMacro(m, SF_SNAPSHOT)) return NULL;
558
559 if (PyModule_AddIntConstant(m, "ST_MODE", 0)) return NULL;
560 if (PyModule_AddIntConstant(m, "ST_INO", 1)) return NULL;
561 if (PyModule_AddIntConstant(m, "ST_DEV", 2)) return NULL;
562 if (PyModule_AddIntConstant(m, "ST_NLINK", 3)) return NULL;
563 if (PyModule_AddIntConstant(m, "ST_UID", 4)) return NULL;
564 if (PyModule_AddIntConstant(m, "ST_GID", 5)) return NULL;
565 if (PyModule_AddIntConstant(m, "ST_SIZE", 6)) return NULL;
566 if (PyModule_AddIntConstant(m, "ST_ATIME", 7)) return NULL;
567 if (PyModule_AddIntConstant(m, "ST_MTIME", 8)) return NULL;
568 if (PyModule_AddIntConstant(m, "ST_CTIME", 9)) return NULL;
569
570 return m;
571}
572
573#ifdef __cplusplus
574}
575#endif