You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

563 lines
14 KiB

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