aboutsummaryrefslogtreecommitdiffstats
path: root/library/cpp/getopt/small/posix_getopt.cpp
blob: bd06f3499f7f5c06c7c68035f1d65b8f85efb3c0 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
#include "posix_getopt.h"

#include <util/generic/ptr.h>

#include <ctype.h>

namespace NLastGetopt {
    char* optarg;
    int optind;
    int optopt;
    int opterr;
    int optreset;

    static THolder<TOpts> Opts;
    static THolder<TOptsParser> OptsParser;

    int getopt_long_impl(int argc, char* const* argv, const char* optstring,
                         const struct option* longopts, int* longindex, bool long_only) {
        if (!Opts || optreset == 1) {
            optarg = nullptr;
            optind = 1;
            opterr = 1;
            optreset = 0;
            Opts.Reset(new TOpts(TOpts::Default(optstring)));

            Opts->AllowSingleDashForLong_ = long_only;

            for (const struct option* o = longopts; o != nullptr && o->name != nullptr; ++o) {
                TOpt* opt;
                if ((unsigned)o->val < 0x80 && isalnum(o->val)) {
                    opt = &Opts->CharOption(char(o->val));
                    opt->AddLongName(o->name);
                } else {
                    Opts->AddLongOption(o->name);
                    opt = const_cast<TOpt*>(&Opts->GetLongOption(o->name));
                }
                opt->HasArg_ = EHasArg(o->has_arg);
                opt->UserValue(o->flag);
            }

            OptsParser.Reset(new TOptsParser(&*Opts, argc, (const char**)argv));
        }

        optarg = nullptr;

        try {
            if (!OptsParser->Next()) {
                return -1;
            } else {
                optarg = (char*)OptsParser->CurVal();
                optind = (int)OptsParser->Pos_;
                if (longindex && OptsParser->CurOpt())
                    *longindex = (int)Opts->IndexOf(OptsParser->CurOpt());
                return OptsParser->CurOpt() ? OptsParser->CurOpt()->GetCharOr0() : 1;
            }
        } catch (const NLastGetopt::TException&) {
            return '?';
        }
    }

    int getopt_long(int argc, char* const* argv, const char* optstring,
                    const struct option* longopts, int* longindex) {
        return getopt_long_impl(argc, argv, optstring, longopts, longindex, false);
    }

    int getopt_long_only(int argc, char* const* argv, const char* optstring,
                         const struct option* longopts, int* longindex) {
        return getopt_long_impl(argc, argv, optstring, longopts, longindex, true);
    }

    // XXX: leading colon is not supported
    // XXX: updating optind by client is not supported
    int getopt(int argc, char* const* argv, const char* optstring) {
        return getopt_long(argc, argv, optstring, nullptr, nullptr);
    }

}