blob: ba5de6d678f71d18fbb2476dced4f237d3d9b28b [file] [log] [blame]
/*
* The PCI Library -- ID to Name Cache
*
* Copyright (c) 2008--2009 Martin Mares <mj@ucw.cz>
*
* Can be freely distributed and used under the terms of the GNU GPL.
*/
#include "internal.h"
#include "names.h"
#ifdef PCI_USE_DNS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <pwd.h>
#include <unistd.h>
static const char cache_version[] = "#PCI-CACHE-1.0";
static char *get_cache_name(struct pci_access *a)
{
char *name, *buf;
name = pci_get_param(a, "net.cache_name");
if (!name || !name[0])
return NULL;
if (strncmp(name, "~/", 2))
return name;
uid_t uid = getuid();
struct passwd *pw = getpwuid(uid);
if (!pw)
return name;
buf = pci_malloc(a, strlen(pw->pw_dir) + strlen(name+1) + 1);
sprintf(buf, "%s%s", pw->pw_dir, name+1);
pci_set_param_internal(a, "net.cache_name", buf, 1);
pci_mfree(buf);
return pci_get_param(a, "net.cache_name");
}
int
pci_id_cache_load(struct pci_access *a, int flags)
{
char *name;
char line[MAX_LINE];
FILE *f;
int lino;
a->id_cache_status = 1;
name = get_cache_name(a);
if (!name)
return 0;
a->debug("Using cache %s\n", name);
if (flags & PCI_LOOKUP_REFRESH_CACHE)
{
a->debug("Not loading cache, will refresh everything\n");
a->id_cache_status = 2;
return 0;
}
f = fopen(name, "rb");
if (!f)
{
a->debug("Cache file does not exist\n");
return 0;
}
/* FIXME: Compare timestamp with the pci.ids file? */
lino = 0;
while (fgets(line, sizeof(line), f))
{
char *p = strchr(line, '\n');
lino++;
if (p)
{
*p = 0;
if (lino == 1)
{
if (strcmp(line, cache_version))
{
a->debug("Unrecognized cache version %s, ignoring\n", line);
break;
}
continue;
}
else
{
int cat, id1, id2, id3, id4, cnt;
if (sscanf(line, "%d%x%x%x%x%n", &cat, &id1, &id2, &id3, &id4, &cnt) >= 5)
{
p = line + cnt;
while (*p && *p == ' ')
p++;
pci_id_insert(a, cat, id1, id2, id3, id4, p, SRC_CACHE);
continue;
}
}
}
a->warning("Malformed cache file %s (line %d), ignoring", name, lino);
break;
}
if (ferror(f))
a->warning("Error while reading %s", name);
fclose(f);
return 1;
}
void
pci_id_cache_flush(struct pci_access *a)
{
int orig_status = a->id_cache_status;
FILE *f;
unsigned int h;
struct id_entry *e, *e2;
char hostname[256], *tmpname, *name;
int this_pid;
a->id_cache_status = 0;
if (orig_status < 2)
return;
name = get_cache_name(a);
if (!name)
return;
this_pid = getpid();
if (gethostname(hostname, sizeof(hostname)) < 0)
hostname[0] = 0;
else
hostname[sizeof(hostname)-1] = 0;
tmpname = pci_malloc(a, strlen(name) + strlen(hostname) + 64);
sprintf(tmpname, "%s.tmp-%s-%d", name, hostname, this_pid);
f = fopen(tmpname, "wb");
if (!f)
{
a->warning("Cannot write to %s: %s", name, strerror(errno));
pci_mfree(tmpname);
return;
}
a->debug("Writing cache to %s\n", name);
fprintf(f, "%s\n", cache_version);
for (h=0; h<HASH_SIZE; h++)
for (e=a->id_hash[h]; e; e=e->next)
if (e->src == SRC_CACHE || e->src == SRC_NET)
{
/* Negative entries are not written */
if (!e->name[0])
continue;
/* Verify that every entry is written at most once */
for (e2=a->id_hash[h]; e2 != e; e2=e2->next)
if ((e2->src == SRC_CACHE || e2->src == SRC_NET) &&
e2->cat == e->cat &&
e2->id12 == e->id12 && e2->id34 == e->id34)
break;
if (e2 == e)
fprintf(f, "%d %x %x %x %x %s\n",
e->cat,
pair_first(e->id12), pair_second(e->id12),
pair_first(e->id34), pair_second(e->id34),
e->name);
}
fflush(f);
if (ferror(f))
a->warning("Error writing %s", name);
fclose(f);
if (rename(tmpname, name) < 0)
{
a->warning("Cannot rename %s to %s: %s", tmpname, name, strerror(errno));
unlink(tmpname);
}
pci_mfree(tmpname);
}
#else
int pci_id_cache_load(struct pci_access *a UNUSED, int flags UNUSED)
{
a->id_cache_status = 1;
return 0;
}
void pci_id_cache_flush(struct pci_access *a)
{
a->id_cache_status = 0;
}
#endif
void
pci_id_cache_dirty(struct pci_access *a)
{
if (a->id_cache_status >= 1)
a->id_cache_status = 2;
}