File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / curl / src / tool_getparam.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Wed Jun 3 10:01:16 2020 UTC (4 years, 10 months ago) by misho
Branches: curl, MAIN
CVS tags: v7_70_0p4, HEAD
curl

    1: /***************************************************************************
    2:  *                                  _   _ ____  _
    3:  *  Project                     ___| | | |  _ \| |
    4:  *                             / __| | | | |_) | |
    5:  *                            | (__| |_| |  _ <| |___
    6:  *                             \___|\___/|_| \_\_____|
    7:  *
    8:  * Copyright (C) 1998 - 2020, Daniel Stenberg, <daniel@haxx.se>, et al.
    9:  *
   10:  * This software is licensed as described in the file COPYING, which
   11:  * you should have received as part of this distribution. The terms
   12:  * are also available at https://curl.haxx.se/docs/copyright.html.
   13:  *
   14:  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
   15:  * copies of the Software, and permit persons to whom the Software is
   16:  * furnished to do so, under the terms of the COPYING file.
   17:  *
   18:  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
   19:  * KIND, either express or implied.
   20:  *
   21:  ***************************************************************************/
   22: #include "tool_setup.h"
   23: 
   24: #include "strcase.h"
   25: 
   26: #define ENABLE_CURLX_PRINTF
   27: /* use our own printf() functions */
   28: #include "curlx.h"
   29: 
   30: #include "tool_binmode.h"
   31: #include "tool_cfgable.h"
   32: #include "tool_cb_prg.h"
   33: #include "tool_convert.h"
   34: #include "tool_filetime.h"
   35: #include "tool_formparse.h"
   36: #include "tool_getparam.h"
   37: #include "tool_helpers.h"
   38: #include "tool_libinfo.h"
   39: #include "tool_metalink.h"
   40: #include "tool_msgs.h"
   41: #include "tool_paramhlp.h"
   42: #include "tool_parsecfg.h"
   43: #include "tool_main.h"
   44: 
   45: #include "memdebug.h" /* keep this as LAST include */
   46: 
   47: #ifdef MSDOS
   48: #  define USE_WATT32
   49: #endif
   50: 
   51: #define GetStr(str,val) do { \
   52:   if(*(str)) { \
   53:     free(*(str)); \
   54:     *(str) = NULL; \
   55:   } \
   56:   if((val)) {              \
   57:     *(str) = strdup((val)); \
   58:     if(!(*(str)))          \
   59:       return PARAM_NO_MEM; \
   60:   } \
   61: } while(0)
   62: 
   63: struct LongShort {
   64:   const char *letter; /* short name option */
   65:   const char *lname;  /* long name option */
   66:   enum {
   67:     ARG_NONE,   /* stand-alone but not a boolean */
   68:     ARG_BOOL,   /* accepts a --no-[name] prefix */
   69:     ARG_STRING, /* requires an argument */
   70:     ARG_FILENAME /* requires an argument, usually a file name */
   71:   } desc;
   72: };
   73: 
   74: static const struct LongShort aliases[]= {
   75:   /* 'letter' strings with more than one character have *no* short option to
   76:      mention. */
   77:   {"*@", "url",                      ARG_STRING},
   78:   {"*4", "dns-ipv4-addr",            ARG_STRING},
   79:   {"*6", "dns-ipv6-addr",            ARG_STRING},
   80:   {"*a", "random-file",              ARG_FILENAME},
   81:   {"*b", "egd-file",                 ARG_STRING},
   82:   {"*B", "oauth2-bearer",            ARG_STRING},
   83:   {"*c", "connect-timeout",          ARG_STRING},
   84:   {"*C", "doh-url"        ,          ARG_STRING},
   85:   {"*d", "ciphers",                  ARG_STRING},
   86:   {"*D", "dns-interface",            ARG_STRING},
   87:   {"*e", "disable-epsv",             ARG_BOOL},
   88:   {"*f", "disallow-username-in-url", ARG_BOOL},
   89:   {"*E", "epsv",                     ARG_BOOL},
   90:          /* 'epsv' made like this to make --no-epsv and --epsv to work
   91:              although --disable-epsv is the documented option */
   92:   {"*F", "dns-servers",              ARG_STRING},
   93:   {"*g", "trace",                    ARG_FILENAME},
   94:   {"*G", "npn",                      ARG_BOOL},
   95:   {"*h", "trace-ascii",              ARG_FILENAME},
   96:   {"*H", "alpn",                     ARG_BOOL},
   97:   {"*i", "limit-rate",               ARG_STRING},
   98:   {"*j", "compressed",               ARG_BOOL},
   99:   {"*J", "tr-encoding",              ARG_BOOL},
  100:   {"*k", "digest",                   ARG_BOOL},
  101:   {"*l", "negotiate",                ARG_BOOL},
  102:   {"*m", "ntlm",                     ARG_BOOL},
  103:   {"*M", "ntlm-wb",                  ARG_BOOL},
  104:   {"*n", "basic",                    ARG_BOOL},
  105:   {"*o", "anyauth",                  ARG_BOOL},
  106: #ifdef USE_WATT32
  107:   {"*p", "wdebug",                   ARG_BOOL},
  108: #endif
  109:   {"*q", "ftp-create-dirs",          ARG_BOOL},
  110:   {"*r", "create-dirs",              ARG_BOOL},
  111:   {"*s", "max-redirs",               ARG_STRING},
  112:   {"*t", "proxy-ntlm",               ARG_BOOL},
  113:   {"*u", "crlf",                     ARG_BOOL},
  114:   {"*v", "stderr",                   ARG_FILENAME},
  115:   {"*w", "interface",                ARG_STRING},
  116:   {"*x", "krb",                      ARG_STRING},
  117:   {"*x", "krb4",                     ARG_STRING},
  118:          /* 'krb4' is the previous name */
  119:   {"*X", "haproxy-protocol",         ARG_BOOL},
  120:   {"*y", "max-filesize",             ARG_STRING},
  121:   {"*z", "disable-eprt",             ARG_BOOL},
  122:   {"*Z", "eprt",                     ARG_BOOL},
  123:          /* 'eprt' made like this to make --no-eprt and --eprt to work
  124:              although --disable-eprt is the documented option */
  125:   {"*~", "xattr",                    ARG_BOOL},
  126:   {"$a", "ftp-ssl",                  ARG_BOOL},
  127:          /* 'ftp-ssl' deprecated name since 7.20.0 */
  128:   {"$a", "ssl",                      ARG_BOOL},
  129:          /* 'ssl' new option name in 7.20.0, previously this was ftp-ssl */
  130:   {"$b", "ftp-pasv",                 ARG_BOOL},
  131:   {"$c", "socks5",                   ARG_STRING},
  132:   {"$d", "tcp-nodelay",              ARG_BOOL},
  133:   {"$e", "proxy-digest",             ARG_BOOL},
  134:   {"$f", "proxy-basic",              ARG_BOOL},
  135:   {"$g", "retry",                    ARG_STRING},
  136:   {"$V", "retry-connrefused",        ARG_BOOL},
  137:   {"$h", "retry-delay",              ARG_STRING},
  138:   {"$i", "retry-max-time",           ARG_STRING},
  139:   {"$k", "proxy-negotiate",          ARG_BOOL},
  140:   {"$m", "ftp-account",              ARG_STRING},
  141:   {"$n", "proxy-anyauth",            ARG_BOOL},
  142:   {"$o", "trace-time",               ARG_BOOL},
  143:   {"$p", "ignore-content-length",    ARG_BOOL},
  144:   {"$q", "ftp-skip-pasv-ip",         ARG_BOOL},
  145:   {"$r", "ftp-method",               ARG_STRING},
  146:   {"$s", "local-port",               ARG_STRING},
  147:   {"$t", "socks4",                   ARG_STRING},
  148:   {"$T", "socks4a",                  ARG_STRING},
  149:   {"$u", "ftp-alternative-to-user",  ARG_STRING},
  150:   {"$v", "ftp-ssl-reqd",             ARG_BOOL},
  151:          /* 'ftp-ssl-reqd' deprecated name since 7.20.0 */
  152:   {"$v", "ssl-reqd",                 ARG_BOOL},
  153:          /* 'ssl-reqd' new in 7.20.0, previously this was ftp-ssl-reqd */
  154:   {"$w", "sessionid",                ARG_BOOL},
  155:          /* 'sessionid' listed as --no-sessionid in the help */
  156:   {"$x", "ftp-ssl-control",          ARG_BOOL},
  157:   {"$y", "ftp-ssl-ccc",              ARG_BOOL},
  158:   {"$j", "ftp-ssl-ccc-mode",         ARG_STRING},
  159:   {"$z", "libcurl",                  ARG_STRING},
  160:   {"$#", "raw",                      ARG_BOOL},
  161:   {"$0", "post301",                  ARG_BOOL},
  162:   {"$1", "keepalive",                ARG_BOOL},
  163:          /* 'keepalive' listed as --no-keepalive in the help */
  164:   {"$2", "socks5-hostname",          ARG_STRING},
  165:   {"$3", "keepalive-time",           ARG_STRING},
  166:   {"$4", "post302",                  ARG_BOOL},
  167:   {"$5", "noproxy",                  ARG_STRING},
  168:   {"$7", "socks5-gssapi-nec",        ARG_BOOL},
  169:   {"$8", "proxy1.0",                 ARG_STRING},
  170:   {"$9", "tftp-blksize",             ARG_STRING},
  171:   {"$A", "mail-from",                ARG_STRING},
  172:   {"$B", "mail-rcpt",                ARG_STRING},
  173:   {"$C", "ftp-pret",                 ARG_BOOL},
  174:   {"$D", "proto",                    ARG_STRING},
  175:   {"$E", "proto-redir",              ARG_STRING},
  176:   {"$F", "resolve",                  ARG_STRING},
  177:   {"$G", "delegation",               ARG_STRING},
  178:   {"$H", "mail-auth",                ARG_STRING},
  179:   {"$I", "post303",                  ARG_BOOL},
  180:   {"$J", "metalink",                 ARG_BOOL},
  181:   {"$6", "sasl-authzid",             ARG_STRING},
  182:   {"$K", "sasl-ir",                  ARG_BOOL },
  183:   {"$L", "test-event",               ARG_BOOL},
  184:   {"$M", "unix-socket",              ARG_FILENAME},
  185:   {"$N", "path-as-is",               ARG_BOOL},
  186:   {"$O", "socks5-gssapi-service",    ARG_STRING},
  187:          /* 'socks5-gssapi-service' merged with'proxy-service-name' and
  188:             deprecated since 7.49.0 */
  189:   {"$O", "proxy-service-name",       ARG_STRING},
  190:   {"$P", "service-name",             ARG_STRING},
  191:   {"$Q", "proto-default",            ARG_STRING},
  192:   {"$R", "expect100-timeout",        ARG_STRING},
  193:   {"$S", "tftp-no-options",          ARG_BOOL},
  194:   {"$U", "connect-to",               ARG_STRING},
  195:   {"$W", "abstract-unix-socket",     ARG_FILENAME},
  196:   {"$X", "tls-max",                  ARG_STRING},
  197:   {"$Y", "suppress-connect-headers", ARG_BOOL},
  198:   {"$Z", "compressed-ssh",           ARG_BOOL},
  199:   {"$~", "happy-eyeballs-timeout-ms", ARG_STRING},
  200:   {"0",   "http1.0",                 ARG_NONE},
  201:   {"01",  "http1.1",                 ARG_NONE},
  202:   {"02",  "http2",                   ARG_NONE},
  203:   {"03",  "http2-prior-knowledge",   ARG_NONE},
  204:   {"04",  "http3",                   ARG_NONE},
  205:   {"09",  "http0.9",                 ARG_BOOL},
  206:   {"1",  "tlsv1",                    ARG_NONE},
  207:   {"10",  "tlsv1.0",                 ARG_NONE},
  208:   {"11",  "tlsv1.1",                 ARG_NONE},
  209:   {"12",  "tlsv1.2",                 ARG_NONE},
  210:   {"13",  "tlsv1.3",                 ARG_NONE},
  211:   {"1A", "tls13-ciphers",            ARG_STRING},
  212:   {"1B", "proxy-tls13-ciphers",      ARG_STRING},
  213:   {"2",  "sslv2",                    ARG_NONE},
  214:   {"3",  "sslv3",                    ARG_NONE},
  215:   {"4",  "ipv4",                     ARG_NONE},
  216:   {"6",  "ipv6",                     ARG_NONE},
  217:   {"a",  "append",                   ARG_BOOL},
  218:   {"A",  "user-agent",               ARG_STRING},
  219:   {"b",  "cookie",                   ARG_STRING},
  220:   {"ba", "alt-svc",                  ARG_STRING},
  221:   {"B",  "use-ascii",                ARG_BOOL},
  222:   {"c",  "cookie-jar",               ARG_STRING},
  223:   {"C",  "continue-at",              ARG_STRING},
  224:   {"d",  "data",                     ARG_STRING},
  225:   {"dr", "data-raw",                 ARG_STRING},
  226:   {"da", "data-ascii",               ARG_STRING},
  227:   {"db", "data-binary",              ARG_STRING},
  228:   {"de", "data-urlencode",           ARG_STRING},
  229:   {"D",  "dump-header",              ARG_FILENAME},
  230:   {"e",  "referer",                  ARG_STRING},
  231:   {"E",  "cert",                     ARG_FILENAME},
  232:   {"Ea", "cacert",                   ARG_FILENAME},
  233:   {"Eb", "cert-type",                ARG_STRING},
  234:   {"Ec", "key",                      ARG_FILENAME},
  235:   {"Ed", "key-type",                 ARG_STRING},
  236:   {"Ee", "pass",                     ARG_STRING},
  237:   {"Ef", "engine",                   ARG_STRING},
  238:   {"Eg", "capath",                   ARG_FILENAME},
  239:   {"Eh", "pubkey",                   ARG_STRING},
  240:   {"Ei", "hostpubmd5",               ARG_STRING},
  241:   {"Ej", "crlfile",                  ARG_FILENAME},
  242:   {"Ek", "tlsuser",                  ARG_STRING},
  243:   {"El", "tlspassword",              ARG_STRING},
  244:   {"Em", "tlsauthtype",              ARG_STRING},
  245:   {"En", "ssl-allow-beast",          ARG_BOOL},
  246:   /* Eo */
  247:   {"Ep", "pinnedpubkey",             ARG_STRING},
  248:   {"EP", "proxy-pinnedpubkey",       ARG_STRING},
  249:   {"Eq", "cert-status",              ARG_BOOL},
  250:   {"Er", "false-start",              ARG_BOOL},
  251:   {"Es", "ssl-no-revoke",            ARG_BOOL},
  252:   {"ES", "ssl-revoke-best-effort",   ARG_BOOL},
  253:   {"Et", "tcp-fastopen",             ARG_BOOL},
  254:   {"Eu", "proxy-tlsuser",            ARG_STRING},
  255:   {"Ev", "proxy-tlspassword",        ARG_STRING},
  256:   {"Ew", "proxy-tlsauthtype",        ARG_STRING},
  257:   {"Ex", "proxy-cert",               ARG_FILENAME},
  258:   {"Ey", "proxy-cert-type",          ARG_STRING},
  259:   {"Ez", "proxy-key",                ARG_FILENAME},
  260:   {"E0", "proxy-key-type",           ARG_STRING},
  261:   {"E1", "proxy-pass",               ARG_STRING},
  262:   {"E2", "proxy-ciphers",            ARG_STRING},
  263:   {"E3", "proxy-crlfile",            ARG_FILENAME},
  264:   {"E4", "proxy-ssl-allow-beast",    ARG_BOOL},
  265:   {"E5", "login-options",            ARG_STRING},
  266:   {"E6", "proxy-cacert",             ARG_FILENAME},
  267:   {"E7", "proxy-capath",             ARG_FILENAME},
  268:   {"E8", "proxy-insecure",           ARG_BOOL},
  269:   {"E9", "proxy-tlsv1",              ARG_NONE},
  270:   {"EA", "socks5-basic",             ARG_BOOL},
  271:   {"EB", "socks5-gssapi",            ARG_BOOL},
  272:   {"EC", "etag-save",                ARG_FILENAME},
  273:   {"ED", "etag-compare",             ARG_FILENAME},
  274:   {"f",  "fail",                     ARG_BOOL},
  275:   {"fa", "fail-early",               ARG_BOOL},
  276:   {"fb", "styled-output",            ARG_BOOL},
  277:   {"fc", "mail-rcpt-allowfails",     ARG_BOOL},
  278:   {"F",  "form",                     ARG_STRING},
  279:   {"Fs", "form-string",              ARG_STRING},
  280:   {"g",  "globoff",                  ARG_BOOL},
  281:   {"G",  "get",                      ARG_NONE},
  282:   {"Ga", "request-target",           ARG_STRING},
  283:   {"h",  "help",                     ARG_BOOL},
  284:   {"H",  "header",                   ARG_STRING},
  285:   {"Hp", "proxy-header",             ARG_STRING},
  286:   {"i",  "include",                  ARG_BOOL},
  287:   {"I",  "head",                     ARG_BOOL},
  288:   {"j",  "junk-session-cookies",     ARG_BOOL},
  289:   {"J",  "remote-header-name",       ARG_BOOL},
  290:   {"k",  "insecure",                 ARG_BOOL},
  291:   {"K",  "config",                   ARG_FILENAME},
  292:   {"l",  "list-only",                ARG_BOOL},
  293:   {"L",  "location",                 ARG_BOOL},
  294:   {"Lt", "location-trusted",         ARG_BOOL},
  295:   {"m",  "max-time",                 ARG_STRING},
  296:   {"M",  "manual",                   ARG_BOOL},
  297:   {"n",  "netrc",                    ARG_BOOL},
  298:   {"no", "netrc-optional",           ARG_BOOL},
  299:   {"ne", "netrc-file",               ARG_FILENAME},
  300:   {"N",  "buffer",                   ARG_BOOL},
  301:          /* 'buffer' listed as --no-buffer in the help */
  302:   {"o",  "output",                   ARG_FILENAME},
  303:   {"O",  "remote-name",              ARG_NONE},
  304:   {"Oa", "remote-name-all",          ARG_BOOL},
  305:   {"p",  "proxytunnel",              ARG_BOOL},
  306:   {"P",  "ftp-port",                 ARG_STRING},
  307:   {"q",  "disable",                  ARG_BOOL},
  308:   {"Q",  "quote",                    ARG_STRING},
  309:   {"r",  "range",                    ARG_STRING},
  310:   {"R",  "remote-time",              ARG_BOOL},
  311:   {"s",  "silent",                   ARG_BOOL},
  312:   {"S",  "show-error",               ARG_BOOL},
  313:   {"t",  "telnet-option",            ARG_STRING},
  314:   {"T",  "upload-file",              ARG_FILENAME},
  315:   {"u",  "user",                     ARG_STRING},
  316:   {"U",  "proxy-user",               ARG_STRING},
  317:   {"v",  "verbose",                  ARG_BOOL},
  318:   {"V",  "version",                  ARG_BOOL},
  319:   {"w",  "write-out",                ARG_STRING},
  320:   {"x",  "proxy",                    ARG_STRING},
  321:   {"xa", "preproxy",                 ARG_STRING},
  322:   {"X",  "request",                  ARG_STRING},
  323:   {"Y",  "speed-limit",              ARG_STRING},
  324:   {"y",  "speed-time",               ARG_STRING},
  325:   {"z",  "time-cond",                ARG_STRING},
  326:   {"Z",  "parallel",                 ARG_BOOL},
  327:   {"Zb", "parallel-max",             ARG_STRING},
  328:   {"Zc", "parallel-immediate",       ARG_BOOL},
  329:   {"#",  "progress-bar",             ARG_BOOL},
  330:   {"#m", "progress-meter",           ARG_BOOL},
  331:   {":",  "next",                     ARG_NONE},
  332: };
  333: 
  334: /* Split the argument of -E to 'certname' and 'passphrase' separated by colon.
  335:  * We allow ':' and '\' to be escaped by '\' so that we can use certificate
  336:  * nicknames containing ':'.  See <https://sourceforge.net/p/curl/bugs/1196/>
  337:  * for details. */
  338: #ifndef UNITTESTS
  339: static
  340: #endif
  341: void parse_cert_parameter(const char *cert_parameter,
  342:                           char **certname,
  343:                           char **passphrase)
  344: {
  345:   size_t param_length = strlen(cert_parameter);
  346:   size_t span;
  347:   const char *param_place = NULL;
  348:   char *certname_place = NULL;
  349:   *certname = NULL;
  350:   *passphrase = NULL;
  351: 
  352:   /* most trivial assumption: cert_parameter is empty */
  353:   if(param_length == 0)
  354:     return;
  355: 
  356:   /* next less trivial: cert_parameter starts 'pkcs11:' and thus
  357:    * looks like a RFC7512 PKCS#11 URI which can be used as-is.
  358:    * Also if cert_parameter contains no colon nor backslash, this
  359:    * means no passphrase was given and no characters escaped */
  360:   if(curl_strnequal(cert_parameter, "pkcs11:", 7) ||
  361:      !strpbrk(cert_parameter, ":\\")) {
  362:     *certname = strdup(cert_parameter);
  363:     return;
  364:   }
  365:   /* deal with escaped chars; find unescaped colon if it exists */
  366:   certname_place = malloc(param_length + 1);
  367:   if(!certname_place)
  368:     return;
  369: 
  370:   *certname = certname_place;
  371:   param_place = cert_parameter;
  372:   while(*param_place) {
  373:     span = strcspn(param_place, ":\\");
  374:     strncpy(certname_place, param_place, span);
  375:     param_place += span;
  376:     certname_place += span;
  377:     /* we just ate all the non-special chars. now we're on either a special
  378:      * char or the end of the string. */
  379:     switch(*param_place) {
  380:     case '\0':
  381:       break;
  382:     case '\\':
  383:       param_place++;
  384:       switch(*param_place) {
  385:         case '\0':
  386:           *certname_place++ = '\\';
  387:           break;
  388:         case '\\':
  389:           *certname_place++ = '\\';
  390:           param_place++;
  391:           break;
  392:         case ':':
  393:           *certname_place++ = ':';
  394:           param_place++;
  395:           break;
  396:         default:
  397:           *certname_place++ = '\\';
  398:           *certname_place++ = *param_place;
  399:           param_place++;
  400:           break;
  401:       }
  402:       break;
  403:     case ':':
  404:       /* Since we live in a world of weirdness and confusion, the win32
  405:          dudes can use : when using drive letters and thus c:\file:password
  406:          needs to work. In order not to break compatibility, we still use : as
  407:          separator, but we try to detect when it is used for a file name! On
  408:          windows. */
  409: #ifdef WIN32
  410:       if(param_place &&
  411:           (param_place == &cert_parameter[1]) &&
  412:           (cert_parameter[2] == '\\' || cert_parameter[2] == '/') &&
  413:           (ISALPHA(cert_parameter[0])) ) {
  414:         /* colon in the second column, followed by a backslash, and the
  415:            first character is an alphabetic letter:
  416: 
  417:            this is a drive letter colon */
  418:         *certname_place++ = ':';
  419:         param_place++;
  420:         break;
  421:       }
  422: #endif
  423:       /* escaped colons and Windows drive letter colons were handled
  424:        * above; if we're still here, this is a separating colon */
  425:       param_place++;
  426:       if(*param_place) {
  427:         *passphrase = strdup(param_place);
  428:       }
  429:       goto done;
  430:     }
  431:   }
  432: done:
  433:   *certname_place = '\0';
  434: }
  435: 
  436: static void
  437: GetFileAndPassword(char *nextarg, char **file, char **password)
  438: {
  439:   char *certname, *passphrase;
  440:   parse_cert_parameter(nextarg, &certname, &passphrase);
  441:   Curl_safefree(*file);
  442:   *file = certname;
  443:   if(passphrase) {
  444:     Curl_safefree(*password);
  445:     *password = passphrase;
  446:   }
  447:   cleanarg(nextarg);
  448: }
  449: 
  450: /* Get a size parameter for '--limit-rate' or '--max-filesize'.
  451:  * We support a 'G', 'M' or 'K' suffix too.
  452:   */
  453: static ParameterError GetSizeParameter(struct GlobalConfig *global,
  454:                                        const char *arg,
  455:                                        const char *which,
  456:                                        curl_off_t *value_out)
  457: {
  458:   char *unit;
  459:   curl_off_t value;
  460: 
  461:   if(curlx_strtoofft(arg, &unit, 0, &value)) {
  462:     warnf(global, "invalid number specified for %s\n", which);
  463:     return PARAM_BAD_USE;
  464:   }
  465: 
  466:   if(!*unit)
  467:     unit = (char *)"b";
  468:   else if(strlen(unit) > 1)
  469:     unit = (char *)"w"; /* unsupported */
  470: 
  471:   switch(*unit) {
  472:   case 'G':
  473:   case 'g':
  474:     if(value > (CURL_OFF_T_MAX / (1024*1024*1024)))
  475:       return PARAM_NUMBER_TOO_LARGE;
  476:     value *= 1024*1024*1024;
  477:     break;
  478:   case 'M':
  479:   case 'm':
  480:     if(value > (CURL_OFF_T_MAX / (1024*1024)))
  481:       return PARAM_NUMBER_TOO_LARGE;
  482:     value *= 1024*1024;
  483:     break;
  484:   case 'K':
  485:   case 'k':
  486:     if(value > (CURL_OFF_T_MAX / 1024))
  487:       return PARAM_NUMBER_TOO_LARGE;
  488:     value *= 1024;
  489:     break;
  490:   case 'b':
  491:   case 'B':
  492:     /* for plain bytes, leave as-is */
  493:     break;
  494:   default:
  495:     warnf(global, "unsupported %s unit. Use G, M, K or B!\n", which);
  496:     return PARAM_BAD_USE;
  497:   }
  498:   *value_out = value;
  499:   return PARAM_OK;
  500: }
  501: 
  502: ParameterError getparameter(const char *flag, /* f or -long-flag */
  503:                             char *nextarg,    /* NULL if unset */
  504:                             bool *usedarg,    /* set to TRUE if the arg
  505:                                                  has been used */
  506:                             struct GlobalConfig *global,
  507:                             struct OperationConfig *config)
  508: {
  509:   char letter;
  510:   char subletter = '\0'; /* subletters can only occur on long options */
  511:   int rc;
  512:   const char *parse = NULL;
  513:   unsigned int j;
  514:   time_t now;
  515:   int hit = -1;
  516:   bool longopt = FALSE;
  517:   bool singleopt = FALSE; /* when true means '-o foo' used '-ofoo' */
  518:   ParameterError err;
  519:   bool toggle = TRUE; /* how to switch boolean options, on or off. Controlled
  520:                          by using --OPTION or --no-OPTION */
  521: 
  522:   *usedarg = FALSE; /* default is that we don't use the arg */
  523: 
  524:   if(('-' != flag[0]) || ('-' == flag[1])) {
  525:     /* this should be a long name */
  526:     const char *word = ('-' == flag[0]) ? flag + 2 : flag;
  527:     size_t fnam = strlen(word);
  528:     int numhits = 0;
  529:     bool noflagged = FALSE;
  530: 
  531:     if(!strncmp(word, "no-", 3)) {
  532:       /* disable this option but ignore the "no-" part when looking for it */
  533:       word += 3;
  534:       toggle = FALSE;
  535:       noflagged = TRUE;
  536:     }
  537: 
  538:     for(j = 0; j < sizeof(aliases)/sizeof(aliases[0]); j++) {
  539:       if(curl_strnequal(aliases[j].lname, word, fnam)) {
  540:         longopt = TRUE;
  541:         numhits++;
  542:         if(curl_strequal(aliases[j].lname, word)) {
  543:           parse = aliases[j].letter;
  544:           hit = j;
  545:           numhits = 1; /* a single unique hit */
  546:           break;
  547:         }
  548:         parse = aliases[j].letter;
  549:         hit = j;
  550:       }
  551:     }
  552:     if(numhits > 1) {
  553:       /* this is at least the second match! */
  554:       return PARAM_OPTION_AMBIGUOUS;
  555:     }
  556:     if(hit < 0) {
  557:       return PARAM_OPTION_UNKNOWN;
  558:     }
  559:     if(noflagged && (aliases[hit].desc != ARG_BOOL))
  560:       /* --no- prefixed an option that isn't boolean! */
  561:       return PARAM_NO_NOT_BOOLEAN;
  562:   }
  563:   else {
  564:     flag++; /* prefixed with one dash, pass it */
  565:     hit = -1;
  566:     parse = flag;
  567:   }
  568: 
  569:   do {
  570:     /* we can loop here if we have multiple single-letters */
  571: 
  572:     if(!longopt) {
  573:       letter = (char)*parse;
  574:       subletter = '\0';
  575:     }
  576:     else {
  577:       letter = parse[0];
  578:       subletter = parse[1];
  579:     }
  580: 
  581:     if(hit < 0) {
  582:       for(j = 0; j < sizeof(aliases)/sizeof(aliases[0]); j++) {
  583:         if(letter == aliases[j].letter[0]) {
  584:           hit = j;
  585:           break;
  586:         }
  587:       }
  588:       if(hit < 0) {
  589:         return PARAM_OPTION_UNKNOWN;
  590:       }
  591:     }
  592: 
  593:     if(aliases[hit].desc >= ARG_STRING) {
  594:       /* this option requires an extra parameter */
  595:       if(!longopt && parse[1]) {
  596:         nextarg = (char *)&parse[1]; /* this is the actual extra parameter */
  597:         singleopt = TRUE;   /* don't loop anymore after this */
  598:       }
  599:       else if(!nextarg)
  600:         return PARAM_REQUIRES_PARAMETER;
  601:       else
  602:         *usedarg = TRUE; /* mark it as used */
  603: 
  604:       if((aliases[hit].desc == ARG_FILENAME) &&
  605:          (nextarg[0] == '-') && nextarg[1]) {
  606:         /* if the file name looks like a command line option */
  607:         warnf(global, "The file name argument '%s' looks like a flag.\n",
  608:               nextarg);
  609:       }
  610:     }
  611:     else if((aliases[hit].desc == ARG_NONE) && !toggle)
  612:       return PARAM_NO_PREFIX;
  613: 
  614:     switch(letter) {
  615:     case '*': /* options without a short option */
  616:       switch(subletter) {
  617:       case '4': /* --dns-ipv4-addr */
  618:         /* addr in dot notation */
  619:         GetStr(&config->dns_ipv4_addr, nextarg);
  620:         break;
  621:       case '6': /* --dns-ipv6-addr */
  622:         /* addr in dot notation */
  623:         GetStr(&config->dns_ipv6_addr, nextarg);
  624:         break;
  625:       case 'a': /* random-file */
  626:         GetStr(&config->random_file, nextarg);
  627:         break;
  628:       case 'b': /* egd-file */
  629:         GetStr(&config->egd_file, nextarg);
  630:         break;
  631:       case 'B': /* OAuth 2.0 bearer token */
  632:         GetStr(&config->oauth_bearer, nextarg);
  633:         config->authtype |= CURLAUTH_BEARER;
  634:         break;
  635:       case 'c': /* connect-timeout */
  636:         err = str2udouble(&config->connecttimeout, nextarg,
  637:                           LONG_MAX/1000);
  638:         if(err)
  639:           return err;
  640:         break;
  641:       case 'C': /* doh-url */
  642:         GetStr(&config->doh_url, nextarg);
  643:         break;
  644:       case 'd': /* ciphers */
  645:         GetStr(&config->cipher_list, nextarg);
  646:         break;
  647:       case 'D': /* --dns-interface */
  648:         /* interface name */
  649:         GetStr(&config->dns_interface, nextarg);
  650:         break;
  651:       case 'e': /* --disable-epsv */
  652:         config->disable_epsv = toggle;
  653:         break;
  654:       case 'f': /* --disallow-username-in-url */
  655:         config->disallow_username_in_url = toggle;
  656:         break;
  657:       case 'E': /* --epsv */
  658:         config->disable_epsv = (!toggle)?TRUE:FALSE;
  659:         break;
  660:       case 'F': /* --dns-servers */
  661:         /* IP addrs of DNS servers */
  662:         GetStr(&config->dns_servers, nextarg);
  663:         break;
  664:       case 'g': /* --trace */
  665:         GetStr(&global->trace_dump, nextarg);
  666:         if(global->tracetype && (global->tracetype != TRACE_BIN))
  667:           warnf(global, "--trace overrides an earlier trace/verbose option\n");
  668:         global->tracetype = TRACE_BIN;
  669:         break;
  670:       case 'G': /* --npn */
  671:         config->nonpn = (!toggle)?TRUE:FALSE;
  672:         break;
  673:       case 'h': /* --trace-ascii */
  674:         GetStr(&global->trace_dump, nextarg);
  675:         if(global->tracetype && (global->tracetype != TRACE_ASCII))
  676:           warnf(global,
  677:                 "--trace-ascii overrides an earlier trace/verbose option\n");
  678:         global->tracetype = TRACE_ASCII;
  679:         break;
  680:       case 'H': /* --alpn */
  681:         config->noalpn = (!toggle)?TRUE:FALSE;
  682:         break;
  683:       case 'i': /* --limit-rate */
  684:       {
  685:         curl_off_t value;
  686:         ParameterError pe = GetSizeParameter(global, nextarg, "rate", &value);
  687: 
  688:         if(pe != PARAM_OK)
  689:            return pe;
  690:         config->recvpersecond = value;
  691:         config->sendpersecond = value;
  692:       }
  693:       break;
  694: 
  695:       case 'j': /* --compressed */
  696:         if(toggle &&
  697:            !(curlinfo->features & (CURL_VERSION_LIBZ | CURL_VERSION_BROTLI)))
  698:           return PARAM_LIBCURL_DOESNT_SUPPORT;
  699:         config->encoding = toggle;
  700:         break;
  701: 
  702:       case 'J': /* --tr-encoding */
  703:         config->tr_encoding = toggle;
  704:         break;
  705: 
  706:       case 'k': /* --digest */
  707:         if(toggle)
  708:           config->authtype |= CURLAUTH_DIGEST;
  709:         else
  710:           config->authtype &= ~CURLAUTH_DIGEST;
  711:         break;
  712: 
  713:       case 'l': /* --negotiate */
  714:         if(toggle) {
  715:           if(curlinfo->features & CURL_VERSION_SPNEGO)
  716:             config->authtype |= CURLAUTH_NEGOTIATE;
  717:           else
  718:             return PARAM_LIBCURL_DOESNT_SUPPORT;
  719:         }
  720:         else
  721:           config->authtype &= ~CURLAUTH_NEGOTIATE;
  722:         break;
  723: 
  724:       case 'm': /* --ntlm */
  725:         if(toggle) {
  726:           if(curlinfo->features & CURL_VERSION_NTLM)
  727:             config->authtype |= CURLAUTH_NTLM;
  728:           else
  729:             return PARAM_LIBCURL_DOESNT_SUPPORT;
  730:         }
  731:         else
  732:           config->authtype &= ~CURLAUTH_NTLM;
  733:         break;
  734: 
  735:       case 'M': /* --ntlm-wb */
  736:         if(toggle) {
  737:           if(curlinfo->features & CURL_VERSION_NTLM_WB)
  738:             config->authtype |= CURLAUTH_NTLM_WB;
  739:           else
  740:             return PARAM_LIBCURL_DOESNT_SUPPORT;
  741:         }
  742:         else
  743:           config->authtype &= ~CURLAUTH_NTLM_WB;
  744:         break;
  745: 
  746:       case 'n': /* --basic for completeness */
  747:         if(toggle)
  748:           config->authtype |= CURLAUTH_BASIC;
  749:         else
  750:           config->authtype &= ~CURLAUTH_BASIC;
  751:         break;
  752: 
  753:       case 'o': /* --anyauth, let libcurl pick it */
  754:         if(toggle)
  755:           config->authtype = CURLAUTH_ANY;
  756:         /* --no-anyauth simply doesn't touch it */
  757:         break;
  758: 
  759: #ifdef USE_WATT32
  760:       case 'p': /* --wdebug */
  761:         dbug_init();
  762:         break;
  763: #endif
  764:       case 'q': /* --ftp-create-dirs */
  765:         config->ftp_create_dirs = toggle;
  766:         break;
  767: 
  768:       case 'r': /* --create-dirs */
  769:         config->create_dirs = toggle;
  770:         break;
  771: 
  772:       case 's': /* --max-redirs */
  773:         /* specified max no of redirects (http(s)), this accepts -1 as a
  774:            special condition */
  775:         err = str2num(&config->maxredirs, nextarg);
  776:         if(err)
  777:           return err;
  778:         if(config->maxredirs < -1)
  779:           return PARAM_BAD_NUMERIC;
  780:         break;
  781: 
  782:       case 't': /* --proxy-ntlm */
  783:         if(curlinfo->features & CURL_VERSION_NTLM)
  784:           config->proxyntlm = toggle;
  785:         else
  786:           return PARAM_LIBCURL_DOESNT_SUPPORT;
  787:         break;
  788: 
  789:       case 'u': /* --crlf */
  790:         /* LF -> CRLF conversion? */
  791:         config->crlf = toggle;
  792:         break;
  793: 
  794:       case 'v': /* --stderr */
  795:         if(strcmp(nextarg, "-")) {
  796:           FILE *newfile = fopen(nextarg, FOPEN_WRITETEXT);
  797:           if(!newfile)
  798:             warnf(global, "Failed to open %s!\n", nextarg);
  799:           else {
  800:             if(global->errors_fopened)
  801:               fclose(global->errors);
  802:             global->errors = newfile;
  803:             global->errors_fopened = TRUE;
  804:           }
  805:         }
  806:         else
  807:           global->errors = stdout;
  808:         break;
  809:       case 'w': /* --interface */
  810:         /* interface */
  811:         GetStr(&config->iface, nextarg);
  812:         break;
  813:       case 'x': /* --krb */
  814:         /* kerberos level string */
  815:         if(curlinfo->features & CURL_VERSION_KERBEROS4)
  816:           GetStr(&config->krblevel, nextarg);
  817:         else
  818:           return PARAM_LIBCURL_DOESNT_SUPPORT;
  819:         break;
  820:       case 'X': /* --haproxy-protocol */
  821:         config->haproxy_protocol = toggle;
  822:         break;
  823:       case 'y': /* --max-filesize */
  824:         {
  825:           curl_off_t value;
  826:           ParameterError pe =
  827:             GetSizeParameter(global, nextarg, "max-filesize", &value);
  828: 
  829:           if(pe != PARAM_OK)
  830:              return pe;
  831:           config->max_filesize = value;
  832:         }
  833:         break;
  834:       case 'z': /* --disable-eprt */
  835:         config->disable_eprt = toggle;
  836:         break;
  837:       case 'Z': /* --eprt */
  838:         config->disable_eprt = (!toggle)?TRUE:FALSE;
  839:         break;
  840:       case '~': /* --xattr */
  841:         config->xattr = toggle;
  842:         break;
  843:       case '@': /* the URL! */
  844:       {
  845:         struct getout *url;
  846: 
  847:         if(!config->url_get)
  848:           config->url_get = config->url_list;
  849: 
  850:         if(config->url_get) {
  851:           /* there's a node here, if it already is filled-in continue to find
  852:              an "empty" node */
  853:           while(config->url_get && (config->url_get->flags & GETOUT_URL))
  854:             config->url_get = config->url_get->next;
  855:         }
  856: 
  857:         /* now there might or might not be an available node to fill in! */
  858: 
  859:         if(config->url_get)
  860:           /* existing node */
  861:           url = config->url_get;
  862:         else
  863:           /* there was no free node, create one! */
  864:           config->url_get = url = new_getout(config);
  865: 
  866:         if(!url)
  867:           return PARAM_NO_MEM;
  868: 
  869:         /* fill in the URL */
  870:         GetStr(&url->url, nextarg);
  871:         url->flags |= GETOUT_URL;
  872:       }
  873:       }
  874:       break;
  875:     case '$': /* more options without a short option */
  876:       switch(subletter) {
  877:       case 'a': /* --ssl */
  878:         if(toggle && !(curlinfo->features & CURL_VERSION_SSL))
  879:           return PARAM_LIBCURL_DOESNT_SUPPORT;
  880:         config->ftp_ssl = toggle;
  881:         break;
  882:       case 'b': /* --ftp-pasv */
  883:         Curl_safefree(config->ftpport);
  884:         break;
  885:       case 'c': /* --socks5 specifies a socks5 proxy to use, and resolves
  886:                    the name locally and passes on the resolved address */
  887:         GetStr(&config->proxy, nextarg);
  888:         config->proxyver = CURLPROXY_SOCKS5;
  889:         break;
  890:       case 't': /* --socks4 specifies a socks4 proxy to use */
  891:         GetStr(&config->proxy, nextarg);
  892:         config->proxyver = CURLPROXY_SOCKS4;
  893:         break;
  894:       case 'T': /* --socks4a specifies a socks4a proxy to use */
  895:         GetStr(&config->proxy, nextarg);
  896:         config->proxyver = CURLPROXY_SOCKS4A;
  897:         break;
  898:       case '2': /* --socks5-hostname specifies a socks5 proxy and enables name
  899:                    resolving with the proxy */
  900:         GetStr(&config->proxy, nextarg);
  901:         config->proxyver = CURLPROXY_SOCKS5_HOSTNAME;
  902:         break;
  903:       case 'd': /* --tcp-nodelay option */
  904:         config->tcp_nodelay = toggle;
  905:         break;
  906:       case 'e': /* --proxy-digest */
  907:         config->proxydigest = toggle;
  908:         break;
  909:       case 'f': /* --proxy-basic */
  910:         config->proxybasic = toggle;
  911:         break;
  912:       case 'g': /* --retry */
  913:         err = str2unum(&config->req_retry, nextarg);
  914:         if(err)
  915:           return err;
  916:         break;
  917:       case 'V': /* --retry-connrefused */
  918:         config->retry_connrefused = toggle;
  919:         break;
  920:       case 'h': /* --retry-delay */
  921:         err = str2unummax(&config->retry_delay, nextarg, LONG_MAX/1000);
  922:         if(err)
  923:           return err;
  924:         break;
  925:       case 'i': /* --retry-max-time */
  926:         err = str2unummax(&config->retry_maxtime, nextarg, LONG_MAX/1000);
  927:         if(err)
  928:           return err;
  929:         break;
  930: 
  931:       case 'k': /* --proxy-negotiate */
  932:         if(curlinfo->features & CURL_VERSION_SPNEGO)
  933:           config->proxynegotiate = toggle;
  934:         else
  935:           return PARAM_LIBCURL_DOESNT_SUPPORT;
  936:         break;
  937: 
  938:       case 'm': /* --ftp-account */
  939:         GetStr(&config->ftp_account, nextarg);
  940:         break;
  941:       case 'n': /* --proxy-anyauth */
  942:         config->proxyanyauth = toggle;
  943:         break;
  944:       case 'o': /* --trace-time */
  945:         global->tracetime = toggle;
  946:         break;
  947:       case 'p': /* --ignore-content-length */
  948:         config->ignorecl = toggle;
  949:         break;
  950:       case 'q': /* --ftp-skip-pasv-ip */
  951:         config->ftp_skip_ip = toggle;
  952:         break;
  953:       case 'r': /* --ftp-method (undocumented at this point) */
  954:         config->ftp_filemethod = ftpfilemethod(config, nextarg);
  955:         break;
  956:       case 's': { /* --local-port */
  957:         char lrange[7];  /* 16bit base 10 is 5 digits, but we allow 6 so that
  958:                             this catches overflows, not just truncates */
  959:         char *p = nextarg;
  960:         while(ISDIGIT(*p))
  961:           p++;
  962:         if(*p) {
  963:           /* if there's anything more than a plain decimal number */
  964:           rc = sscanf(p, " - %6s", lrange);
  965:           *p = 0; /* zero terminate to make str2unum() work below */
  966:         }
  967:         else
  968:           rc = 0;
  969: 
  970:         err = str2unum(&config->localport, nextarg);
  971:         if(err || (config->localport > 65535))
  972:           return PARAM_BAD_USE;
  973:         if(!rc)
  974:           config->localportrange = 1; /* default number of ports to try */
  975:         else {
  976:           err = str2unum(&config->localportrange, lrange);
  977:           if(err || (config->localportrange > 65535))
  978:             return PARAM_BAD_USE;
  979:           config->localportrange -= (config->localport-1);
  980:           if(config->localportrange < 1)
  981:             return PARAM_BAD_USE;
  982:         }
  983:         break;
  984:       }
  985:       case 'u': /* --ftp-alternative-to-user */
  986:         GetStr(&config->ftp_alternative_to_user, nextarg);
  987:         break;
  988:       case 'v': /* --ssl-reqd */
  989:         if(toggle && !(curlinfo->features & CURL_VERSION_SSL))
  990:           return PARAM_LIBCURL_DOESNT_SUPPORT;
  991:         config->ftp_ssl_reqd = toggle;
  992:         break;
  993:       case 'w': /* --no-sessionid */
  994:         config->disable_sessionid = (!toggle)?TRUE:FALSE;
  995:         break;
  996:       case 'x': /* --ftp-ssl-control */
  997:         if(toggle && !(curlinfo->features & CURL_VERSION_SSL))
  998:           return PARAM_LIBCURL_DOESNT_SUPPORT;
  999:         config->ftp_ssl_control = toggle;
 1000:         break;
 1001:       case 'y': /* --ftp-ssl-ccc */
 1002:         config->ftp_ssl_ccc = toggle;
 1003:         if(!config->ftp_ssl_ccc_mode)
 1004:           config->ftp_ssl_ccc_mode = CURLFTPSSL_CCC_PASSIVE;
 1005:         break;
 1006:       case 'j': /* --ftp-ssl-ccc-mode */
 1007:         config->ftp_ssl_ccc = TRUE;
 1008:         config->ftp_ssl_ccc_mode = ftpcccmethod(config, nextarg);
 1009:         break;
 1010:       case 'z': /* --libcurl */
 1011: #ifdef CURL_DISABLE_LIBCURL_OPTION
 1012:         warnf(global,
 1013:               "--libcurl option was disabled at build-time!\n");
 1014:         return PARAM_OPTION_UNKNOWN;
 1015: #else
 1016:         GetStr(&global->libcurl, nextarg);
 1017:         break;
 1018: #endif
 1019:       case '#': /* --raw */
 1020:         config->raw = toggle;
 1021:         break;
 1022:       case '0': /* --post301 */
 1023:         config->post301 = toggle;
 1024:         break;
 1025:       case '1': /* --no-keepalive */
 1026:         config->nokeepalive = (!toggle)?TRUE:FALSE;
 1027:         break;
 1028:       case '3': /* --keepalive-time */
 1029:         err = str2unum(&config->alivetime, nextarg);
 1030:         if(err)
 1031:           return err;
 1032:         break;
 1033:       case '4': /* --post302 */
 1034:         config->post302 = toggle;
 1035:         break;
 1036:       case 'I': /* --post303 */
 1037:         config->post303 = toggle;
 1038:         break;
 1039:       case '5': /* --noproxy */
 1040:         /* This specifies the noproxy list */
 1041:         GetStr(&config->noproxy, nextarg);
 1042:         break;
 1043:        case '7': /* --socks5-gssapi-nec*/
 1044:         config->socks5_gssapi_nec = toggle;
 1045:         break;
 1046:       case '8': /* --proxy1.0 */
 1047:         /* http 1.0 proxy */
 1048:         GetStr(&config->proxy, nextarg);
 1049:         config->proxyver = CURLPROXY_HTTP_1_0;
 1050:         break;
 1051:       case '9': /* --tftp-blksize */
 1052:         err = str2unum(&config->tftp_blksize, nextarg);
 1053:         if(err)
 1054:           return err;
 1055:         break;
 1056:       case 'A': /* --mail-from */
 1057:         GetStr(&config->mail_from, nextarg);
 1058:         break;
 1059:       case 'B': /* --mail-rcpt */
 1060:         /* append receiver to a list */
 1061:         err = add2list(&config->mail_rcpt, nextarg);
 1062:         if(err)
 1063:           return err;
 1064:         break;
 1065:       case 'C': /* --ftp-pret */
 1066:         config->ftp_pret = toggle;
 1067:         break;
 1068:       case 'D': /* --proto */
 1069:         config->proto_present = TRUE;
 1070:         if(proto2num(config, &config->proto, nextarg))
 1071:           return PARAM_BAD_USE;
 1072:         break;
 1073:       case 'E': /* --proto-redir */
 1074:         config->proto_redir_present = TRUE;
 1075:         if(proto2num(config, &config->proto_redir, nextarg))
 1076:           return PARAM_BAD_USE;
 1077:         break;
 1078:       case 'F': /* --resolve */
 1079:         err = add2list(&config->resolve, nextarg);
 1080:         if(err)
 1081:           return err;
 1082:         break;
 1083:       case 'G': /* --delegation LEVEL */
 1084:         config->gssapi_delegation = delegation(config, nextarg);
 1085:         break;
 1086:       case 'H': /* --mail-auth */
 1087:         GetStr(&config->mail_auth, nextarg);
 1088:         break;
 1089:       case 'J': /* --metalink */
 1090:         {
 1091: #ifdef USE_METALINK
 1092:           int mlmaj, mlmin, mlpatch;
 1093:           metalink_get_version(&mlmaj, &mlmin, &mlpatch);
 1094:           if((mlmaj*10000)+(mlmin*100) + mlpatch < CURL_REQ_LIBMETALINK_VERS) {
 1095:             warnf(global,
 1096:                   "--metalink option cannot be used because the version of "
 1097:                   "the linked libmetalink library is too old. "
 1098:                   "Required: %d.%d.%d, found %d.%d.%d\n",
 1099:                   CURL_REQ_LIBMETALINK_MAJOR,
 1100:                   CURL_REQ_LIBMETALINK_MINOR,
 1101:                   CURL_REQ_LIBMETALINK_PATCH,
 1102:                   mlmaj, mlmin, mlpatch);
 1103:             return PARAM_BAD_USE;
 1104:           }
 1105:           else
 1106:             config->use_metalink = toggle;
 1107: #else
 1108:           warnf(global, "--metalink option is ignored because the binary is "
 1109:                 "built without the Metalink support.\n");
 1110: #endif
 1111:           break;
 1112:         }
 1113:       case '6': /* --sasl-authzid */
 1114:         GetStr(&config->sasl_authzid, nextarg);
 1115:         break;
 1116:       case 'K': /* --sasl-ir */
 1117:         config->sasl_ir = toggle;
 1118:         break;
 1119:       case 'L': /* --test-event */
 1120: #ifdef CURLDEBUG
 1121:         global->test_event_based = toggle;
 1122: #else
 1123:         warnf(global, "--test-event is ignored unless a debug build!\n");
 1124: #endif
 1125:         break;
 1126:       case 'M': /* --unix-socket */
 1127:         config->abstract_unix_socket = FALSE;
 1128:         GetStr(&config->unix_socket_path, nextarg);
 1129:         break;
 1130:       case 'N': /* --path-as-is */
 1131:         config->path_as_is = toggle;
 1132:         break;
 1133:       case 'O': /* --proxy-service-name */
 1134:         GetStr(&config->proxy_service_name, nextarg);
 1135:         break;
 1136:       case 'P': /* --service-name */
 1137:         GetStr(&config->service_name, nextarg);
 1138:         break;
 1139:       case 'Q': /* --proto-default */
 1140:         GetStr(&config->proto_default, nextarg);
 1141:         err = check_protocol(config->proto_default);
 1142:         if(err)
 1143:           return err;
 1144:         break;
 1145:       case 'R': /* --expect100-timeout */
 1146:         err = str2udouble(&config->expect100timeout, nextarg, LONG_MAX/1000);
 1147:         if(err)
 1148:           return err;
 1149:         break;
 1150:       case 'S': /* --tftp-no-options */
 1151:         config->tftp_no_options = toggle;
 1152:         break;
 1153:       case 'U': /* --connect-to */
 1154:         err = add2list(&config->connect_to, nextarg);
 1155:         if(err)
 1156:           return err;
 1157:         break;
 1158:       case 'W': /* --abstract-unix-socket */
 1159:         config->abstract_unix_socket = TRUE;
 1160:         GetStr(&config->unix_socket_path, nextarg);
 1161:         break;
 1162:       case 'X': /* --tls-max */
 1163:         err = str2tls_max(&config->ssl_version_max, nextarg);
 1164:         if(err)
 1165:           return err;
 1166:         break;
 1167:       case 'Y': /* --suppress-connect-headers */
 1168:         config->suppress_connect_headers = toggle;
 1169:         break;
 1170:       case 'Z': /* --compressed-ssh */
 1171:         config->ssh_compression = toggle;
 1172:         break;
 1173:       case '~': /* --happy-eyeballs-timeout-ms */
 1174:         err = str2unum(&config->happy_eyeballs_timeout_ms, nextarg);
 1175:         if(err)
 1176:           return err;
 1177:         /* 0 is a valid value for this timeout */
 1178:         break;
 1179:       }
 1180:       break;
 1181:     case '#':
 1182:       switch(subletter) {
 1183:       case 'm': /* --progress-meter */
 1184:         global->noprogress = !toggle;
 1185:         break;
 1186:       default:  /* --progress-bar */
 1187:         global->progressmode =
 1188:           toggle ? CURL_PROGRESS_BAR : CURL_PROGRESS_STATS;
 1189:         break;
 1190:       }
 1191:       break;
 1192:     case ':': /* --next */
 1193:       return PARAM_NEXT_OPERATION;
 1194:     case '0': /* --http* options */
 1195:       switch(subletter) {
 1196:       case '\0':
 1197:         /* HTTP version 1.0 */
 1198:         config->httpversion = CURL_HTTP_VERSION_1_0;
 1199:         break;
 1200:       case '1':
 1201:         /* HTTP version 1.1 */
 1202:         config->httpversion = CURL_HTTP_VERSION_1_1;
 1203:         break;
 1204:       case '2':
 1205:         /* HTTP version 2.0 */
 1206:         config->httpversion = CURL_HTTP_VERSION_2_0;
 1207:         break;
 1208:       case '3': /* --http2-prior-knowledge */
 1209:         /* HTTP version 2.0 over clean TCP*/
 1210:         config->httpversion = CURL_HTTP_VERSION_2_PRIOR_KNOWLEDGE;
 1211:         break;
 1212:       case '4': /* --http3 */
 1213:         /* HTTP version 3 go over QUIC - at once */
 1214:         if(curlinfo->features & CURL_VERSION_HTTP3)
 1215:           config->httpversion = CURL_HTTP_VERSION_3;
 1216:         else
 1217:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1218:         break;
 1219:       case '9':
 1220:         /* Allow HTTP/0.9 responses! */
 1221:         config->http09_allowed = toggle;
 1222:         break;
 1223:       }
 1224:       break;
 1225:     case '1': /* --tlsv1* options */
 1226:       switch(subletter) {
 1227:       case '\0':
 1228:         /* TLS version 1.x */
 1229:         config->ssl_version = CURL_SSLVERSION_TLSv1;
 1230:         break;
 1231:       case '0':
 1232:         /* TLS version 1.0 */
 1233:         config->ssl_version = CURL_SSLVERSION_TLSv1_0;
 1234:         break;
 1235:       case '1':
 1236:         /* TLS version 1.1 */
 1237:         config->ssl_version = CURL_SSLVERSION_TLSv1_1;
 1238:         break;
 1239:       case '2':
 1240:         /* TLS version 1.2 */
 1241:         config->ssl_version = CURL_SSLVERSION_TLSv1_2;
 1242:         break;
 1243:       case '3':
 1244:         /* TLS version 1.3 */
 1245:         config->ssl_version = CURL_SSLVERSION_TLSv1_3;
 1246:         break;
 1247:       case 'A': /* --tls13-ciphers */
 1248:         GetStr(&config->cipher13_list, nextarg);
 1249:         break;
 1250:       case 'B': /* --proxy-tls13-ciphers */
 1251:         GetStr(&config->proxy_cipher13_list, nextarg);
 1252:         break;
 1253:       }
 1254:       break;
 1255:     case '2':
 1256:       /* SSL version 2 */
 1257:       config->ssl_version = CURL_SSLVERSION_SSLv2;
 1258:       break;
 1259:     case '3':
 1260:       /* SSL version 3 */
 1261:       config->ssl_version = CURL_SSLVERSION_SSLv3;
 1262:       break;
 1263:     case '4':
 1264:       /* IPv4 */
 1265:       config->ip_version = 4;
 1266:       break;
 1267:     case '6':
 1268:       /* IPv6 */
 1269:       config->ip_version = 6;
 1270:       break;
 1271:     case 'a':
 1272:       /* This makes the FTP sessions use APPE instead of STOR */
 1273:       config->ftp_append = toggle;
 1274:       break;
 1275:     case 'A':
 1276:       /* This specifies the User-Agent name */
 1277:       GetStr(&config->useragent, nextarg);
 1278:       break;
 1279:     case 'b':
 1280:       switch(subletter) {
 1281:       case 'a': /* --alt-svc */
 1282:         if(curlinfo->features & CURL_VERSION_ALTSVC)
 1283:           GetStr(&config->altsvc, nextarg);
 1284:         else
 1285:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1286:         break;
 1287:       default:  /* --cookie string coming up: */
 1288:         if(nextarg[0] == '@') {
 1289:           nextarg++;
 1290:         }
 1291:         else if(strchr(nextarg, '=')) {
 1292:           /* A cookie string must have a =-letter */
 1293:           GetStr(&config->cookie, nextarg);
 1294:           break;
 1295:         }
 1296:         /* We have a cookie file to read from! */
 1297:         GetStr(&config->cookiefile, nextarg);
 1298:       }
 1299:       break;
 1300:     case 'B':
 1301:       /* use ASCII/text when transferring */
 1302:       config->use_ascii = toggle;
 1303:       break;
 1304:     case 'c':
 1305:       /* get the file name to dump all cookies in */
 1306:       GetStr(&config->cookiejar, nextarg);
 1307:       break;
 1308:     case 'C':
 1309:       /* This makes us continue an ftp transfer at given position */
 1310:       if(strcmp(nextarg, "-")) {
 1311:         err = str2offset(&config->resume_from, nextarg);
 1312:         if(err)
 1313:           return err;
 1314:         config->resume_from_current = FALSE;
 1315:       }
 1316:       else {
 1317:         config->resume_from_current = TRUE;
 1318:         config->resume_from = 0;
 1319:       }
 1320:       config->use_resume = TRUE;
 1321:       break;
 1322:     case 'd':
 1323:       /* postfield data */
 1324:     {
 1325:       char *postdata = NULL;
 1326:       FILE *file;
 1327:       size_t size = 0;
 1328:       bool raw_mode = (subletter == 'r');
 1329: 
 1330:       if(subletter == 'e') { /* --data-urlencode*/
 1331:         /* [name]=[content], we encode the content part only
 1332:          * [name]@[file name]
 1333:          *
 1334:          * Case 2: we first load the file using that name and then encode
 1335:          * the content.
 1336:          */
 1337:         const char *p = strchr(nextarg, '=');
 1338:         size_t nlen;
 1339:         char is_file;
 1340:         if(!p)
 1341:           /* there was no '=' letter, check for a '@' instead */
 1342:           p = strchr(nextarg, '@');
 1343:         if(p) {
 1344:           nlen = p - nextarg; /* length of the name part */
 1345:           is_file = *p++; /* pass the separator */
 1346:         }
 1347:         else {
 1348:           /* neither @ nor =, so no name and it isn't a file */
 1349:           nlen = is_file = 0;
 1350:           p = nextarg;
 1351:         }
 1352:         if('@' == is_file) {
 1353:           /* a '@' letter, it means that a file name or - (stdin) follows */
 1354:           if(!strcmp("-", p)) {
 1355:             file = stdin;
 1356:             set_binmode(stdin);
 1357:           }
 1358:           else {
 1359:             file = fopen(p, "rb");
 1360:             if(!file)
 1361:               warnf(global,
 1362:                     "Couldn't read data from file \"%s\", this makes "
 1363:                     "an empty POST.\n", nextarg);
 1364:           }
 1365: 
 1366:           err = file2memory(&postdata, &size, file);
 1367: 
 1368:           if(file && (file != stdin))
 1369:             fclose(file);
 1370:           if(err)
 1371:             return err;
 1372:         }
 1373:         else {
 1374:           GetStr(&postdata, p);
 1375:           if(postdata)
 1376:             size = strlen(postdata);
 1377:         }
 1378: 
 1379:         if(!postdata) {
 1380:           /* no data from the file, point to a zero byte string to make this
 1381:              get sent as a POST anyway */
 1382:           postdata = strdup("");
 1383:           if(!postdata)
 1384:             return PARAM_NO_MEM;
 1385:           size = 0;
 1386:         }
 1387:         else {
 1388:           char *enc = curl_easy_escape(NULL, postdata, (int)size);
 1389:           Curl_safefree(postdata); /* no matter if it worked or not */
 1390:           if(enc) {
 1391:             /* now make a string with the name from above and append the
 1392:                encoded string */
 1393:             size_t outlen = nlen + strlen(enc) + 2;
 1394:             char *n = malloc(outlen);
 1395:             if(!n) {
 1396:               curl_free(enc);
 1397:               return PARAM_NO_MEM;
 1398:             }
 1399:             if(nlen > 0) { /* only append '=' if we have a name */
 1400:               msnprintf(n, outlen, "%.*s=%s", nlen, nextarg, enc);
 1401:               size = outlen-1;
 1402:             }
 1403:             else {
 1404:               strcpy(n, enc);
 1405:               size = outlen-2; /* since no '=' was inserted */
 1406:             }
 1407:             curl_free(enc);
 1408:             postdata = n;
 1409:           }
 1410:           else
 1411:             return PARAM_NO_MEM;
 1412:         }
 1413:       }
 1414:       else if('@' == *nextarg && !raw_mode) {
 1415:         /* the data begins with a '@' letter, it means that a file name
 1416:            or - (stdin) follows */
 1417:         nextarg++; /* pass the @ */
 1418: 
 1419:         if(!strcmp("-", nextarg)) {
 1420:           file = stdin;
 1421:           if(subletter == 'b') /* forced data-binary */
 1422:             set_binmode(stdin);
 1423:         }
 1424:         else {
 1425:           file = fopen(nextarg, "rb");
 1426:           if(!file)
 1427:             warnf(global, "Couldn't read data from file \"%s\", this makes "
 1428:                   "an empty POST.\n", nextarg);
 1429:         }
 1430: 
 1431:         if(subletter == 'b')
 1432:           /* forced binary */
 1433:           err = file2memory(&postdata, &size, file);
 1434:         else {
 1435:           err = file2string(&postdata, file);
 1436:           if(postdata)
 1437:             size = strlen(postdata);
 1438:         }
 1439: 
 1440:         if(file && (file != stdin))
 1441:           fclose(file);
 1442:         if(err)
 1443:           return err;
 1444: 
 1445:         if(!postdata) {
 1446:           /* no data from the file, point to a zero byte string to make this
 1447:              get sent as a POST anyway */
 1448:           postdata = strdup("");
 1449:           if(!postdata)
 1450:             return PARAM_NO_MEM;
 1451:         }
 1452:       }
 1453:       else {
 1454:         GetStr(&postdata, nextarg);
 1455:         if(postdata)
 1456:           size = strlen(postdata);
 1457:       }
 1458: 
 1459: #ifdef CURL_DOES_CONVERSIONS
 1460:       if(subletter != 'b') {
 1461:         /* NOT forced binary, convert to ASCII */
 1462:         if(convert_to_network(postdata, strlen(postdata))) {
 1463:           Curl_safefree(postdata);
 1464:           return PARAM_NO_MEM;
 1465:         }
 1466:       }
 1467: #endif
 1468: 
 1469:       if(config->postfields) {
 1470:         /* we already have a string, we append this one with a separating
 1471:            &-letter */
 1472:         char *oldpost = config->postfields;
 1473:         curl_off_t oldlen = config->postfieldsize;
 1474:         curl_off_t newlen = oldlen + curlx_uztoso(size) + 2;
 1475:         config->postfields = malloc((size_t)newlen);
 1476:         if(!config->postfields) {
 1477:           Curl_safefree(oldpost);
 1478:           Curl_safefree(postdata);
 1479:           return PARAM_NO_MEM;
 1480:         }
 1481:         memcpy(config->postfields, oldpost, (size_t)oldlen);
 1482:         /* use byte value 0x26 for '&' to accommodate non-ASCII platforms */
 1483:         config->postfields[oldlen] = '\x26';
 1484:         memcpy(&config->postfields[oldlen + 1], postdata, size);
 1485:         config->postfields[oldlen + 1 + size] = '\0';
 1486:         Curl_safefree(oldpost);
 1487:         Curl_safefree(postdata);
 1488:         config->postfieldsize += size + 1;
 1489:       }
 1490:       else {
 1491:         config->postfields = postdata;
 1492:         config->postfieldsize = curlx_uztoso(size);
 1493:       }
 1494:     }
 1495:     /*
 1496:       We can't set the request type here, as this data might be used in
 1497:       a simple GET if -G is used. Already or soon.
 1498: 
 1499:       if(SetHTTPrequest(HTTPREQ_SIMPLEPOST, &config->httpreq)) {
 1500:         Curl_safefree(postdata);
 1501:         return PARAM_BAD_USE;
 1502:       }
 1503:     */
 1504:     break;
 1505:     case 'D':
 1506:       /* dump-header to given file name */
 1507:       GetStr(&config->headerfile, nextarg);
 1508:       break;
 1509:     case 'e':
 1510:     {
 1511:       char *ptr = strstr(nextarg, ";auto");
 1512:       if(ptr) {
 1513:         /* Automatic referer requested, this may be combined with a
 1514:            set initial one */
 1515:         config->autoreferer = TRUE;
 1516:         *ptr = 0; /* zero terminate here */
 1517:       }
 1518:       else
 1519:         config->autoreferer = FALSE;
 1520:       GetStr(&config->referer, nextarg);
 1521:     }
 1522:     break;
 1523:     case 'E':
 1524:       switch(subletter) {
 1525:       case '\0': /* certificate file */
 1526:         GetFileAndPassword(nextarg, &config->cert, &config->key_passwd);
 1527:         break;
 1528:       case 'a': /* CA info PEM file */
 1529:         GetStr(&config->cacert, nextarg);
 1530:         break;
 1531:       case 'b': /* cert file type */
 1532:         GetStr(&config->cert_type, nextarg);
 1533:         break;
 1534:       case 'c': /* private key file */
 1535:         GetStr(&config->key, nextarg);
 1536:         break;
 1537:       case 'd': /* private key file type */
 1538:         GetStr(&config->key_type, nextarg);
 1539:         break;
 1540:       case 'e': /* private key passphrase */
 1541:         GetStr(&config->key_passwd, nextarg);
 1542:         cleanarg(nextarg);
 1543:         break;
 1544:       case 'f': /* crypto engine */
 1545:         GetStr(&config->engine, nextarg);
 1546:         if(config->engine && curl_strequal(config->engine, "list"))
 1547:           return PARAM_ENGINES_REQUESTED;
 1548:         break;
 1549:       case 'g': /* CA cert directory */
 1550:         GetStr(&config->capath, nextarg);
 1551:         break;
 1552:       case 'h': /* --pubkey public key file */
 1553:         GetStr(&config->pubkey, nextarg);
 1554:         break;
 1555:       case 'i': /* --hostpubmd5 md5 of the host public key */
 1556:         GetStr(&config->hostpubmd5, nextarg);
 1557:         if(!config->hostpubmd5 || strlen(config->hostpubmd5) != 32)
 1558:           return PARAM_BAD_USE;
 1559:         break;
 1560:       case 'j': /* CRL file */
 1561:         GetStr(&config->crlfile, nextarg);
 1562:         break;
 1563:       case 'k': /* TLS username */
 1564:         if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
 1565:           GetStr(&config->tls_username, nextarg);
 1566:         else
 1567:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1568:         break;
 1569:       case 'l': /* TLS password */
 1570:         if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
 1571:           GetStr(&config->tls_password, nextarg);
 1572:         else
 1573:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1574:         break;
 1575:       case 'm': /* TLS authentication type */
 1576:         if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP) {
 1577:           GetStr(&config->tls_authtype, nextarg);
 1578:           if(!curl_strequal(config->tls_authtype, "SRP"))
 1579:             return PARAM_LIBCURL_DOESNT_SUPPORT; /* only support TLS-SRP */
 1580:         }
 1581:         else
 1582:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1583:         break;
 1584:       case 'n': /* no empty SSL fragments, --ssl-allow-beast */
 1585:         if(curlinfo->features & CURL_VERSION_SSL)
 1586:           config->ssl_allow_beast = toggle;
 1587:         break;
 1588: 
 1589:       case 'p': /* Pinned public key DER file */
 1590:         GetStr(&config->pinnedpubkey, nextarg);
 1591:         break;
 1592: 
 1593:       case 'P': /* proxy pinned public key */
 1594:         GetStr(&config->proxy_pinnedpubkey, nextarg);
 1595:         break;
 1596: 
 1597:       case 'q': /* --cert-status */
 1598:         config->verifystatus = TRUE;
 1599:         break;
 1600: 
 1601:       case 'r': /* --false-start */
 1602:         config->falsestart = TRUE;
 1603:         break;
 1604: 
 1605:       case 's': /* --ssl-no-revoke */
 1606:         if(curlinfo->features & CURL_VERSION_SSL)
 1607:           config->ssl_no_revoke = TRUE;
 1608:         break;
 1609: 
 1610:       case 'S': /* --ssl-revoke-best-effort */
 1611:         if(curlinfo->features & CURL_VERSION_SSL)
 1612:           config->ssl_revoke_best_effort = TRUE;
 1613:         break;
 1614: 
 1615:       case 't': /* --tcp-fastopen */
 1616:         config->tcp_fastopen = TRUE;
 1617:         break;
 1618: 
 1619:       case 'u': /* TLS username for proxy */
 1620:         if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
 1621:           GetStr(&config->proxy_tls_username, nextarg);
 1622:         else
 1623:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1624:         break;
 1625: 
 1626:       case 'v': /* TLS password for proxy */
 1627:         if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
 1628:           GetStr(&config->proxy_tls_password, nextarg);
 1629:         else
 1630:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1631:         break;
 1632: 
 1633:       case 'w': /* TLS authentication type for proxy */
 1634:         if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP) {
 1635:           GetStr(&config->proxy_tls_authtype, nextarg);
 1636:           if(!curl_strequal(config->proxy_tls_authtype, "SRP"))
 1637:             return PARAM_LIBCURL_DOESNT_SUPPORT; /* only support TLS-SRP */
 1638:         }
 1639:         else
 1640:           return PARAM_LIBCURL_DOESNT_SUPPORT;
 1641:         break;
 1642: 
 1643:       case 'x': /* certificate file for proxy */
 1644:         GetFileAndPassword(nextarg, &config->proxy_cert,
 1645:                            &config->proxy_key_passwd);
 1646:         break;
 1647: 
 1648:       case 'y': /* cert file type for proxy */
 1649:         GetStr(&config->proxy_cert_type, nextarg);
 1650:         break;
 1651: 
 1652:       case 'z': /* private key file for proxy */
 1653:         GetStr(&config->proxy_key, nextarg);
 1654:         break;
 1655: 
 1656:       case '0': /* private key file type for proxy */
 1657:         GetStr(&config->proxy_key_type, nextarg);
 1658:         break;
 1659: 
 1660:       case '1': /* private key passphrase for proxy */
 1661:         GetStr(&config->proxy_key_passwd, nextarg);
 1662:         cleanarg(nextarg);
 1663:         break;
 1664: 
 1665:       case '2': /* ciphers for proxy */
 1666:         GetStr(&config->proxy_cipher_list, nextarg);
 1667:         break;
 1668: 
 1669:       case '3': /* CRL file for proxy */
 1670:         GetStr(&config->proxy_crlfile, nextarg);
 1671:         break;
 1672: 
 1673:       case '4': /* no empty SSL fragments for proxy */
 1674:         if(curlinfo->features & CURL_VERSION_SSL)
 1675:           config->proxy_ssl_allow_beast = toggle;
 1676:         break;
 1677: 
 1678:       case '5': /* --login-options */
 1679:         GetStr(&config->login_options, nextarg);
 1680:         break;
 1681: 
 1682:       case '6': /* CA info PEM file for proxy */
 1683:         GetStr(&config->proxy_cacert, nextarg);
 1684:         break;
 1685: 
 1686:       case '7': /* CA cert directory for proxy */
 1687:         GetStr(&config->proxy_capath, nextarg);
 1688:         break;
 1689: 
 1690:       case '8': /* allow insecure SSL connects for proxy */
 1691:         config->proxy_insecure_ok = toggle;
 1692:         break;
 1693: 
 1694:       case '9': /* --proxy-tlsv1 */
 1695:         /* TLS version 1 for proxy */
 1696:         config->proxy_ssl_version = CURL_SSLVERSION_TLSv1;
 1697:         break;
 1698: 
 1699:       case 'A':
 1700:         /* --socks5-basic */
 1701:         if(toggle)
 1702:           config->socks5_auth |= CURLAUTH_BASIC;
 1703:         else
 1704:           config->socks5_auth &= ~CURLAUTH_BASIC;
 1705:         break;
 1706: 
 1707:       case 'B':
 1708:         /* --socks5-gssapi */
 1709:         if(toggle)
 1710:           config->socks5_auth |= CURLAUTH_GSSAPI;
 1711:         else
 1712:           config->socks5_auth &= ~CURLAUTH_GSSAPI;
 1713:         break;
 1714: 
 1715:       case 'C':
 1716:         GetStr(&config->etag_save_file, nextarg);
 1717:         break;
 1718: 
 1719:       case 'D':
 1720:         GetStr(&config->etag_compare_file, nextarg);
 1721:         break;
 1722: 
 1723:       default: /* unknown flag */
 1724:         return PARAM_OPTION_UNKNOWN;
 1725:       }
 1726:       break;
 1727:     case 'f':
 1728:       switch(subletter) {
 1729:       case 'a': /* --fail-early */
 1730:         global->fail_early = toggle;
 1731:         break;
 1732:       case 'b': /* --styled-output */
 1733:         global->styled_output = toggle;
 1734:         break;
 1735:       case 'c': /* --mail-rcpt-allowfails */
 1736:         config->mail_rcpt_allowfails = toggle;
 1737:         break;
 1738:       default: /* --fail (hard on errors)  */
 1739:         config->failonerror = toggle;
 1740:       }
 1741:       break;
 1742:     case 'F':
 1743:       /* "form data" simulation, this is a little advanced so lets do our best
 1744:          to sort this out slowly and carefully */
 1745:       if(formparse(config,
 1746:                    nextarg,
 1747:                    &config->mimeroot,
 1748:                    &config->mimecurrent,
 1749:                    (subletter == 's')?TRUE:FALSE)) /* 's' is literal string */
 1750:         return PARAM_BAD_USE;
 1751:       if(SetHTTPrequest(config, HTTPREQ_MIMEPOST, &config->httpreq))
 1752:         return PARAM_BAD_USE;
 1753:       break;
 1754: 
 1755:     case 'g': /* g disables URLglobbing */
 1756:       config->globoff = toggle;
 1757:       break;
 1758: 
 1759:     case 'G': /* HTTP GET */
 1760:       if(subletter == 'a') { /* --request-target */
 1761:         GetStr(&config->request_target, nextarg);
 1762:       }
 1763:       else
 1764:         config->use_httpget = TRUE;
 1765:       break;
 1766: 
 1767:     case 'h': /* h for help */
 1768:       if(toggle) {
 1769:         return PARAM_HELP_REQUESTED;
 1770:       }
 1771:       /* we now actually support --no-help too! */
 1772:       break;
 1773:     case 'H':
 1774:       /* A custom header to append to a list */
 1775:       if(nextarg[0] == '@') {
 1776:         /* read many headers from a file or stdin */
 1777:         char *string;
 1778:         size_t len;
 1779:         bool use_stdin = !strcmp(&nextarg[1], "-");
 1780:         FILE *file = use_stdin?stdin:fopen(&nextarg[1], FOPEN_READTEXT);
 1781:         if(!file)
 1782:           warnf(global, "Failed to open %s!\n", &nextarg[1]);
 1783:         else {
 1784:           err = file2memory(&string, &len, file);
 1785:           if(!err && string) {
 1786:             /* Allow strtok() here since this isn't used threaded */
 1787:             /* !checksrc! disable BANNEDFUNC 2 */
 1788:             char *h = strtok(string, "\r\n");
 1789:             while(h) {
 1790:               if(subletter == 'p') /* --proxy-header */
 1791:                 err = add2list(&config->proxyheaders, h);
 1792:               else
 1793:                 err = add2list(&config->headers, h);
 1794:               if(err)
 1795:                 break;
 1796:               h = strtok(NULL, "\r\n");
 1797:             }
 1798:             free(string);
 1799:           }
 1800:           if(!use_stdin)
 1801:             fclose(file);
 1802:           if(err)
 1803:             return err;
 1804:         }
 1805:       }
 1806:       else {
 1807:         if(subletter == 'p') /* --proxy-header */
 1808:           err = add2list(&config->proxyheaders, nextarg);
 1809:         else
 1810:           err = add2list(&config->headers, nextarg);
 1811:         if(err)
 1812:           return err;
 1813:       }
 1814:       break;
 1815:     case 'i':
 1816:       config->show_headers = toggle; /* show the headers as well in the
 1817:                                         general output stream */
 1818:       break;
 1819:     case 'j':
 1820:       config->cookiesession = toggle;
 1821:       break;
 1822:     case 'I': /* --head */
 1823:       config->no_body = toggle;
 1824:       config->show_headers = toggle;
 1825:       if(SetHTTPrequest(config,
 1826:                         (config->no_body)?HTTPREQ_HEAD:HTTPREQ_GET,
 1827:                         &config->httpreq))
 1828:         return PARAM_BAD_USE;
 1829:       break;
 1830:     case 'J': /* --remote-header-name */
 1831:       if(config->show_headers) {
 1832:         warnf(global,
 1833:               "--include and --remote-header-name cannot be combined.\n");
 1834:         return PARAM_BAD_USE;
 1835:       }
 1836:       config->content_disposition = toggle;
 1837:       break;
 1838:     case 'k': /* allow insecure SSL connects */
 1839:       config->insecure_ok = toggle;
 1840:       break;
 1841:     case 'K': /* parse config file */
 1842:       if(parseconfig(nextarg, global))
 1843:         warnf(global, "error trying read config from the '%s' file\n",
 1844:               nextarg);
 1845:       break;
 1846:     case 'l':
 1847:       config->dirlistonly = toggle; /* only list the names of the FTP dir */
 1848:       break;
 1849:     case 'L':
 1850:       config->followlocation = toggle; /* Follow Location: HTTP headers */
 1851:       switch(subletter) {
 1852:       case 't':
 1853:         /* Continue to send authentication (user+password) when following
 1854:          * locations, even when hostname changed */
 1855:         config->unrestricted_auth = toggle;
 1856:         break;
 1857:       }
 1858:       break;
 1859:     case 'm':
 1860:       /* specified max time */
 1861:       err = str2udouble(&config->timeout, nextarg, LONG_MAX/1000);
 1862:       if(err)
 1863:         return err;
 1864:       break;
 1865:     case 'M': /* M for manual, huge help */
 1866:       if(toggle) { /* --no-manual shows no manual... */
 1867: #ifdef USE_MANUAL
 1868:         return PARAM_MANUAL_REQUESTED;
 1869: #else
 1870:         warnf(global,
 1871:               "built-in manual was disabled at build-time!\n");
 1872:         return PARAM_OPTION_UNKNOWN;
 1873: #endif
 1874:       }
 1875:       break;
 1876:     case 'n':
 1877:       switch(subletter) {
 1878:       case 'o': /* use .netrc or URL */
 1879:         config->netrc_opt = toggle;
 1880:         break;
 1881:       case 'e': /* netrc-file */
 1882:         GetStr(&config->netrc_file, nextarg);
 1883:         break;
 1884:       default:
 1885:         /* pick info from .netrc, if this is used for http, curl will
 1886:            automatically enfore user+password with the request */
 1887:         config->netrc = toggle;
 1888:         break;
 1889:       }
 1890:       break;
 1891:     case 'N':
 1892:       /* disable the output I/O buffering. note that the option is called
 1893:          --buffer but is mostly used in the negative form: --no-buffer */
 1894:       if(longopt)
 1895:         config->nobuffer = (!toggle)?TRUE:FALSE;
 1896:       else
 1897:         config->nobuffer = toggle;
 1898:       break;
 1899:     case 'O': /* --remote-name */
 1900:       if(subletter == 'a') { /* --remote-name-all */
 1901:         config->default_node_flags = toggle?GETOUT_USEREMOTE:0;
 1902:         break;
 1903:       }
 1904:       /* FALLTHROUGH */
 1905:     case 'o': /* --output */
 1906:       /* output file */
 1907:     {
 1908:       struct getout *url;
 1909:       if(!config->url_out)
 1910:         config->url_out = config->url_list;
 1911:       if(config->url_out) {
 1912:         /* there's a node here, if it already is filled-in continue to find
 1913:            an "empty" node */
 1914:         while(config->url_out && (config->url_out->flags & GETOUT_OUTFILE))
 1915:           config->url_out = config->url_out->next;
 1916:       }
 1917: 
 1918:       /* now there might or might not be an available node to fill in! */
 1919: 
 1920:       if(config->url_out)
 1921:         /* existing node */
 1922:         url = config->url_out;
 1923:       else
 1924:         /* there was no free node, create one! */
 1925:         config->url_out = url = new_getout(config);
 1926: 
 1927:       if(!url)
 1928:         return PARAM_NO_MEM;
 1929: 
 1930:       /* fill in the outfile */
 1931:       if('o' == letter) {
 1932:         GetStr(&url->outfile, nextarg);
 1933:         url->flags &= ~GETOUT_USEREMOTE; /* switch off */
 1934:       }
 1935:       else {
 1936:         url->outfile = NULL; /* leave it */
 1937:         if(toggle)
 1938:           url->flags |= GETOUT_USEREMOTE;  /* switch on */
 1939:         else
 1940:           url->flags &= ~GETOUT_USEREMOTE; /* switch off */
 1941:       }
 1942:       url->flags |= GETOUT_OUTFILE;
 1943:     }
 1944:     break;
 1945:     case 'P':
 1946:       /* This makes the FTP sessions use PORT instead of PASV */
 1947:       /* use <eth0> or <192.168.10.10> style addresses. Anything except
 1948:          this will make us try to get the "default" address.
 1949:          NOTE: this is a changed behaviour since the released 4.1!
 1950:       */
 1951:       GetStr(&config->ftpport, nextarg);
 1952:       break;
 1953:     case 'p':
 1954:       /* proxy tunnel for non-http protocols */
 1955:       config->proxytunnel = toggle;
 1956:       break;
 1957: 
 1958:     case 'q': /* if used first, already taken care of, we do it like
 1959:                  this so we don't cause an error! */
 1960:       break;
 1961:     case 'Q':
 1962:       /* QUOTE command to send to FTP server */
 1963:       switch(nextarg[0]) {
 1964:       case '-':
 1965:         /* prefixed with a dash makes it a POST TRANSFER one */
 1966:         nextarg++;
 1967:         err = add2list(&config->postquote, nextarg);
 1968:         break;
 1969:       case '+':
 1970:         /* prefixed with a plus makes it a just-before-transfer one */
 1971:         nextarg++;
 1972:         err = add2list(&config->prequote, nextarg);
 1973:         break;
 1974:       default:
 1975:         err = add2list(&config->quote, nextarg);
 1976:         break;
 1977:       }
 1978:       if(err)
 1979:         return err;
 1980:       break;
 1981:     case 'r':
 1982:       /* Specifying a range WITHOUT A DASH will create an illegal HTTP range
 1983:          (and won't actually be range by definition). The man page previously
 1984:          claimed that to be a good way, why this code is added to work-around
 1985:          it. */
 1986:       if(ISDIGIT(*nextarg) && !strchr(nextarg, '-')) {
 1987:         char buffer[32];
 1988:         curl_off_t off;
 1989:         if(curlx_strtoofft(nextarg, NULL, 10, &off)) {
 1990:           warnf(global, "unsupported range point\n");
 1991:           return PARAM_BAD_USE;
 1992:         }
 1993:         warnf(global,
 1994:               "A specified range MUST include at least one dash (-). "
 1995:               "Appending one for you!\n");
 1996:         msnprintf(buffer, sizeof(buffer), "%" CURL_FORMAT_CURL_OFF_T "-", off);
 1997:         Curl_safefree(config->range);
 1998:         config->range = strdup(buffer);
 1999:         if(!config->range)
 2000:           return PARAM_NO_MEM;
 2001:       }
 2002:       {
 2003:         /* byte range requested */
 2004:         const char *tmp_range = nextarg;
 2005:         while(*tmp_range != '\0') {
 2006:           if(!ISDIGIT(*tmp_range) && *tmp_range != '-' && *tmp_range != ',') {
 2007:             warnf(global, "Invalid character is found in given range. "
 2008:                   "A specified range MUST have only digits in "
 2009:                   "\'start\'-\'stop\'. The server's response to this "
 2010:                   "request is uncertain.\n");
 2011:             break;
 2012:           }
 2013:           tmp_range++;
 2014:         }
 2015:         /* byte range requested */
 2016:         GetStr(&config->range, nextarg);
 2017:       }
 2018:       break;
 2019:     case 'R':
 2020:       /* use remote file's time */
 2021:       config->remote_time = toggle;
 2022:       break;
 2023:     case 's':
 2024:       /* don't show progress meter, don't show errors : */
 2025:       if(toggle)
 2026:         global->mute = global->noprogress = TRUE;
 2027:       else
 2028:         global->mute = global->noprogress = FALSE;
 2029:       if(global->showerror < 0)
 2030:         /* if still on the default value, set showerror to the reverse of
 2031:            toggle. This is to allow -S and -s to be used in an independent
 2032:            order but still have the same effect. */
 2033:         global->showerror = (!toggle)?TRUE:FALSE; /* toggle off */
 2034:       break;
 2035:     case 'S':
 2036:       /* show errors */
 2037:       global->showerror = toggle?1:0; /* toggle on if used with -s */
 2038:       break;
 2039:     case 't':
 2040:       /* Telnet options */
 2041:       err = add2list(&config->telnet_options, nextarg);
 2042:       if(err)
 2043:         return err;
 2044:       break;
 2045:     case 'T':
 2046:       /* we are uploading */
 2047:     {
 2048:       struct getout *url;
 2049:       if(!config->url_ul)
 2050:         config->url_ul = config->url_list;
 2051:       if(config->url_ul) {
 2052:         /* there's a node here, if it already is filled-in continue to find
 2053:            an "empty" node */
 2054:         while(config->url_ul && (config->url_ul->flags & GETOUT_UPLOAD))
 2055:           config->url_ul = config->url_ul->next;
 2056:       }
 2057: 
 2058:       /* now there might or might not be an available node to fill in! */
 2059: 
 2060:       if(config->url_ul)
 2061:         /* existing node */
 2062:         url = config->url_ul;
 2063:       else
 2064:         /* there was no free node, create one! */
 2065:         config->url_ul = url = new_getout(config);
 2066: 
 2067:       if(!url)
 2068:         return PARAM_NO_MEM;
 2069: 
 2070:       url->flags |= GETOUT_UPLOAD; /* mark -T used */
 2071:       if(!*nextarg)
 2072:         url->flags |= GETOUT_NOUPLOAD;
 2073:       else {
 2074:         /* "-" equals stdin, but keep the string around for now */
 2075:         GetStr(&url->infile, nextarg);
 2076:       }
 2077:     }
 2078:     break;
 2079:     case 'u':
 2080:       /* user:password  */
 2081:       GetStr(&config->userpwd, nextarg);
 2082:       cleanarg(nextarg);
 2083:       break;
 2084:     case 'U':
 2085:       /* Proxy user:password  */
 2086:       GetStr(&config->proxyuserpwd, nextarg);
 2087:       cleanarg(nextarg);
 2088:       break;
 2089:     case 'v':
 2090:       if(toggle) {
 2091:         /* the '%' thing here will cause the trace get sent to stderr */
 2092:         Curl_safefree(global->trace_dump);
 2093:         global->trace_dump = strdup("%");
 2094:         if(!global->trace_dump)
 2095:           return PARAM_NO_MEM;
 2096:         if(global->tracetype && (global->tracetype != TRACE_PLAIN))
 2097:           warnf(global,
 2098:                 "-v, --verbose overrides an earlier trace/verbose option\n");
 2099:         global->tracetype = TRACE_PLAIN;
 2100:       }
 2101:       else
 2102:         /* verbose is disabled here */
 2103:         global->tracetype = TRACE_NONE;
 2104:       break;
 2105:     case 'V':
 2106:       if(toggle)    /* --no-version yields no output! */
 2107:         return PARAM_VERSION_INFO_REQUESTED;
 2108:       break;
 2109: 
 2110:     case 'w':
 2111:       /* get the output string */
 2112:       if('@' == *nextarg) {
 2113:         /* the data begins with a '@' letter, it means that a file name
 2114:            or - (stdin) follows */
 2115:         FILE *file;
 2116:         const char *fname;
 2117:         nextarg++; /* pass the @ */
 2118:         if(!strcmp("-", nextarg)) {
 2119:           fname = "<stdin>";
 2120:           file = stdin;
 2121:         }
 2122:         else {
 2123:           fname = nextarg;
 2124:           file = fopen(nextarg, FOPEN_READTEXT);
 2125:         }
 2126:         Curl_safefree(config->writeout);
 2127:         err = file2string(&config->writeout, file);
 2128:         if(file && (file != stdin))
 2129:           fclose(file);
 2130:         if(err)
 2131:           return err;
 2132:         if(!config->writeout)
 2133:           warnf(global, "Failed to read %s", fname);
 2134:       }
 2135:       else
 2136:         GetStr(&config->writeout, nextarg);
 2137:       break;
 2138:     case 'x':
 2139:       switch(subletter) {
 2140:       case 'a': /* --preproxy */
 2141:         GetStr(&config->preproxy, nextarg);
 2142:         break;
 2143:       default:
 2144:         /* --proxy */
 2145:         GetStr(&config->proxy, nextarg);
 2146:         config->proxyver = CURLPROXY_HTTP;
 2147:         break;
 2148:       }
 2149:       break;
 2150:     case 'X':
 2151:       /* set custom request */
 2152:       GetStr(&config->customrequest, nextarg);
 2153:       break;
 2154:     case 'y':
 2155:       /* low speed time */
 2156:       err = str2unum(&config->low_speed_time, nextarg);
 2157:       if(err)
 2158:         return err;
 2159:       if(!config->low_speed_limit)
 2160:         config->low_speed_limit = 1;
 2161:       break;
 2162:     case 'Y':
 2163:       /* low speed limit */
 2164:       err = str2unum(&config->low_speed_limit, nextarg);
 2165:       if(err)
 2166:         return err;
 2167:       if(!config->low_speed_time)
 2168:         config->low_speed_time = 30;
 2169:       break;
 2170:     case 'Z':
 2171:       switch(subletter) {
 2172:       case '\0':  /* --parallel */
 2173:         global->parallel = toggle;
 2174:         break;
 2175:       case 'b':   /* --parallel-max */
 2176:         err = str2unum(&global->parallel_max, nextarg);
 2177:         if(err)
 2178:           return err;
 2179:         if((global->parallel_max > MAX_PARALLEL) ||
 2180:            (global->parallel_max < 1))
 2181:           global->parallel_max = PARALLEL_DEFAULT;
 2182:         break;
 2183:       case 'c':   /* --parallel-connect */
 2184:         global->parallel_connect = toggle;
 2185:         break;
 2186:       }
 2187:       break;
 2188:     case 'z': /* time condition coming up */
 2189:       switch(*nextarg) {
 2190:       case '+':
 2191:         nextarg++;
 2192:         /* FALLTHROUGH */
 2193:       default:
 2194:         /* If-Modified-Since: (section 14.28 in RFC2068) */
 2195:         config->timecond = CURL_TIMECOND_IFMODSINCE;
 2196:         break;
 2197:       case '-':
 2198:         /* If-Unmodified-Since:  (section 14.24 in RFC2068) */
 2199:         config->timecond = CURL_TIMECOND_IFUNMODSINCE;
 2200:         nextarg++;
 2201:         break;
 2202:       case '=':
 2203:         /* Last-Modified:  (section 14.29 in RFC2068) */
 2204:         config->timecond = CURL_TIMECOND_LASTMOD;
 2205:         nextarg++;
 2206:         break;
 2207:       }
 2208:       now = time(NULL);
 2209:       config->condtime = (curl_off_t)curl_getdate(nextarg, &now);
 2210:       if(-1 == config->condtime) {
 2211:         /* now let's see if it is a file name to get the time from instead! */
 2212:         curl_off_t filetime = getfiletime(nextarg, config->global->errors);
 2213:         if(filetime >= 0) {
 2214:           /* pull the time out from the file */
 2215:           config->condtime = filetime;
 2216:         }
 2217:         else {
 2218:           /* failed, remove time condition */
 2219:           config->timecond = CURL_TIMECOND_NONE;
 2220:           warnf(global,
 2221:                 "Illegal date format for -z, --time-cond (and not "
 2222:                 "a file name). Disabling time condition. "
 2223:                 "See curl_getdate(3) for valid date syntax.\n");
 2224:         }
 2225:       }
 2226:       break;
 2227:     default: /* unknown flag */
 2228:       return PARAM_OPTION_UNKNOWN;
 2229:     }
 2230:     hit = -1;
 2231: 
 2232:   } while(!longopt && !singleopt && *++parse && !*usedarg);
 2233: 
 2234:   return PARAM_OK;
 2235: }
 2236: 
 2237: ParameterError parse_args(struct GlobalConfig *global, int argc,
 2238:                           argv_item_t argv[])
 2239: {
 2240:   int i;
 2241:   bool stillflags;
 2242:   char *orig_opt = NULL;
 2243:   ParameterError result = PARAM_OK;
 2244:   struct OperationConfig *config = global->first;
 2245: 
 2246:   for(i = 1, stillflags = TRUE; i < argc && !result; i++) {
 2247:     orig_opt = argv[i];
 2248: 
 2249:     if(stillflags && ('-' == argv[i][0])) {
 2250:       bool passarg;
 2251:       char *flag = argv[i];
 2252: 
 2253:       if(!strcmp("--", argv[i]))
 2254:         /* This indicates the end of the flags and thus enables the
 2255:            following (URL) argument to start with -. */
 2256:         stillflags = FALSE;
 2257:       else {
 2258:         char *nextarg = (i < (argc - 1)) ? argv[i + 1] : NULL;
 2259: 
 2260:         result = getparameter(flag, nextarg, &passarg, global, config);
 2261:         config = global->last;
 2262:         if(result == PARAM_NEXT_OPERATION) {
 2263:           /* Reset result as PARAM_NEXT_OPERATION is only used here and not
 2264:              returned from this function */
 2265:           result = PARAM_OK;
 2266: 
 2267:           if(config->url_list && config->url_list->url) {
 2268:             /* Allocate the next config */
 2269:             config->next = malloc(sizeof(struct OperationConfig));
 2270:             if(config->next) {
 2271:               /* Initialise the newly created config */
 2272:               config_init(config->next);
 2273: 
 2274:               /* Set the global config pointer */
 2275:               config->next->global = global;
 2276: 
 2277:               /* Update the last config pointer */
 2278:               global->last = config->next;
 2279: 
 2280:               /* Move onto the new config */
 2281:               config->next->prev = config;
 2282:               config = config->next;
 2283:             }
 2284:             else
 2285:               result = PARAM_NO_MEM;
 2286:           }
 2287:         }
 2288:         else if(!result && passarg)
 2289:           i++; /* we're supposed to skip this */
 2290:       }
 2291:     }
 2292:     else {
 2293:       bool used;
 2294: 
 2295:       /* Just add the URL please */
 2296:       result = getparameter((char *)"--url", argv[i], &used, global,
 2297:                             config);
 2298:     }
 2299:   }
 2300: 
 2301:   if(result && result != PARAM_HELP_REQUESTED &&
 2302:      result != PARAM_MANUAL_REQUESTED &&
 2303:      result != PARAM_VERSION_INFO_REQUESTED &&
 2304:      result != PARAM_ENGINES_REQUESTED) {
 2305:     const char *reason = param2text(result);
 2306: 
 2307:     if(orig_opt && strcmp(":", orig_opt))
 2308:       helpf(global->errors, "option %s: %s\n", orig_opt, reason);
 2309:     else
 2310:       helpf(global->errors, "%s\n", reason);
 2311:   }
 2312: 
 2313:   return result;
 2314: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>