logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/

mknod.c (3190B)


  1. // utils-std: Collection of commonly available Unix tools
  2. // SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me>
  3. // SPDX-License-Identifier: MPL-2.0
  4. #define _POSIX_C_SOURCE 200809L
  5. #define _XOPEN_SOURCE 700 // mknod is in XSI
  6. #include "../lib/getopt_nolong.h"
  7. #include "../lib/mode.h"
  8. #include <errno.h>
  9. #include <stdio.h> // fprintf
  10. #include <stdlib.h> // abort, exit
  11. #include <string.h> // strerror, strrchr
  12. #include <sys/stat.h> // mknod
  13. #include <sys/sysmacros.h> // makedev
  14. #include <unistd.h> // getopt
  15. mode_t filemask;
  16. const char *argv0 = "mknod";
  17. static void
  18. usage(void)
  19. {
  20. fprintf(stderr, "Usage: mknod [-m mode] file <b|c|u|p> [major minor]\n");
  21. }
  22. static unsigned int
  23. strtodev(char *arg)
  24. {
  25. char *endptr = NULL;
  26. long dev = strtol(arg, &endptr, 0);
  27. if(errno != 0)
  28. {
  29. fprintf(stderr, "mknod: error: Failed parsing '%s' into a number: %s\n", arg, strerror(errno));
  30. exit(1);
  31. }
  32. if(endptr != NULL && endptr[0] != 0)
  33. {
  34. fprintf(stderr,
  35. "mknod: error: Extraneous characters in '%s' while parsing it as a number: '%s'\n",
  36. arg,
  37. endptr);
  38. exit(1);
  39. }
  40. return (unsigned int)dev;
  41. }
  42. int
  43. main(int argc, char *argv[])
  44. {
  45. mode_t mode = 0666;
  46. const char *errstr = NULL;
  47. for(int c = -1; (c = getopt_nolong(argc, argv, ":m:")) != -1;)
  48. {
  49. switch(c)
  50. {
  51. case 'm':
  52. mode = new_mode(optarg, 0666, &errstr);
  53. if(errstr != NULL)
  54. {
  55. fprintf(stderr, "mknod: error: Failed parsing mode '%s': %s\n", optarg, errstr);
  56. return 1;
  57. }
  58. break;
  59. case ':':
  60. fprintf(stderr, "mknod: error: Missing operand for option: '-%c'\n", optopt);
  61. usage();
  62. return 1;
  63. case '?':
  64. if(!got_long_opt) fprintf(stderr, "mknod: error: Unrecognised option: '-%c'\n", optopt);
  65. usage();
  66. return 1;
  67. default:
  68. abort();
  69. }
  70. }
  71. argc -= optind;
  72. argv += optind;
  73. if(argc < 2)
  74. {
  75. fprintf(stderr, "mknod: error: Missing operands\n");
  76. usage();
  77. return 1;
  78. }
  79. char *file = argv[0];
  80. char *type = argv[1];
  81. if(type[0] == 0 || type[1] != 0)
  82. {
  83. fprintf(stderr, "mknod: error: Invalid type '%s'\n", type);
  84. usage();
  85. return 1;
  86. }
  87. if(type[0] == 'p')
  88. {
  89. if(mknod(file, mode | S_IFIFO, 0) != 0)
  90. {
  91. fprintf(stderr, "mknod: error: Failed creating FIFO at '%s': %s\n", file, strerror(errno));
  92. return 1;
  93. }
  94. }
  95. else
  96. {
  97. if(argc != 4)
  98. {
  99. fprintf(stderr,
  100. "mknod: error: non-FIFO device types requires major minor arguments to be given\n");
  101. return 1;
  102. }
  103. unsigned int maj = strtodev(argv[2]);
  104. unsigned int min = strtodev(argv[3]);
  105. const char *type_name;
  106. switch(type[0])
  107. {
  108. case 'b':
  109. mode |= S_IFBLK;
  110. type_name = "block-special";
  111. break;
  112. case 'c':
  113. mode |= S_IFCHR;
  114. type_name = "character-special";
  115. break;
  116. case 'u':
  117. mode |= S_IFCHR;
  118. type_name = "character-special";
  119. break;
  120. default:
  121. fprintf(stderr, "mknod: error: Invalid type '%c'\n", type[0]);
  122. usage();
  123. return 1;
  124. }
  125. if(mknod(file, mode, makedev(maj, min)) != 0)
  126. {
  127. fprintf(stderr,
  128. "mknod: error: Failed creating %s at '%s': %s\n",
  129. type_name,
  130. file,
  131. strerror(errno));
  132. return 1;
  133. }
  134. }
  135. return 0;
  136. }