Vladimir D. Seleznev
443060d11a
After commit f4d876f93 rpm-build may generate dependencies beginning with a dot character, but spec parser refuses such dependencies. rpmrebuild fails to rebuild packages with such dependencies because it builds packages with rpmbuild from generated spec files.
205 lines
5.2 KiB
C
205 lines
5.2 KiB
C
/** \ingroup rpmbuild
|
|
* \file build/parseReqs.c
|
|
* Parse dependency tag from spec file or from auto-dependency generator.
|
|
*/
|
|
|
|
#include "system.h"
|
|
|
|
#include "rpmbuild.h"
|
|
#include "debug.h"
|
|
|
|
/**
|
|
*/
|
|
/*@observer@*/ /*@unchecked@*/
|
|
static struct ReqComp {
|
|
/*@observer@*/ /*@null@*/ const char * token;
|
|
rpmsenseFlags sense;
|
|
} ReqComparisons[] = {
|
|
{ "<=", RPMSENSE_LESS | RPMSENSE_EQUAL},
|
|
{ "=<", RPMSENSE_LESS | RPMSENSE_EQUAL},
|
|
{ "<", RPMSENSE_LESS},
|
|
|
|
{ "==", RPMSENSE_EQUAL},
|
|
{ "=", RPMSENSE_EQUAL},
|
|
|
|
{ ">=", RPMSENSE_GREATER | RPMSENSE_EQUAL},
|
|
{ "=>", RPMSENSE_GREATER | RPMSENSE_EQUAL},
|
|
{ ">", RPMSENSE_GREATER},
|
|
|
|
{ NULL, 0 },
|
|
};
|
|
|
|
#define SKIPWHITE(_x) {while(*(_x) && (xisspace(*_x) || *(_x) == ',')) (_x)++;}
|
|
#define SKIPNONWHITE(_x){while(*(_x) &&!(xisspace(*_x) || *(_x) == ',')) (_x)++;}
|
|
|
|
int parseRCPOT(Spec spec, Package pkg, const char *field, int tag,
|
|
int index, rpmsenseFlags tagflags)
|
|
{
|
|
static int allowbdot = -1;
|
|
const char *r, *re, *v, *ve;
|
|
char * req, * version;
|
|
Header h;
|
|
rpmsenseFlags flags;
|
|
|
|
switch (tag) {
|
|
case RPMTAG_PROVIDEFLAGS:
|
|
tagflags |= RPMSENSE_PROVIDES;
|
|
h = pkg->header;
|
|
break;
|
|
case RPMTAG_OBSOLETEFLAGS:
|
|
tagflags |= RPMSENSE_OBSOLETES;
|
|
h = pkg->header;
|
|
break;
|
|
case RPMTAG_CONFLICTFLAGS:
|
|
tagflags |= RPMSENSE_CONFLICTS;
|
|
h = pkg->header;
|
|
break;
|
|
case RPMTAG_BUILDCONFLICTS:
|
|
tagflags |= RPMSENSE_CONFLICTS;
|
|
h = spec->buildRestrictions;
|
|
break;
|
|
case RPMTAG_PREREQ:
|
|
tagflags |= RPMSENSE_PREREQ;
|
|
h = pkg->header;
|
|
break;
|
|
case RPMTAG_BUILDPREREQ:
|
|
tagflags |= RPMSENSE_PREREQ;
|
|
h = spec->buildRestrictions;
|
|
break;
|
|
case RPMTAG_TRIGGERIN:
|
|
tagflags |= RPMSENSE_TRIGGERIN;
|
|
h = pkg->header;
|
|
break;
|
|
case RPMTAG_TRIGGERPOSTUN:
|
|
tagflags |= RPMSENSE_TRIGGERPOSTUN;
|
|
h = pkg->header;
|
|
break;
|
|
case RPMTAG_TRIGGERUN:
|
|
tagflags |= RPMSENSE_TRIGGERUN;
|
|
h = pkg->header;
|
|
break;
|
|
case RPMTAG_BUILDREQUIRES:
|
|
tagflags |= RPMSENSE_ANY;
|
|
h = spec->buildRestrictions;
|
|
break;
|
|
default:
|
|
case RPMTAG_REQUIREFLAGS:
|
|
tagflags |= RPMSENSE_ANY;
|
|
h = pkg->header;
|
|
break;
|
|
}
|
|
|
|
for (r = field; *r != '\0'; r = re) {
|
|
SKIPWHITE(r);
|
|
if (*r == '\0')
|
|
break;
|
|
|
|
flags = (tagflags & ~RPMSENSE_SENSEMASK);
|
|
|
|
/* Tokens must begin with alphanumeric, _, or /
|
|
It also can begin with dot symbol if _allow_deps_with_beginning_dot macro is set */
|
|
if (allowbdot < 0)
|
|
allowbdot = rpmExpandNumeric("%{?_allow_deps_with_beginning_dot}");
|
|
if (!(xisalnum(r[0]) || r[0] == '_' || r[0] == '/' || (allowbdot && r[0] == '.'))) {
|
|
rpmError(RPMERR_BADSPEC,
|
|
_("line %d: Dependency tokens must begin with alpha-numeric, '_' or '/': %s\n"),
|
|
spec->lineNum, (spec->fileStack ? spec->line : field));
|
|
return RPMERR_BADSPEC;
|
|
}
|
|
|
|
/* Don't permit file names as args for certain tags */
|
|
switch (tag) {
|
|
case RPMTAG_OBSOLETEFLAGS:
|
|
case RPMTAG_CONFLICTFLAGS:
|
|
case RPMTAG_BUILDCONFLICTS:
|
|
if (r[0] == '/') {
|
|
rpmError(RPMERR_BADSPEC,_("line %d: File name not permitted: %s\n"),
|
|
spec->lineNum, (spec->fileStack ? spec->line : field));
|
|
return RPMERR_BADSPEC;
|
|
}
|
|
/*@switchbreak@*/ break;
|
|
default:
|
|
/*@switchbreak@*/ break;
|
|
}
|
|
|
|
re = r;
|
|
SKIPNONWHITE(re);
|
|
req = xmalloc((re-r) + 1);
|
|
strncpy(req, r, (re-r));
|
|
req[re-r] = '\0';
|
|
|
|
/* Names must not contain '%<=>' symbols - it's common error */
|
|
if (strpbrk (req, "%<=>")) {
|
|
rpmError(RPMERR_BADSPEC,
|
|
_("line %d: Dependency tokens must not contain '%%<=>' symbols: %s\n"),
|
|
spec->lineNum, (spec->fileStack ? spec->line : field));
|
|
return RPMERR_BADSPEC;
|
|
}
|
|
|
|
/* Parse version */
|
|
v = re;
|
|
SKIPWHITE(v);
|
|
ve = v;
|
|
SKIPNONWHITE(ve);
|
|
|
|
re = v; /* ==> next token (if no version found) starts here */
|
|
|
|
/* Check for possible logical operator */
|
|
if (ve > v) {
|
|
struct ReqComp *rc;
|
|
for (rc = ReqComparisons; rc->token != NULL; rc++) {
|
|
if ((ve-v) != strlen(rc->token) || strncmp(v, rc->token, (ve-v)))
|
|
/*@innercontinue@*/ continue;
|
|
|
|
flags |= rc->sense;
|
|
|
|
/* now parse version */
|
|
v = ve;
|
|
SKIPWHITE(v);
|
|
ve = v;
|
|
SKIPNONWHITE(ve);
|
|
/*@innerbreak@*/ break;
|
|
}
|
|
}
|
|
|
|
/*@-branchstate@*/
|
|
if (flags & RPMSENSE_SENSEMASK) {
|
|
if (*v == '\0' || ve == v) {
|
|
rpmError(RPMERR_BADSPEC, _("line %d: Version required: %s\n"),
|
|
spec->lineNum, (spec->fileStack ? spec->line : field));
|
|
return RPMERR_BADSPEC;
|
|
}
|
|
version = xmalloc((ve-v) + 1);
|
|
strncpy(version, v, (ve-v));
|
|
version[ve-v] = '\0';
|
|
re = ve; /* ==> next token after version string starts here */
|
|
} else
|
|
version = NULL;
|
|
/*@=branchstate@*/
|
|
|
|
/* Versions must not contain '%<=>' symbols - it's common error */
|
|
if (version && strpbrk (version, "%<=>")) {
|
|
rpmError(RPMERR_BADSPEC,
|
|
_("line %d: Dependency tokens must not contain '%%<=>' symbols: %s\n"),
|
|
spec->lineNum, (spec->fileStack ? spec->line : field));
|
|
req = _free(req);
|
|
version = _free(version);
|
|
return RPMERR_BADSPEC;
|
|
}
|
|
|
|
if (addReqProv(spec, h, flags, req, version, index) == 0) {
|
|
// Set-versions need their rpmlib feature.
|
|
if (version && strncmp(version, "set:", 4) == 0)
|
|
// Except for provides, to facilitate bootstrap.
|
|
if (tag != RPMTAG_PROVIDEFLAGS)
|
|
rpmlibNeedsFeature(h, "SetVersions", NULL);
|
|
}
|
|
|
|
req = _free(req);
|
|
version = _free(version);
|
|
|
|
}
|
|
|
|
return 0;
|
|
}
|