Files
ocserv/libopts/check.c
2013-01-30 01:39:14 +01:00

167 lines
4.6 KiB
C

/**
* @file check.c
*
* @brief consistency checks.
*
* Time-stamp: "2012-03-31 13:46:35 bkorb"
*
* This file contains the routines that deal with processing quoted strings
* into an internal format.
*
* This file is part of AutoOpts, a companion to AutoGen.
* AutoOpts is free software.
* AutoOpts is Copyright (c) 1992-2012 by Bruce Korb - all rights reserved
*
* AutoOpts is available under any one of two licenses. The license
* in use must be one of these two and the choice is under the control
* of the user of the license.
*
* The GNU Lesser General Public License, version 3 or later
* See the files "COPYING.lgplv3" and "COPYING.gplv3"
*
* The Modified Berkeley Software Distribution License
* See the file "COPYING.mbsd"
*
* These files have the following md5sums:
*
* 43b91e8ca915626ed3818ffb1b71248b pkg/libopts/COPYING.gplv3
* 06a1a2e4760c90ea5e1dad8dfaac4d39 pkg/libopts/COPYING.lgplv3
* 66a5cedaf62c4b2637025f049f9b826f pkg/libopts/COPYING.mbsd
*/
/**
* Check for conflicts based on "must" and "cannot" attributes.
*/
static bool
has_conflict(tOptions * pOpts, tOptDesc * pOD)
{
if (pOD->pOptMust != NULL) {
int const * pMust = pOD->pOptMust;
while (*pMust != NO_EQUIVALENT) {
tOptDesc * p = pOpts->pOptDesc + *(pMust++);
if (UNUSED_OPT(p)) {
const tOptDesc * pN = pOpts->pOptDesc + pMust[-1];
fprintf(stderr, zReqFmt, pOD->pz_Name, pN->pz_Name);
return true;
}
}
}
if (pOD->pOptCant != NULL) {
int const * pCant = pOD->pOptCant;
while (*pCant != NO_EQUIVALENT) {
tOptDesc * p = pOpts->pOptDesc + *(pCant++);
if (SELECTED_OPT(p)) {
const tOptDesc* pN = pOpts->pOptDesc + pCant[-1];
fprintf(stderr, zCantFmt, pOD->pz_Name, pN->pz_Name);
return true;
}
}
}
return false;
}
/**
* Check that the option occurs often enough. Too often is already checked.
*/
static bool
occurs_enough(tOptions * pOpts, tOptDesc * pOD)
{
(void)pOpts;
/*
* IF the occurrence counts have been satisfied,
* THEN there is no problem.
*/
if (pOD->optOccCt >= pOD->optMinCt)
return true;
/*
* IF MUST_SET means SET and PRESET are okay,
* so min occurrence count doesn't count
*/
if ( (pOD->fOptState & OPTST_MUST_SET)
&& (pOD->fOptState & (OPTST_PRESET | OPTST_SET)) )
return true;
if (pOD->optMinCt > 1)
fprintf(stderr, zNotEnough, pOD->pz_Name, pOD->optMinCt);
else fprintf(stderr, zNeedOne, pOD->pz_Name);
return false;
}
/**
* Verify option consistency.
*
* Make sure that the argument list passes our consistency tests.
*/
LOCAL bool
is_consistent(tOptions * pOpts)
{
tOptDesc * pOD = pOpts->pOptDesc;
int oCt = pOpts->presetOptCt;
/*
* FOR each of "oCt" options, ...
*/
for (;;) {
/*
* IF the current option was provided on the command line
* THEN ensure that any "MUST" requirements are not
* "DEFAULT" (unspecified) *AND* ensure that any
* "CANT" options have not been SET or DEFINED.
*/
if (SELECTED_OPT(pOD)) {
if (has_conflict(pOpts, pOD))
return false;
}
/*
* IF this option is not equivalenced to another,
* OR it is equivalenced to itself (is the equiv. root)
* THEN we need to make sure it occurs often enough.
*/
if ( (pOD->optEquivIndex == NO_EQUIVALENT)
|| (pOD->optEquivIndex == pOD->optIndex) )
if (! occurs_enough(pOpts, pOD))
return false;
if (--oCt <= 0)
break;
pOD++;
}
/*
* IF we are stopping on errors, check to see if any remaining
* arguments are required to be there or prohibited from being there.
*/
if ((pOpts->fOptSet & OPTPROC_ERRSTOP) != 0) {
/*
* Check for prohibition
*/
if ((pOpts->fOptSet & OPTPROC_NO_ARGS) != 0) {
if (pOpts->origArgCt > pOpts->curOptIdx) {
fprintf(stderr, zNoArgs, pOpts->pzProgName);
return false;
}
}
/*
* ELSE not prohibited, check for being required
*/
else if ((pOpts->fOptSet & OPTPROC_ARGS_REQ) != 0) {
if (pOpts->origArgCt <= pOpts->curOptIdx) {
fprintf(stderr, zArgsMust, pOpts->pzProgName);
return false;
}
}
}
return true;
}