getopt.c 1.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576
  1. /*
  2. * getopt.c
  3. */
  4. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  5. #include <linux/kernel.h>
  6. #include <linux/string.h>
  7. #include <asm/errno.h>
  8. #include "getopt.h"
  9. /**
  10. * ncp_getopt - option parser
  11. * @caller: name of the caller, for error messages
  12. * @options: the options string
  13. * @opts: an array of &struct option entries controlling parser operations
  14. * @optopt: output; will contain the current option
  15. * @optarg: output; will contain the value (if one exists)
  16. * @value: output; may be NULL; will be overwritten with the integer value
  17. * of the current argument.
  18. *
  19. * Helper to parse options on the format used by mount ("a=b,c=d,e,f").
  20. * Returns opts->val if a matching entry in the 'opts' array is found,
  21. * 0 when no more tokens are found, -1 if an error is encountered.
  22. */
  23. int ncp_getopt(const char *caller, char **options, const struct ncp_option *opts,
  24. char **optopt, char **optarg, unsigned long *value)
  25. {
  26. char *token;
  27. char *val;
  28. do {
  29. if ((token = strsep(options, ",")) == NULL)
  30. return 0;
  31. } while (*token == '\0');
  32. if (optopt)
  33. *optopt = token;
  34. if ((val = strchr (token, '=')) != NULL) {
  35. *val++ = 0;
  36. }
  37. *optarg = val;
  38. for (; opts->name; opts++) {
  39. if (!strcmp(opts->name, token)) {
  40. if (!val) {
  41. if (opts->has_arg & OPT_NOPARAM) {
  42. return opts->val;
  43. }
  44. pr_info("%s: the %s option requires an argument\n",
  45. caller, token);
  46. return -EINVAL;
  47. }
  48. if (opts->has_arg & OPT_INT) {
  49. int rc = kstrtoul(val, 0, value);
  50. if (rc) {
  51. pr_info("%s: invalid numeric value in %s=%s\n",
  52. caller, token, val);
  53. return rc;
  54. }
  55. return opts->val;
  56. }
  57. if (opts->has_arg & OPT_STRING) {
  58. return opts->val;
  59. }
  60. pr_info("%s: unexpected argument %s to the %s option\n",
  61. caller, val, token);
  62. return -EINVAL;
  63. }
  64. }
  65. pr_info("%s: Unrecognized mount option %s\n", caller, token);
  66. return -EOPNOTSUPP;
  67. }