1757 lines
43 KiB
C
1757 lines
43 KiB
C
/*@-mods@*/
|
|
#include "system.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#if defined(__linux__) && defined(__powerpc__)
|
|
#include <setjmp.h>
|
|
#endif
|
|
|
|
#if HAVE_SYS_SYSTEMCFG_H
|
|
#include <sys/systemcfg.h>
|
|
#else
|
|
#define __power_pc() 0
|
|
#endif
|
|
|
|
#include "rpmlib.h"
|
|
#include "rpmmacro.h"
|
|
|
|
#include "misc.h"
|
|
#include "debug.h"
|
|
|
|
/*@access FD_t@*/ /* compared with NULL */
|
|
|
|
/*@observer@*/ /*@unchecked@*/
|
|
static const char *defrcfiles = LIBRPMRC_FILENAME ":/etc/rpmrc:~/.rpmrc";
|
|
|
|
/*@observer@*/ /*@checked@*/
|
|
const char * macrofiles = MACROFILES;
|
|
|
|
typedef /*@owned@*/ const char * cptr_t;
|
|
|
|
typedef struct machCacheEntry_s {
|
|
const char * name;
|
|
int count;
|
|
cptr_t * equivs;
|
|
int visited;
|
|
} * machCacheEntry;
|
|
|
|
typedef struct machCache_s {
|
|
machCacheEntry cache;
|
|
int size;
|
|
} * machCache;
|
|
|
|
typedef struct machEquivInfo_s {
|
|
const char * name;
|
|
int score;
|
|
} * machEquivInfo;
|
|
|
|
typedef struct machEquivTable_s {
|
|
int count;
|
|
machEquivInfo list;
|
|
} * machEquivTable;
|
|
|
|
struct rpmvarValue {
|
|
const char * value;
|
|
/* eventually, this arch will be replaced with a generic condition */
|
|
const char * arch;
|
|
/*@only@*/ /*@null@*/ struct rpmvarValue * next;
|
|
};
|
|
|
|
struct rpmOption {
|
|
const char * name;
|
|
int var;
|
|
int archSpecific;
|
|
/*@unused@*/ int required;
|
|
int macroize;
|
|
int localize;
|
|
/*@unused@*/ struct rpmOptionValue * value;
|
|
};
|
|
|
|
typedef struct defaultEntry_s {
|
|
/*@owned@*/ /*@null@*/ const char * name;
|
|
/*@owned@*/ /*@null@*/ const char * defName;
|
|
} * defaultEntry;
|
|
|
|
typedef struct canonEntry_s {
|
|
/*@owned@*/ const char * name;
|
|
/*@owned@*/ const char * short_name;
|
|
short num;
|
|
} * canonEntry;
|
|
|
|
/* tags are 'key'canon, 'key'translate, 'key'compat
|
|
*
|
|
* for giggles, 'key'_canon, 'key'_compat, and 'key'_canon will also work
|
|
*/
|
|
typedef struct tableType_s {
|
|
/*@observer@*/ const char * const key;
|
|
const int hasCanon;
|
|
const int hasTranslate;
|
|
struct machEquivTable_s equiv;
|
|
struct machCache_s cache;
|
|
defaultEntry defaults;
|
|
canonEntry canons;
|
|
int defaultsLength;
|
|
int canonsLength;
|
|
} * tableType;
|
|
|
|
/*@-fullinitblock@*/
|
|
/*@unchecked@*/
|
|
static struct tableType_s tables[RPM_MACHTABLE_COUNT] = {
|
|
{ "arch", 1, 0 },
|
|
{ "os", 1, 0 },
|
|
{ "buildarch", 0, 1 },
|
|
{ "buildos", 0, 1 }
|
|
};
|
|
|
|
/* this *must* be kept in alphabetical order */
|
|
/* The order of the flags is archSpecific, required, macroize, localize */
|
|
|
|
/*@unchecked@*/
|
|
static struct rpmOption optionTable[] = {
|
|
{ "include", RPMVAR_INCLUDE, 0, 1, 0, 2 },
|
|
{ "macrofiles", RPMVAR_MACROFILES, 0, 0, 0, 1 },
|
|
{ "optflags", RPMVAR_OPTFLAGS, 1, 0, 1, 0 },
|
|
{ "provides", RPMVAR_PROVIDES, 0, 0, 0, 0 },
|
|
};
|
|
/*@=fullinitblock@*/
|
|
|
|
/*@unchecked@*/
|
|
static int optionTableSize = sizeof(optionTable) / sizeof(*optionTable);
|
|
|
|
#define OS 0
|
|
#define ARCH 1
|
|
|
|
/*@unchecked@*/
|
|
static cptr_t current[2];
|
|
|
|
/*@unchecked@*/
|
|
static int currTables[2] = { RPM_MACHTABLE_INSTOS, RPM_MACHTABLE_INSTARCH };
|
|
|
|
/*@unchecked@*/
|
|
static struct rpmvarValue values[RPMVAR_NUM];
|
|
|
|
/*@unchecked@*/
|
|
static int defaultsInitialized = 0;
|
|
|
|
/* prototypes */
|
|
static int doReadRC( /*@killref@*/ FD_t fd, const char * urlfn)
|
|
/*@globals rpmGlobalMacroContext,
|
|
fileSystem, internalState @*/
|
|
/*@modifies fd, fileSystem, internalState @*/;
|
|
|
|
static void rpmSetVarArch(int var, const char * val,
|
|
/*@null@*/ const char * arch)
|
|
/*@globals internalState @*/
|
|
/*@modifies internalState @*/;
|
|
|
|
static void rebuildCompatTables(int type, const char * name)
|
|
/*@globals internalState @*/
|
|
/*@modifies internalState @*/;
|
|
|
|
static void rpmRebuildTargetVars(/*@null@*/ const char **target, /*@null@*/ const char ** canontarget)
|
|
/*@globals rpmGlobalMacroContext,
|
|
fileSystem, internalState @*/
|
|
/*@modifies *canontarget, fileSystem, internalState @*/;
|
|
|
|
static int optionCompare(const void * a, const void * b)
|
|
/*@*/
|
|
{
|
|
return xstrcasecmp(((struct rpmOption *) a)->name,
|
|
((struct rpmOption *) b)->name);
|
|
}
|
|
|
|
static /*@observer@*/ /*@null@*/ machCacheEntry
|
|
machCacheFindEntry(const machCache cache, const char * key)
|
|
/*@*/
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < cache->size; i++)
|
|
if (!strcmp(cache->cache[i].name, key)) return cache->cache + i;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static int machCompatCacheAdd(char * name, const char * fn, int linenum,
|
|
machCache cache)
|
|
/*@globals internalState @*/
|
|
/*@modifies *name, cache->cache, cache->size, internalState @*/
|
|
{
|
|
machCacheEntry entry = NULL;
|
|
char * chptr;
|
|
char * equivs;
|
|
int delEntry = 0;
|
|
int i;
|
|
|
|
while (*name && xisspace(*name)) name++;
|
|
|
|
chptr = name;
|
|
while (*chptr && *chptr != ':') chptr++;
|
|
if (!*chptr) {
|
|
rpmError(RPMERR_RPMRC, _("missing second ':' at %s:%d\n"), fn, linenum);
|
|
return 1;
|
|
} else if (chptr == name) {
|
|
rpmError(RPMERR_RPMRC, _("missing architecture name at %s:%d\n"), fn,
|
|
linenum);
|
|
return 1;
|
|
}
|
|
|
|
while (*chptr == ':' || xisspace(*chptr)) chptr--;
|
|
*(++chptr) = '\0';
|
|
equivs = chptr + 1;
|
|
while (*equivs && xisspace(*equivs)) equivs++;
|
|
if (!*equivs) {
|
|
delEntry = 1;
|
|
}
|
|
|
|
if (cache->size) {
|
|
entry = machCacheFindEntry(cache, name);
|
|
if (entry) {
|
|
for (i = 0; i < entry->count; i++)
|
|
entry->equivs[i] = _free(entry->equivs[i]);
|
|
entry->equivs = _free(entry->equivs);
|
|
entry->count = 0;
|
|
}
|
|
}
|
|
|
|
if (!entry) {
|
|
cache->cache = xrealloc(cache->cache,
|
|
(cache->size + 1) * sizeof(*cache->cache));
|
|
entry = cache->cache + cache->size++;
|
|
entry->name = xstrdup(name);
|
|
entry->count = 0;
|
|
entry->visited = 0;
|
|
}
|
|
|
|
if (delEntry) return 0;
|
|
|
|
while ((chptr = strtok(equivs, " ")) != NULL) {
|
|
equivs = NULL;
|
|
if (chptr[0] == '\0') /* does strtok() return "" ever?? */
|
|
continue;
|
|
if (entry->count)
|
|
entry->equivs = xrealloc(entry->equivs, sizeof(*entry->equivs)
|
|
* (entry->count + 1));
|
|
else
|
|
entry->equivs = xmalloc(sizeof(*entry->equivs));
|
|
|
|
entry->equivs[entry->count] = xstrdup(chptr);
|
|
entry->count++;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static /*@observer@*/ /*@null@*/ machEquivInfo
|
|
machEquivSearch(const machEquivTable table, const char * name)
|
|
/*@*/
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < table->count; i++)
|
|
if (!xstrcasecmp(table->list[i].name, name))
|
|
return table->list + i;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void machAddEquiv(machEquivTable table, const char * name,
|
|
int distance)
|
|
/*@modifies table->list, table->count @*/
|
|
{
|
|
machEquivInfo equiv;
|
|
|
|
equiv = machEquivSearch(table, name);
|
|
if (!equiv) {
|
|
if (table->count)
|
|
table->list = xrealloc(table->list, (table->count + 1)
|
|
* sizeof(*table->list));
|
|
else
|
|
table->list = xmalloc(sizeof(*table->list));
|
|
|
|
table->list[table->count].name = xstrdup(name);
|
|
table->list[table->count++].score = distance;
|
|
}
|
|
}
|
|
|
|
static void machCacheEntryVisit(machCache cache,
|
|
machEquivTable table, const char * name, int distance)
|
|
/*@modifies table->list, table->count @*/
|
|
{
|
|
machCacheEntry entry;
|
|
int i;
|
|
|
|
entry = machCacheFindEntry(cache, name);
|
|
if (!entry || entry->visited) return;
|
|
|
|
entry->visited = 1;
|
|
|
|
for (i = 0; i < entry->count; i++) {
|
|
machAddEquiv(table, entry->equivs[i], distance);
|
|
}
|
|
|
|
for (i = 0; i < entry->count; i++) {
|
|
machCacheEntryVisit(cache, table, entry->equivs[i], distance + 1);
|
|
}
|
|
}
|
|
|
|
static void machFindEquivs(machCache cache, machEquivTable table,
|
|
const char * key)
|
|
/*@modifies cache->cache, table->list, table->count @*/
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < cache->size; i++)
|
|
cache->cache[i].visited = 0;
|
|
|
|
while (table->count > 0) {
|
|
--table->count;
|
|
table->list[table->count].name = _free(table->list[table->count].name);
|
|
}
|
|
table->count = 0;
|
|
table->list = _free(table->list);
|
|
|
|
/*
|
|
* We have a general graph built using strings instead of pointers.
|
|
* Yuck. We have to start at a point at traverse it, remembering how
|
|
* far away everything is.
|
|
*/
|
|
/*@-nullstate@*/ /* FIX: table->list may be NULL. */
|
|
machAddEquiv(table, key, 1);
|
|
machCacheEntryVisit(cache, table, key, 2);
|
|
return;
|
|
/*@=nullstate@*/
|
|
}
|
|
|
|
static int addCanon(canonEntry * table, int * tableLen, char * line,
|
|
const char * fn, int lineNum)
|
|
/*@globals internalState @*/
|
|
/*@modifies *table, *tableLen, *line, internalState @*/
|
|
{
|
|
canonEntry t;
|
|
char *s, *s1;
|
|
const char * tname;
|
|
const char * tshort_name;
|
|
int tnum;
|
|
|
|
(*tableLen) += 2;
|
|
/*@-unqualifiedtrans@*/
|
|
*table = xrealloc(*table, sizeof(**table) * (*tableLen));
|
|
/*@=unqualifiedtrans@*/
|
|
|
|
t = & ((*table)[*tableLen - 2]);
|
|
|
|
tname = strtok(line, ": \t");
|
|
tshort_name = strtok(NULL, " \t");
|
|
s = strtok(NULL, " \t");
|
|
if (! (tname && tshort_name && s)) {
|
|
rpmError(RPMERR_RPMRC, _("Incomplete data line at %s:%d\n"),
|
|
fn, lineNum);
|
|
return RPMERR_RPMRC;
|
|
}
|
|
if (strtok(NULL, " \t")) {
|
|
rpmError(RPMERR_RPMRC, _("Too many args in data line at %s:%d\n"),
|
|
fn, lineNum);
|
|
return RPMERR_RPMRC;
|
|
}
|
|
|
|
/*@-nullpass@*/ /* LCL: s != NULL here. */
|
|
tnum = strtoul(s, &s1, 10);
|
|
if ((*s1) || (s1 == s) || (tnum == ULONG_MAX)) {
|
|
rpmError(RPMERR_RPMRC, _("Bad arch/os number: %s (%s:%d)\n"), s,
|
|
fn, lineNum);
|
|
return(RPMERR_RPMRC);
|
|
}
|
|
/*@=nullpass@*/
|
|
|
|
t[0].name = xstrdup(tname);
|
|
t[0].short_name = (tshort_name ? xstrdup(tshort_name) : xstrdup(""));
|
|
t[0].num = tnum;
|
|
|
|
/* From A B C entry */
|
|
/* Add B B C entry */
|
|
t[1].name = (tshort_name ? xstrdup(tshort_name) : xstrdup(""));
|
|
t[1].short_name = (tshort_name ? xstrdup(tshort_name) : xstrdup(""));
|
|
t[1].num = tnum;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int addDefault(defaultEntry * table, int * tableLen, char * line,
|
|
const char * fn, int lineNum)
|
|
/*@globals internalState @*/
|
|
/*@modifies *table, *tableLen, *line, internalState @*/
|
|
{
|
|
defaultEntry t;
|
|
|
|
(*tableLen)++;
|
|
/*@-unqualifiedtrans@*/
|
|
*table = xrealloc(*table, sizeof(**table) * (*tableLen));
|
|
/*@=unqualifiedtrans@*/
|
|
|
|
t = & ((*table)[*tableLen - 1]);
|
|
|
|
/*@-temptrans@*/
|
|
t->name = strtok(line, ": \t");
|
|
t->defName = strtok(NULL, " \t");
|
|
if (! (t->name && t->defName)) {
|
|
rpmError(RPMERR_RPMRC, _("Incomplete default line at %s:%d\n"),
|
|
fn, lineNum);
|
|
return RPMERR_RPMRC;
|
|
}
|
|
if (strtok(NULL, " \t")) {
|
|
rpmError(RPMERR_RPMRC, _("Too many args in default line at %s:%d\n"),
|
|
fn, lineNum);
|
|
return RPMERR_RPMRC;
|
|
}
|
|
|
|
t->name = xstrdup(t->name);
|
|
t->defName = (t->defName ? xstrdup(t->defName) : NULL);
|
|
/*@=temptrans@*/
|
|
|
|
return 0;
|
|
}
|
|
|
|
static /*@null@*/ canonEntry lookupInCanonTable(const char * name,
|
|
const canonEntry table, int tableLen)
|
|
/*@*/
|
|
{
|
|
while (tableLen) {
|
|
tableLen--;
|
|
if (strcmp(name, table[tableLen].name))
|
|
continue;
|
|
/*@-immediatetrans -retalias@*/
|
|
return &(table[tableLen]);
|
|
/*@=immediatetrans =retalias@*/
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static /*@observer@*/ /*@null@*/
|
|
const char * lookupInDefaultTable(const char * name,
|
|
const defaultEntry table, int tableLen)
|
|
/*@*/
|
|
{
|
|
while (tableLen) {
|
|
tableLen--;
|
|
if (table[tableLen].name && !strcmp(name, table[tableLen].name))
|
|
return table[tableLen].defName;
|
|
}
|
|
|
|
return name;
|
|
}
|
|
|
|
static void setVarDefault(int var, const char * macroname, const char * val,
|
|
/*@null@*/ const char * body)
|
|
/*@globals rpmGlobalMacroContext,
|
|
internalState @*/
|
|
/*@modifies internalState @*/
|
|
{
|
|
if (var >= 0) { /* XXX Dying ... */
|
|
if (rpmGetVar(var)) return;
|
|
rpmSetVar(var, val);
|
|
}
|
|
if (body == NULL)
|
|
body = val;
|
|
addMacro(NULL, macroname, NULL, body, RMIL_DEFAULT);
|
|
}
|
|
|
|
static void setVar( const char *macroname, const char *body )
|
|
/*@globals rpmGlobalMacroContext,
|
|
internalState @*/
|
|
/*@modifies internalState @*/
|
|
{
|
|
if ( macroname && body )
|
|
addMacro( NULL, macroname, NULL, body, RMIL_DEFAULT );
|
|
}
|
|
|
|
/*@observer@*/ /*@unchecked@*/
|
|
static const char * prescriptenviron = "\n\
|
|
RPM_SOURCE_DIR=\"%{_sourcedir}\"\n\
|
|
RPM_BUILD_DIR=\"%{_builddir}\"\n\
|
|
RPM_DOC_DIR=\"%{_docdir}\"\n\
|
|
export RPM_SOURCE_DIR RPM_BUILD_DIR RPM_DOC_DIR\n\
|
|
RPM_OPT_FLAGS=\"%{optflags}\"\n\
|
|
RPM_ARCH=\"%{_arch}\"\n\
|
|
RPM_OS=\"%{_os}\"\n\
|
|
export RPM_OPT_FLAGS RPM_ARCH RPM_OS\n\
|
|
RPM_PACKAGE_NAME=\"%{name}\"\n\
|
|
RPM_PACKAGE_VERSION=\"%{version}\"\n\
|
|
RPM_PACKAGE_RELEASE=\"%{release}\"\n\
|
|
export RPM_PACKAGE_NAME RPM_PACKAGE_VERSION RPM_PACKAGE_RELEASE\n\
|
|
%{?buildroot:RPM_BUILD_ROOT=\"%{buildroot}\"\n\
|
|
export RPM_BUILD_ROOT\n}\
|
|
";
|
|
|
|
static void rpmSetDefaults(void)
|
|
/*@globals rpmGlobalMacroContext,
|
|
internalState @*/
|
|
/*@modifies internalState @*/
|
|
{
|
|
if ( defaultsInitialized )
|
|
return;
|
|
else
|
|
{
|
|
struct passwd *pw = getpwuid( geteuid() );
|
|
|
|
setVar( "_usr", "/usr" );
|
|
setVar( "_var", "/var" );
|
|
setVar( "_preScriptEnvironment", prescriptenviron );
|
|
setVar( "_username", pw ? pw->pw_name : 0 );
|
|
setVar( "_homedir", pw ? pw->pw_dir : 0 );
|
|
|
|
setVar( "_topdir", "%{_usr}/src/RPM" );
|
|
setVar( "_tmppath", "%{_var}/tmp" );
|
|
setVar( "_dbpath", "%{_var}/lib/rpm" );
|
|
setVar( "_defaultdocdir", "%{_usr}/share/doc" );
|
|
|
|
setVar( "_rpmfilename", "%%{ARCH}/%%{NAME}-%%{VERSION}-%%{RELEASE}.%%{ARCH}.rpm" );
|
|
|
|
setVar( "_signature", "none" );
|
|
setVar( "_buildshell", "/bin/sh" );
|
|
|
|
setVar( "_topsrcdir", "%{_topdir}" );
|
|
setVar( "_builddir", "%{_topdir}/BUILD" );
|
|
setVar( "_rpmdir", "%{_topdir}/RPMS" );
|
|
setVar( "_srcrpmdir", "%{_topdir}/SRPMS" );
|
|
setVar( "_sourcedir", "%{_topsrcdir}/SOURCES" );
|
|
setVar( "_specdir", "%{_topsrcdir}/SPECS" );
|
|
|
|
setVarDefault( RPMVAR_OPTFLAGS, "optflags", "-O2", NULL );
|
|
|
|
defaultsInitialized = 1;
|
|
}
|
|
}
|
|
|
|
/*@-usedef@*/ /*@ FIX: se usage inconsistent, W2DO? */
|
|
static int doReadRC( /*@killref@*/ FD_t fd, const char * urlfn)
|
|
/*@globals rpmGlobalMacroContext,
|
|
fileSystem, internalState @*/
|
|
/*@modifies fd, fileSystem, internalState @*/
|
|
{
|
|
const char *s;
|
|
char *se, *next;
|
|
int linenum = 0;
|
|
struct rpmOption searchOption, * option;
|
|
int rc;
|
|
|
|
/* XXX really need rc = Slurp(fd, const char * filename, char ** buf) */
|
|
{ off_t size = fdSize(fd);
|
|
size_t nb = (size >= 0 ? size : (8*BUFSIZ - 2));
|
|
if (nb == 0) {
|
|
(void) Fclose(fd);
|
|
return 0;
|
|
}
|
|
next = alloca(nb + 2);
|
|
next[0] = '\0';
|
|
rc = Fread(next, sizeof(*next), nb, fd);
|
|
if (Ferror(fd) || (size > 0 && rc != nb)) { /* XXX Feof(fd) */
|
|
rpmError(RPMERR_RPMRC, _("Failed to read %s: %s.\n"), urlfn,
|
|
Fstrerror(fd));
|
|
rc = 1;
|
|
} else
|
|
rc = 0;
|
|
(void) Fclose(fd);
|
|
if (rc) return rc;
|
|
next[nb] = '\n';
|
|
next[nb + 1] = '\0';
|
|
}
|
|
|
|
/*@-branchstate@*/
|
|
while (*next != '\0') {
|
|
linenum++;
|
|
|
|
s = se = next;
|
|
|
|
/* Find end-of-line. */
|
|
while (*se && *se != '\n') se++;
|
|
if (*se != '\0') *se++ = '\0';
|
|
next = se;
|
|
|
|
/* Trim leading spaces */
|
|
while (*s && xisspace(*s)) s++;
|
|
|
|
/* We used to allow comments to begin anywhere, but not anymore. */
|
|
if (*s == '#' || *s == '\0') continue;
|
|
|
|
/* Find end-of-keyword. */
|
|
se = (char *)s;
|
|
while (*se && !xisspace(*se) && *se != ':') se++;
|
|
|
|
if (xisspace(*se)) {
|
|
*se++ = '\0';
|
|
while (*se && xisspace(*se) && *se != ':') se++;
|
|
}
|
|
|
|
if (*se != ':') {
|
|
rpmError(RPMERR_RPMRC, _("missing ':' (found 0x%02x) at %s:%d\n"),
|
|
(unsigned)(0xff & *se), urlfn, linenum);
|
|
return 1;
|
|
}
|
|
*se++ = '\0'; /* terminate keyword or option, point to value */
|
|
while (*se && xisspace(*se)) se++;
|
|
|
|
/* Find keyword in table */
|
|
searchOption.name = s;
|
|
option = bsearch(&searchOption, optionTable, optionTableSize,
|
|
sizeof(optionTable[0]), optionCompare);
|
|
|
|
if (option) { /* For configuration variables ... */
|
|
const char *arch, *val, *fn;
|
|
|
|
arch = val = fn = NULL;
|
|
if (*se == '\0') {
|
|
rpmError(RPMERR_RPMRC, _("missing argument for %s at %s:%d\n"),
|
|
option->name, urlfn, linenum);
|
|
return 1;
|
|
}
|
|
|
|
switch (option->var) {
|
|
case RPMVAR_INCLUDE:
|
|
{ FD_t fdinc;
|
|
|
|
s = se;
|
|
while (*se && !xisspace(*se)) se++;
|
|
if (*se != '\0') *se++ = '\0';
|
|
|
|
rpmRebuildTargetVars(NULL, NULL);
|
|
|
|
fn = rpmGetPath(s, NULL);
|
|
if (fn == NULL || *fn == '\0') {
|
|
rpmError(RPMERR_RPMRC, _("%s expansion failed at %s:%d \"%s\"\n"),
|
|
option->name, urlfn, linenum, s);
|
|
fn = _free(fn);
|
|
return 1;
|
|
/*@notreached@*/
|
|
}
|
|
|
|
fdinc = Fopen(fn, "r.fpio");
|
|
if (fdinc == NULL || Ferror(fdinc)) {
|
|
rpmError(RPMERR_RPMRC, _("cannot open %s at %s:%d: %s\n"),
|
|
fn, urlfn, linenum, Fstrerror(fdinc));
|
|
rc = 1;
|
|
} else {
|
|
rc = doReadRC(fdinc, fn);
|
|
}
|
|
fn = _free(fn);
|
|
if (rc) return rc;
|
|
continue; /* XXX don't save include value as var/macro */
|
|
} /*@notreached@*/ /*@switchbreak@*/ break;
|
|
case RPMVAR_MACROFILES:
|
|
fn = rpmGetPath(se, NULL);
|
|
if (fn == NULL || *fn == '\0') {
|
|
rpmError(RPMERR_RPMRC, _("%s expansion failed at %s:%d \"%s\"\n"),
|
|
option->name, urlfn, linenum, fn);
|
|
fn = _free(fn);
|
|
return 1;
|
|
}
|
|
se = (char *)fn;
|
|
/*@switchbreak@*/ break;
|
|
case RPMVAR_PROVIDES:
|
|
{ char *t;
|
|
s = rpmGetVar(RPMVAR_PROVIDES);
|
|
if (s == NULL) s = "";
|
|
fn = t = xmalloc(strlen(s) + strlen(se) + 2);
|
|
while (*s != '\0') *t++ = *s++;
|
|
*t++ = ' ';
|
|
while (*se != '\0') *t++ = *se++;
|
|
*t++ = '\0';
|
|
se = (char *)fn;
|
|
} /*@switchbreak@*/ break;
|
|
default:
|
|
/*@switchbreak@*/ break;
|
|
}
|
|
|
|
if (option->archSpecific) {
|
|
arch = se;
|
|
while (*se && !xisspace(*se)) se++;
|
|
if (*se == '\0') {
|
|
rpmError(RPMERR_RPMRC,
|
|
_("missing architecture for %s at %s:%d\n"),
|
|
option->name, urlfn, linenum);
|
|
return 1;
|
|
}
|
|
*se++ = '\0';
|
|
while (*se && xisspace(*se)) se++;
|
|
if (*se == '\0') {
|
|
rpmError(RPMERR_RPMRC,
|
|
_("missing argument for %s at %s:%d\n"),
|
|
option->name, urlfn, linenum);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
val = se;
|
|
|
|
/* Only add macros if appropriate for this arch */
|
|
if (option->macroize &&
|
|
(arch == NULL || !strcmp(arch, current[ARCH]))) {
|
|
char *n, *name;
|
|
n = name = xmalloc(strlen(option->name)+2);
|
|
if (option->localize)
|
|
*n++ = '_';
|
|
strcpy(n, option->name);
|
|
addMacro(NULL, name, NULL, val, RMIL_RPMRC);
|
|
free(name);
|
|
}
|
|
rpmSetVarArch(option->var, val, arch);
|
|
fn = _free(fn);
|
|
|
|
} else { /* For arch/os compatibilty tables ... */
|
|
int gotit;
|
|
int i;
|
|
|
|
gotit = 0;
|
|
|
|
for (i = 0; i < RPM_MACHTABLE_COUNT; i++) {
|
|
if (!strncmp(tables[i].key, s, strlen(tables[i].key)))
|
|
/*@innerbreak@*/ break;
|
|
}
|
|
|
|
if (i < RPM_MACHTABLE_COUNT) {
|
|
const char *rest = s + strlen(tables[i].key);
|
|
if (*rest == '_') rest++;
|
|
|
|
if (!strcmp(rest, "compat")) {
|
|
if (machCompatCacheAdd(se, urlfn, linenum,
|
|
&tables[i].cache))
|
|
return 1;
|
|
gotit = 1;
|
|
} else if (tables[i].hasTranslate &&
|
|
!strcmp(rest, "translate")) {
|
|
if (addDefault(&tables[i].defaults,
|
|
&tables[i].defaultsLength,
|
|
se, urlfn, linenum))
|
|
return 1;
|
|
gotit = 1;
|
|
} else if (tables[i].hasCanon &&
|
|
!strcmp(rest, "canon")) {
|
|
if (addCanon(&tables[i].canons, &tables[i].canonsLength,
|
|
se, urlfn, linenum))
|
|
return 1;
|
|
gotit = 1;
|
|
}
|
|
}
|
|
|
|
if (!gotit) {
|
|
rpmError(RPMERR_RPMRC, _("bad option '%s' at %s:%d\n"),
|
|
s, urlfn, linenum);
|
|
}
|
|
}
|
|
}
|
|
/*@=branchstate@*/
|
|
|
|
return 0;
|
|
}
|
|
/*@=usedef@*/
|
|
|
|
# if defined(__linux__) && defined(__i386__)
|
|
#include <setjmp.h>
|
|
#include <signal.h>
|
|
|
|
/*
|
|
* Generic CPUID function
|
|
*/
|
|
static inline void cpuid(int op, int *eax, int *ebx, int *ecx, int *edx)
|
|
/*@modifies *eax, *ebx, *ecx, *edx @*/
|
|
{
|
|
#ifdef __LCLINT__
|
|
*eax = *ebx = *ecx = *edx = 0;
|
|
#endif
|
|
#ifdef PIC
|
|
__asm__("pushl %%ebx; cpuid; movl %%ebx,%1; popl %%ebx"
|
|
: "=a"(*eax), "=g"(*ebx), "=&c"(*ecx), "=&d"(*edx)
|
|
: "a" (op));
|
|
#else
|
|
__asm__("cpuid"
|
|
: "=a" (*eax), "=b" (*ebx), "=c" (*ecx), "=d" (*edx)
|
|
: "a" (op));
|
|
#endif
|
|
|
|
}
|
|
|
|
/*
|
|
* CPUID functions returning a single datum
|
|
*/
|
|
static inline unsigned int cpuid_eax(unsigned int op)
|
|
/*@*/
|
|
{
|
|
unsigned int val;
|
|
|
|
#ifdef PIC
|
|
__asm__("pushl %%ebx; cpuid; popl %%ebx"
|
|
: "=a" (val) : "a" (op) : "ecx", "edx");
|
|
#else
|
|
__asm__("cpuid"
|
|
: "=a" (val) : "a" (op) : "ebx", "ecx", "edx");
|
|
#endif
|
|
return val;
|
|
}
|
|
|
|
static inline unsigned int cpuid_ebx(unsigned int op)
|
|
/*@*/
|
|
{
|
|
unsigned int tmp, val;
|
|
|
|
#ifdef PIC
|
|
__asm__("pushl %%ebx; cpuid; movl %%ebx,%1; popl %%ebx"
|
|
: "=a" (tmp), "=g" (val) : "a" (op) : "ecx", "edx");
|
|
#else
|
|
__asm__("cpuid"
|
|
: "=a" (tmp), "=b" (val) : "a" (op) : "ecx", "edx");
|
|
#endif
|
|
return val;
|
|
}
|
|
|
|
static inline unsigned int cpuid_ecx(unsigned int op)
|
|
/*@*/
|
|
{
|
|
unsigned int tmp, val;
|
|
#ifdef PIC
|
|
__asm__("pushl %%ebx; cpuid; popl %%ebx"
|
|
: "=a" (tmp), "=c" (val) : "a" (op) : "edx");
|
|
#else
|
|
__asm__("cpuid"
|
|
: "=a" (tmp), "=c" (val) : "a" (op) : "ebx", "edx");
|
|
#endif
|
|
return val;
|
|
|
|
}
|
|
|
|
static inline unsigned int cpuid_edx(unsigned int op)
|
|
/*@*/
|
|
{
|
|
unsigned int tmp, val;
|
|
#ifdef PIC
|
|
__asm__("pushl %%ebx; cpuid; popl %%ebx"
|
|
: "=a" (tmp), "=d" (val) : "a" (op) : "ecx");
|
|
#else
|
|
__asm__("cpuid"
|
|
: "=a" (tmp), "=d" (val) : "a" (op) : "ebx", "ecx");
|
|
#endif
|
|
return val;
|
|
|
|
}
|
|
|
|
/*@unchecked@*/
|
|
static sigjmp_buf jenv;
|
|
|
|
static inline void model3(int _unused)
|
|
/*@globals internalState @*/
|
|
/*@modifies internalState @*/
|
|
{
|
|
siglongjmp(jenv, 1);
|
|
}
|
|
|
|
static inline int RPMClass(void)
|
|
/*@globals internalState @*/
|
|
/*@modifies internalState @*/
|
|
{
|
|
int cpu;
|
|
unsigned int tfms, junk, cap;
|
|
|
|
signal(SIGILL, model3);
|
|
|
|
if(sigsetjmp(jenv, 1))
|
|
return 3;
|
|
|
|
if(cpuid_eax(0x000000000)==0)
|
|
return 4;
|
|
cpuid(0x000000001, &tfms, &junk, &junk, &cap);
|
|
|
|
cpu = (tfms>>8)&15;
|
|
|
|
if(cpu < 6)
|
|
return cpu;
|
|
|
|
if(cap & (1<<15))
|
|
return 6;
|
|
|
|
return 5;
|
|
}
|
|
|
|
/* should only be called for model 6 CPU's */
|
|
static int is_athlon(void)
|
|
/*@*/
|
|
{
|
|
unsigned int eax, ebx, ecx, edx;
|
|
char vendor[16];
|
|
int i;
|
|
|
|
cpuid (0, &eax, &ebx, &ecx, &edx);
|
|
|
|
/* If you care about space, you can just check ebx, ecx and edx directly
|
|
instead of forming a string first and then doing a strcmp */
|
|
memset(vendor, 0, sizeof(vendor));
|
|
|
|
for (i=0; i<4; i++)
|
|
vendor[i] = (unsigned char) (ebx >>(8*i));
|
|
for (i=0; i<4; i++)
|
|
vendor[4+i] = (unsigned char) (edx >>(8*i));
|
|
for (i=0; i<4; i++)
|
|
vendor[8+i] = (unsigned char) (ecx >>(8*i));
|
|
|
|
if (strcmp(vendor, "AuthenticAMD") != 0)
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined(__linux__) && defined(__powerpc__)
|
|
static jmp_buf mfspr_jmpbuf;
|
|
|
|
static void mfspr_ill(int notused)
|
|
{
|
|
longjmp(mfspr_jmpbuf, -1);
|
|
}
|
|
#endif
|
|
|
|
static const char *checkAMD( void )
|
|
{
|
|
int fd = open( "/proc/cpuinfo", O_RDONLY );
|
|
if ( !fd )
|
|
return 0;
|
|
else
|
|
{
|
|
char buffer[ 1 + getpagesize() ];
|
|
|
|
memset( buffer, 0, sizeof buffer );
|
|
read( fd, buffer, sizeof buffer - 1 );
|
|
close( fd );
|
|
|
|
if ( !strstr( buffer, "AMD" ) )
|
|
return 0;
|
|
|
|
if ( strstr( buffer, "Athlon" ) || strstr( buffer, "Duron" ) )
|
|
return "athlon";
|
|
|
|
if ( strstr( buffer, "K6" ) )
|
|
return "k6";
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static void defaultMachine(/*@out@*/ const char ** arch,
|
|
/*@out@*/ const char ** os)
|
|
/*@globals fileSystem@*/
|
|
/*@modifies *arch, *os, fileSystem @*/
|
|
{
|
|
static struct utsname un;
|
|
static int gotDefaults = 0;
|
|
char * chptr;
|
|
canonEntry canon;
|
|
int rc;
|
|
|
|
if (!gotDefaults) {
|
|
rc = uname(&un);
|
|
if (rc < 0) return;
|
|
|
|
#if !defined(__linux__)
|
|
#ifdef SNI
|
|
/* USUALLY un.sysname on sinix does start with the word "SINIX"
|
|
* let's be absolutely sure
|
|
*/
|
|
strncpy(un.sysname, "SINIX", sizeof(un.sysname));
|
|
#endif
|
|
/*@-nullpass@*/
|
|
if (!strcmp(un.sysname, "AIX")) {
|
|
strcpy(un.machine, __power_pc() ? "ppc" : "rs6000");
|
|
sprintf(un.sysname,"aix%s.%s", un.version, un.release);
|
|
}
|
|
else if (!strcmp(un.sysname, "SunOS")) {
|
|
if (!strncmp(un.release,"4", 1)) /* SunOS 4.x */ {
|
|
int fd;
|
|
for (fd = 0;
|
|
(un.release[fd] != 0 && (fd < sizeof(un.release)));
|
|
fd++) {
|
|
if (!xisdigit(un.release[fd]) && (un.release[fd] != '.')) {
|
|
un.release[fd] = 0;
|
|
break;
|
|
}
|
|
}
|
|
sprintf(un.sysname,"sunos%s",un.release);
|
|
}
|
|
|
|
else /* Solaris 2.x: n.x.x becomes n-3.x.x */
|
|
sprintf(un.sysname, "solaris%1d%s", atoi(un.release)-3,
|
|
un.release+1+(atoi(un.release)/10));
|
|
|
|
/* Solaris on Intel hardware reports i86pc instead of i386
|
|
* (at least on 2.6 and 2.8)
|
|
*/
|
|
if (!strcmp(un.machine, "i86pc"))
|
|
sprintf(un.machine, "i386");
|
|
}
|
|
else if (!strcmp(un.sysname, "HP-UX"))
|
|
/*make un.sysname look like hpux9.05 for example*/
|
|
sprintf(un.sysname, "hpux%s", strpbrk(un.release, "123456789"));
|
|
else if (!strcmp(un.sysname, "OSF1"))
|
|
/*make un.sysname look like osf3.2 for example*/
|
|
sprintf(un.sysname, "osf%s", strpbrk(un.release, "123456789"));
|
|
else if (!strncmp(un.sysname, "IP", 2))
|
|
un.sysname[2] = '\0';
|
|
else if (!strncmp(un.sysname, "SINIX", 5)) {
|
|
sprintf(un.sysname, "sinix%s",un.release);
|
|
if (!strncmp(un.machine, "RM", 2))
|
|
sprintf(un.machine, "mips");
|
|
}
|
|
else if ((!strncmp(un.machine, "34", 2) ||
|
|
!strncmp(un.machine, "33", 2)) && \
|
|
!strncmp(un.release, "4.0", 3))
|
|
{
|
|
/* we are on ncr-sysv4 */
|
|
char * prelid = NULL;
|
|
FD_t fd = Fopen("/etc/.relid", "r.fdio");
|
|
int gotit = 0;
|
|
/*@-branchstate@*/
|
|
if (fd != NULL && !Ferror(fd)) {
|
|
chptr = xcalloc(1, 256);
|
|
{ int irelid = Fread(chptr, sizeof(*chptr), 256, fd);
|
|
(void) Fclose(fd);
|
|
/* example: "112393 RELEASE 020200 Version 01 OS" */
|
|
if (irelid > 0) {
|
|
if ((prelid = strstr(chptr, "RELEASE "))){
|
|
prelid += strlen("RELEASE ")+1;
|
|
sprintf(un.sysname,"ncr-sysv4.%.*s",1,prelid);
|
|
gotit = 1;
|
|
}
|
|
}
|
|
}
|
|
chptr = _free (chptr);
|
|
}
|
|
/*@=branchstate@*/
|
|
if (!gotit) /* parsing /etc/.relid file failed? */
|
|
strcpy(un.sysname,"ncr-sysv4");
|
|
/* wrong, just for now, find out how to look for i586 later*/
|
|
strcpy(un.machine,"i486");
|
|
}
|
|
/*@=nullpass@*/
|
|
#endif /* __linux__ */
|
|
|
|
/* get rid of the hyphens in the sysname */
|
|
for (chptr = un.machine; *chptr != '\0'; chptr++)
|
|
if (*chptr == '/') *chptr = '-';
|
|
|
|
# if defined(__MIPSEL__) || defined(__MIPSEL) || defined(_MIPSEL)
|
|
/* little endian */
|
|
strcpy(un.machine, "mipsel");
|
|
# elif defined(__MIPSEB__) || defined(__MIPSEB) || defined(_MIPSEB)
|
|
/* big endian */
|
|
strcpy(un.machine, "mips");
|
|
# endif
|
|
|
|
# if defined(__hpux) && defined(_SC_CPU_VERSION)
|
|
{
|
|
# if !defined(CPU_PA_RISC1_2)
|
|
# define CPU_PA_RISC1_2 0x211 /* HP PA-RISC1.2 */
|
|
# endif
|
|
# if !defined(CPU_PA_RISC2_0)
|
|
# define CPU_PA_RISC2_0 0x214 /* HP PA-RISC2.0 */
|
|
# endif
|
|
int cpu_version = sysconf(_SC_CPU_VERSION);
|
|
|
|
# if defined(CPU_HP_MC68020)
|
|
if (cpu_version == CPU_HP_MC68020)
|
|
strcpy(un.machine, "m68k");
|
|
# endif
|
|
# if defined(CPU_HP_MC68030)
|
|
if (cpu_version == CPU_HP_MC68030)
|
|
strcpy(un.machine, "m68k");
|
|
# endif
|
|
# if defined(CPU_HP_MC68040)
|
|
if (cpu_version == CPU_HP_MC68040)
|
|
strcpy(un.machine, "m68k");
|
|
# endif
|
|
|
|
# if defined(CPU_PA_RISC1_0)
|
|
if (cpu_version == CPU_PA_RISC1_0)
|
|
strcpy(un.machine, "hppa1.0");
|
|
# endif
|
|
# if defined(CPU_PA_RISC1_1)
|
|
if (cpu_version == CPU_PA_RISC1_1)
|
|
strcpy(un.machine, "hppa1.1");
|
|
# endif
|
|
# if defined(CPU_PA_RISC1_2)
|
|
if (cpu_version == CPU_PA_RISC1_2)
|
|
strcpy(un.machine, "hppa1.2");
|
|
# endif
|
|
# if defined(CPU_PA_RISC2_0)
|
|
if (cpu_version == CPU_PA_RISC2_0)
|
|
strcpy(un.machine, "hppa2.0");
|
|
# endif
|
|
}
|
|
# endif /* hpux */
|
|
|
|
# if HAVE_PERSONALITY && defined(__linux__) && defined(__sparc__)
|
|
if (!strcmp(un.machine, "sparc")) {
|
|
#define PERS_LINUX 0x00000000
|
|
#define PERS_LINUX_32BIT 0x00800000
|
|
#define PERS_LINUX32 0x00000008
|
|
|
|
extern int personality(unsigned long);
|
|
int oldpers;
|
|
|
|
oldpers = personality(PERS_LINUX_32BIT);
|
|
if (oldpers != -1) {
|
|
if (personality(PERS_LINUX) != -1) {
|
|
uname(&un);
|
|
if (! strcmp(un.machine, "sparc64")) {
|
|
strcpy(un.machine, "sparcv9");
|
|
oldpers = PERS_LINUX32;
|
|
}
|
|
}
|
|
personality(oldpers);
|
|
}
|
|
}
|
|
# endif /* sparc*-linux */
|
|
|
|
# if defined(__GNUC__) && defined(__alpha__)
|
|
{
|
|
unsigned long amask, implver;
|
|
register long v0 __asm__("$0") = -1;
|
|
__asm__ (".long 0x47e00c20" : "=r"(v0) : "0"(v0));
|
|
amask = ~v0;
|
|
__asm__ (".long 0x47e03d80" : "=r"(v0));
|
|
implver = v0;
|
|
switch (implver) {
|
|
case 1:
|
|
switch (amask) {
|
|
case 0: strcpy(un.machine, "alphaev5"); break;
|
|
case 1: strcpy(un.machine, "alphaev56"); break;
|
|
case 0x101: strcpy(un.machine, "alphapca56"); break;
|
|
}
|
|
break;
|
|
case 2:
|
|
switch (amask) {
|
|
case 0x303: strcpy(un.machine, "alphaev6"); break;
|
|
case 0x307: strcpy(un.machine, "alphaev67"); break;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
# endif
|
|
|
|
# if defined(__linux__) && defined(__i386__)
|
|
#if 0
|
|
{
|
|
char class = (char) (RPMClass() | '0');
|
|
|
|
if (class == '6' && is_athlon())
|
|
strcpy(un.machine, "athlon");
|
|
else if (strchr("3456", un.machine[1]) && un.machine[1] != class)
|
|
un.machine[1] = class;
|
|
}
|
|
#endif
|
|
|
|
if ( !strcmp( un.machine, "i586" ) || !strcmp( un.machine, "i686" ) )
|
|
{
|
|
const char *amd = checkAMD();
|
|
if ( amd )
|
|
strcpy( un.machine, amd );
|
|
}
|
|
|
|
# endif
|
|
|
|
# if defined(__linux__) && defined(__powerpc__)
|
|
{
|
|
unsigned pvr = 0;
|
|
__sighandler_t oldh = signal(SIGILL, mfspr_ill);
|
|
if (setjmp(mfspr_jmpbuf) == 0) {
|
|
__asm__ __volatile__ ("mfspr %0, 287" : "=r" (pvr));
|
|
}
|
|
signal(SIGILL, oldh);
|
|
|
|
if ( pvr ) {
|
|
pvr >>= 16;
|
|
if ( pvr >= 0x40)
|
|
strcpy(un.machine, "ppcpseries");
|
|
else if ( (pvr == 0x36) || (pvr == 0x37) )
|
|
strcpy(un.machine, "ppciseries");
|
|
else
|
|
strcpy(un.machine, "ppc");
|
|
}
|
|
}
|
|
# endif
|
|
|
|
/* the uname() result goes through the arch_canon table */
|
|
canon = lookupInCanonTable(un.machine,
|
|
tables[RPM_MACHTABLE_INSTARCH].canons,
|
|
tables[RPM_MACHTABLE_INSTARCH].canonsLength);
|
|
if (canon)
|
|
strcpy(un.machine, canon->short_name);
|
|
|
|
canon = lookupInCanonTable(un.sysname,
|
|
tables[RPM_MACHTABLE_INSTOS].canons,
|
|
tables[RPM_MACHTABLE_INSTOS].canonsLength);
|
|
if (canon)
|
|
strcpy(un.sysname, canon->short_name);
|
|
gotDefaults = 1;
|
|
}
|
|
|
|
if (arch) *arch = un.machine;
|
|
if (os) *os = un.sysname;
|
|
}
|
|
|
|
static /*@observer@*/ /*@null@*/
|
|
const char * rpmGetVarArch(int var, /*@null@*/ const char * arch)
|
|
/*@*/
|
|
{
|
|
const struct rpmvarValue * next;
|
|
|
|
if (arch == NULL) arch = current[ARCH];
|
|
|
|
if (arch) {
|
|
next = &values[var];
|
|
while (next) {
|
|
if (next->arch && !strcmp(next->arch, arch)) return next->value;
|
|
next = next->next;
|
|
}
|
|
}
|
|
|
|
next = values + var;
|
|
while (next && next->arch) next = next->next;
|
|
|
|
return next ? next->value : NULL;
|
|
}
|
|
|
|
const char *rpmGetVar(int var)
|
|
{
|
|
return rpmGetVarArch(var, NULL);
|
|
}
|
|
|
|
/* this doesn't free the passed pointer! */
|
|
static void freeRpmVar(/*@only@*/ struct rpmvarValue * orig)
|
|
/*@modifies *orig @*/
|
|
{
|
|
struct rpmvarValue * next, * var = orig;
|
|
|
|
while (var) {
|
|
next = var->next;
|
|
var->arch = _free(var->arch);
|
|
var->value = _free(var->value);
|
|
|
|
/*@-branchstate@*/
|
|
if (var != orig) var = _free(var);
|
|
/*@=branchstate@*/
|
|
var = next;
|
|
}
|
|
}
|
|
|
|
void rpmSetVar(int var, const char * val)
|
|
{
|
|
/*@-immediatetrans@*/
|
|
freeRpmVar(&values[var]);
|
|
/*@=immediatetrans@*/
|
|
values[var].value = (val ? xstrdup(val) : NULL);
|
|
}
|
|
|
|
static void rpmSetVarArch(int var, const char * val, const char * arch)
|
|
/*@*/
|
|
{
|
|
struct rpmvarValue * next = values + var;
|
|
|
|
if (next->value) {
|
|
if (arch) {
|
|
while (next->next) {
|
|
if (next->arch && !strcmp(next->arch, arch)) break;
|
|
next = next->next;
|
|
}
|
|
} else {
|
|
while (next->next) {
|
|
if (!next->arch) break;
|
|
next = next->next;
|
|
}
|
|
}
|
|
|
|
/*@-nullpass@*/ /* LCL: arch != NULL here. */
|
|
if (next->arch && arch && !strcmp(next->arch, arch)) {
|
|
/*@=nullpass@*/
|
|
next->value = _free(next->value);
|
|
next->arch = _free(next->arch);
|
|
} else if (next->arch || arch) {
|
|
next->next = xmalloc(sizeof(*next->next));
|
|
next = next->next;
|
|
next->value = NULL;
|
|
next->arch = NULL;
|
|
next->next = NULL;
|
|
}
|
|
}
|
|
|
|
next->value = xstrdup(val); /* XXX memory leak, hard to plug */
|
|
next->arch = (arch ? xstrdup(arch) : NULL);
|
|
}
|
|
|
|
void rpmSetTables(int archTable, int osTable)
|
|
{
|
|
const char * arch, * os;
|
|
|
|
defaultMachine(&arch, &os);
|
|
|
|
if (currTables[ARCH] != archTable) {
|
|
currTables[ARCH] = archTable;
|
|
rebuildCompatTables(ARCH, arch);
|
|
}
|
|
|
|
if (currTables[OS] != osTable) {
|
|
currTables[OS] = osTable;
|
|
rebuildCompatTables(OS, os);
|
|
}
|
|
}
|
|
|
|
int rpmMachineScore(int type, const char * name)
|
|
{
|
|
machEquivInfo info = machEquivSearch(&tables[type].equiv, name);
|
|
return (info != NULL ? info->score : 0);
|
|
}
|
|
|
|
void rpmGetMachine(const char ** arch, const char ** os)
|
|
{
|
|
if (arch)
|
|
*arch = current[ARCH];
|
|
|
|
if (os)
|
|
*os = current[OS];
|
|
}
|
|
|
|
void rpmSetMachine(const char * arch, const char * os)
|
|
{
|
|
const char * host_cpu, * host_os;
|
|
|
|
defaultMachine(&host_cpu, &host_os);
|
|
|
|
if (arch == NULL) {
|
|
arch = host_cpu;
|
|
if (tables[currTables[ARCH]].hasTranslate)
|
|
arch = lookupInDefaultTable(arch,
|
|
tables[currTables[ARCH]].defaults,
|
|
tables[currTables[ARCH]].defaultsLength);
|
|
}
|
|
if (arch == NULL) return; /* XXX can't happen */
|
|
|
|
if (os == NULL) {
|
|
os = host_os;
|
|
if (tables[currTables[OS]].hasTranslate)
|
|
os = lookupInDefaultTable(os,
|
|
tables[currTables[OS]].defaults,
|
|
tables[currTables[OS]].defaultsLength);
|
|
}
|
|
if (os == NULL) return; /* XXX can't happen */
|
|
|
|
if (!current[ARCH] || strcmp(arch, current[ARCH])) {
|
|
current[ARCH] = _free(current[ARCH]);
|
|
current[ARCH] = xstrdup(arch);
|
|
rebuildCompatTables(ARCH, host_cpu);
|
|
}
|
|
|
|
if (!current[OS] || strcmp(os, current[OS])) {
|
|
char * t = xstrdup(os);
|
|
current[OS] = _free(current[OS]);
|
|
/*
|
|
* XXX Capitalizing the 'L' is needed to insure that old
|
|
* XXX os-from-uname (e.g. "Linux") is compatible with the new
|
|
* XXX os-from-platform (e.g "linux" from "sparc-*-linux").
|
|
* XXX A copy of this string is embedded in headers and is
|
|
* XXX used by rpmInstallPackage->{os,arch}Okay->rpmMachineScore->
|
|
* XXX to verify correct arch/os from headers.
|
|
*/
|
|
if (!strcmp(t, "linux"))
|
|
*t = 'L';
|
|
current[OS] = t;
|
|
|
|
rebuildCompatTables(OS, host_os);
|
|
}
|
|
}
|
|
|
|
static void rebuildCompatTables(int type, const char * name)
|
|
/*@*/
|
|
{
|
|
machFindEquivs(&tables[currTables[type]].cache,
|
|
&tables[currTables[type]].equiv,
|
|
name);
|
|
}
|
|
|
|
static void getMachineInfo(int type, /*@null@*/ /*@out@*/ const char ** name,
|
|
/*@null@*/ /*@out@*/int * num)
|
|
/*@modifies *name, *num @*/
|
|
{
|
|
canonEntry canon;
|
|
int which = currTables[type];
|
|
|
|
/* use the normal canon tables, even if we're looking up build stuff */
|
|
if (which >= 2) which -= 2;
|
|
|
|
canon = lookupInCanonTable(current[type],
|
|
tables[which].canons,
|
|
tables[which].canonsLength);
|
|
|
|
if (canon) {
|
|
if (num) *num = canon->num;
|
|
if (name) *name = canon->short_name;
|
|
} else {
|
|
if (num) *num = 255;
|
|
if (name) *name = current[type];
|
|
|
|
if (tables[currTables[type]].hasCanon) {
|
|
rpmMessage(RPMMESS_WARNING, _("Unknown system: %s\n"), current[type]);
|
|
rpmMessage(RPMMESS_WARNING, _("Please contact rpm-list@redhat.com\n"));
|
|
}
|
|
}
|
|
}
|
|
|
|
void rpmGetArchInfo(const char ** name, int * num)
|
|
{
|
|
getMachineInfo(ARCH, name, num);
|
|
}
|
|
|
|
void rpmGetOsInfo(const char ** name, int * num)
|
|
{
|
|
getMachineInfo(OS, name, num);
|
|
}
|
|
|
|
static void rpmRebuildTargetVars(const char ** target, const char ** canontarget)
|
|
{
|
|
|
|
char *ca = NULL, *co = NULL, *ct = NULL;
|
|
int x;
|
|
|
|
/* Rebuild the compat table to recalculate the current target arch. */
|
|
|
|
rpmSetMachine(NULL, NULL);
|
|
rpmSetTables(RPM_MACHTABLE_INSTARCH, RPM_MACHTABLE_INSTOS);
|
|
rpmSetTables(RPM_MACHTABLE_BUILDARCH, RPM_MACHTABLE_BUILDOS);
|
|
|
|
/*@-branchstate@*/
|
|
if (target && *target) {
|
|
char *c;
|
|
/* Set arch and os from specified build target */
|
|
ca = xstrdup(*target);
|
|
if ((c = strchr(ca, '-')) != NULL) {
|
|
*c++ = '\0';
|
|
|
|
if ((co = strrchr(c, '-')) == NULL) {
|
|
co = c;
|
|
} else {
|
|
if (!xstrcasecmp(co, "-gnu"))
|
|
*co = '\0';
|
|
if ((co = strrchr(c, '-')) == NULL)
|
|
co = c;
|
|
else
|
|
co++;
|
|
}
|
|
if (co != NULL) co = xstrdup(co);
|
|
}
|
|
} else {
|
|
const char *a = NULL;
|
|
const char *o = NULL;
|
|
/* Set build target from rpm arch and os */
|
|
rpmGetArchInfo(&a, NULL);
|
|
ca = (a) ? xstrdup(a) : NULL;
|
|
rpmGetOsInfo(&o, NULL);
|
|
co = (o) ? xstrdup(o) : NULL;
|
|
}
|
|
/*@=branchstate@*/
|
|
|
|
/* If still not set, Set target arch/os from default uname(2) values */
|
|
if (ca == NULL) {
|
|
const char *a = NULL;
|
|
defaultMachine(&a, NULL);
|
|
ca = (a) ? xstrdup(a) : NULL;
|
|
}
|
|
for (x = 0; ca[x] != '\0'; x++)
|
|
ca[x] = xtolower(ca[x]);
|
|
|
|
if (co == NULL) {
|
|
const char *o = NULL;
|
|
defaultMachine(NULL, &o);
|
|
co = (o) ? xstrdup(o) : NULL;
|
|
}
|
|
for (x = 0; co[x] != '\0'; x++)
|
|
co[x] = xtolower(co[x]);
|
|
|
|
/* XXX For now, set canonical target to arch-os */
|
|
if (ct == NULL) {
|
|
ct = xmalloc(strlen(ca) + sizeof("-") + strlen(co));
|
|
sprintf(ct, "%s-%s", ca, co);
|
|
}
|
|
|
|
/*
|
|
* XXX All this macro pokery/jiggery could be achieved by doing a delayed
|
|
* rpmInitMacros(NULL, PER-PLATFORM-MACRO-FILE-NAMES);
|
|
*/
|
|
delMacro(NULL, "_target");
|
|
addMacro(NULL, "_target", NULL, ct, RMIL_RPMRC);
|
|
delMacro(NULL, "_target_cpu");
|
|
addMacro(NULL, "_target_cpu", NULL, ca, RMIL_RPMRC);
|
|
delMacro(NULL, "_target_os");
|
|
addMacro(NULL, "_target_os", NULL, co, RMIL_RPMRC);
|
|
/*
|
|
* XXX Make sure that per-arch optflags is initialized correctly.
|
|
*/
|
|
{ const char *optflags = rpmGetVarArch(RPMVAR_OPTFLAGS, ca);
|
|
if (optflags != NULL) {
|
|
delMacro(NULL, "optflags");
|
|
addMacro(NULL, "optflags", NULL, optflags, RMIL_RPMRC);
|
|
}
|
|
}
|
|
|
|
/*@-branchstate@*/
|
|
if (canontarget)
|
|
*canontarget = ct;
|
|
else
|
|
ct = _free(ct);
|
|
/*@=branchstate@*/
|
|
ca = _free(ca);
|
|
/*@-usereleased@*/
|
|
co = _free(co);
|
|
/*@=usereleased@*/
|
|
}
|
|
|
|
void rpmFreeRpmrc(void)
|
|
{
|
|
int i, j, k;
|
|
|
|
for (i = 0; i < RPM_MACHTABLE_COUNT; i++) {
|
|
tableType t;
|
|
t = tables + i;
|
|
if (t->equiv.list) {
|
|
for (j = 0; j < t->equiv.count; j++)
|
|
t->equiv.list[j].name = _free(t->equiv.list[j].name);
|
|
t->equiv.list = _free(t->equiv.list);
|
|
t->equiv.count = 0;
|
|
}
|
|
if (t->cache.cache) {
|
|
for (j = 0; j < t->cache.size; j++) {
|
|
machCacheEntry e;
|
|
e = t->cache.cache + j;
|
|
if (e == NULL)
|
|
/*@innercontinue@*/ continue;
|
|
e->name = _free(e->name);
|
|
if (e->equivs) {
|
|
for (k = 0; k < e->count; k++)
|
|
e->equivs[k] = _free(e->equivs[k]);
|
|
e->equivs = _free(e->equivs);
|
|
}
|
|
}
|
|
t->cache.cache = _free(t->cache.cache);
|
|
t->cache.size = 0;
|
|
}
|
|
if (t->defaults) {
|
|
for (j = 0; j < t->defaultsLength; j++) {
|
|
t->defaults[j].name = _free(t->defaults[j].name);
|
|
t->defaults[j].defName = _free(t->defaults[j].defName);
|
|
}
|
|
t->defaults = _free(t->defaults);
|
|
t->defaultsLength = 0;
|
|
}
|
|
if (t->canons) {
|
|
for (j = 0; j < t->canonsLength; j++) {
|
|
t->canons[j].name = _free(t->canons[j].name);
|
|
t->canons[j].short_name = _free(t->canons[j].short_name);
|
|
}
|
|
t->canons = _free(t->canons);
|
|
t->canonsLength = 0;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < RPMVAR_NUM; i++) {
|
|
/*@only@*/ /*@null@*/ struct rpmvarValue * vp;
|
|
while ((vp = values[i].next) != NULL) {
|
|
values[i].next = vp->next;
|
|
vp->value = _free(vp->value);
|
|
vp->arch = _free(vp->arch);
|
|
vp = _free(vp);
|
|
}
|
|
values[i].value = _free(values[i].value);
|
|
values[i].arch = _free(values[i].arch);
|
|
}
|
|
current[OS] = _free(current[OS]);
|
|
current[ARCH] = _free(current[ARCH]);
|
|
defaultsInitialized = 0;
|
|
/*@-nullstate@*/ /* FIX: current may be NULL */
|
|
return;
|
|
/*@=nullstate@*/
|
|
}
|
|
|
|
/** \ingroup rpmrc
|
|
* Read rpmrc (and macro) configuration file(s).
|
|
* @param rcfiles colon separated files to read (NULL uses default)
|
|
* @return 0 on succes
|
|
*/
|
|
static int rpmReadRC(/*@null@*/ const char * rcfiles)
|
|
/*@globals rpmGlobalMacroContext, rpmCLIMacroContext,
|
|
fileSystem, internalState @*/
|
|
/*@modifies rpmGlobalMacroContext,
|
|
fileSystem, internalState @*/
|
|
{
|
|
char *myrcfiles, *r, *re;
|
|
int rc;
|
|
|
|
rpmSetDefaults();
|
|
|
|
if (rcfiles == NULL)
|
|
rcfiles = defrcfiles;
|
|
|
|
/* Read each file in rcfiles. */
|
|
rc = 0;
|
|
for (r = myrcfiles = xstrdup(rcfiles); r && *r != '\0'; r = re) {
|
|
char fn[4096];
|
|
FD_t fd;
|
|
|
|
/* Get pointer to rest of files */
|
|
for (re = r; (re = strchr(re, ':')) != NULL; re++) {
|
|
if (!(re[1] == '/' && re[2] == '/'))
|
|
/*@innerbreak@*/ break;
|
|
}
|
|
if (re && *re == ':')
|
|
*re++ = '\0';
|
|
else
|
|
re = r + strlen(r);
|
|
|
|
/* Expand ~/ to $HOME/ */
|
|
fn[0] = '\0';
|
|
if (r[0] == '~' && r[1] == '/') {
|
|
const char * home = getenv("HOME");
|
|
if (home == NULL) {
|
|
/* XXX Only /usr/lib/rpm/rpmrc must exist in default rcfiles list */
|
|
if (rcfiles == defrcfiles && myrcfiles != r)
|
|
continue;
|
|
rpmError(RPMERR_RPMRC, _("Cannot expand %s\n"), r);
|
|
rc = 1;
|
|
break;
|
|
}
|
|
if (strlen(home) > (sizeof(fn) - strlen(r))) {
|
|
rpmError(RPMERR_RPMRC, _("Cannot read %s, HOME is too large.\n"),
|
|
r);
|
|
rc = 1;
|
|
break;
|
|
}
|
|
strcpy(fn, home);
|
|
r++;
|
|
}
|
|
strncat(fn, r, sizeof(fn) - (strlen(fn) + 1));
|
|
fn[sizeof(fn)-1] = '\0';
|
|
|
|
/* Read another rcfile */
|
|
fd = Fopen(fn, "r.fpio");
|
|
if (fd == NULL || Ferror(fd)) {
|
|
/* XXX Only /usr/lib/rpm/rpmrc must exist in default rcfiles list */
|
|
if (rcfiles == defrcfiles && myrcfiles != r)
|
|
continue;
|
|
rpmError(RPMERR_RPMRC, _("Unable to open %s for reading: %s.\n"),
|
|
fn, Fstrerror(fd));
|
|
rc = 1;
|
|
break;
|
|
} else {
|
|
rc = doReadRC(fd, fn);
|
|
}
|
|
if (rc) break;
|
|
}
|
|
myrcfiles = _free(myrcfiles);
|
|
if (rc)
|
|
return rc;
|
|
|
|
rpmSetMachine(NULL, NULL); /* XXX WTFO? Why bother? */
|
|
|
|
{ const char *mfpath;
|
|
/*@-branchstate@*/
|
|
if ((mfpath = rpmGetVar(RPMVAR_MACROFILES)) != NULL) {
|
|
mfpath = xstrdup(mfpath);
|
|
rpmInitMacros(NULL, mfpath);
|
|
mfpath = _free(mfpath);
|
|
}
|
|
/*@=branchstate@*/
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
int rpmReadConfigFiles(const char * file, const char * target)
|
|
{
|
|
|
|
/* Preset target macros */
|
|
/*@-nullstate@*/ /* FIX: target can be NULL */
|
|
rpmRebuildTargetVars(&target, NULL);
|
|
|
|
/* Read the files */
|
|
if (rpmReadRC(file)) return -1;
|
|
|
|
/* Reset target macros */
|
|
rpmRebuildTargetVars(&target, NULL);
|
|
/*@=nullstate@*/
|
|
|
|
/* Finally set target platform */
|
|
{ const char *cpu = rpmExpand("%{_target_cpu}", NULL);
|
|
const char *os = rpmExpand("%{_target_os}", NULL);
|
|
rpmSetMachine(cpu, os);
|
|
cpu = _free(cpu);
|
|
os = _free(os);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int rpmShowRC(FILE * fp)
|
|
{
|
|
struct rpmOption *opt;
|
|
int i;
|
|
machEquivTable equivTable;
|
|
|
|
/* the caller may set the build arch which should be printed here */
|
|
fprintf(fp, "ARCHITECTURE AND OS:\n");
|
|
fprintf(fp, "build arch : %s\n", current[ARCH]);
|
|
|
|
fprintf(fp, "compatible build archs:");
|
|
equivTable = &tables[RPM_MACHTABLE_BUILDARCH].equiv;
|
|
for (i = 0; i < equivTable->count; i++)
|
|
fprintf(fp," %s", equivTable->list[i].name);
|
|
fprintf(fp, "\n");
|
|
|
|
fprintf(fp, "build os : %s\n", current[OS]);
|
|
|
|
fprintf(fp, "compatible build os's :");
|
|
equivTable = &tables[RPM_MACHTABLE_BUILDOS].equiv;
|
|
for (i = 0; i < equivTable->count; i++)
|
|
fprintf(fp," %s", equivTable->list[i].name);
|
|
fprintf(fp, "\n");
|
|
|
|
rpmSetTables(RPM_MACHTABLE_INSTARCH, RPM_MACHTABLE_INSTOS);
|
|
rpmSetMachine(NULL, NULL); /* XXX WTFO? Why bother? */
|
|
|
|
fprintf(fp, "install arch : %s\n", current[ARCH]);
|
|
fprintf(fp, "install os : %s\n", current[OS]);
|
|
|
|
fprintf(fp, "compatible archs :");
|
|
equivTable = &tables[RPM_MACHTABLE_INSTARCH].equiv;
|
|
for (i = 0; i < equivTable->count; i++)
|
|
fprintf(fp," %s", equivTable->list[i].name);
|
|
fprintf(fp, "\n");
|
|
|
|
fprintf(fp, "compatible os's :");
|
|
equivTable = &tables[RPM_MACHTABLE_INSTOS].equiv;
|
|
for (i = 0; i < equivTable->count; i++)
|
|
fprintf(fp," %s", equivTable->list[i].name);
|
|
fprintf(fp, "\n");
|
|
|
|
fprintf(fp, "\nRPMRC VALUES:\n");
|
|
for (i = 0, opt = optionTable; i < optionTableSize; i++, opt++) {
|
|
const char *s = rpmGetVar(opt->var);
|
|
if (s != NULL || rpmIsVerbose())
|
|
fprintf(fp, "%-21s : %s\n", opt->name, s ? s : "(not set)");
|
|
}
|
|
fprintf(fp, "\n");
|
|
|
|
fprintf(fp, "Features supported by rpmlib:\n");
|
|
rpmShowRpmlibProvides(fp);
|
|
fprintf(fp, "\n");
|
|
|
|
rpmDumpMacroTable(NULL, fp);
|
|
|
|
return 0;
|
|
}
|
|
/*@=mods@*/
|