nasm/rdoff/rdlib.c

291 lines
8.0 KiB
C
Raw Normal View History

/* ----------------------------------------------------------------------- *
*
* Copyright 1996-2014 The NASM Authors - All Rights Reserved
* See the file AUTHORS included with the NASM distribution for
* the specific copyright holders.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following
* conditions are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* ----------------------------------------------------------------------- */
2002-12-04 08:49:37 +08:00
/*
* rdlib.c - routines for manipulating RDOFF libraries (.rdl)
*/
2005-01-16 06:15:51 +08:00
#include "compiler.h"
2002-05-01 04:52:08 +08:00
#include <stdio.h>
#include <stdlib.h>
2002-05-01 04:58:18 +08:00
#include <string.h>
2002-05-01 04:52:08 +08:00
#include "rdfutils.h"
2002-05-01 04:52:08 +08:00
#include "rdlib.h"
2002-12-04 08:49:37 +08:00
#include "rdlar.h"
2002-05-01 04:52:08 +08:00
2002-12-04 08:49:37 +08:00
/* See Texinfo documentation about new RDOFF libraries format */
2002-05-01 04:58:18 +08:00
2002-05-01 04:52:08 +08:00
int rdl_error = 0;
char *rdl_errors[5] = {
2005-01-16 06:15:51 +08:00
"no error", "could not open file", "invalid file structure",
2002-05-01 04:58:18 +08:00
"file contains modules of an unsupported RDOFF version",
"module not found"
2002-05-01 04:52:08 +08:00
};
2002-12-04 08:49:37 +08:00
int rdl_verify(const char *filename)
2002-05-01 04:58:18 +08:00
{
FILE *fp;
char buf[257];
2005-01-16 06:15:51 +08:00
int i;
int32_t length;
static char lastverified[256];
2002-05-01 04:58:18 +08:00
static int lastresult = -1;
if (lastresult != -1 && !strcmp(filename, lastverified))
2005-01-16 06:15:51 +08:00
return lastresult;
2002-05-01 04:58:18 +08:00
fp = fopen(filename, "rb");
2002-05-01 04:58:18 +08:00
strcpy(lastverified, filename);
if (!fp)
2005-01-16 06:15:51 +08:00
return (rdl_error = lastresult = 1);
while (!feof(fp)) {
i = 0;
while (fread(buf + i, 1, 1, fp) == 1 && i < 257 && buf[i])
2005-01-16 06:15:51 +08:00
i++;
if (feof(fp))
break;
if (buf[0] == '.') {
/*
* A special module, eg a signature block or a directory.
* Format of such a module is defined to be:
* six char type identifier
* int32_t count bytes content
2005-01-16 06:15:51 +08:00
* content
* so we can handle it uniformaly with RDOFF2 modules.
*/
nasm_read(buf, 6, fp);
2005-01-16 06:15:51 +08:00
buf[6] = 0;
/* Currently, nothing useful to do with signature block.. */
} else {
nasm_read(buf, 6, fp);
2005-01-16 06:15:51 +08:00
buf[6] = 0;
if (strncmp(buf, "RDOFF", 5)) {
fclose(fp);
2005-01-16 06:15:51 +08:00
return rdl_error = lastresult = 2;
} else if (buf[5] != '2') {
fclose(fp);
2005-01-16 06:15:51 +08:00
return rdl_error = lastresult = 3;
}
}
nasm_read(&length, 4, fp);
2005-01-16 06:15:51 +08:00
fseek(fp, length, SEEK_CUR); /* skip over the module */
2002-05-01 04:58:18 +08:00
}
fclose(fp);
2005-01-16 06:15:51 +08:00
return lastresult = 0; /* library in correct format */
2002-05-01 04:58:18 +08:00
}
int rdl_open(struct librarynode *lib, const char *name)
2002-05-01 04:58:18 +08:00
{
int i = rdl_verify(name);
2005-01-16 06:15:51 +08:00
if (i)
return i;
2002-05-01 04:58:18 +08:00
lib->fp = NULL;
lib->name = nasm_strdup(name);
2002-05-01 04:58:18 +08:00
lib->referenced = 0;
lib->next = NULL;
return 0;
}
int rdl_searchlib(struct librarynode *lib, const char *label, rdffile * f)
2002-05-01 04:52:08 +08:00
{
char buf[512];
2005-01-16 06:15:51 +08:00
int i, t;
void *hdr;
rdfheaderrec *r;
int32_t l;
2002-05-01 04:52:08 +08:00
rdl_error = 0;
2005-01-16 06:15:51 +08:00
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)) {
/*
* read the module name from the file, and prepend
2005-01-16 06:15:51 +08:00
* the library name and '.' to it.
*/
strcpy(buf, lib->name);
i = strlen(lib->name);
buf[i++] = '.';
t = i;
while (fread(buf + i, 1, 1, lib->fp) == 1 && i < 512 && buf[i])
2005-01-16 06:15:51 +08:00
i++;
buf[i] = 0;
if (feof(lib->fp))
break;
if (!strcmp(buf + t, ".dir")) { /* skip over directory */
nasm_read(&l, 4, lib->fp);
2005-01-16 06:15:51 +08:00
fseek(lib->fp, l, SEEK_CUR);
continue;
}
/*
* open the RDOFF module
*/
if (rdfopenhere(f, lib->fp, &lib->referenced, buf)) {
rdl_error = 16 * rdf_errno;
return 0;
}
/*
* read in the header, and scan for exported symbols
*/
hdr = nasm_malloc(f->header_len);
2005-01-16 06:15:51 +08:00
rdfloadseg(f, RDOFF_HEADER, hdr);
while ((r = rdfgetheaderrec(f))) {
if (r->type != 3) /* not an export */
continue;
if (!strcmp(r->e.label, label)) { /* match! */
nasm_free(hdr); /* reset to 'just open' */
2005-01-16 06:15:51 +08:00
f->header_loc = NULL; /* state... */
f->header_fp = 0;
return 1;
}
}
/* find start of next module... */
i = f->eof_offset;
rdfclose(f);
fseek(lib->fp, i, SEEK_SET);
2002-05-01 04:52:08 +08:00
}
2002-05-01 04:58:18 +08:00
/*
* close the file if nobody else is using it
*/
2005-01-16 06:15:51 +08:00
lib->referenced--;
if (!lib->referenced) {
fclose(lib->fp);
lib->fp = NULL;
2002-05-01 04:52:08 +08:00
}
return 0;
}
2005-01-16 06:15:51 +08:00
int rdl_openmodule(struct librarynode *lib, int moduleno, rdffile * f)
2002-05-01 04:58:18 +08:00
{
char buf[512];
2005-01-16 06:15:51 +08:00
int i, cmod, t;
int32_t length;
2002-05-01 04:58:18 +08:00
lib->referenced++;
2005-01-16 06:15:51 +08:00
if (!lib->fp) {
lib->fp = fopen(lib->name, "rb");
if (!lib->fp) {
lib->referenced--;
return (rdl_error = 1);
}
} else
rewind(lib->fp);
2002-05-01 04:58:18 +08:00
cmod = -1;
2005-01-16 06:15:51 +08:00
while (!feof(lib->fp)) {
strcpy(buf, lib->name);
i = strlen(buf);
buf[i++] = '.';
t = i;
while (fread(buf + i, 1, 1, lib->fp) == 1 && i < 512 && buf[i])
2005-01-16 06:15:51 +08:00
i++;
buf[i] = 0;
if (feof(lib->fp))
break;
if (buf[t] != '.') /* special module - not counted in the numbering */
cmod++; /* of RDOFF modules - must be referred to by name */
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;
}
nasm_read(buf, 6, lib->fp);
2005-01-16 06:15:51 +08:00
buf[6] = 0;
if (buf[t] == '.') {
/* do nothing */
} 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;
}
nasm_read(&length, 4, lib->fp);
2005-01-16 06:15:51 +08:00
fseek(lib->fp, length, SEEK_CUR); /* skip over the module */
2002-05-01 04:58:18 +08:00
}
2005-01-16 06:15:51 +08:00
if (!--lib->referenced) {
fclose(lib->fp);
lib->fp = NULL;
2002-05-01 04:58:18 +08:00
}
2005-01-16 06:15:51 +08:00
return rdl_error = 4; /* module not found */
2002-05-01 04:58:18 +08:00
}
void rdl_perror(const char *apname, const char *filename)
2002-05-01 04:52:08 +08:00
{
2002-05-01 04:58:18 +08:00
if (rdl_error >= 16)
2005-01-16 06:15:51 +08:00
rdfperror(apname, filename);
2002-05-01 04:58:18 +08:00
else
2005-01-16 06:15:51 +08:00
fprintf(stderr, "%s:%s:%s\n", apname, filename,
rdl_errors[rdl_error]);
2002-05-01 04:52:08 +08:00
}