#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define RDOFF_UTILS
#include "rdoff.h"
#include "rdlib.h"
#include "rdlar.h"
int rdl_error = 0;
char *rdl_errors[5] = {
"no error", "could not open file", "invalid file structure",
"file contains modules of an unsupported RDOFF version",
"module not found"
};
int rdl_verify(const char *filename)
{
FILE *fp = fopen(filename, "rb");
char buf[257];
int i;
long length;
static char lastverified[256];
static int lastresult = -1;
if (lastresult != -1 && !strcmp(filename, lastverified))
return lastresult;
strcpy(lastverified, filename);
if (!fp)
return (rdl_error = lastresult = 1);
while (!feof(fp)) {
i = 0;
while (fread(buf + i, 1, 1, fp) == 1 && buf[i] && i < 257)
i++;
if (feof(fp))
break;
if (buf[0] == '.') {
fread(buf, 6, 1, fp);
buf[6] = 0;
} else {
fread(buf, 6, 1, fp);
buf[6] = 0;
if (strncmp(buf, "RDOFF", 5)) {
return rdl_error = lastresult = 2;
} else if (buf[5] != '2') {
return rdl_error = lastresult = 3;
}
}
fread(&length, 4, 1, fp);
fseek(fp, length, SEEK_CUR);
}
fclose(fp);
return lastresult = 0;
}
int rdl_open(struct librarynode *lib, const char *name)
{
int i = rdl_verify(name);
if (i)
return i;
lib->fp = NULL;
lib->name = strdup(name);
lib->referenced = 0;
lib->next = NULL;
return 0;
}
void rdl_close(struct librarynode *lib)
{
if (lib->fp)
fclose(lib->fp);
free(lib->name);
}
int rdl_searchlib(struct librarynode *lib, const char *label, rdffile * f)
{
char buf[512];
int i, t;
void *hdr;
rdfheaderrec *r;
long l;
rdl_error = 0;
lib->referenced++;
if (!lib->fp) {
lib->fp = fopen(lib->name, "rb");
if (!lib->fp) {
rdl_error = 1;
return 0;
}
} else
rewind(lib->fp);
while (!feof(lib->fp)) {
strcpy(buf, lib->name);
i = strlen(lib->name);
buf[i++] = '.';
t = i;
while (fread(buf + i, 1, 1, lib->fp) == 1 && buf[i] && i < 512)
i++;
buf[i] = 0;
if (feof(lib->fp))
break;
if (!strcmp(buf + t, ".dir")) {
fread(&l, 4, 1, lib->fp);
fseek(lib->fp, l, SEEK_CUR);
continue;
}
if (rdfopenhere(f, lib->fp, &lib->referenced, buf)) {
rdl_error = 16 * rdf_errno;
return 0;
}
hdr = malloc(f->header_len);
rdfloadseg(f, RDOFF_HEADER, hdr);
while ((r = rdfgetheaderrec(f))) {
if (r->type != 3)
continue;
if (!strcmp(r->e.label, label)) {
free(hdr);
f->header_loc = NULL;
f->header_fp = 0;
return 1;
}
}
i = f->eof_offset;
rdfclose(f);
fseek(lib->fp, i, SEEK_SET);
}
lib->referenced--;
if (!lib->referenced) {
fclose(lib->fp);
lib->fp = NULL;
}
return 0;
}
int rdl_openmodule(struct librarynode *lib, int moduleno, rdffile * f)
{
char buf[512];
int i, cmod, t;
long length;
lib->referenced++;
if (!lib->fp) {
lib->fp = fopen(lib->name, "rb");
if (!lib->fp) {
lib->referenced--;
return (rdl_error = 1);
}
} else
rewind(lib->fp);
cmod = -1;
while (!feof(lib->fp)) {
strcpy(buf, lib->name);
i = strlen(buf);
buf[i++] = '.';
t = i;
while (fread(buf + i, 1, 1, lib->fp) == 1 && buf[i] && i < 512)
i++;
buf[i] = 0;
if (feof(lib->fp))
break;
if (buf[t] != '.')
cmod++;
if (cmod == moduleno) {
rdl_error = 16 *
rdfopenhere(f, lib->fp, &lib->referenced, buf);
lib->referenced--;
if (!lib->referenced) {
fclose(lib->fp);
lib->fp = NULL;
}
return rdl_error;
}
fread(buf, 6, 1, lib->fp);
buf[6] = 0;
if (buf[t] == '.') {
} else if (strncmp(buf, "RDOFF", 5)) {
if (!--lib->referenced) {
fclose(lib->fp);
lib->fp = NULL;
}
return rdl_error = 2;
} else if (buf[5] != '2') {
if (!--lib->referenced) {
fclose(lib->fp);
lib->fp = NULL;
}
return rdl_error = 3;
}
fread(&length, 4, 1, lib->fp);
fseek(lib->fp, length, SEEK_CUR);
}
if (!--lib->referenced) {
fclose(lib->fp);
lib->fp = NULL;
}
return rdl_error = 4;
}
void rdl_perror(const char *apname, const char *filename)
{
if (rdl_error >= 16)
rdfperror(apname, filename);
else
fprintf(stderr, "%s:%s:%s\n", apname, filename,
rdl_errors[rdl_error]);
}