mntent.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <mntent.h>
  4. #include <errno.h>
  5. #include <limits.h>
  6. static char *internal_buf;
  7. static size_t internal_bufsize;
  8. #define SENTINEL (char *)&internal_buf
  9. FILE *setmntent(const char *name, const char *mode)
  10. {
  11. return fopen(name, mode);
  12. }
  13. int endmntent(FILE *f)
  14. {
  15. if (f) fclose(f);
  16. return 1;
  17. }
  18. static char *unescape_ent(char *beg)
  19. {
  20. char *dest = beg;
  21. const char *src = beg;
  22. while (*src) {
  23. const char *val;
  24. unsigned char cval = 0;
  25. if (*src != '\\') {
  26. *dest++ = *src++;
  27. continue;
  28. }
  29. if (src[1] == '\\') {
  30. ++src;
  31. *dest++ = *src++;
  32. continue;
  33. }
  34. val = src + 1;
  35. for (int i = 0; i < 3; ++i) {
  36. if (*val >= '0' && *val <= '7') {
  37. cval <<= 3;
  38. cval += *val++ - '0';
  39. } else {
  40. break;
  41. }
  42. }
  43. if (cval) {
  44. *dest++ = cval;
  45. src = val;
  46. } else {
  47. *dest++ = *src++;
  48. }
  49. }
  50. *dest = 0;
  51. return beg;
  52. }
  53. struct mntent *getmntent_r(FILE *f, struct mntent *mnt, char *linebuf, int buflen)
  54. {
  55. int n[8], use_internal = (linebuf == SENTINEL);
  56. size_t len, i;
  57. mnt->mnt_freq = 0;
  58. mnt->mnt_passno = 0;
  59. do {
  60. if (use_internal) {
  61. getline(&internal_buf, &internal_bufsize, f);
  62. linebuf = internal_buf;
  63. } else {
  64. fgets(linebuf, buflen, f);
  65. }
  66. if (feof(f) || ferror(f)) return 0;
  67. if (!strchr(linebuf, '\n')) {
  68. fscanf(f, "%*[^\n]%*[\n]");
  69. errno = ERANGE;
  70. return 0;
  71. }
  72. len = strlen(linebuf);
  73. if (len > INT_MAX) continue;
  74. for (i = 0; i < sizeof n / sizeof *n; i++) n[i] = len;
  75. sscanf(linebuf, " %n%*[^ \t\n]%n %n%*[^ \t\n]%n %n%*[^ \t\n]%n %n%*[^ \t\n]%n %d %d",
  76. n, n+1, n+2, n+3, n+4, n+5, n+6, n+7,
  77. &mnt->mnt_freq, &mnt->mnt_passno);
  78. } while (linebuf[n[0]] == '#' || n[1]==len);
  79. linebuf[n[1]] = 0;
  80. linebuf[n[3]] = 0;
  81. linebuf[n[5]] = 0;
  82. linebuf[n[7]] = 0;
  83. mnt->mnt_fsname = unescape_ent(linebuf+n[0]);
  84. mnt->mnt_dir = unescape_ent(linebuf+n[2]);
  85. mnt->mnt_type = unescape_ent(linebuf+n[4]);
  86. mnt->mnt_opts = unescape_ent(linebuf+n[6]);
  87. return mnt;
  88. }
  89. struct mntent *getmntent(FILE *f)
  90. {
  91. static struct mntent mnt;
  92. return getmntent_r(f, &mnt, SENTINEL, 0);
  93. }
  94. int addmntent(FILE *f, const struct mntent *mnt)
  95. {
  96. if (fseek(f, 0, SEEK_END)) return 1;
  97. return fprintf(f, "%s\t%s\t%s\t%s\t%d\t%d\n",
  98. mnt->mnt_fsname, mnt->mnt_dir, mnt->mnt_type, mnt->mnt_opts,
  99. mnt->mnt_freq, mnt->mnt_passno) < 0;
  100. }
  101. char *hasmntopt(const struct mntent *mnt, const char *opt)
  102. {
  103. size_t l = strlen(opt);
  104. char *p = mnt->mnt_opts;
  105. for (;;) {
  106. if (!strncmp(p, opt, l) && (!p[l] || p[l]==',' || p[l]=='='))
  107. return p;
  108. p = strchr(p, ',');
  109. if (!p) return 0;
  110. p++;
  111. }
  112. }