#include #include #include #include #include #include #include uint8_t gargoyle_digest_argv(uint16_t optc, struct gargoyle_opt *optv, int *argc, char ***argv, struct gargoyle_err *err, gargoyle_flag_type flags) { if(!(flags & GARGOYLE_FLG_GREED)) { *argc -= 1; *argv += 1; } if(flags & GARGOYLE_FLG_BCASE) { flags |= GARGOYLE_FLG_ICASE; } gargoyle_cpy(err->func, __func__, GARGOYLE_MAX_FUNC_SZ, GARGOYLE_FLG_FILL0); while(1) { const char *arg = **argv; if(!arg || *arg != '-' || !*(arg + 1)) { return GARGOYLE_ERR_SUCCESS; } else if(*arg == '-' && *(arg + 1) == '-' && !*(arg + 2)) { *argc -= 1; *argv += 1; return GARGOYLE_ERR_SUCCESS; } struct gargoyle_opt *opt = NULL; if(*(arg + 1) == '-') { const char *brand = arg + 2; opt = gargoyle_find_brand(optc, optv, brand, NULL, 0, flags); if(!opt) { if(flags & GARGOYLE_FLG_STRCT) { err->code = GARGOYLE_ERR_UNKNOWN_OPT; char *eql = index(brand, '='); snprintf(err->msg, GARGOYLE_MAX_ERR_SZ, "unknown option --%.*s", (int) (eql - brand), brand); return GARGOYLE_ERR_UNKNOWN_OPT; } else { *argc -= 1; *argv += 1; continue; } } uint8_t res = 0; if(opt->type & GARGOYLE_TYPE_BOOL) { res = gargoyle_from_bool(opt, brand, "no-", 3, flags); } else if(*(brand + opt->brand_sz) == '=') { res = gargoyle_from_rope(opt, brand + opt->brand_sz + 1, flags); } else { if(!*(*argv + 1)) { return GARGOYLE_ERR_VALUE_REQUIRED; } *argc -= 1; *argv += 1; res = gargoyle_from_rope(opt, **argv, flags); } if(res) { return res; } } else { const char *idx = arg + 1; for(; *idx; idx += 1) { opt = gargoyle_find_emblem(optc, optv, *idx, flags); if(!opt) { if(flags & GARGOYLE_FLG_STRCT) { return GARGOYLE_ERR_UNKNOWN_OPT; } else { *argc -= 1; *argv += 1; continue; } } uint8_t res = 0; if(opt->type & GARGOYLE_TYPE_BOOL) { res = gargoyle_from_bool(opt, "", "*", 1, flags); } else if(*(idx + 1)) { res = gargoyle_from_rope(opt, idx + 1, flags); break; } else { if(!*(*argv + 1)) { return GARGOYLE_ERR_VALUE_REQUIRED; } *argc -= 1; *argv += 1; res = gargoyle_from_rope(opt, **argv, flags); } } } *argc -= 1; *argv += 1; } return GARGOYLE_ERR_SUCCESS; } uint8_t gargoyle_digest_envh(uint16_t optc, struct gargoyle_opt *optv, const char *prefix, uint16_t prefix_sz, char ***envh, struct gargoyle_err *err, gargoyle_flag_type flags) { if(flags & GARGOYLE_FLG_BCASE) { flags |= GARGOYLE_FLG_ICASE; } for(; **envh; *envh += 1) { const char *var = **envh; if(!gargoyle_cmp(prefix, var, prefix_sz, flags) || *(var + prefix_sz) != '_') { continue; } const char *brand = var + prefix_sz + 1; struct gargoyle_opt *opt = gargoyle_find_brand(optc, optv, brand, "NO_", 3, flags); if(!opt) { if(flags & GARGOYLE_FLG_STRCT) { err->code = GARGOYLE_ERR_UNKNOWN_OPT; char *eql = index(brand, '='); snprintf(err->msg, GARGOYLE_MAX_ERR_SZ, "unknown option %.*s", (int) (eql - brand), brand); return GARGOYLE_ERR_UNKNOWN_OPT; } else { continue; } } uint8_t res = 0; if(opt->type & GARGOYLE_TYPE_BOOL) { res = gargoyle_from_bool(opt, brand, "NO_", 3, flags); } else { res = gargoyle_from_rope(opt, brand + opt->brand_sz + 1, flags); } if(res) { return res; } } return GARGOYLE_ERR_SUCCESS; }