logo

utils-std

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

seq.c (2897B)


  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. #include "../lib/getopt_nolong.h"
  6. #include <ctype.h> // isdigit
  7. #include <errno.h> // errno
  8. #include <math.h> // fabs
  9. #include <stdbool.h> // bool, true, false
  10. #include <stdio.h> // fprintf
  11. #include <stdlib.h> // exit, strtod
  12. #include <string.h> // strerror
  13. #include <unistd.h> // getopt, optarg, optind
  14. const char *argv0 = "seq";
  15. const char *sep = "\n";
  16. const char *term = "\n";
  17. int len = 0;
  18. bool opt_w = false;
  19. double g_last = 1.0;
  20. static void
  21. seq(double i, double step, double last)
  22. {
  23. printf("%0*g", len, i);
  24. if(i < last)
  25. {
  26. while((i + step) <= last)
  27. printf("%s%0*g", sep, len, i += step);
  28. }
  29. else if(i > last)
  30. {
  31. while((i - step) >= last)
  32. printf("%s%0*g", sep, len, i -= step);
  33. }
  34. printf("%s", term);
  35. if(opt_w && i != g_last)
  36. fprintf(stderr, "seq: error: Failed to guess last, got: %g; expected: %g\n", i, g_last);
  37. }
  38. static double
  39. get_num(char *str)
  40. {
  41. errno = 0;
  42. char *endptr = NULL;
  43. double num = strtod(str, &endptr);
  44. if(errno != 0)
  45. {
  46. fprintf(stderr, "seq: error: strtod(\"%s\"): %s\n", str, strerror(errno));
  47. exit(1);
  48. }
  49. if(!(endptr == NULL || endptr[0] == 0))
  50. {
  51. fprintf(stderr,
  52. "seq: error: strtod(\"%s\"), invalid remaining characters found '%s'\n",
  53. str,
  54. endptr);
  55. exit(1);
  56. }
  57. return num;
  58. }
  59. static void
  60. usage(void)
  61. {
  62. fprintf(stderr, "usage: seq [-w] [-s separator] [first [step]] last\n");
  63. }
  64. int
  65. main(int argc, char *argv[])
  66. {
  67. for(int c = -1; (c = getopt_nolong(argc, argv, ":ws:t:")) != -1;)
  68. {
  69. switch(c)
  70. {
  71. case 'w':
  72. opt_w = true;
  73. break;
  74. case 's':
  75. sep = optarg;
  76. break;
  77. case 't':
  78. term = optarg;
  79. break;
  80. case ':':
  81. fprintf(stderr, "seq: error: Option -%c requires an operand\n", optopt);
  82. usage();
  83. return 1;
  84. case '?':
  85. if(!got_long_opt)
  86. if(isdigit(optopt))
  87. {
  88. fprintf(stderr,
  89. "seq: warning: Pass -- if the first non-option argument starts with a dash(-)\n");
  90. optind--;
  91. goto getopt_end;
  92. }
  93. usage();
  94. return 1;
  95. }
  96. }
  97. getopt_end:
  98. argc -= optind;
  99. argv += optind;
  100. double first = 1.0;
  101. double step = 1.0;
  102. double last = 1.0;
  103. switch(argc)
  104. {
  105. case 1:
  106. last = get_num(argv[0]);
  107. break;
  108. case 2:
  109. first = get_num(argv[0]);
  110. last = get_num(argv[1]);
  111. break;
  112. case 3:
  113. first = get_num(argv[0]);
  114. step = fabs(get_num(argv[1]));
  115. last = get_num(argv[2]);
  116. break;
  117. default:
  118. usage();
  119. return 1;
  120. }
  121. if(opt_w)
  122. {
  123. double f_mod = fmod(first, step);
  124. double l_mod = fmod(last, step);
  125. g_last = last - l_mod + f_mod;
  126. int f_len = snprintf(NULL, 0, "%g", first);
  127. int l_len = snprintf(NULL, 0, "%g", g_last);
  128. len = f_len > l_len ? f_len : l_len;
  129. }
  130. seq(first, step, last);
  131. return 0;
  132. }